Материальная угловая бесконечная прокрутка с запросом $http

Я использую директиву md-virtual-repeat для Angular Material для бесконечной прокрутки, и мне нужно заменить ее демо Функция $timeout с запросом $http. Но я не могу найти правильное решение. В приведенном ниже коде бесконечная прокрутка работает нормально, но не отображает данные из http-запроса. Проблема в том, что я не знаю, как связать результат $http с infiniteItems.

Вот плункер.

Index.html

<body ng-app="infiniteScrolling" class="virtualRepeatdemoInfiniteScroll">
<div ng-controller="AppCtrl as ctrl" ng-cloak>
    <md-content layout="column">
        <md-virtual-repeat-container id="vertical-container" flex>
            <div md-virtual-repeat="item in ctrl.infiniteItems" md-on-demand
                 class="repeated-item" flex>
                {{item.id}}
            </div>
        </md-virtual-repeat-container>
    </md-content>
</div>
</body>

JS:

(function () {
'use strict';
angular
  .module('infiniteScrolling', ['ngMaterial'])
  .controller('AppCtrl', function ($timeout,$scope,$http) {
     this.infiniteItems = {
          numLoaded_: 0,
          toLoad_: 0,
          items:[],
          getItemAtIndex: function (index) {
              if (index > this.numLoaded_) {
                  this.fetchMoreItems_(index);
                  return null;
              }
              return index;
          },
          getLength: function () {
              return this.numLoaded_ + 5;
          },
          fetchMoreItems_: function (index) {
               if (this.toLoad_ < index) {
                  this.toLoad_ += 20;

                  $http.get('items.json').success(function (data) {
                      var items = data;
                      for (var i = 0; i < items.length; i++) {
                          this.items.push(items[i].data);
                      }
                      this.numLoaded_ = this.toLoad_;
                  }.bind(this));
              }
          }
      };
   });
})();

person Asieh Mokarian    schedule 17.11.2015    source источник


Ответы (3)


Этот работает:

plnkr

  • getItemAtIndex вернул индекс, а не элемент
  • если бы вы проверили, что вы нажали, вы бы увидели, что в строке 33 в моем plunkr я concat obj.data, а не обычный obj
(function () {
    'use strict';
    angular.module('infiniteScrolling', ['ngMaterial'])
      .controller('AppCtrl', function ($scope, $http) {
          // In this example, we set up our model using a plain object.
          // Using a class works too. All that matters is that we implement
          // getItemAtIndex and getLength.
          var vm = this;
          vm.infiniteItems = {
              numLoaded_: 0,
              toLoad_: 0,
              items: [],

              // Required.
              getItemAtIndex: function (index) {
                  if (index > this.numLoaded_) {
                      this.fetchMoreItems_(index);
                      return null;
                  }
                  return this.items[index];
              },

              // Required.
              getLength: function () {
                  return this.numLoaded_ + 5;
              },

              fetchMoreItems_: function (index) {
                  if (this.toLoad_ < index) {
                      this.toLoad_ += 5;
                      $http.get('items.json').then(angular.bind(this, function (obj) {
                          this.items = this.items.concat(obj.data);
                          this.numLoaded_ = this.toLoad_;
                      }));
                  }
              }
          }
      })
})();
person Bhudjo    schedule 17.11.2015
comment
Спасибо, это работает. Но я не знаю, почему для toLoad_+=20 , когда я иду вниз по прокрутке, данные загружаются с большой задержкой, очень вниз никогда не загружаются не то же самое для toLoad_+=10 . - person Asieh Mokarian; 17.11.2015
comment
Ну, на самом деле он создает элемент в DOM, но поскольку ваш API возвращает 5 элементов, iirc, у него недостаточно данных для инициализации, поэтому они пусты. - person Bhudjo; 18.11.2015
comment
Спасибо, вы правы. Я просто хочу добавить еще один момент: я думаю, что замена data на obj.data не является частью решения, поскольку я использовал success вместо then, и он возвращает данные, а не результат, но использование concat является частью решения. Потому что когда я удаляю этот код, он работает неправильно. У него есть недостаток, поскольку он хранит все данные (все данные вне поля зрения) в памяти, а не только данные в поле зрения. - person Asieh Mokarian; 18.11.2015
comment
@A.M О решении success vs then, которое я представил: success и error были устарели - person Bhudjo; 20.11.2015
comment
Это действительно работает для меня, за исключением того, что я хотел бы, чтобы список перестал прокручиваться после отображения всех данных. Это решение снова повторяет json.data с самого начала. - person lunacafu; 07.03.2016
comment
Кто-нибудь разобрался, как остановить прокрутку? - person rcheuk; 21.04.2016

