Как бороться с замораживанием браузера из-за вложенного ng-repeat

Я создал вложенное дерево, которое может содержать 1-5000 элементов, я могу заставить его работать, но он замораживает мой браузер\загрузка spinner в течение нескольких секунд перед показом дерева.

Как я могу сделать его гладким, чтобы браузер никогда не зависал?

Как я могу узнать, когда angularjs закончил создание или рендеринг или вычисление (не уверенное правильное слово) всего списка, чтобы я мог удалить загрузку, тогда как вы можете видеть область. $last не будет работать, как мы вложенное ng-repeat и такое же для области. $parent. $last

Вот плункер, который я создал, но с демо-данными -

http://plnkr.co/edit/GSZEpHjt5YVxqpg386k5?p=preview

Пример набора данных - http://pastebin.com/YggqE2MK

Это не так уж плохо в этом примере, но в точках мой браузер зависает более чем на 10 секунд для около 4000 элементов в моей настройке OWN со всеми другими компонентами.

То, что я уже рассмотрел

  • библиотека bindonce.js, но без особого успеха
  • используется "::" единственное связывание без особого успеха.
  • загружает только первый уровень, а затем выводит следующий уровень, когда пользователь расширяет категорию, но мое дерево может быть очень случайным, возможно, у меня есть только один единственный корень node с дочерними узлами 100s.
  • разбиение на страницы не идеально подходит для сценария

HTML

  <script type="text/ng-template" id="tree_item">
    <div ng-init="category.expanded=true">
      <div ng-class="{'selected': category.ID==selectedCategoryID}">
        <div class="icon icon16" ng-if="category.Children.length>0" ng-click="$parent.category.expanded=!$parent.category.expanded" ng-class="$parent.category.expanded?'col':'exp'"></div>
        <div class="icon icon-category" ng-class="'icon-category-'+category.TypeType" ng-style="{'border-color':category.Colour}" ng-attr-title="{{category.Status?Res['wpOPT_Status'+category.Status]:''}}"></div>
        <a ng-href="#id={{category.ID}}" ng-class="{'pending-text': category.PendingChange}">{{category.Name}}</a>
      </div>
      <ul class="emlist" ng-show="category.expanded">
        <li ng-repeat="category in category.Children | orderBy:'Name'" ng-include="'tree_item'" ng-class="{'selected': category.ID==selectedCategoryID}" e2-tree-item is-selected="category.ID==selectedCategoryID">
        </li>
      </ul>
    </div>
  </script>

  <div id="CategoryListContainer" class="dragmenu-container initial-el-height">
    <div class="spinner" data-ng-show="status=='loading'"></div>
    <ul id="CategoryList" class="dragmenu-list ng-cloak" ng-show="status=='loaded'">
      <li ng-repeat="category in tree | orderBy:'Name'" ng-include="'tree_item'" e2-tree-item is-selected="category.ID==selectedCategoryID"></li>
    </ul>
  </div>

JS

var app = angular.module('recursionDemo', []);

