-
Notifications
You must be signed in to change notification settings - Fork 43
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(helper): add
getHeaders()
(#189)
Co-authored-by: Mister-Hope <mister-hope@outlook.com>
- Loading branch information
1 parent
03646b9
commit 421fece
Showing
4 changed files
with
340 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,139 @@ | ||
export interface Header { | ||
/** | ||
* The level of the header | ||
* | ||
* `1` to `6` for `<h1>` to `<h6>` | ||
*/ | ||
level: number | ||
/** | ||
* The title of the header | ||
*/ | ||
title: string | ||
/** | ||
* The slug of the header | ||
* | ||
* Typically the `id` attr of the header anchor | ||
*/ | ||
slug: string | ||
/** | ||
* Link of the header | ||
* | ||
* Typically using `#${slug}` as the anchor hash | ||
*/ | ||
link: string | ||
/** | ||
* The children of the header | ||
*/ | ||
children: Header[] | ||
} | ||
|
||
export type HeaderLevels = false | number | [number, number] | 'deep' | ||
|
||
export type MenuItem = Omit<Header, 'slug' | 'children'> & { | ||
element: HTMLHeadElement | ||
children?: MenuItem[] | ||
} | ||
|
||
export interface GetHeadersOptions { | ||
/** | ||
* The selector of the headers. | ||
* | ||
* It will be passed as an argument to `document.querySelectorAll(selector)`, | ||
* so you should pass a `CSS Selector` string. | ||
* | ||
* @default '#vp-content h1, #vp-content h2, #vp-content h3, #vp-content h4, #vp-content h5, #vp-content h6' | ||
*/ | ||
selector?: string | ||
/** | ||
* Ignore specific elements within the header. | ||
* | ||
* The Array of `CSS Selector` | ||
* | ||
* @default [] | ||
*/ | ||
ignore?: string[] | ||
/** | ||
* The levels of the headers. | ||
* | ||
* `1` to `6` for `<h1>` to `<h6>` | ||
* | ||
* - `false`: No headers. | ||
* - `number`: only headings of that level will be displayed. | ||
* - `[number, number]: headings level tuple, where the first number should be less than the second number, for example, `[2, 4]` which means all headings from `<h2>` to `<h4>` will be displayed. | ||
* - `deep`: same as `[2, 6]`, which means all headings from `<h2>` to `<h6>` will be displayed. | ||
* | ||
* @default 2 | ||
*/ | ||
levels?: HeaderLevels | ||
} | ||
|
||
export const getHeaders = ({ | ||
selector = [...new Array(6)].map((_, i) => `#vp-content h${i + 1}`).join(','), | ||
levels = 2, | ||
ignore = [], | ||
}: GetHeadersOptions = {}): MenuItem[] => { | ||
const headers = Array.from(document.querySelectorAll(selector)) | ||
.filter((el) => el.id && el.hasChildNodes()) | ||
.map((el) => { | ||
const level = Number(el.tagName[1]) | ||
return { | ||
element: el as HTMLHeadElement, | ||
title: serializeHeader(el, ignore), | ||
link: '#' + el.id, | ||
slug: el.id, | ||
level, | ||
} | ||
}) | ||
return resolveHeaders(headers, levels) | ||
} | ||
|
||
const serializeHeader = (h: Element, ignore: string[] = []): string => { | ||
let text = '' | ||
if (ignore.length) { | ||
const clone = h.cloneNode(true) as Element | ||
clone.querySelectorAll(ignore.join(',')).forEach((el) => el.remove()) | ||
text = clone.textContent || '' | ||
} else { | ||
text = h.textContent || '' | ||
} | ||
return text.trim() | ||
} | ||
|
||
export const resolveHeaders = ( | ||
headers: MenuItem[], | ||
levels: HeaderLevels = 2, | ||
): MenuItem[] => { | ||
if (levels === false) { | ||
return [] | ||
} | ||
|
||
const [high, low]: [number, number] = | ||
typeof levels === 'number' | ||
? [levels, levels] | ||
: levels === 'deep' | ||
? [2, 6] | ||
: levels | ||
|
||
headers = headers.filter((h) => h.level >= high && h.level <= low) | ||
|
||
const res: MenuItem[] = [] | ||
// eslint-disable-next-line no-labels | ||
outer: for (let i = 0; i < headers.length; i++) { | ||
const cur = headers[i] | ||
if (i === 0) { | ||
res.push(cur) | ||
} else { | ||
for (let j = i - 1; j >= 0; j--) { | ||
const prev = headers[j] | ||
if (prev.level < cur.level) { | ||
;(prev.children ??= []).push(cur) | ||
// eslint-disable-next-line no-labels | ||
continue outer | ||
} | ||
} | ||
res.push(cur) | ||
} | ||
} | ||
|
||
return res | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,4 @@ | ||
export * from './data.js' | ||
export * from './hasGlobalComponent.js' | ||
export * from './wait.js' | ||
export * from './getHeaders.js' |