Skip to content

Commit 9f4c0e7

Browse files
authoredSep 23, 2021
Highlight Lines: Expose highlightLines function as Prism.plugins.highlightLines (#3086)
1 parent e6e1d5a commit 9f4c0e7

File tree

2 files changed

+113
-110
lines changed

2 files changed

+113
-110
lines changed
 

‎plugins/line-highlight/prism-line-highlight.js

+112-109
Original file line numberDiff line numberDiff line change
@@ -112,137 +112,140 @@
112112

113113
var scrollIntoView = true;
114114

115-
/**
116-
* Highlights the lines of the given pre.
117-
*
118-
* This function is split into a DOM measuring and mutate phase to improve performance.
119-
* The returned function mutates the DOM when called.
120-
*
121-
* @param {HTMLElement} pre
122-
* @param {string | null} [lines]
123-
* @param {string} [classes='']
124-
* @returns {() => void}
125-
*/
126-
function highlightLines(pre, lines, classes) {
127-
lines = typeof lines === 'string' ? lines : (pre.getAttribute('data-line') || '');
128-
129-
var ranges = lines.replace(/\s+/g, '').split(',').filter(Boolean);
130-
var offset = +pre.getAttribute('data-line-offset') || 0;
131-
132-
var parseMethod = isLineHeightRounded() ? parseInt : parseFloat;
133-
var lineHeight = parseMethod(getComputedStyle(pre).lineHeight);
134-
var hasLineNumbers = Prism.util.isActive(pre, LINE_NUMBERS_CLASS);
135-
var codeElement = pre.querySelector('code');
136-
var parentElement = hasLineNumbers ? pre : codeElement || pre;
137-
var mutateActions = /** @type {(() => void)[]} */ ([]);
138-
115+
Prism.plugins.lineHighlight = {
139116
/**
140-
* The top offset between the content box of the <code> element and the content box of the parent element of
141-
* the line highlight element (either `<pre>` or `<code>`).
117+
* Highlights the lines of the given pre.
142118
*
143-
* This offset might not be zero for some themes where the <code> element has a top margin. Some plugins
144-
* (or users) might also add element above the <code> element. Because the line highlight is aligned relative
145-
* to the <pre> element, we have to take this into account.
119+
* This function is split into a DOM measuring and mutate phase to improve performance.
120+
* The returned function mutates the DOM when called.
146121
*
147-
* This offset will be 0 if the parent element of the line highlight element is the `<code>` element.
122+
* @param {HTMLElement} pre
123+
* @param {string | null} [lines]
124+
* @param {string} [classes='']
125+
* @returns {() => void}
148126
*/
149-
var codePreOffset = !codeElement || parentElement == codeElement ? 0 : getContentBoxTopOffset(pre, codeElement);
127+
highlightLines: function highlightLines(pre, lines, classes) {
128+
lines = typeof lines === 'string' ? lines : (pre.getAttribute('data-line') || '');
129+
130+
var ranges = lines.replace(/\s+/g, '').split(',').filter(Boolean);
131+
var offset = +pre.getAttribute('data-line-offset') || 0;
132+
133+
var parseMethod = isLineHeightRounded() ? parseInt : parseFloat;
134+
var lineHeight = parseMethod(getComputedStyle(pre).lineHeight);
135+
var hasLineNumbers = Prism.util.isActive(pre, LINE_NUMBERS_CLASS);
136+
var codeElement = pre.querySelector('code');
137+
var parentElement = hasLineNumbers ? pre : codeElement || pre;
138+
var mutateActions = /** @type {(() => void)[]} */ ([]);
139+
140+
/**
141+
* The top offset between the content box of the <code> element and the content box of the parent element of
142+
* the line highlight element (either `<pre>` or `<code>`).
143+
*
144+
* This offset might not be zero for some themes where the <code> element has a top margin. Some plugins
145+
* (or users) might also add element above the <code> element. Because the line highlight is aligned relative
146+
* to the <pre> element, we have to take this into account.
147+
*
148+
* This offset will be 0 if the parent element of the line highlight element is the `<code>` element.
149+
*/
150+
var codePreOffset = !codeElement || parentElement == codeElement ? 0 : getContentBoxTopOffset(pre, codeElement);
151+
152+
ranges.forEach(function (currentRange) {
153+
var range = currentRange.split('-');
154+
155+
var start = +range[0];
156+
var end = +range[1] || start;
157+
158+
/** @type {HTMLElement} */
159+
var line = pre.querySelector('.line-highlight[data-range="' + currentRange + '"]') || document.createElement('div');
150160

151-
ranges.forEach(function (currentRange) {
152-
var range = currentRange.split('-');
161+
mutateActions.push(function () {
162+
line.setAttribute('aria-hidden', 'true');
163+
line.setAttribute('data-range', currentRange);
164+
line.className = (classes || '') + ' line-highlight';
165+
});
153166

154-
var start = +range[0];
155-
var end = +range[1] || start;
167+
// if the line-numbers plugin is enabled, then there is no reason for this plugin to display the line numbers
168+
if (hasLineNumbers && Prism.plugins.lineNumbers) {
169+
var startNode = Prism.plugins.lineNumbers.getLine(pre, start);
170+
var endNode = Prism.plugins.lineNumbers.getLine(pre, end);
156171

157-
/** @type {HTMLElement} */
158-
var line = pre.querySelector('.line-highlight[data-range="' + currentRange + '"]') || document.createElement('div');
172+
if (startNode) {
173+
var top = startNode.offsetTop + codePreOffset + 'px';
174+
mutateActions.push(function () {
175+
line.style.top = top;
176+
});
177+
}
159178

160-
mutateActions.push(function () {
161-
line.setAttribute('aria-hidden', 'true');
162-
line.setAttribute('data-range', currentRange);
163-
line.className = (classes || '') + ' line-highlight';
164-
});
179+
if (endNode) {
180+
var height = (endNode.offsetTop - startNode.offsetTop) + endNode.offsetHeight + 'px';
181+
mutateActions.push(function () {
182+
line.style.height = height;
183+
});
184+
}
185+
} else {
186+
mutateActions.push(function () {
187+
line.setAttribute('data-start', String(start));
165188

166-
// if the line-numbers plugin is enabled, then there is no reason for this plugin to display the line numbers
167-
if (hasLineNumbers && Prism.plugins.lineNumbers) {
168-
var startNode = Prism.plugins.lineNumbers.getLine(pre, start);
169-
var endNode = Prism.plugins.lineNumbers.getLine(pre, end);
189+
if (end > start) {
190+
line.setAttribute('data-end', String(end));
191+
}
170192

171-
if (startNode) {
172-
var top = startNode.offsetTop + codePreOffset + 'px';
173-
mutateActions.push(function () {
174-
line.style.top = top;
175-
});
176-
}
193+
line.style.top = (start - offset - 1) * lineHeight + codePreOffset + 'px';
177194

178-
if (endNode) {
179-
var height = (endNode.offsetTop - startNode.offsetTop) + endNode.offsetHeight + 'px';
180-
mutateActions.push(function () {
181-
line.style.height = height;
195+
line.textContent = new Array(end - start + 2).join(' \n');
182196
});
183197
}
184-
} else {
185-
mutateActions.push(function () {
186-
line.setAttribute('data-start', String(start));
187198

188-
if (end > start) {
189-
line.setAttribute('data-end', String(end));
190-
}
191-
192-
line.style.top = (start - offset - 1) * lineHeight + codePreOffset + 'px';
193-
194-
line.textContent = new Array(end - start + 2).join(' \n');
199+
mutateActions.push(function () {
200+
line.style.width = pre.scrollWidth + 'px';
195201
});
196-
}
197202

198-
mutateActions.push(function () {
199-
line.style.width = pre.scrollWidth + 'px';
203+
mutateActions.push(function () {
204+
// allow this to play nicely with the line-numbers plugin
205+
// need to attack to pre as when line-numbers is enabled, the code tag is relatively which screws up the positioning
206+
parentElement.appendChild(line);
207+
});
200208
});
201209

202-
mutateActions.push(function () {
203-
// allow this to play nicely with the line-numbers plugin
204-
// need to attack to pre as when line-numbers is enabled, the code tag is relatively which screws up the positioning
205-
parentElement.appendChild(line);
206-
});
207-
});
210+
var id = pre.id;
211+
if (hasLineNumbers && Prism.util.isActive(pre, LINKABLE_LINE_NUMBERS_CLASS) && id) {
212+
// This implements linkable line numbers. Linkable line numbers use Line Highlight to create a link to a
213+
// specific line. For this to work, the pre element has to:
214+
// 1) have line numbers,
215+
// 2) have the `linkable-line-numbers` class or an ascendant that has that class, and
216+
// 3) have an id.
208217

209-
var id = pre.id;
210-
if (hasLineNumbers && Prism.util.isActive(pre, LINKABLE_LINE_NUMBERS_CLASS) && id) {
211-
// This implements linkable line numbers. Linkable line numbers use Line Highlight to create a link to a
212-
// specific line. For this to work, the pre element has to:
213-
// 1) have line numbers,
214-
// 2) have the `linkable-line-numbers` class or an ascendant that has that class, and
215-
// 3) have an id.
218+
if (!hasClass(pre, LINKABLE_LINE_NUMBERS_CLASS)) {
219+
// add class to pre
220+
mutateActions.push(function () {
221+
pre.classList.add(LINKABLE_LINE_NUMBERS_CLASS);
222+
});
223+
}
216224

217-
if (!hasClass(pre, LINKABLE_LINE_NUMBERS_CLASS)) {
218-
// add class to pre
219-
mutateActions.push(function () {
220-
pre.classList.add(LINKABLE_LINE_NUMBERS_CLASS);
225+
var start = parseInt(pre.getAttribute('data-start') || '1');
226+
227+
// iterate all line number spans
228+
$$('.line-numbers-rows > span', pre).forEach(function (lineSpan, i) {
229+
var lineNumber = i + start;
230+
lineSpan.onclick = function () {
231+
var hash = id + '.' + lineNumber;
232+
233+
// this will prevent scrolling since the span is obviously in view
234+
scrollIntoView = false;
235+
location.hash = hash;
236+
setTimeout(function () {
237+
scrollIntoView = true;
238+
}, 1);
239+
};
221240
});
222241
}
223242

224-
var start = parseInt(pre.getAttribute('data-start') || '1');
225-
226-
// iterate all line number spans
227-
$$('.line-numbers-rows > span', pre).forEach(function (lineSpan, i) {
228-
var lineNumber = i + start;
229-
lineSpan.onclick = function () {
230-
var hash = id + '.' + lineNumber;
231-
232-
// this will prevent scrolling since the span is obviously in view
233-
scrollIntoView = false;
234-
location.hash = hash;
235-
setTimeout(function () {
236-
scrollIntoView = true;
237-
}, 1);
238-
};
239-
});
243+
return function () {
244+
mutateActions.forEach(callFunction);
245+
};
240246
}
247+
};
241248

242-
return function () {
243-
mutateActions.forEach(callFunction);
244-
};
245-
}
246249

247250
function applyHash() {
248251
var hash = location.hash.slice(1);
@@ -269,7 +272,7 @@
269272
pre.setAttribute('data-line', '');
270273
}
271274

272-
var mutateDom = highlightLines(pre, range, 'temporary ');
275+
var mutateDom = Prism.plugins.lineHighlight.highlightLines(pre, range, 'temporary ');
273276
mutateDom();
274277

275278
if (scrollIntoView) {
@@ -317,7 +320,7 @@
317320
if (hasClass(pre, LINE_NUMBERS_CLASS) && hasLineNumbers && !isLineNumbersLoaded) {
318321
Prism.hooks.add('line-numbers', completeHook);
319322
} else {
320-
var mutateDom = highlightLines(pre);
323+
var mutateDom = Prism.plugins.lineHighlight.highlightLines(pre);
321324
mutateDom();
322325
fakeTimer = setTimeout(applyHash, 1);
323326
}
@@ -328,7 +331,7 @@
328331
var actions = $$('pre')
329332
.filter(isActiveFor)
330333
.map(function (pre) {
331-
return highlightLines(pre);
334+
return Prism.plugins.lineHighlight.highlightLines(pre);
332335
});
333336
actions.forEach(callFunction);
334337
});

‎plugins/line-highlight/prism-line-highlight.min.js

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

0 commit comments

Comments
 (0)
Please sign in to comment.