1
0
mirror of https://github.com/squidfunk/mkdocs-material.git synced 2025-01-12 14:12:13 +01:00

113 lines
3.4 KiB
TypeScript
Raw Normal View History

/*
* Copyright (c) 2016-2020 Martin Donath <martin.donath@squidfunk.com>
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to
* deal in the Software without restriction, including without limitation the
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
* sell copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
* IN THE SOFTWARE.
*/
import { identity } from "ramda"
2020-02-19 08:57:36 +01:00
import { Observable, animationFrameScheduler, fromEvent, merge, of } from "rxjs"
import {
filter,
map,
2020-02-19 08:57:36 +01:00
observeOn,
shareReplay,
2020-02-19 08:57:36 +01:00
switchMap,
switchMapTo,
tap
} from "rxjs/operators"
2020-02-19 08:57:36 +01:00
import { getElement, getElements, watchMedia } from "observables"
/* ----------------------------------------------------------------------------
* Helper types
* ------------------------------------------------------------------------- */
/**
* Mount options
*/
interface MountOptions {
document$: Observable<Document> /* Document observable */
2020-02-19 08:57:36 +01:00
hash$: Observable<string> /* Location hash observable */
}
/* ----------------------------------------------------------------------------
* Functions
* ------------------------------------------------------------------------- */
/**
* Patch all `details` elements
*
* This function will ensure that all `details` tags are opened prior to
* printing, so the whole content of the page is included.
*
* @param options - Options
*
* @return Details elements observable
*/
export function patchDetails(
2020-02-19 08:57:36 +01:00
{ document$, hash$ }: MountOptions
): Observable<HTMLDetailsElement[]> {
2020-02-19 08:57:36 +01:00
const els$ = document$
.pipe(
map(() => getElements<HTMLDetailsElement>("details")),
shareReplay(1)
)
/* Open all details before printing */
merge(
watchMedia("print").pipe(filter(identity)), // Webkit
fromEvent(window, "beforeprint") // IE, FF
)
.pipe(
2020-02-19 08:57:36 +01:00
switchMapTo(els$)
)
2020-02-19 08:57:36 +01:00
.subscribe(els => {
for (const el of els)
el.setAttribute("open", "")
})
/* Open details before anchor jump */
merge(hash$, of(location.hash))
.pipe(
filter(hash => !!hash.length),
switchMap(hash => of(getElement<HTMLElement>(hash) || document.body)
.pipe(
map(el => el.closest("details")!),
filter(el => el && !el.open),
/* Open details and temporarily reset anchor */
tap(el => {
el.setAttribute("open", "")
location.hash = ""
}),
/* Defer anchor jump to next animation frame */
observeOn(animationFrameScheduler),
tap(() => {
location.hash = hash
})
)
)
)
.subscribe()
/* Return details elements */
return els$
}