|
112 | 112 |
|
113 | 113 | var scrollIntoView = true;
|
114 | 114 |
|
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 = { |
139 | 116 | /**
|
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. |
142 | 118 | *
|
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. |
146 | 121 | *
|
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} |
148 | 126 | */
|
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'); |
150 | 160 |
|
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 | + }); |
153 | 166 |
|
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); |
156 | 171 |
|
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 | + } |
159 | 178 |
|
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)); |
165 | 188 |
|
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 | + } |
170 | 192 |
|
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'; |
177 | 194 |
|
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'); |
182 | 196 | });
|
183 | 197 | }
|
184 |
| - } else { |
185 |
| - mutateActions.push(function () { |
186 |
| - line.setAttribute('data-start', String(start)); |
187 | 198 |
|
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'; |
195 | 201 | });
|
196 |
| - } |
197 | 202 |
|
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 | + }); |
200 | 208 | });
|
201 | 209 |
|
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. |
208 | 217 |
|
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 | + } |
216 | 224 |
|
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 | + }; |
221 | 240 | });
|
222 | 241 | }
|
223 | 242 |
|
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 | + }; |
240 | 246 | }
|
| 247 | + }; |
241 | 248 |
|
242 |
| - return function () { |
243 |
| - mutateActions.forEach(callFunction); |
244 |
| - }; |
245 |
| - } |
246 | 249 |
|
247 | 250 | function applyHash() {
|
248 | 251 | var hash = location.hash.slice(1);
|
|
269 | 272 | pre.setAttribute('data-line', '');
|
270 | 273 | }
|
271 | 274 |
|
272 |
| - var mutateDom = highlightLines(pre, range, 'temporary '); |
| 275 | + var mutateDom = Prism.plugins.lineHighlight.highlightLines(pre, range, 'temporary '); |
273 | 276 | mutateDom();
|
274 | 277 |
|
275 | 278 | if (scrollIntoView) {
|
|
317 | 320 | if (hasClass(pre, LINE_NUMBERS_CLASS) && hasLineNumbers && !isLineNumbersLoaded) {
|
318 | 321 | Prism.hooks.add('line-numbers', completeHook);
|
319 | 322 | } else {
|
320 |
| - var mutateDom = highlightLines(pre); |
| 323 | + var mutateDom = Prism.plugins.lineHighlight.highlightLines(pre); |
321 | 324 | mutateDom();
|
322 | 325 | fakeTimer = setTimeout(applyHash, 1);
|
323 | 326 | }
|
|
328 | 331 | var actions = $$('pre')
|
329 | 332 | .filter(isActiveFor)
|
330 | 333 | .map(function (pre) {
|
331 |
| - return highlightLines(pre); |
| 334 | + return Prism.plugins.lineHighlight.highlightLines(pre); |
332 | 335 | });
|
333 | 336 | actions.forEach(callFunction);
|
334 | 337 | });
|
|
0 commit comments