editor/performance: Fix performance for large pads (#4267)

* Fix line numbers top padding

This old rule was conflicting with new css rules introduced in 1.8.4

* Fixes #4228 Performance degradation for long pads 

Due to layout trashing when calculating new heights
This commit is contained in:
Sebastian Castro 2020-09-08 15:52:26 +02:00 committed by GitHub
parent da459888dc
commit 818194da90
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
2 changed files with 49 additions and 63 deletions

View file

@ -38,10 +38,6 @@ html.inner-editor {
word-wrap: break-word; word-wrap: break-word;
overflow-wrap: break-word; overflow-wrap: break-word;
} }
#innerdocbody, #sidediv {
padding-top: 10px; /* Both must have same top padding, so line number are aligned with the rows */
padding-bottom: 10px; /* some space when we scroll to the bottom */
}
#innerdocbody a { #innerdocbody a {
color: #2e96f3; color: #2e96f3;

View file

@ -5224,53 +5224,53 @@ function Ace2Inner(){
$(sideDiv).addClass("sidediv"); $(sideDiv).addClass("sidediv");
} }
// We apply the height of a line in the doc body, to the corresponding sidediv line number
function updateLineNumbers() function updateLineNumbers()
{ {
if (!currentCallStack || currentCallStack && !currentCallStack.domClean) return;
// Refs #4228, to avoid layout trashing, we need to first calculate all the heights,
// and then apply at once all new height to div elements
var lineHeights = [];
var docLine = doc.body.firstChild;
var currentLine = 0;
var h = null;
// First loop to calculate the heights from doc body
while (docLine)
{
if (docLine.nextSibling) {
if (currentLine === 0) {
// It's the first line. For line number alignment purposes, its
// height is taken to be the top offset of the next line. If we
// didn't do this special case, we would miss out on any top margin
// included on the first line. The default stylesheet doesn't add
// extra margins/padding, but plugins might.
h = docLine.nextSibling.offsetTop - parseInt(window.getComputedStyle(doc.body).getPropertyValue("padding-top").split('px')[0]);
} else {
h = docLine.nextSibling.offsetTop - docLine.offsetTop;
}
} else {
// last line
h = (docLine.clientHeight || docLine.offsetHeight);
}
lineHeights.push(h)
docLine = docLine.nextSibling;
currentLine++;
}
var newNumLines = rep.lines.length(); var newNumLines = rep.lines.length();
if (newNumLines < 1) newNumLines = 1; if (newNumLines < 1) newNumLines = 1;
//update height of all current line numbers var sidebarLine = sideDivInner.firstChild;
var a = sideDivInner.firstChild; // Apply height to existing sidediv lines
var b = doc.body.firstChild; currentLine = 0
var n = 0; while (sidebarLine && currentLine <= lineNumbersShown) {
if (lineHeights[currentLine]) {
if (currentCallStack && currentCallStack.domClean) sidebarLine.style.height = lineHeights[currentLine] + "px";
{
while (a && b)
{
if(n > lineNumbersShown) //all updated, break
break;
var h = (b.clientHeight || b.offsetHeight);
if (b.nextSibling)
{
// when text is zoomed in mozilla, divs have fractional
// heights (though the properties are always integers)
// and the line-numbers don't line up unless we pay
// attention to where the divs are actually placed...
// (also: padding on TTs/SPANs in IE...)
if (b === doc.body.firstChild) {
// It's the first line. For line number alignment purposes, its
// height is taken to be the top offset of the next line. If we
// didn't do this special case, we would miss out on any top margin
// included on the first line. The default stylesheet doesn't add
// extra margins/padding, but plugins might.
h = b.nextSibling.offsetTop - parseInt(window.getComputedStyle(doc.body).getPropertyValue("padding-top").split('px')[0]);
} else {
h = b.nextSibling.offsetTop - b.offsetTop;
}
}
if (h)
{
var hpx = h + "px";
if (a.style.height != hpx) {
a.style.height = hpx;
}
}
a = a.nextSibling;
b = b.nextSibling;
n++;
} }
sidebarLine = sidebarLine.nextSibling;
currentLine++;
} }
if (newNumLines != lineNumbersShown) if (newNumLines != lineNumbersShown)
@ -5278,32 +5278,22 @@ function Ace2Inner(){
var container = sideDivInner; var container = sideDivInner;
var odoc = outerWin.document; var odoc = outerWin.document;
var fragment = odoc.createDocumentFragment(); var fragment = odoc.createDocumentFragment();
// Create missing line and apply height
while (lineNumbersShown < newNumLines) while (lineNumbersShown < newNumLines)
{ {
lineNumbersShown++; lineNumbersShown++;
var n = lineNumbersShown;
var div = odoc.createElement("DIV"); var div = odoc.createElement("DIV");
//calculate height for new line number if (lineHeights[currentLine]) {
if(b){ div.style.height = lineHeights[currentLine] +"px";
var h = (b.clientHeight || b.offsetHeight);
if (b.nextSibling){
h = b.nextSibling.offsetTop - b.offsetTop;
}
} }
$(div).append($("<span class='line-number'>" + String(lineNumbersShown) + "</span>"));
if(h){ // apply style to div
div.style.height = h +"px";
}
$(div).append($("<span class='line-number'>" + String(n) + "</span>"));
fragment.appendChild(div); fragment.appendChild(div);
if(b){ currentLine++;
b = b.nextSibling;
}
} }
container.appendChild(fragment); container.appendChild(fragment);
// Remove extra lines
while (lineNumbersShown > newNumLines) while (lineNumbersShown > newNumLines)
{ {
container.removeChild(container.lastChild); container.removeChild(container.lastChild);