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

103 lines
3.1 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 11:07:34 +01:00
import { NEVER, Observable, fromEvent, merge, of } from "rxjs"
import {
2020-02-19 11:07:34 +01:00
catchError,
filter,
map,
2020-02-19 08:57:36 +01:00
switchMap,
switchMapTo,
} from "rxjs/operators"
2020-02-19 11:07:34 +01:00
import {
getElementOrThrow,
getElements,
getLocationHash,
setLocationHash,
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
*/
export function patchDetails(
2020-02-19 08:57:36 +01:00
{ document$, hash$ }: MountOptions
): void {
2020-02-19 08:57:36 +01:00
const els$ = document$
.pipe(
map(() => getElements<HTMLDetailsElement>("details"))
2020-02-19 08:57:36 +01:00
)
/* 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 parent details before anchor jump */
2020-02-19 11:07:34 +01:00
merge(hash$, of(getLocationHash()))
2020-02-19 08:57:36 +01:00
.pipe(
filter(hash => !!hash.length),
2020-02-19 11:07:34 +01:00
switchMap(hash => of(getElementOrThrow<HTMLElement>(hash))
2020-02-19 08:57:36 +01:00
.pipe(
map(el => [el.closest("details")!, hash] as const),
filter(([el]) => el && !el.open)
2020-02-19 08:57:36 +01:00
)
2020-02-19 11:07:34 +01:00
),
catchError(() => NEVER)
2020-02-19 08:57:36 +01:00
)
2020-02-19 11:07:34 +01:00
.subscribe(([el, hash]) => {
el.setAttribute("open", "")
setLocationHash(hash)
})
}