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

96 lines
2.9 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-20 14:44:41 +01:00
import { Observable, fromEvent, merge } from "rxjs"
import {
filter,
map,
switchMapTo,
2020-02-20 14:44:41 +01:00
tap
} from "rxjs/operators"
2020-02-19 11:07:34 +01:00
import {
2020-02-20 14:44:41 +01:00
getElement,
2020-02-19 11:07:34 +01:00
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
*/
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(
2020-02-20 14:44:41 +01:00
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", "")
})
2020-02-20 14:44:41 +01:00
/* Open parent details and fix anchor jump */
hash$
2020-02-19 08:57:36 +01:00
.pipe(
2020-02-20 17:42:46 +01:00
map(id => getElement(`[id="${id}"]`)!),
2020-02-20 14:44:41 +01:00
filter(el => typeof el !== "undefined"),
tap(el => {
const details = el.closest("details")
if (details && !details.open)
details.setAttribute("open", "")
})
2020-02-20 14:44:41 +01:00
)
.subscribe(el => el.scrollIntoView())
}