app.controller("TreeController", function($scope, $timeout) {
  $scope.status = "loading";

  $timeout(function() {
        var result = {
      "GetCategoryTreeResult": [{ // too big data set so I pasted it here http://pastebin.com/YggqE2MK }];
    $scope.tree = result.GetCategoryTreeResult;

    $scope.status = "loaded";
  }, 3000);
});

app.directive('e2TreeItem', function($timeout) {
  function link(scope, element, ngModel) {
    scope.$watch('isSelected', function(oldVal, newVal) {
      if (scope.isSelected === true) {
        element.parentsUntil('#CategoryListContainer', 'li').each(function(index, item) {
          angular.element(item).scope().category.expanded = true;
        });
      }
    });

    // not working
    //if (scope.$parent.$last) {
    //  console.log("last has been caught");
    //  var appElement = document.querySelector('[ng-app=recursionDemo]');
    //  angular.element(appElement).scope().status = "loaded";
    //}
  }
  return {
    link: link,
    scope: {
      isSelected: '=?'
    }
  };
});

Ответ 1

UPDATE:

DEMO: http://plnkr.co/edit/bHMIU8Mx5grht9nod1CW?p=preview

Вы можете легко загрузить свои данные, изменив L10901 app.js, как показано ниже. Но вы должны заметить, что если вы все еще хотите заказать его по имени и хотите, чтобы он был устойчивым, вы должны просто отсортировать исходные данные, но не сортировать в angular с "order by". (Вы хотите, чтобы какой-то код сортировал исходные данные?)

    function getDataGradually(data, childKey, gap, count, updateCb, finishCb) {
        const lastPositon = [];
        const linearData = [];
        const ret = [];


        function getLinearData(arr, position) {
            arr.forEach(function (obj, index) {
                const pos = position.concat([index]);
                if (obj[childKey] && obj[childKey].length) {
                    var children = obj[childKey];
                    obj[childKey] = [];
                    linearData.push({
                        obj,
                        pos
                    });
                    getLinearData(children, pos);
                } else {
                    linearData.push({
                        obj,
                        pos
                    });
                }
            });
        }
        getLinearData(data, []);

        function insertData({
            obj,
            pos
        }) {

            let target = ret;
            pos.forEach(function (i, index) {


                if (index === pos.length - 1) {
                    target[i] = obj;

                } else {
                    target = target[i][childKey];
                }
            });

        }
        let handled = 0;

        function doInsert() {
            let start = handled;
            let end;
            if (handled + count > linearData.length) {
                end = linearData.length;
            } else {
                end = handled + count;
            }
            for (let i = start; i < end; i++) {
                insertData(linearData[i]);
            }
            handled += count;
            if (handled < linearData.length) {
                setTimeout(function () {
                    doInsert();
                    updateCb && updateCb();
                }, gap);
            } else {
                finishCb && finishCb();
            }
        }
        doInsert();



        return ret;
    }

    $scope.tree = getDataGradually(result.GetCategoryTreeResult, "Children", 0, 30, function () {
      $scope.$digest();
    }, function () {
      //finished 
    });
    //$scope.tree = result.GetCategoryTreeResult;
    $scope.status = "loaded";
  }, 3000);

Старый ответ:

Вы можете получить время рендеринга, используя $timeout. Попробуйте изменить L10901 app.js, как показано ниже.

        $scope.tree = result.GetCategoryTreeResult;
        console.time("A")
        $timeout(function(){
          console.timeEnd("A");
        });
        $scope.status = "loaded";
      }, 3000);

Кстати, я получил "1931.881мс" на моем компьютере.

Ответ 2

Как я справляюсь с этим:

В контроллере:

// Consider results to be my main array, which contains 5000 items
var results = [{ ... }, { ... }, ..., ..., { ... }];

$scope.bindResults = results.slice(0, 5);

// Here you can start loading
$scope.status = "loading";

// Inject $interval in controller
var interval = $interval(function() {
    $scope.bindResults = results.slice(0, $scope.bindResults.length + 5);

    if($scope.bindResults.length >= results.length) {
         $interval.cancel(interval);

        // Here you can end loading
        $scope.status = "loading";
    }
}, 10);

В HTML:

<span ng-repeat="x in bindResults"></span>

Это не повредит браузер и будет хранить данные каждые 10 мс, вы можете увеличить это число по мере необходимости.

Ответ 3

Из моего опыта:

Также это старая, но хорошая статья о производительности angularjs с ng-repeat Оптимизация AngularJS: от 1200 мс до 35 мс

Ответ 4

Поскольку вы привязываете большие данные к HTML, он внутренне строит так много наблюдателей. Это, безусловно, снизит производительность и может зависнуть в системах с низкой конфигурацией.

Создайте эту директиву как элемент, а не атрибут. Затем выполните манипуляции с DOM в директиве. Это определенно решит проблему.