mokuji.js 4.4.0 → 4.4.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.cjs +2 -2
- package/dist/index.cjs.map +1 -1
- package/dist/index.d.ts +2 -2
- package/dist/index.js +2 -2
- package/dist/index.js.map +1 -1
- package/package.json +1 -1
package/dist/index.cjs
CHANGED
|
@@ -1,8 +1,8 @@
|
|
|
1
1
|
'use strict';
|
|
2
2
|
|
|
3
|
-
var g=(t,n)=>!t||!n?!1:n.contains(t),u=t=>t.querySelectorAll("h1, h2, h3, h4, h5, h6"),h=t=>document.createElement(t);var L=new Set,E=t=>t.replaceAll(/\s+/g,"_"),A=t=>encodeURIComponent(t).replaceAll(/%+/g,"."),T=(t,n="")=>{let e=n,o=1;for(;o<=t.length;){let r=o===1?e:`${e}_${o}`;if(!L.has(r)){e=r,L.add(e);break}o++;}return e},M=(t,n)=>{let e=E(t);return e=e.replaceAll(/&+/g,"").replaceAll(/&+/g,""),n&&(e=A(e)),e};var H="data-mokuji-anchor",k={anchorType:!0,anchorLink:!1,anchorLinkSymbol:"#",anchorLinkBefore:!0,anchorLinkClassName:"",anchorContainerTagName:"ol"},N=t=>{let n=new Map;for(let e=0;e<t.length;e++){let o=t[e].hash.replace("#","");n.set(o,t[e]);}return n},x=(t,n,e)=>{let o=h("a");o.setAttribute(H,""),e.anchorLinkClassName&&o.classList.add(e.anchorLinkClassName);for(let r=0;r<t.length;r++){let c=t[r],l=n.get(c.id);if(!l)continue;let s=o.cloneNode(!1);s.setAttribute("href",l.hash),e.anchorLinkSymbol&&(s.textContent=e.anchorLinkSymbol),e.anchorLinkBefore?c.insertBefore(s,c.firstChild):c.append(s);}},y=(t,n)=>{for(let e=0;e<n.length;e++){let o=n[e],r=o.textContent,c=o.hash,l=t.filter(a=>a.id===r);if(l.length===1)continue;let s=0;for(let a=0;a<l.length;a++){let f=l[a],
|
|
3
|
+
var g=(t,n)=>!t||!n?!1:n.contains(t),u=t=>t.querySelectorAll("h1, h2, h3, h4, h5, h6"),h=t=>document.createElement(t);var L=new Set,E=t=>t.replaceAll(/\s+/g,"_"),A=t=>encodeURIComponent(t).replaceAll(/%+/g,"."),T=(t,n="")=>{let e=n,o=1;for(;o<=t.length;){let r=o===1?e:`${e}_${o}`;if(!L.has(r)){e=r,L.add(e);break}o++;}return e},M=(t,n)=>{let e=E(t);return e=e.replaceAll(/&+/g,"").replaceAll(/&+/g,""),n&&(e=A(e)),e};var H="data-mokuji-anchor",k={anchorType:!0,anchorLink:!1,anchorLinkSymbol:"#",anchorLinkBefore:!0,anchorLinkClassName:"",anchorContainerTagName:"ol"},N=t=>{let n=new Map;for(let e=0;e<t.length;e++){let o=t[e].hash.replace("#","");n.set(o,t[e]);}return n},x=(t,n,e)=>{let o=h("a");o.setAttribute(H,""),e.anchorLinkClassName&&o.classList.add(e.anchorLinkClassName);for(let r=0;r<t.length;r++){let c=t[r],l=n.get(c.id);if(!l)continue;let s=o.cloneNode(!1);s.setAttribute("href",l.hash),e.anchorLinkSymbol&&(s.textContent=e.anchorLinkSymbol),e.anchorLinkBefore?c.insertBefore(s,c.firstChild):c.append(s);}},y=(t,n)=>{for(let e=0;e<n.length;e++){let o=n[e],r=o.textContent,c=o.hash,l=t.filter(a=>a.id===r);if(l.length===1)continue;let s=0;for(let a=0;a<l.length;a++){let f=l[a],m=`${f.id}-${s}`;for(let i=0;i<n.length;i++){let d=n[i];if(d.hash===c){d.href=`#${m}`;break}}f.id=m,s++;}}},C=(t,n,e)=>{let o=0,r=h("li"),c=h("a");for(let l=0;l<t.length;l++){let s=t[l],a=Number(s.tagName[1]);if(o!==0&&o<a){let p=h("ol");n.lastChild.append(p),n=p;}else if(o!==0&&o>a)for(let p=0;p<o-a;p++)g(n,n.parentNode)&&(n=n.parentNode?.parentNode);let f=T(t,s.textContent||""),m=M(f,e);s.id=m;let i=c.cloneNode(!1);i.href=`#${m}`,i.textContent=s.textContent;let d=r.cloneNode(!1);d.append(i),n.append(d),o=a;}},I=(t,n)=>{if(!t)return;let e={...k,...n},o=[...u(t)];if(o.length===0)return;let r=h(e.anchorContainerTagName);C(o,r,e.anchorType);let c=[...r.querySelectorAll("a")];if(c.length!==0){if(y(o,c),e.anchorLink){let l=N(c);x(o,l,e);}return r}},_=()=>{let t=document.querySelectorAll(`[${H}]`);for(let n=t.length-1;n>=0;n--)t[n].remove();};
|
|
4
4
|
|
|
5
|
+
exports.Destroy = _;
|
|
5
6
|
exports.Mokuji = I;
|
|
6
|
-
exports.destory = _;
|
|
7
7
|
//# sourceMappingURL=out.js.map
|
|
8
8
|
//# sourceMappingURL=index.cjs.map
|
package/dist/index.cjs.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/dom.ts","../src/text.ts","../src/index.ts"],"names":["hasParentNode","element","parent","getHeadingsElement","createElement","tagName","storeIds","replaceSpacesWithUnderscores","text","convert2WikipediaStyleAnchor","anchor","censorshipId","headings","textContent","id","suffix_count","tmp_id","generateAnchorText","isConvertToWikipediaStyleAnchor","ANCHOR_DATASET_ATTRIBUTE","defaultOptions","generateAnchorsMap","anchors","anchorMap","i","anchorId","insertAnchorToHeadings","options","a","heading","matchedAnchor","removeDuplicateIds","hash","matchedHeadings","count","j","heading_id","k","generateHierarchyList","elementContainer","number","elementListClone","elementAnchorClone","currentNumber","nextElementOListClone","anchorText","elementAnchor","elementList","Mokuji","externalOptions","anchorsMap","destory","mokujiAnchor"],"mappings":"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","sourcesContent":["/**\n * 親要素内に要素が存在するか判定する\n */\nexport const hasParentNode = (element: Node | null, parent: Node | null) => {\n if (!element || !parent) {\n return false;\n }\n\n return parent.contains(element);\n};\n\n/**\n * 見出し要素をすべて取得する\n */\nexport const getHeadingsElement = (element: Element) => {\n return element.querySelectorAll<HTMLHeadingElement>('h1, h2, h3, h4, h5, h6');\n};\n\n/**\n * 要素を作成する\n */\nexport const createElement = <T extends keyof HTMLElementTagNameMap>(tagName: T): HTMLElementTagNameMap[T] => {\n return document.createElement(tagName);\n};\n","const storeIds = new Set<string>();\n\nconst replaceSpacesWithUnderscores = (text: string) => {\n return text.replaceAll(/\\s+/g, '_');\n};\n\nconst convert2WikipediaStyleAnchor = (anchor: string) => {\n return encodeURIComponent(anchor).replaceAll(/%+/g, '.');\n};\n\nexport const censorshipId = (headings: HTMLHeadingElement[], textContent = '') => {\n let id = textContent;\n let suffix_count = 1;\n\n // IDが重複していた場合はsuffixを付ける\n while (suffix_count <= headings.length) {\n const tmp_id = suffix_count === 1 ? id : `${id}_${suffix_count}`;\n\n if (!storeIds.has(tmp_id)) {\n id = tmp_id;\n storeIds.add(id);\n break;\n }\n\n suffix_count++;\n }\n\n return id;\n};\n\nexport const generateAnchorText = (text: string, isConvertToWikipediaStyleAnchor: boolean) => {\n // convert spaces to _\n let anchor = replaceSpacesWithUnderscores(text);\n\n // remove &\n anchor = anchor.replaceAll(/&+/g, '').replaceAll(/&+/g, '');\n\n if (isConvertToWikipediaStyleAnchor) {\n anchor = convert2WikipediaStyleAnchor(anchor);\n }\n\n return anchor;\n};\n","import { hasParentNode, getHeadingsElement, createElement } from './dom';\nimport type { MokujiOption } from './types';\nimport { censorshipId, generateAnchorText } from './text';\n\nexport { MokujiOption };\n\nconst ANCHOR_DATASET_ATTRIBUTE = 'data-mokuji-anchor';\n\nconst defaultOptions = {\n anchorType: true,\n anchorLink: false,\n anchorLinkSymbol: '#',\n anchorLinkBefore: true,\n anchorLinkClassName: '',\n anchorContainerTagName: 'ol',\n} as const;\n\nconst generateAnchorsMap = (anchors: HTMLAnchorElement[]) => {\n const anchorMap = new Map<string, HTMLAnchorElement>();\n\n for (let i = 0; i < anchors.length; i++) {\n const anchorId = anchors[i].hash.replace('#', '');\n anchorMap.set(anchorId, anchors[i]);\n }\n\n return anchorMap;\n};\n\nconst insertAnchorToHeadings = (\n headings: HTMLHeadingElement[],\n anchorMap: Map<string, HTMLAnchorElement>,\n options: MokujiOption,\n) => {\n const a = createElement('a');\n a.setAttribute(ANCHOR_DATASET_ATTRIBUTE, '');\n\n if (options.anchorLinkClassName) {\n a.classList.add(options.anchorLinkClassName);\n }\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const matchedAnchor = anchorMap.get(heading.id);\n\n if (!matchedAnchor) {\n continue;\n }\n\n // create anchor\n const anchor = a.cloneNode(false) as HTMLAnchorElement;\n anchor.setAttribute('href', matchedAnchor.hash);\n\n if (options.anchorLinkSymbol) {\n anchor.textContent = options.anchorLinkSymbol;\n }\n\n // insert anchor into headings\n if (options.anchorLinkBefore) {\n // before\n heading.insertBefore(anchor, heading.firstChild);\n } else {\n // after\n heading.append(anchor);\n }\n }\n};\n\nconst removeDuplicateIds = (headings: HTMLHeadingElement[], anchors: HTMLAnchorElement[]) => {\n for (let i = 0; i < anchors.length; i++) {\n const anchor = anchors[i];\n const id = anchor.textContent;\n const hash = anchor.hash;\n const matchedHeadings = headings.filter((heading) => heading.id === id);\n\n if (matchedHeadings.length === 1) {\n continue;\n }\n\n // duplicated id\n let count = 0;\n\n for (let j = 0; j < matchedHeadings.length; j++) {\n const heading = matchedHeadings[j];\n const heading_id = `${heading.id}-${count}`;\n\n // search duplicate list\n for (let k = 0; k < anchors.length; k++) {\n const anchor = anchors[k];\n if (anchor.hash === hash) {\n // update hash\n anchor.href = `#${heading_id}`;\n break;\n }\n }\n\n // update id\n heading.id = heading_id;\n count++;\n }\n }\n};\n\nconst generateHierarchyList = (\n headings: HTMLHeadingElement[],\n elementContainer: HTMLUListElement | HTMLOListElement,\n isConvertToWikipediaStyleAnchor: boolean,\n) => {\n let number = 0;\n const elementListClone = createElement('li');\n const elementAnchorClone = createElement('a');\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const currentNumber = Number(heading.tagName[1]);\n\n // check list hierarchy\n if (number !== 0 && number < currentNumber) {\n // number of the heading is large (small as heading)\n const nextElementOListClone = createElement('ol');\n // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n // @ts-ignore\n elementContainer.lastChild.append(nextElementOListClone);\n elementContainer = nextElementOListClone;\n } else if (number !== 0 && number > currentNumber) {\n // number of heading is small (large as heading)\n for (let i = 0; i < number - currentNumber; i++) {\n if (hasParentNode(elementContainer, elementContainer.parentNode)) {\n elementContainer = elementContainer.parentNode?.parentNode as HTMLUListElement | HTMLOListElement;\n }\n }\n }\n\n const textContent = censorshipId(headings, heading.textContent || '');\n\n // headingへidを付与\n const anchorText = generateAnchorText(textContent, isConvertToWikipediaStyleAnchor);\n heading.id = anchorText;\n\n // add to wrapper\n const elementAnchor = elementAnchorClone.cloneNode(false) as HTMLAnchorElement;\n elementAnchor.href = `#${anchorText}`;\n elementAnchor.textContent = heading.textContent;\n const elementList = elementListClone.cloneNode(false) as HTMLLIElement;\n elementList.append(elementAnchor);\n\n elementContainer.append(elementList);\n\n // upadte current number\n number = currentNumber;\n }\n};\n\nexport const Mokuji = (\n element: HTMLElement | null,\n externalOptions?: MokujiOption,\n): HTMLUListElement | HTMLOListElement | undefined => {\n if (!element) {\n return;\n }\n\n // Merge the default options with the external options.\n const options = {\n ...defaultOptions,\n ...externalOptions,\n };\n\n const headings = [...getHeadingsElement(element)];\n\n if (headings.length === 0) {\n return;\n }\n\n // mokuji start\n const elementContainer = createElement(options.anchorContainerTagName);\n\n // generate mokuji list\n generateHierarchyList(headings, elementContainer, options.anchorType);\n\n const anchors = [...elementContainer.querySelectorAll('a')];\n\n if (anchors.length === 0) {\n return;\n }\n\n // remove duplicates by adding suffix\n removeDuplicateIds(headings, anchors);\n\n // setup anchor link\n if (options.anchorLink) {\n const anchorsMap = generateAnchorsMap(anchors);\n insertAnchorToHeadings(headings, anchorsMap, options);\n }\n\n return elementContainer;\n};\n\n// [data-mokuji-anchor]要素をすべて破棄する\nexport const destory = () => {\n const mokujiAnchor = document.querySelectorAll(`[${ANCHOR_DATASET_ATTRIBUTE}]`);\n for (let i = mokujiAnchor.length - 1; i >= 0; i--) {\n const element = mokujiAnchor[i];\n element.remove();\n }\n};\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/dom.ts","../src/text.ts","../src/index.ts"],"names":["hasParentNode","element","parent","getHeadingsElement","createElement","tagName","storeIds","replaceSpacesWithUnderscores","text","convert2WikipediaStyleAnchor","anchor","censorshipId","headings","textContent","id","suffix_count","tmp_id","generateAnchorText","isConvertToWikipediaStyleAnchor","ANCHOR_DATASET_ATTRIBUTE","defaultOptions","generateAnchorsMap","anchors","anchorMap","i","anchorId","insertAnchorToHeadings","options","a","heading","matchedAnchor","removeDuplicateIds","hash","matchedHeadings","count","j","heading_id","k","generateHierarchyList","elementContainer","number","elementListClone","elementAnchorClone","currentNumber","nextElementOListClone","anchorText","elementAnchor","elementList","Mokuji","externalOptions","anchorsMap","Destroy","mokujiAnchor"],"mappings":"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","sourcesContent":["/**\n * 親要素内に要素が存在するか判定する\n */\nexport const hasParentNode = (element: Node | null, parent: Node | null) => {\n if (!element || !parent) {\n return false;\n }\n\n return parent.contains(element);\n};\n\n/**\n * 見出し要素をすべて取得する\n */\nexport const getHeadingsElement = (element: Element) => {\n return element.querySelectorAll<HTMLHeadingElement>('h1, h2, h3, h4, h5, h6');\n};\n\n/**\n * 要素を作成する\n */\nexport const createElement = <T extends keyof HTMLElementTagNameMap>(tagName: T): HTMLElementTagNameMap[T] => {\n return document.createElement(tagName);\n};\n","const storeIds = new Set<string>();\n\nconst replaceSpacesWithUnderscores = (text: string) => {\n return text.replaceAll(/\\s+/g, '_');\n};\n\nconst convert2WikipediaStyleAnchor = (anchor: string) => {\n return encodeURIComponent(anchor).replaceAll(/%+/g, '.');\n};\n\nexport const censorshipId = (headings: HTMLHeadingElement[], textContent = '') => {\n let id = textContent;\n let suffix_count = 1;\n\n // IDが重複していた場合はsuffixを付ける\n while (suffix_count <= headings.length) {\n const tmp_id = suffix_count === 1 ? id : `${id}_${suffix_count}`;\n\n if (!storeIds.has(tmp_id)) {\n id = tmp_id;\n storeIds.add(id);\n break;\n }\n\n suffix_count++;\n }\n\n return id;\n};\n\nexport const generateAnchorText = (text: string, isConvertToWikipediaStyleAnchor: boolean) => {\n // convert spaces to _\n let anchor = replaceSpacesWithUnderscores(text);\n\n // remove &\n anchor = anchor.replaceAll(/&+/g, '').replaceAll(/&+/g, '');\n\n if (isConvertToWikipediaStyleAnchor) {\n anchor = convert2WikipediaStyleAnchor(anchor);\n }\n\n return anchor;\n};\n","import { hasParentNode, getHeadingsElement, createElement } from './dom';\nimport type { MokujiOption } from './types';\nimport { censorshipId, generateAnchorText } from './text';\n\nexport { MokujiOption };\n\nconst ANCHOR_DATASET_ATTRIBUTE = 'data-mokuji-anchor';\n\nconst defaultOptions = {\n anchorType: true,\n anchorLink: false,\n anchorLinkSymbol: '#',\n anchorLinkBefore: true,\n anchorLinkClassName: '',\n anchorContainerTagName: 'ol',\n} as const;\n\nconst generateAnchorsMap = (anchors: HTMLAnchorElement[]) => {\n const anchorMap = new Map<string, HTMLAnchorElement>();\n\n for (let i = 0; i < anchors.length; i++) {\n const anchorId = anchors[i].hash.replace('#', '');\n anchorMap.set(anchorId, anchors[i]);\n }\n\n return anchorMap;\n};\n\nconst insertAnchorToHeadings = (\n headings: HTMLHeadingElement[],\n anchorMap: Map<string, HTMLAnchorElement>,\n options: MokujiOption,\n) => {\n const a = createElement('a');\n a.setAttribute(ANCHOR_DATASET_ATTRIBUTE, '');\n\n if (options.anchorLinkClassName) {\n a.classList.add(options.anchorLinkClassName);\n }\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const matchedAnchor = anchorMap.get(heading.id);\n\n if (!matchedAnchor) {\n continue;\n }\n\n // create anchor\n const anchor = a.cloneNode(false) as HTMLAnchorElement;\n anchor.setAttribute('href', matchedAnchor.hash);\n\n if (options.anchorLinkSymbol) {\n anchor.textContent = options.anchorLinkSymbol;\n }\n\n // insert anchor into headings\n if (options.anchorLinkBefore) {\n // before\n heading.insertBefore(anchor, heading.firstChild);\n } else {\n // after\n heading.append(anchor);\n }\n }\n};\n\nconst removeDuplicateIds = (headings: HTMLHeadingElement[], anchors: HTMLAnchorElement[]) => {\n for (let i = 0; i < anchors.length; i++) {\n const anchor = anchors[i];\n const id = anchor.textContent;\n const hash = anchor.hash;\n const matchedHeadings = headings.filter((heading) => heading.id === id);\n\n if (matchedHeadings.length === 1) {\n continue;\n }\n\n // duplicated id\n let count = 0;\n\n for (let j = 0; j < matchedHeadings.length; j++) {\n const heading = matchedHeadings[j];\n const heading_id = `${heading.id}-${count}`;\n\n // search duplicate list\n for (let k = 0; k < anchors.length; k++) {\n const anchor = anchors[k];\n if (anchor.hash === hash) {\n // update hash\n anchor.href = `#${heading_id}`;\n break;\n }\n }\n\n // update id\n heading.id = heading_id;\n count++;\n }\n }\n};\n\nconst generateHierarchyList = (\n headings: HTMLHeadingElement[],\n elementContainer: HTMLUListElement | HTMLOListElement,\n isConvertToWikipediaStyleAnchor: boolean,\n) => {\n let number = 0;\n const elementListClone = createElement('li');\n const elementAnchorClone = createElement('a');\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const currentNumber = Number(heading.tagName[1]);\n\n // check list hierarchy\n if (number !== 0 && number < currentNumber) {\n // number of the heading is large (small as heading)\n const nextElementOListClone = createElement('ol');\n // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n // @ts-ignore\n elementContainer.lastChild.append(nextElementOListClone);\n elementContainer = nextElementOListClone;\n } else if (number !== 0 && number > currentNumber) {\n // number of heading is small (large as heading)\n for (let i = 0; i < number - currentNumber; i++) {\n if (hasParentNode(elementContainer, elementContainer.parentNode)) {\n elementContainer = elementContainer.parentNode?.parentNode as HTMLUListElement | HTMLOListElement;\n }\n }\n }\n\n const textContent = censorshipId(headings, heading.textContent || '');\n\n // headingへidを付与\n const anchorText = generateAnchorText(textContent, isConvertToWikipediaStyleAnchor);\n heading.id = anchorText;\n\n // add to wrapper\n const elementAnchor = elementAnchorClone.cloneNode(false) as HTMLAnchorElement;\n elementAnchor.href = `#${anchorText}`;\n elementAnchor.textContent = heading.textContent;\n const elementList = elementListClone.cloneNode(false) as HTMLLIElement;\n elementList.append(elementAnchor);\n\n elementContainer.append(elementList);\n\n // upadte current number\n number = currentNumber;\n }\n};\n\nexport const Mokuji = (\n element: HTMLElement | null,\n externalOptions?: MokujiOption,\n): HTMLUListElement | HTMLOListElement | undefined => {\n if (!element) {\n return;\n }\n\n // Merge the default options with the external options.\n const options = {\n ...defaultOptions,\n ...externalOptions,\n };\n\n const headings = [...getHeadingsElement(element)];\n\n if (headings.length === 0) {\n return;\n }\n\n // mokuji start\n const elementContainer = createElement(options.anchorContainerTagName);\n\n // generate mokuji list\n generateHierarchyList(headings, elementContainer, options.anchorType);\n\n const anchors = [...elementContainer.querySelectorAll('a')];\n\n if (anchors.length === 0) {\n return;\n }\n\n // remove duplicates by adding suffix\n removeDuplicateIds(headings, anchors);\n\n // setup anchor link\n if (options.anchorLink) {\n const anchorsMap = generateAnchorsMap(anchors);\n insertAnchorToHeadings(headings, anchorsMap, options);\n }\n\n return elementContainer;\n};\n\n// [data-mokuji-anchor]要素をすべて破棄する\nexport const Destroy = () => {\n const mokujiAnchor = document.querySelectorAll(`[${ANCHOR_DATASET_ATTRIBUTE}]`);\n for (let i = mokujiAnchor.length - 1; i >= 0; i--) {\n const element = mokujiAnchor[i];\n element.remove();\n }\n};\n"]}
|
package/dist/index.d.ts
CHANGED
|
@@ -9,6 +9,6 @@ type MokujiOption = {
|
|
|
9
9
|
};
|
|
10
10
|
|
|
11
11
|
declare const Mokuji: (element: HTMLElement | null, externalOptions?: MokujiOption) => HTMLUListElement | HTMLOListElement | undefined;
|
|
12
|
-
declare const
|
|
12
|
+
declare const Destroy: () => void;
|
|
13
13
|
|
|
14
|
-
export { Mokuji, MokujiOption
|
|
14
|
+
export { Destroy, Mokuji, MokujiOption };
|
package/dist/index.js
CHANGED
|
@@ -1,5 +1,5 @@
|
|
|
1
|
-
var g=(t,n)=>!t||!n?!1:n.contains(t),u=t=>t.querySelectorAll("h1, h2, h3, h4, h5, h6"),h=t=>document.createElement(t);var L=new Set,E=t=>t.replaceAll(/\s+/g,"_"),A=t=>encodeURIComponent(t).replaceAll(/%+/g,"."),T=(t,n="")=>{let e=n,o=1;for(;o<=t.length;){let r=o===1?e:`${e}_${o}`;if(!L.has(r)){e=r,L.add(e);break}o++;}return e},M=(t,n)=>{let e=E(t);return e=e.replaceAll(/&+/g,"").replaceAll(/&+/g,""),n&&(e=A(e)),e};var H="data-mokuji-anchor",k={anchorType:!0,anchorLink:!1,anchorLinkSymbol:"#",anchorLinkBefore:!0,anchorLinkClassName:"",anchorContainerTagName:"ol"},N=t=>{let n=new Map;for(let e=0;e<t.length;e++){let o=t[e].hash.replace("#","");n.set(o,t[e]);}return n},x=(t,n,e)=>{let o=h("a");o.setAttribute(H,""),e.anchorLinkClassName&&o.classList.add(e.anchorLinkClassName);for(let r=0;r<t.length;r++){let c=t[r],l=n.get(c.id);if(!l)continue;let s=o.cloneNode(!1);s.setAttribute("href",l.hash),e.anchorLinkSymbol&&(s.textContent=e.anchorLinkSymbol),e.anchorLinkBefore?c.insertBefore(s,c.firstChild):c.append(s);}},y=(t,n)=>{for(let e=0;e<n.length;e++){let o=n[e],r=o.textContent,c=o.hash,l=t.filter(a=>a.id===r);if(l.length===1)continue;let s=0;for(let a=0;a<l.length;a++){let f=l[a],
|
|
1
|
+
var g=(t,n)=>!t||!n?!1:n.contains(t),u=t=>t.querySelectorAll("h1, h2, h3, h4, h5, h6"),h=t=>document.createElement(t);var L=new Set,E=t=>t.replaceAll(/\s+/g,"_"),A=t=>encodeURIComponent(t).replaceAll(/%+/g,"."),T=(t,n="")=>{let e=n,o=1;for(;o<=t.length;){let r=o===1?e:`${e}_${o}`;if(!L.has(r)){e=r,L.add(e);break}o++;}return e},M=(t,n)=>{let e=E(t);return e=e.replaceAll(/&+/g,"").replaceAll(/&+/g,""),n&&(e=A(e)),e};var H="data-mokuji-anchor",k={anchorType:!0,anchorLink:!1,anchorLinkSymbol:"#",anchorLinkBefore:!0,anchorLinkClassName:"",anchorContainerTagName:"ol"},N=t=>{let n=new Map;for(let e=0;e<t.length;e++){let o=t[e].hash.replace("#","");n.set(o,t[e]);}return n},x=(t,n,e)=>{let o=h("a");o.setAttribute(H,""),e.anchorLinkClassName&&o.classList.add(e.anchorLinkClassName);for(let r=0;r<t.length;r++){let c=t[r],l=n.get(c.id);if(!l)continue;let s=o.cloneNode(!1);s.setAttribute("href",l.hash),e.anchorLinkSymbol&&(s.textContent=e.anchorLinkSymbol),e.anchorLinkBefore?c.insertBefore(s,c.firstChild):c.append(s);}},y=(t,n)=>{for(let e=0;e<n.length;e++){let o=n[e],r=o.textContent,c=o.hash,l=t.filter(a=>a.id===r);if(l.length===1)continue;let s=0;for(let a=0;a<l.length;a++){let f=l[a],m=`${f.id}-${s}`;for(let i=0;i<n.length;i++){let d=n[i];if(d.hash===c){d.href=`#${m}`;break}}f.id=m,s++;}}},C=(t,n,e)=>{let o=0,r=h("li"),c=h("a");for(let l=0;l<t.length;l++){let s=t[l],a=Number(s.tagName[1]);if(o!==0&&o<a){let p=h("ol");n.lastChild.append(p),n=p;}else if(o!==0&&o>a)for(let p=0;p<o-a;p++)g(n,n.parentNode)&&(n=n.parentNode?.parentNode);let f=T(t,s.textContent||""),m=M(f,e);s.id=m;let i=c.cloneNode(!1);i.href=`#${m}`,i.textContent=s.textContent;let d=r.cloneNode(!1);d.append(i),n.append(d),o=a;}},I=(t,n)=>{if(!t)return;let e={...k,...n},o=[...u(t)];if(o.length===0)return;let r=h(e.anchorContainerTagName);C(o,r,e.anchorType);let c=[...r.querySelectorAll("a")];if(c.length!==0){if(y(o,c),e.anchorLink){let l=N(c);x(o,l,e);}return r}},_=()=>{let t=document.querySelectorAll(`[${H}]`);for(let n=t.length-1;n>=0;n--)t[n].remove();};
|
|
2
2
|
|
|
3
|
-
export {
|
|
3
|
+
export { _ as Destroy, I as Mokuji };
|
|
4
4
|
//# sourceMappingURL=out.js.map
|
|
5
5
|
//# sourceMappingURL=index.js.map
|
package/dist/index.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/dom.ts","../src/text.ts","../src/index.ts"],"names":["hasParentNode","element","parent","getHeadingsElement","createElement","tagName","storeIds","replaceSpacesWithUnderscores","text","convert2WikipediaStyleAnchor","anchor","censorshipId","headings","textContent","id","suffix_count","tmp_id","generateAnchorText","isConvertToWikipediaStyleAnchor","ANCHOR_DATASET_ATTRIBUTE","defaultOptions","generateAnchorsMap","anchors","anchorMap","i","anchorId","insertAnchorToHeadings","options","a","heading","matchedAnchor","removeDuplicateIds","hash","matchedHeadings","count","j","heading_id","k","generateHierarchyList","elementContainer","number","elementListClone","elementAnchorClone","currentNumber","nextElementOListClone","anchorText","elementAnchor","elementList","Mokuji","externalOptions","anchorsMap","destory","mokujiAnchor"],"mappings":"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","sourcesContent":["/**\n * 親要素内に要素が存在するか判定する\n */\nexport const hasParentNode = (element: Node | null, parent: Node | null) => {\n if (!element || !parent) {\n return false;\n }\n\n return parent.contains(element);\n};\n\n/**\n * 見出し要素をすべて取得する\n */\nexport const getHeadingsElement = (element: Element) => {\n return element.querySelectorAll<HTMLHeadingElement>('h1, h2, h3, h4, h5, h6');\n};\n\n/**\n * 要素を作成する\n */\nexport const createElement = <T extends keyof HTMLElementTagNameMap>(tagName: T): HTMLElementTagNameMap[T] => {\n return document.createElement(tagName);\n};\n","const storeIds = new Set<string>();\n\nconst replaceSpacesWithUnderscores = (text: string) => {\n return text.replaceAll(/\\s+/g, '_');\n};\n\nconst convert2WikipediaStyleAnchor = (anchor: string) => {\n return encodeURIComponent(anchor).replaceAll(/%+/g, '.');\n};\n\nexport const censorshipId = (headings: HTMLHeadingElement[], textContent = '') => {\n let id = textContent;\n let suffix_count = 1;\n\n // IDが重複していた場合はsuffixを付ける\n while (suffix_count <= headings.length) {\n const tmp_id = suffix_count === 1 ? id : `${id}_${suffix_count}`;\n\n if (!storeIds.has(tmp_id)) {\n id = tmp_id;\n storeIds.add(id);\n break;\n }\n\n suffix_count++;\n }\n\n return id;\n};\n\nexport const generateAnchorText = (text: string, isConvertToWikipediaStyleAnchor: boolean) => {\n // convert spaces to _\n let anchor = replaceSpacesWithUnderscores(text);\n\n // remove &\n anchor = anchor.replaceAll(/&+/g, '').replaceAll(/&+/g, '');\n\n if (isConvertToWikipediaStyleAnchor) {\n anchor = convert2WikipediaStyleAnchor(anchor);\n }\n\n return anchor;\n};\n","import { hasParentNode, getHeadingsElement, createElement } from './dom';\nimport type { MokujiOption } from './types';\nimport { censorshipId, generateAnchorText } from './text';\n\nexport { MokujiOption };\n\nconst ANCHOR_DATASET_ATTRIBUTE = 'data-mokuji-anchor';\n\nconst defaultOptions = {\n anchorType: true,\n anchorLink: false,\n anchorLinkSymbol: '#',\n anchorLinkBefore: true,\n anchorLinkClassName: '',\n anchorContainerTagName: 'ol',\n} as const;\n\nconst generateAnchorsMap = (anchors: HTMLAnchorElement[]) => {\n const anchorMap = new Map<string, HTMLAnchorElement>();\n\n for (let i = 0; i < anchors.length; i++) {\n const anchorId = anchors[i].hash.replace('#', '');\n anchorMap.set(anchorId, anchors[i]);\n }\n\n return anchorMap;\n};\n\nconst insertAnchorToHeadings = (\n headings: HTMLHeadingElement[],\n anchorMap: Map<string, HTMLAnchorElement>,\n options: MokujiOption,\n) => {\n const a = createElement('a');\n a.setAttribute(ANCHOR_DATASET_ATTRIBUTE, '');\n\n if (options.anchorLinkClassName) {\n a.classList.add(options.anchorLinkClassName);\n }\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const matchedAnchor = anchorMap.get(heading.id);\n\n if (!matchedAnchor) {\n continue;\n }\n\n // create anchor\n const anchor = a.cloneNode(false) as HTMLAnchorElement;\n anchor.setAttribute('href', matchedAnchor.hash);\n\n if (options.anchorLinkSymbol) {\n anchor.textContent = options.anchorLinkSymbol;\n }\n\n // insert anchor into headings\n if (options.anchorLinkBefore) {\n // before\n heading.insertBefore(anchor, heading.firstChild);\n } else {\n // after\n heading.append(anchor);\n }\n }\n};\n\nconst removeDuplicateIds = (headings: HTMLHeadingElement[], anchors: HTMLAnchorElement[]) => {\n for (let i = 0; i < anchors.length; i++) {\n const anchor = anchors[i];\n const id = anchor.textContent;\n const hash = anchor.hash;\n const matchedHeadings = headings.filter((heading) => heading.id === id);\n\n if (matchedHeadings.length === 1) {\n continue;\n }\n\n // duplicated id\n let count = 0;\n\n for (let j = 0; j < matchedHeadings.length; j++) {\n const heading = matchedHeadings[j];\n const heading_id = `${heading.id}-${count}`;\n\n // search duplicate list\n for (let k = 0; k < anchors.length; k++) {\n const anchor = anchors[k];\n if (anchor.hash === hash) {\n // update hash\n anchor.href = `#${heading_id}`;\n break;\n }\n }\n\n // update id\n heading.id = heading_id;\n count++;\n }\n }\n};\n\nconst generateHierarchyList = (\n headings: HTMLHeadingElement[],\n elementContainer: HTMLUListElement | HTMLOListElement,\n isConvertToWikipediaStyleAnchor: boolean,\n) => {\n let number = 0;\n const elementListClone = createElement('li');\n const elementAnchorClone = createElement('a');\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const currentNumber = Number(heading.tagName[1]);\n\n // check list hierarchy\n if (number !== 0 && number < currentNumber) {\n // number of the heading is large (small as heading)\n const nextElementOListClone = createElement('ol');\n // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n // @ts-ignore\n elementContainer.lastChild.append(nextElementOListClone);\n elementContainer = nextElementOListClone;\n } else if (number !== 0 && number > currentNumber) {\n // number of heading is small (large as heading)\n for (let i = 0; i < number - currentNumber; i++) {\n if (hasParentNode(elementContainer, elementContainer.parentNode)) {\n elementContainer = elementContainer.parentNode?.parentNode as HTMLUListElement | HTMLOListElement;\n }\n }\n }\n\n const textContent = censorshipId(headings, heading.textContent || '');\n\n // headingへidを付与\n const anchorText = generateAnchorText(textContent, isConvertToWikipediaStyleAnchor);\n heading.id = anchorText;\n\n // add to wrapper\n const elementAnchor = elementAnchorClone.cloneNode(false) as HTMLAnchorElement;\n elementAnchor.href = `#${anchorText}`;\n elementAnchor.textContent = heading.textContent;\n const elementList = elementListClone.cloneNode(false) as HTMLLIElement;\n elementList.append(elementAnchor);\n\n elementContainer.append(elementList);\n\n // upadte current number\n number = currentNumber;\n }\n};\n\nexport const Mokuji = (\n element: HTMLElement | null,\n externalOptions?: MokujiOption,\n): HTMLUListElement | HTMLOListElement | undefined => {\n if (!element) {\n return;\n }\n\n // Merge the default options with the external options.\n const options = {\n ...defaultOptions,\n ...externalOptions,\n };\n\n const headings = [...getHeadingsElement(element)];\n\n if (headings.length === 0) {\n return;\n }\n\n // mokuji start\n const elementContainer = createElement(options.anchorContainerTagName);\n\n // generate mokuji list\n generateHierarchyList(headings, elementContainer, options.anchorType);\n\n const anchors = [...elementContainer.querySelectorAll('a')];\n\n if (anchors.length === 0) {\n return;\n }\n\n // remove duplicates by adding suffix\n removeDuplicateIds(headings, anchors);\n\n // setup anchor link\n if (options.anchorLink) {\n const anchorsMap = generateAnchorsMap(anchors);\n insertAnchorToHeadings(headings, anchorsMap, options);\n }\n\n return elementContainer;\n};\n\n// [data-mokuji-anchor]要素をすべて破棄する\nexport const destory = () => {\n const mokujiAnchor = document.querySelectorAll(`[${ANCHOR_DATASET_ATTRIBUTE}]`);\n for (let i = mokujiAnchor.length - 1; i >= 0; i--) {\n const element = mokujiAnchor[i];\n element.remove();\n }\n};\n"]}
|
|
1
|
+
{"version":3,"sources":["../src/dom.ts","../src/text.ts","../src/index.ts"],"names":["hasParentNode","element","parent","getHeadingsElement","createElement","tagName","storeIds","replaceSpacesWithUnderscores","text","convert2WikipediaStyleAnchor","anchor","censorshipId","headings","textContent","id","suffix_count","tmp_id","generateAnchorText","isConvertToWikipediaStyleAnchor","ANCHOR_DATASET_ATTRIBUTE","defaultOptions","generateAnchorsMap","anchors","anchorMap","i","anchorId","insertAnchorToHeadings","options","a","heading","matchedAnchor","removeDuplicateIds","hash","matchedHeadings","count","j","heading_id","k","generateHierarchyList","elementContainer","number","elementListClone","elementAnchorClone","currentNumber","nextElementOListClone","anchorText","elementAnchor","elementList","Mokuji","externalOptions","anchorsMap","Destroy","mokujiAnchor"],"mappings":"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","sourcesContent":["/**\n * 親要素内に要素が存在するか判定する\n */\nexport const hasParentNode = (element: Node | null, parent: Node | null) => {\n if (!element || !parent) {\n return false;\n }\n\n return parent.contains(element);\n};\n\n/**\n * 見出し要素をすべて取得する\n */\nexport const getHeadingsElement = (element: Element) => {\n return element.querySelectorAll<HTMLHeadingElement>('h1, h2, h3, h4, h5, h6');\n};\n\n/**\n * 要素を作成する\n */\nexport const createElement = <T extends keyof HTMLElementTagNameMap>(tagName: T): HTMLElementTagNameMap[T] => {\n return document.createElement(tagName);\n};\n","const storeIds = new Set<string>();\n\nconst replaceSpacesWithUnderscores = (text: string) => {\n return text.replaceAll(/\\s+/g, '_');\n};\n\nconst convert2WikipediaStyleAnchor = (anchor: string) => {\n return encodeURIComponent(anchor).replaceAll(/%+/g, '.');\n};\n\nexport const censorshipId = (headings: HTMLHeadingElement[], textContent = '') => {\n let id = textContent;\n let suffix_count = 1;\n\n // IDが重複していた場合はsuffixを付ける\n while (suffix_count <= headings.length) {\n const tmp_id = suffix_count === 1 ? id : `${id}_${suffix_count}`;\n\n if (!storeIds.has(tmp_id)) {\n id = tmp_id;\n storeIds.add(id);\n break;\n }\n\n suffix_count++;\n }\n\n return id;\n};\n\nexport const generateAnchorText = (text: string, isConvertToWikipediaStyleAnchor: boolean) => {\n // convert spaces to _\n let anchor = replaceSpacesWithUnderscores(text);\n\n // remove &\n anchor = anchor.replaceAll(/&+/g, '').replaceAll(/&+/g, '');\n\n if (isConvertToWikipediaStyleAnchor) {\n anchor = convert2WikipediaStyleAnchor(anchor);\n }\n\n return anchor;\n};\n","import { hasParentNode, getHeadingsElement, createElement } from './dom';\nimport type { MokujiOption } from './types';\nimport { censorshipId, generateAnchorText } from './text';\n\nexport { MokujiOption };\n\nconst ANCHOR_DATASET_ATTRIBUTE = 'data-mokuji-anchor';\n\nconst defaultOptions = {\n anchorType: true,\n anchorLink: false,\n anchorLinkSymbol: '#',\n anchorLinkBefore: true,\n anchorLinkClassName: '',\n anchorContainerTagName: 'ol',\n} as const;\n\nconst generateAnchorsMap = (anchors: HTMLAnchorElement[]) => {\n const anchorMap = new Map<string, HTMLAnchorElement>();\n\n for (let i = 0; i < anchors.length; i++) {\n const anchorId = anchors[i].hash.replace('#', '');\n anchorMap.set(anchorId, anchors[i]);\n }\n\n return anchorMap;\n};\n\nconst insertAnchorToHeadings = (\n headings: HTMLHeadingElement[],\n anchorMap: Map<string, HTMLAnchorElement>,\n options: MokujiOption,\n) => {\n const a = createElement('a');\n a.setAttribute(ANCHOR_DATASET_ATTRIBUTE, '');\n\n if (options.anchorLinkClassName) {\n a.classList.add(options.anchorLinkClassName);\n }\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const matchedAnchor = anchorMap.get(heading.id);\n\n if (!matchedAnchor) {\n continue;\n }\n\n // create anchor\n const anchor = a.cloneNode(false) as HTMLAnchorElement;\n anchor.setAttribute('href', matchedAnchor.hash);\n\n if (options.anchorLinkSymbol) {\n anchor.textContent = options.anchorLinkSymbol;\n }\n\n // insert anchor into headings\n if (options.anchorLinkBefore) {\n // before\n heading.insertBefore(anchor, heading.firstChild);\n } else {\n // after\n heading.append(anchor);\n }\n }\n};\n\nconst removeDuplicateIds = (headings: HTMLHeadingElement[], anchors: HTMLAnchorElement[]) => {\n for (let i = 0; i < anchors.length; i++) {\n const anchor = anchors[i];\n const id = anchor.textContent;\n const hash = anchor.hash;\n const matchedHeadings = headings.filter((heading) => heading.id === id);\n\n if (matchedHeadings.length === 1) {\n continue;\n }\n\n // duplicated id\n let count = 0;\n\n for (let j = 0; j < matchedHeadings.length; j++) {\n const heading = matchedHeadings[j];\n const heading_id = `${heading.id}-${count}`;\n\n // search duplicate list\n for (let k = 0; k < anchors.length; k++) {\n const anchor = anchors[k];\n if (anchor.hash === hash) {\n // update hash\n anchor.href = `#${heading_id}`;\n break;\n }\n }\n\n // update id\n heading.id = heading_id;\n count++;\n }\n }\n};\n\nconst generateHierarchyList = (\n headings: HTMLHeadingElement[],\n elementContainer: HTMLUListElement | HTMLOListElement,\n isConvertToWikipediaStyleAnchor: boolean,\n) => {\n let number = 0;\n const elementListClone = createElement('li');\n const elementAnchorClone = createElement('a');\n\n for (let i = 0; i < headings.length; i++) {\n const heading = headings[i];\n const currentNumber = Number(heading.tagName[1]);\n\n // check list hierarchy\n if (number !== 0 && number < currentNumber) {\n // number of the heading is large (small as heading)\n const nextElementOListClone = createElement('ol');\n // eslint-disable-next-line @typescript-eslint/ban-ts-comment\n // @ts-ignore\n elementContainer.lastChild.append(nextElementOListClone);\n elementContainer = nextElementOListClone;\n } else if (number !== 0 && number > currentNumber) {\n // number of heading is small (large as heading)\n for (let i = 0; i < number - currentNumber; i++) {\n if (hasParentNode(elementContainer, elementContainer.parentNode)) {\n elementContainer = elementContainer.parentNode?.parentNode as HTMLUListElement | HTMLOListElement;\n }\n }\n }\n\n const textContent = censorshipId(headings, heading.textContent || '');\n\n // headingへidを付与\n const anchorText = generateAnchorText(textContent, isConvertToWikipediaStyleAnchor);\n heading.id = anchorText;\n\n // add to wrapper\n const elementAnchor = elementAnchorClone.cloneNode(false) as HTMLAnchorElement;\n elementAnchor.href = `#${anchorText}`;\n elementAnchor.textContent = heading.textContent;\n const elementList = elementListClone.cloneNode(false) as HTMLLIElement;\n elementList.append(elementAnchor);\n\n elementContainer.append(elementList);\n\n // upadte current number\n number = currentNumber;\n }\n};\n\nexport const Mokuji = (\n element: HTMLElement | null,\n externalOptions?: MokujiOption,\n): HTMLUListElement | HTMLOListElement | undefined => {\n if (!element) {\n return;\n }\n\n // Merge the default options with the external options.\n const options = {\n ...defaultOptions,\n ...externalOptions,\n };\n\n const headings = [...getHeadingsElement(element)];\n\n if (headings.length === 0) {\n return;\n }\n\n // mokuji start\n const elementContainer = createElement(options.anchorContainerTagName);\n\n // generate mokuji list\n generateHierarchyList(headings, elementContainer, options.anchorType);\n\n const anchors = [...elementContainer.querySelectorAll('a')];\n\n if (anchors.length === 0) {\n return;\n }\n\n // remove duplicates by adding suffix\n removeDuplicateIds(headings, anchors);\n\n // setup anchor link\n if (options.anchorLink) {\n const anchorsMap = generateAnchorsMap(anchors);\n insertAnchorToHeadings(headings, anchorsMap, options);\n }\n\n return elementContainer;\n};\n\n// [data-mokuji-anchor]要素をすべて破棄する\nexport const Destroy = () => {\n const mokujiAnchor = document.querySelectorAll(`[${ANCHOR_DATASET_ATTRIBUTE}]`);\n for (let i = mokujiAnchor.length - 1; i >= 0; i--) {\n const element = mokujiAnchor[i];\n element.remove();\n }\n};\n"]}
|