转换/移动大量DOM元素的最佳方式

LJ *_*ski 10 html javascript css jquery angularjs

我不久前为AngularJS(使用Angular Material)构建了一个树表结构.

我的目标是让它只在大屏幕上工作(1280及更高),但现在我想更新它并使其适用于较小的设备(主要是平板电脑)而不限制数据.由于性能的原因,我希望保持HTML尽可能简单(树表可以有1000多行,因此为单行创建更复杂的HTML将延长追加和呈现表行所需的时间(行是动态的,所以它不仅仅是关于初始渲染)).

我想到我将保持"固定"部分与包含名称的第一个单元格并滚动包含所有指标的第二部分并将同步滚动.

当前单行HTML:

<div class="tb-header layout-row">
  <div class="tb-title"><span>Name</span></div>
  <div class="tb-metrics">
     <div class="layout-row">
        <div class="tb-cell flex-10">812</div>
        <div class="tb-cell flex-7">621</div>
        <div class="tb-cell flex-4">76.5</div>
        <div class="tb-cell flex-7">289</div>
        <div class="tb-cell flex-4">46.5</div>
        <div class="tb-cell flex-7">308</div>
        <div class="tb-cell flex-4">49.6</div>
        <div class="tb-cell flex-7">390</div>
        <div class="tb-cell flex-4">48.0</div>
        <div class="tb-cell flex-7">190</div>
        <div class="tb-cell flex-4">23.4</div>
        <div class="tb-cell flex-7">0</div>
        <div class="tb-cell flex-4">0.0</div>
        <div class="tb-cell flex-8">6.4</div>
        <div class="tb-cell flex-8">0.0</div>
        <div class="tb-cell flex-8"></div>
     </div>
  </div>
Run Code Online (Sandbox Code Playgroud)

我的想法是touchmove在父容器上使用事件(包装整个树并绑定为指令)并检查何时touchmove启动指标部分然后计算我应移动指标的值.那部分工作正常.当我想在上面应用偏移时,问题就开始了.tb-metrics >.

我的第一次尝试是使用jQuery:

function moveMetrics( offset ) {
  var ofx = offset < 0 ? (offset < -getMetricsScrollWidth() ? -getMetricsScrollWidth() : offset) : 0;
  $('.tb-metrics').children().css('transform', 'translateX(' + ofx + 'px)');

  /*...*/

}
Run Code Online (Sandbox Code Playgroud)

不幸的是,当表包含更多行时,此解决方案非常慢(我无法缓存行,因为它们是动态的).

在我的第二次尝试中,我试图避免尽可能多的DOM操作.为了实现这一点,我决定将<script>标签添加到包含适用于的css的dom .metrics > .layout-row.

解:

function moveMetrics( offset ) {
  var ofx = offset < 0 ? (offset < -getMetricsScrollWidth() ? -getMetricsScrollWidth() : offset) : 0
    , style = $( '#tbMetricsVirtualScroll' )
    ;

  if ( !style.length ) {
    body.append( '<style id="tbMetricsVirtualScroll">.tb-metrics > * {transform: translateX(' + ofx + 'px)}</style>' );
    style = $( '#tbMetricsVirtualScroll' );
  } else {
    style.text( '.tb-metrics > * {transform: translateX(' + ofx + 'px)}' );
  }
  /*...*/
}
Run Code Online (Sandbox Code Playgroud)

但是,当表包含大量行时,它似乎不会快得多.所以这不是DOM操作,但渲染/绘制视图似乎是这里的瓶颈.

我尝试创建某种虚拟滚动但是因为树结构对于不同的数据集是不同的,并且可以具有"无限"数量的级别(每行可以包含新的子行ng-repeat),这是一项非常艰巨的任务.

我会很感激有关如何在不使用虚拟滚动的情况下提高该情况下的性能的任何想法.

编辑:

Chrome时间轴的屏幕截图显示大部分滚动时间都是通过渲染消耗的(我猜这是因为复杂的DOM结构)

在此输入图像描述

编辑2:

我不会说我实现了绝对平滑的滚动,但我发现了一些显着的性能改进(其中一些并不明显,结果比我预期的那么小的变化后更好).

  • 简化类选择器: .tb-header > .tb-metrics > .tb-cell.tb-specific-cell它慢得多(似乎需要更多时间来解析更复杂的选择器?)
  • 从转换后的元素中删除不透明度和框阴影
  • 尝试将变换后的元素分发到新层(使用css will-change和/或translateZ(0))

Jah*_*uez -4

所以我自己不是专家,但我确实有一些我认为值得一提的想法。如果我理解正确的话,问题是这个项目需要越来越多的 HTML 元素,这会不断降低性能

这里有 3 个我认为对你有帮助的 js 概念

*动态创建html元素:

let div = document.createElement('div');
Run Code Online (Sandbox Code Playgroud)

*动态创建要用于元素的属性,并动态地分配此类属性

function attributeSetter(atr){
   div.setAttribute('class', atr)
}
Run Code Online (Sandbox Code Playgroud)

这样,您就可以在需要的地方调用该函数,并动态地为其指定任何类名称。此外,您可以根据需要多次使用 div 元素

最终概念。您还可以动态添加预制类,假设它们已经在 css 中预定义,如下所示。

div.classList.add("my-class-name")
Run Code Online (Sandbox Code Playgroud)

另一个可能派上用场的概念是解构

let myClasses = ['flex-10', 'flex-7', 'flex-4', 'flex-9'] //etc
let [flex10, flex7, flex4, flex9] = myClasses;
Run Code Online (Sandbox Code Playgroud)

这会将所有这些字符串转换为您可以轻松迭代的变量,并将它们动态添加到 attributeSetter 函数中,如下所示:

attributeSetter(flex10)
Run Code Online (Sandbox Code Playgroud)

无论你在哪里需要它。我不知道这是否有帮助,或者是否回答了你的问题,但至少有一些想法。祝你好运 :)