spec-up-t 1.1.54 → 1.2.0
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/assets/compiled/body.js +59 -8
- package/assets/compiled/head.css +13 -12
- package/assets/css/adjust-font-size.css +11 -0
- package/assets/css/collapse-meta-info.css +27 -8
- package/assets/css/create-term-filter.css +11 -0
- package/assets/css/index.css +15 -0
- package/assets/css/prism.css +176 -153
- package/assets/css/prism.dark.css +157 -0
- package/assets/css/prism.default.css +180 -0
- package/assets/css/terms-and-definitions.1.css +223 -0
- package/assets/css/terms-and-definitions.css +214 -100
- package/assets/js/addAnchorsToTerms.js +1 -1
- package/assets/js/collapse-definitions.js +2 -1
- package/assets/js/collapse-meta-info.js +25 -11
- package/assets/js/create-term-filter.js +61 -0
- package/assets/js/horizontal-scroll-hint.js +159 -0
- package/assets/js/index.1.js +137 -0
- package/assets/js/index.js +2 -1
- package/assets/js/insert-trefs.js +122 -116
- package/assets/js/insert-xrefs.1.js +372 -0
- package/assets/js/{show-commit-hashes.js → insert-xrefs.js} +67 -7
- package/assets/js/prism.dark.js +24 -0
- package/assets/js/prism.default.js +23 -0
- package/assets/js/prism.js +4 -5
- package/assets/js/search.js +1 -1
- package/assets/js/toggle-dense-info.js +40 -0
- package/branches.md +4 -29
- package/index.js +397 -189
- package/index.new.js +662 -0
- package/package.json +1 -1
- package/src/asset-map.json +9 -5
- package/src/collect-external-references.js +16 -9
- package/src/collectExternalReferences/fetchTermsFromIndex.js +328 -0
- package/src/collectExternalReferences/processXTrefsData.js +73 -18
- package/src/create-pdf.js +385 -89
- package/src/health-check/external-specs-checker.js +207 -0
- package/src/health-check/output-gitignore-checker.js +261 -0
- package/src/health-check/specs-configuration-checker.js +274 -0
- package/src/health-check/term-references-checker.js +191 -0
- package/src/health-check/terms-intro-checker.js +81 -0
- package/src/health-check/tref-term-checker.js +463 -0
- package/src/health-check.js +445 -0
- package/src/install-from-boilerplate/config-scripts-keys.js +2 -0
- package/src/install-from-boilerplate/menu.sh +6 -3
- package/src/markdown-it-extensions.js +134 -103
- package/src/prepare-tref.js +61 -24
- package/src/utils/fetch.js +100 -0
- package/templates/template.html +12 -7
- package/assets/js/css-helper.js +0 -30
- package/src/collectExternalReferences/fetchTermsFromGitHubRepository.js +0 -232
- package/src/collectExternalReferences/fetchTermsFromGitHubRepository.test.js +0 -385
- package/src/collectExternalReferences/octokitClient.js +0 -96
|
@@ -0,0 +1,159 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @file Adds a visual horizontal scroll hint to elements with overflow content
|
|
3
|
+
* @author Kor Dwarshuis
|
|
4
|
+
* @version 1.1.0
|
|
5
|
+
* @license MIT
|
|
6
|
+
* @see https://example.com/more_info_about_horizontal_scroll for more information
|
|
7
|
+
* @since 2022-01-01
|
|
8
|
+
*
|
|
9
|
+
* @description
|
|
10
|
+
* The horizontalScrollHint function improves user experience in specification documents
|
|
11
|
+
* by providing a visual cue when content extends beyond the visible horizontal area.
|
|
12
|
+
* It dynamically adds a hint element with a finger-pointing icon and text to elements
|
|
13
|
+
* that require horizontal scrolling, particularly useful for wide tables, code blocks,
|
|
14
|
+
* and other overflowing content in technical specifications.
|
|
15
|
+
*
|
|
16
|
+
* Features:
|
|
17
|
+
* - Automatically detects when content is wider than its container
|
|
18
|
+
* - Shows a visual hint with a finger pointing icon and text
|
|
19
|
+
* - Hint disappears after the user scrolls horizontally
|
|
20
|
+
* - Adaptive to window resizing (shows/hides based on current overflow state)
|
|
21
|
+
* - Can be applied to single elements, multiple elements, or CSS selectors
|
|
22
|
+
* - Uses ResizeObserver to monitor for dynamic content changes
|
|
23
|
+
*
|
|
24
|
+
* @param {(string|Element|Array<string|Element>)} [elements='.horizontalScroll'] - Target element(s) to apply the scroll hint to:
|
|
25
|
+
* - String: CSS selector for target elements
|
|
26
|
+
* - Element: Direct DOM element reference
|
|
27
|
+
* - Array: Multiple CSS selectors or DOM elements
|
|
28
|
+
* - If omitted, defaults to '.horizontalScroll'
|
|
29
|
+
*
|
|
30
|
+
* @returns {void} This function doesn't return a value
|
|
31
|
+
*
|
|
32
|
+
* @example
|
|
33
|
+
* // Basic usage with default selector ('.horizontalScroll')
|
|
34
|
+
* horizontalScrollHint();
|
|
35
|
+
*
|
|
36
|
+
* @example
|
|
37
|
+
* // Apply to a specific element by selector
|
|
38
|
+
* horizontalScrollHint('#specification-table');
|
|
39
|
+
*
|
|
40
|
+
* @example
|
|
41
|
+
* // Apply to multiple different elements
|
|
42
|
+
* horizontalScrollHint(['.table-responsive', '.code-block', document.querySelector('#algorithm-section')]);
|
|
43
|
+
*
|
|
44
|
+
* @example
|
|
45
|
+
* // In your main script, typically call on DOMContentLoaded
|
|
46
|
+
* document.addEventListener("DOMContentLoaded", function() {
|
|
47
|
+
* horizontalScrollHint('.table-responsive');
|
|
48
|
+
* });
|
|
49
|
+
*/
|
|
50
|
+
|
|
51
|
+
const horizontalScrollHint = (elements) => {
|
|
52
|
+
// Credits: <a href="https://www.flaticon.com/free-icons/hand-cursor" title="hand cursor icons">Hand cursor icons created by Icon Bolt - Flaticon</a>
|
|
53
|
+
const fingerHorizontalScrollingImage =
|
|
54
|
+
'';
|
|
55
|
+
|
|
56
|
+
// Create styles for the horizontal scroll hint
|
|
57
|
+
if (!document.getElementById('horizontalScrollHintStyles')) {
|
|
58
|
+
const style = document.createElement('style');
|
|
59
|
+
style.id = 'horizontalScrollHintStyles';
|
|
60
|
+
style.textContent = `
|
|
61
|
+
@keyframes arrow-tweet-panel-pulse {
|
|
62
|
+
0%,
|
|
63
|
+
to {
|
|
64
|
+
transform: translateZ(0);
|
|
65
|
+
}
|
|
66
|
+
50% {
|
|
67
|
+
transform: translate3d(10px, 0, 0);
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
|
|
71
|
+
.scrollHint {
|
|
72
|
+
position: absolute;
|
|
73
|
+
top: 1em;
|
|
74
|
+
right: 1em;
|
|
75
|
+
background: #EEE;
|
|
76
|
+
padding: 0.5em;
|
|
77
|
+
|
|
78
|
+
/* https://smoothshadows.com/#djEsMSw1LDAuMDgsMjQsMCwwLCMwMzA3MTIsI2YzZjRmNiwjZmZmZmZmLDI%3D */
|
|
79
|
+
box-shadow: 0px 0px 1px rgba(3, 7, 18, 0.02),
|
|
80
|
+
0px 0px 4px rgba(3, 7, 18, 0.03),
|
|
81
|
+
0px 0px 9px rgba(3, 7, 18, 0.05),
|
|
82
|
+
0px 0px 15px rgba(3, 7, 18, 0.06),
|
|
83
|
+
0px 0px 24px rgba(3, 7, 18, 0.08);
|
|
84
|
+
|
|
85
|
+
border-radius: 7px;
|
|
86
|
+
opacity: 0;
|
|
87
|
+
transition: opacity 1s ease-in-out;
|
|
88
|
+
}
|
|
89
|
+
`;
|
|
90
|
+
document.head.appendChild(style);
|
|
91
|
+
}
|
|
92
|
+
|
|
93
|
+
// If no elements provided or null/undefined, use default selector
|
|
94
|
+
if (!elements) {
|
|
95
|
+
elements = '.horizontalScroll';
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
// Convert single element or selector to array
|
|
99
|
+
const elementSelectors = Array.isArray(elements) ? elements : [elements];
|
|
100
|
+
|
|
101
|
+
// Process each element selector
|
|
102
|
+
elementSelectors.forEach(selector => {
|
|
103
|
+
// Handle both direct DOM elements and selector strings
|
|
104
|
+
const targets = typeof selector === 'string'
|
|
105
|
+
? document.querySelectorAll(selector)
|
|
106
|
+
: [selector];
|
|
107
|
+
|
|
108
|
+
// Apply to each matched element
|
|
109
|
+
Array.from(targets).forEach(el => {
|
|
110
|
+
if (el) {
|
|
111
|
+
el.style.position = 'relative';
|
|
112
|
+
|
|
113
|
+
// Create scroll hint element
|
|
114
|
+
const scrollHint = document.createElement('p');
|
|
115
|
+
scrollHint.classList.add('scrollHint');
|
|
116
|
+
scrollHint.innerHTML = `<img style='width: 40px; vertical-align: middle;' src='${fingerHorizontalScrollingImage}' alt='finger scrolling horizontally' /> Scroll to the right`;
|
|
117
|
+
scrollHint.style.animation = 'arrow-tweet-panel-pulse 0.82s ease-in-out infinite';
|
|
118
|
+
|
|
119
|
+
el.appendChild(scrollHint);
|
|
120
|
+
|
|
121
|
+
// Function to show/hide hint based on scroll width
|
|
122
|
+
function showHideScrollHint() {
|
|
123
|
+
if (el.scrollWidth > el.clientWidth) {
|
|
124
|
+
scrollHint.style.opacity = '1';
|
|
125
|
+
} else {
|
|
126
|
+
scrollHint.style.opacity = '0';
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// Hide hint when user scrolls
|
|
131
|
+
function hideScrollHint() {
|
|
132
|
+
scrollHint.style.opacity = '0';
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
// Initialize visibility
|
|
136
|
+
showHideScrollHint();
|
|
137
|
+
|
|
138
|
+
// Watch for element size changes
|
|
139
|
+
new ResizeObserver(showHideScrollHint).observe(el);
|
|
140
|
+
|
|
141
|
+
// Handle scroll events
|
|
142
|
+
el.addEventListener('scroll', hideScrollHint);
|
|
143
|
+
}
|
|
144
|
+
});
|
|
145
|
+
});
|
|
146
|
+
};
|
|
147
|
+
|
|
148
|
+
document.addEventListener("DOMContentLoaded", function () {
|
|
149
|
+
// Apply to a default class
|
|
150
|
+
// horizontalScrollHint();
|
|
151
|
+
|
|
152
|
+
// Apply to a specific selector
|
|
153
|
+
// horizontalScrollHint('#myElement');
|
|
154
|
+
|
|
155
|
+
// Apply to multiple different elements
|
|
156
|
+
// horizontalScrollHint(['.table-container', '.code-block', document.querySelector('#special-element')]);
|
|
157
|
+
|
|
158
|
+
horizontalScrollHint(['.table-responsive']);
|
|
159
|
+
});
|
|
@@ -0,0 +1,137 @@
|
|
|
1
|
+
(function () {
|
|
2
|
+
|
|
3
|
+
var markdown = window.markdownit();
|
|
4
|
+
|
|
5
|
+
|
|
6
|
+
/* GitHub Issues */
|
|
7
|
+
|
|
8
|
+
let source = specConfig.source;
|
|
9
|
+
if (source) {
|
|
10
|
+
if (source.host === 'github') {
|
|
11
|
+
fetch(`https://api.github.com/repos/${source.account + '/' + source.repo}/issues`)
|
|
12
|
+
.then(response => response.json())
|
|
13
|
+
.then(issues => {
|
|
14
|
+
let count = issues.length;
|
|
15
|
+
document.querySelectorAll('[issue-count]').forEach(node => {
|
|
16
|
+
node.setAttribute('issue-count', count)
|
|
17
|
+
});
|
|
18
|
+
repo_issue_list.innerHTML = issues.map(issue => {
|
|
19
|
+
return `<li class="repo-issue">
|
|
20
|
+
<detail-box>
|
|
21
|
+
<section>${markdown.render(issue.body || '')}</section>
|
|
22
|
+
<header class="repo-issue-title">
|
|
23
|
+
<span class="repo-issue-number">${issue.number}</span>
|
|
24
|
+
<span class="repo-issue-link">
|
|
25
|
+
<a href="${issue.html_url}" target="_blank">${issue.title}</a>
|
|
26
|
+
</span>
|
|
27
|
+
<span detail-box-toggle></span>
|
|
28
|
+
</header>
|
|
29
|
+
</detail-box>
|
|
30
|
+
</li>`
|
|
31
|
+
}).join('');
|
|
32
|
+
Prism.highlightAllUnder(repo_issue_list);
|
|
33
|
+
})
|
|
34
|
+
}
|
|
35
|
+
}
|
|
36
|
+
//${markdown.render(issue.body)}
|
|
37
|
+
|
|
38
|
+
/* Mermaid Diagrams */
|
|
39
|
+
|
|
40
|
+
mermaid.initialize({
|
|
41
|
+
startOnLoad: true,
|
|
42
|
+
theme: 'neutral'
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
/* Charts */
|
|
46
|
+
|
|
47
|
+
document.querySelectorAll('.chartjs').forEach(chart => {
|
|
48
|
+
new Chart(chart, JSON.parse(chart.textContent));
|
|
49
|
+
});
|
|
50
|
+
|
|
51
|
+
/* Tooltips */
|
|
52
|
+
let tipMap = new WeakMap();
|
|
53
|
+
delegateEvent('pointerover', '.term-reference, .spec-reference, .x-term-reference', (e, anchor) => {
|
|
54
|
+
const id = anchor.getAttribute('data-local-href') || anchor.getAttribute('href') || '';
|
|
55
|
+
|
|
56
|
+
// Don't process if we've already created a tooltip for this anchor
|
|
57
|
+
if (tipMap.has(anchor)) return;
|
|
58
|
+
|
|
59
|
+
// Extract term ID
|
|
60
|
+
const termId = id.replace('#', '');
|
|
61
|
+
|
|
62
|
+
// Find the term in the document
|
|
63
|
+
let term = document.getElementById(termId);
|
|
64
|
+
|
|
65
|
+
// If term wasn't found directly, and it's an external reference (contains ':')
|
|
66
|
+
if (!term && termId.includes(':')) {
|
|
67
|
+
// Try to find the term in the hidden div with all external references
|
|
68
|
+
// This is the format for external references: term:namespace:name
|
|
69
|
+
const parts = termId.split(':');
|
|
70
|
+
|
|
71
|
+
// For external refs, first look for exact match in the hidden div
|
|
72
|
+
if (parts.length > 1) {
|
|
73
|
+
term = document.getElementById(termId);
|
|
74
|
+
|
|
75
|
+
// If not found, try just looking for the term name (last part)
|
|
76
|
+
if (!term && parts.length > 2) {
|
|
77
|
+
const termName = parts[2]; // The actual term name
|
|
78
|
+
term = document.getElementById(`term:${termName}`);
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
|
|
83
|
+
// If we still haven't found the term definition, give up
|
|
84
|
+
if (!term) return;
|
|
85
|
+
|
|
86
|
+
// Find the container of the term definition
|
|
87
|
+
let container = term.closest('dt, td:first-child');
|
|
88
|
+
|
|
89
|
+
// Special handling for external term references
|
|
90
|
+
// which might be directly in the term span without a container
|
|
91
|
+
if (!container && term.textContent) {
|
|
92
|
+
let tip = {
|
|
93
|
+
allowHTML: true,
|
|
94
|
+
inlinePositioning: true,
|
|
95
|
+
content: term.innerHTML || term.textContent
|
|
96
|
+
};
|
|
97
|
+
tipMap.set(anchor, tippy(anchor, tip));
|
|
98
|
+
return;
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
if (!container) return;
|
|
102
|
+
|
|
103
|
+
let tip = {
|
|
104
|
+
allowHTML: true,
|
|
105
|
+
inlinePositioning: true
|
|
106
|
+
};
|
|
107
|
+
|
|
108
|
+
switch (container.tagName) {
|
|
109
|
+
case 'DT':
|
|
110
|
+
// For definition terms, show the definition text
|
|
111
|
+
tip.content = container.nextElementSibling?.textContent || 'No definition found';
|
|
112
|
+
break;
|
|
113
|
+
case 'TD':
|
|
114
|
+
// For table cells, create a table of properties
|
|
115
|
+
let table = container.closest('table');
|
|
116
|
+
let tds = Array.from(container.closest('tr').children);
|
|
117
|
+
tds.shift();
|
|
118
|
+
if (table) {
|
|
119
|
+
let headings = Array.from(table.querySelectorAll('thead th'));
|
|
120
|
+
headings.shift();
|
|
121
|
+
if (headings.length) {
|
|
122
|
+
tip.content = `
|
|
123
|
+
<header>${container.textContent}</header>
|
|
124
|
+
<table>
|
|
125
|
+
${headings.map((th, i) => {
|
|
126
|
+
return `<tr><td>${th.textContent}:</td><td>${tds[i] ? tds[i].textContent : ''}</td></tr>`
|
|
127
|
+
}).join('')}
|
|
128
|
+
</table>`;
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
break;
|
|
132
|
+
}
|
|
133
|
+
|
|
134
|
+
if (tip.content) tipMap.set(anchor, tippy(anchor, tip));
|
|
135
|
+
}, { passive: true });
|
|
136
|
+
|
|
137
|
+
})();
|
package/assets/js/index.js
CHANGED
|
@@ -60,7 +60,7 @@ delegateEvent('pointerover', '.term-reference, .spec-reference', (e, anchor) =>
|
|
|
60
60
|
let tip = {
|
|
61
61
|
allowHTML: true,
|
|
62
62
|
inlinePositioning: true
|
|
63
|
-
}
|
|
63
|
+
};
|
|
64
64
|
switch (container.tagName) {
|
|
65
65
|
case 'DT':
|
|
66
66
|
tip.content = container.nextElementSibling.textContent;
|
|
@@ -84,6 +84,7 @@ delegateEvent('pointerover', '.term-reference, .spec-reference', (e, anchor) =>
|
|
|
84
84
|
}
|
|
85
85
|
break;
|
|
86
86
|
}
|
|
87
|
+
|
|
87
88
|
if (tip.content) tipMap.set(anchor, tippy(anchor, tip));
|
|
88
89
|
}, { passive: true });
|
|
89
90
|
|
|
@@ -1,136 +1,134 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* @fileoverview
|
|
3
|
-
*
|
|
4
|
-
* This script enhances an HTML document by adding definitions and meta information
|
|
5
|
-
* for terms marked with `<span class="transcluded-xref-term">` inside `<dt>` elements
|
|
6
|
-
* within a `<dl>` structure. It matches these terms against an external data source,
|
|
7
|
-
* renders the corresponding content and meta information from markdown to HTML,
|
|
8
|
-
* and inserts them as `<dd>` elements following each matching `<dt>`. The script
|
|
9
|
-
* executes automatically when the DOM is fully loaded via a `DOMContentLoaded` event listener.
|
|
10
|
-
*
|
|
11
|
-
* ### Dependencies
|
|
12
|
-
* - **`allXTrefs`**: A global object containing the external references to be transcluded.
|
|
13
|
-
* - **`md`**: A markdown renderer object with a `render` method to convert markdown to HTML.
|
|
14
|
-
*
|
|
15
|
-
* ### Data Structure
|
|
16
|
-
* The `allXTrefs` object must have the following structure:
|
|
17
|
-
* ```javascript
|
|
18
|
-
* {
|
|
19
|
-
* xtrefs: [
|
|
20
|
-
* {
|
|
21
|
-
* term: string, // The term to match against the <span> text content
|
|
22
|
-
* content: string, // Markdown content for the definition
|
|
23
|
-
* owner: string, // Owner of the source repository
|
|
24
|
-
* repo: string, // Repository name
|
|
25
|
-
* repoUrl: string, // URL to the repository
|
|
26
|
-
* commitHash: string, // Commit hash of the source
|
|
27
|
-
* avatarUrl: string // URL to the owner's avatar
|
|
28
|
-
* },
|
|
29
|
-
* // Additional reference objects...
|
|
30
|
-
* ]
|
|
31
|
-
* }
|
|
32
|
-
* ```
|
|
33
|
-
*
|
|
34
|
-
* ### Behavior
|
|
35
|
-
* The script:
|
|
36
|
-
* 1. Identifies all `<dt>` elements containing `<span class="transcluded-xref-term">`.
|
|
37
|
-
* 2. Extracts the text content of each `<span>`.
|
|
38
|
-
* 3. Searches for a matching `term` in `allXTrefs.xtrefs`.
|
|
39
|
-
* 4. If a match is found, creates:
|
|
40
|
-
* - A `<dd>` element with the rendered `content` (markdown to HTML).
|
|
41
|
-
* - A `<dd>` element with meta information (e.g., owner, repo, commit hash) also rendered from markdown.
|
|
42
|
-
* 5. Inserts these `<dd>` elements after the corresponding `<dt>`.
|
|
43
|
-
*
|
|
44
|
-
* ### DOM Modifications
|
|
45
|
-
* The script modifies the DOM by appending new `<dd>` elements after each matching `<dt>`.
|
|
46
|
-
*
|
|
47
|
-
* @requires {Object} allXTrefs - The external data source containing the references.
|
|
48
|
-
* @requires {Object} md - A markdown renderer with a `render` method (e.g., marked.js or similar).
|
|
49
|
-
*
|
|
50
|
-
* @example
|
|
51
|
-
* // Define dependencies in a <script> tag before this script:
|
|
52
|
-
* window.allXTrefs = {
|
|
53
|
-
* xtrefs: [
|
|
54
|
-
* {
|
|
55
|
-
* term: "example",
|
|
56
|
-
* content: "This is an **example** definition.",
|
|
57
|
-
* owner: "user",
|
|
58
|
-
* repo: "glossary",
|
|
59
|
-
* repoUrl: "https://github.com/user/glossary",
|
|
60
|
-
* commitHash: "abc123",
|
|
61
|
-
* avatarUrl: "https://github.com/user.png"
|
|
62
|
-
* }
|
|
63
|
-
* ]
|
|
64
|
-
* };
|
|
65
|
-
* window.md = { render: function(markdown) { return "<p>" + markdown + "</p>"; } };
|
|
66
|
-
*
|
|
67
|
-
* // HTML example:
|
|
68
|
-
* // <dl>
|
|
69
|
-
* // <dt><span class="transcluded-xref-term">example</span></dt>
|
|
70
|
-
* // </dl>
|
|
71
|
-
* // After script execution:
|
|
72
|
-
* // <dl>
|
|
73
|
-
* // <dt><span class="transcluded-xref-term">example</span></dt>
|
|
74
|
-
* // <dd> … table with meta info … </dd>
|
|
75
|
-
* // <dd><p>This is an <strong>example</strong> definition.</p></dd>
|
|
76
|
-
* // </dl>
|
|
2
|
+
* @fileoverview Handles the insertion of transcluded external references (xrefs) into HTML documentation.
|
|
3
|
+
* This script processes terms marked with a specific class and adds their definitions from external sources.
|
|
77
4
|
*/
|
|
78
5
|
|
|
6
|
+
/**
|
|
7
|
+
* Inserts transcluded external references into the document.
|
|
8
|
+
* @param {Object} allXTrefs - The object containing all external references data
|
|
9
|
+
* @param {Array} allXTrefs.xtrefs - Array of external reference objects, each containing term definitions
|
|
10
|
+
*/
|
|
79
11
|
function insertTrefs(allXTrefs) { // Pass allXTrefs as a parameter
|
|
12
|
+
/**
|
|
13
|
+
* Processes all terms found in the document and inserts their corresponding content
|
|
14
|
+
* @param {Object} xtrefsData - The object containing xtrefs data
|
|
15
|
+
* @param {Array} xtrefsData.xtrefs - Array of external reference objects
|
|
16
|
+
*/
|
|
80
17
|
function processTerms(xtrefsData) {
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
// Get the text content of the element, excluding its child nodes
|
|
18
|
+
// First collect all terms to ensure consistent processing order
|
|
19
|
+
const allTerms = [];
|
|
20
|
+
document.querySelectorAll('dt span.transcluded-xref-term').forEach(termElement => {
|
|
85
21
|
const textContent = Array.from(termElement.childNodes)
|
|
86
22
|
.filter(node => node.nodeType === Node.TEXT_NODE)
|
|
87
23
|
.map(node => node.textContent.trim())
|
|
88
24
|
.join('');
|
|
89
25
|
|
|
26
|
+
allTerms.push({
|
|
27
|
+
element: termElement,
|
|
28
|
+
textContent: textContent
|
|
29
|
+
});
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
// Then process all terms in a consistent order (from the JS file order)
|
|
33
|
+
allTerms.forEach(termData => {
|
|
34
|
+
const termElement = termData.element;
|
|
35
|
+
const textContent = termData.textContent;
|
|
36
|
+
|
|
90
37
|
// Find the first matching xref to avoid duplicates
|
|
91
38
|
const xref = xtrefsData.xtrefs.find(x => x.term === textContent);
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
const commitHash = xref.commitHash || 'Unknown';
|
|
118
|
-
|
|
119
|
-
const metaInfo = `
|
|
39
|
+
|
|
40
|
+
// Skip if we've already added content for this term (check for existing dd elements)
|
|
41
|
+
const dt = termElement.closest('dt');
|
|
42
|
+
if (dt) {
|
|
43
|
+
const nextElement = dt.nextElementSibling;
|
|
44
|
+
if (nextElement && nextElement.classList.contains('transcluded-xref-term') &&
|
|
45
|
+
nextElement.classList.contains('meta-info-content-wrapper')) {
|
|
46
|
+
return; // Already processed
|
|
47
|
+
}
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
if (xref) {
|
|
51
|
+
const parent = dt.parentNode;
|
|
52
|
+
|
|
53
|
+
// Create and insert meta info element
|
|
54
|
+
const metaInfoEl = document.createElement('dd');
|
|
55
|
+
metaInfoEl.classList.add('transcluded-xref-term', 'meta-info-content-wrapper', 'collapsed');
|
|
56
|
+
|
|
57
|
+
// Generate meta info content
|
|
58
|
+
const avatar = xref.avatarUrl ? `` : '';
|
|
59
|
+
const owner = xref.owner || 'Unknown';
|
|
60
|
+
const repo = xref.repo && xref.repoUrl ? `[${xref.repo}](${xref.repoUrl})` : 'Unknown';
|
|
61
|
+
const commitHash = xref.commitHash || 'Unknown';
|
|
62
|
+
|
|
63
|
+
const metaInfo = `
|
|
120
64
|
| Property | Value |
|
|
121
65
|
| -------- | ----- |
|
|
122
66
|
| Owner | ${avatar} ${owner} |
|
|
123
67
|
| Repo | ${repo} |
|
|
124
68
|
| Commit hash | ${commitHash} |
|
|
125
|
-
|
|
126
|
-
|
|
69
|
+
`;
|
|
70
|
+
metaInfoEl.innerHTML = md.render(metaInfo);
|
|
71
|
+
parent.insertBefore(metaInfoEl, dt.nextSibling);
|
|
127
72
|
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
73
|
+
// Clean up markdown content
|
|
74
|
+
let content = xref.content
|
|
75
|
+
.replace(/\[\[def:[^\]]*?\]\]/g, '') // Remove [[def: ...]] patterns regardless of trailing chars
|
|
76
|
+
.split('\n')
|
|
77
|
+
.map(line => line.replace(/^\s*~\s*/, '')) // Remove leading ~ and spaces
|
|
78
|
+
.join('\n')
|
|
79
|
+
.replace(/\[\[ref:/g, '') // Remove [[ref: ...]]
|
|
80
|
+
.replace(/\]\]/g, '');
|
|
81
|
+
|
|
82
|
+
// Parse the rendered HTML to check for dd elements
|
|
83
|
+
const tempDiv = document.createElement('div');
|
|
84
|
+
tempDiv.innerHTML = md.render(content);
|
|
85
|
+
|
|
86
|
+
// If there are dd elements in the rendered content, insert them directly
|
|
87
|
+
const ddElements = tempDiv.querySelectorAll('dd');
|
|
88
|
+
if (ddElements.length > 0) {
|
|
89
|
+
// Insert each dd element in the correct order
|
|
90
|
+
let insertPosition = metaInfoEl; // Start inserting after the meta info element
|
|
91
|
+
|
|
92
|
+
// Convert NodeList to Array and insert in original order
|
|
93
|
+
Array.from(ddElements).forEach(dd => {
|
|
94
|
+
// Clone the node to avoid removing it from tempDiv during insertion
|
|
95
|
+
const clonedDD = dd.cloneNode(true);
|
|
96
|
+
// Add necessary classes
|
|
97
|
+
clonedDD.classList.add('transcluded-xref-term', 'transcluded-xref-term-embedded');
|
|
98
|
+
// Insert after the previous element
|
|
99
|
+
parent.insertBefore(clonedDD, insertPosition.nextSibling);
|
|
100
|
+
// Update insertion position to be after this newly inserted element
|
|
101
|
+
insertPosition = clonedDD;
|
|
102
|
+
});
|
|
103
|
+
} else {
|
|
104
|
+
// No dd elements found, create one to hold the content
|
|
105
|
+
const contentEl = document.createElement('dd');
|
|
106
|
+
contentEl.classList.add('transcluded-xref-term', 'transcluded-xref-term-embedded');
|
|
107
|
+
contentEl.innerHTML = tempDiv.innerHTML;
|
|
108
|
+
parent.insertBefore(contentEl, metaInfoEl.nextSibling);
|
|
109
|
+
}
|
|
110
|
+
} else {
|
|
111
|
+
// Handle case where xref is not found
|
|
131
112
|
const parent = dt.parentNode;
|
|
132
|
-
|
|
133
|
-
|
|
113
|
+
|
|
114
|
+
// Create and insert meta info for not found case
|
|
115
|
+
const metaInfoEl = document.createElement('dd');
|
|
116
|
+
metaInfoEl.classList.add('transcluded-xref-term', 'meta-info-content-wrapper', 'collapsed');
|
|
117
|
+
const metaInfo = `
|
|
118
|
+
| Property | Value |
|
|
119
|
+
| -------- | ----- |
|
|
120
|
+
| Owner | Unknown |
|
|
121
|
+
| Repo | Unknown |
|
|
122
|
+
| Commit hash | not found |
|
|
123
|
+
`;
|
|
124
|
+
metaInfoEl.innerHTML = md.render(metaInfo);
|
|
125
|
+
parent.insertBefore(metaInfoEl, dt.nextSibling);
|
|
126
|
+
|
|
127
|
+
// Create and insert not found message
|
|
128
|
+
const notFoundEl = document.createElement('dd');
|
|
129
|
+
notFoundEl.classList.add('transcluded-xref-term', 'transcluded-xref-term-embedded');
|
|
130
|
+
notFoundEl.innerHTML = '<p>This term was not found in the external repository.</p>';
|
|
131
|
+
parent.insertBefore(notFoundEl, metaInfoEl.nextSibling);
|
|
134
132
|
}
|
|
135
133
|
});
|
|
136
134
|
}
|
|
@@ -142,7 +140,15 @@ function insertTrefs(allXTrefs) { // Pass allXTrefs as a parameter
|
|
|
142
140
|
}
|
|
143
141
|
}
|
|
144
142
|
|
|
143
|
+
/**
|
|
144
|
+
* Initialize the transcluded references when the DOM is fully loaded.
|
|
145
|
+
* Checks for the global allXTrefs object and calls insertTrefs if available.
|
|
146
|
+
*/
|
|
145
147
|
document.addEventListener('DOMContentLoaded', () => {
|
|
146
|
-
//
|
|
147
|
-
|
|
148
|
+
// Check if allXTrefs is defined in the global scope
|
|
149
|
+
if (typeof allXTrefs !== 'undefined') {
|
|
150
|
+
insertTrefs(allXTrefs);
|
|
151
|
+
} else {
|
|
152
|
+
console.warn('allXTrefs is not available in the global scope. Transcluded references will not be inserted.');
|
|
153
|
+
}
|
|
148
154
|
});
|