При каждом вызове API пытайтесь узнать, есть ли в БД еще несколько записей или нет. и добавьте это условие в функцию fetchMoreItems_.

fetchMoreItems_: function (index) {
              if (this.toLoad_ < index && hasMoreRecords) {
                  this.toLoad_ += 5;

В нашем коде мы получаем такие детали, как

  • sCurrentPage : 3
  • sMore: true ==>> это указывает, что в БД есть больше записей или нет после выборки постраничных данных.
  • Всего страниц: 4
  • sTotalRecords: 36
person Raju Donthula    schedule 03.06.2016
comment
И вы говорите, что это создает элемент в DOM, и если мы получаем меньшие записи, которые показывают пустые записи. Итак, как я могу удалить или остановить создание этих пустых записей. Не могли бы вы помочь... - person Raju Donthula; 03.06.2016

пришел сюда и увидел ответ @alessandro-buggin, который был очень полезным. Мне пришлось немного изменить его, поэтому я решил поделиться им с другими, чтобы помочь. Мне было нужно:

  • чтобы избежать получения запросов на прокрутку при уже восстановленных данных (используя this.hold)
  • остановка запросов, когда все данные были получены от бэкэнда (используя this.stop_)
  • скрытие содержимого при загрузке, чтобы избежать сбоев или пустых элементов (опять же с помощью this.hold). В представлении вам нужно использовать ng-hide для этого элемента, потому что ng-if избегайте существования элемента, поэтому он не будет загружаться в первый раз.
  • реализация метода обновления для перезагрузки данных при изменении параметров/фильтров из внешней формы.

Далеко не идеально, но работает довольно хорошо.

vm.elements = null;
vm.infiniteItems = { // Start of infinte logic

stop_: false,
hold: false,
numLoaded_: 0,
toLoad_: 0,
items: [],

refresh: function() {
    this.stop_ = false;
    this.hold = false;
    this.numLoaded_ = 0;
    this.toLoad_ = 0;
    this.items = [];
},

getItemAtIndex: function (index) {
    if (!this.hold) {
        if (index > this.numLoaded_) {
            this.fetchMoreItems_(index);
            return null;
        }
    }
    return this.items[index];
},

getLength: function () {
    if (this.stop_) {
        return this.items.length;
    }
    return this.numLoaded_ + 5;
},

fetchMoreItems_: function (index) {
    if (this.toLoad_ < index) {

        this.hold = true;
        this.toLoad_ += 5;

        var start = this.numLoaded_;
        if (start > 0) start++;

        MyService.getData(parameters)
         .then(angular.bind(this, function (obj) {

          if (obj && obj.elements > 0) {
            vm.elements = obj.elements;
            this.items = this.items.concat(obj.data);

            if (obj.elements < this.toLoad_) {
                this.stop_ = true;
            }
            this.numLoaded_ = this.items.length;
            this.hold = false;

          } else { // if no data
            vm.elements = 0;
          }
        }));
    }
}

} // End of infinte logic

Примечание: моя служба возвращает объект, составленный следующим образом: obj = {elements: INTEGER, data: ARRAY} где elements сообщают вам длину полного запроса.

person paulie    schedule 27.01.2017