Search code examples
angularjsvirtualscrollui-scroll

How can I use "ui-scroll" with my own data?


I'm trying to create a infinite scroll feature in my application but it feels a bit abstract. I want to use ui-scroll and this fiddle shows a simple example of how it works.

I've read the readme and looked through some examples also I've integrated the example in my project and got it working, but I can't figure out on how to combine this with data from my own database.

I have a database table called movies. The movies have a few values such as title, release_date, image_url

How would I insert that data into the $scope.movieDataSource so I can use it in my view?

$http.get(('/movies.json'), {
    cache: true
  })
  .success(function(data, status, headers, config) {
    if (status == 200) {
      $scope.userMovies = data;
    } else {
      console.error('Error happened while getting the user list.')
    }
    $scope.movieDataSource = {
      get: function(index, count, callback) {
        var i, items = [$scope.userMovies], item;
        var min = 1;
        var max = 1000;

        for (i = index; i < index + count; i++) {
          if (i < min || i > max) {
            continue;
          }
          item = {
            title: $scope.userMovies.title,
            imageURL: $scope.userMovies.poster_path
          };
          items.push(item);
        }
        callback(items);
      }
    }
  });

I've tried to create an example of what I'm trying to get at. I use a http.get to fill my userMovies scope with records from my database and I want to use those records as items in the movieDataSource.

But when I visit the page I that ui-scroll does add results in the container, but it does not show content.

<div class="imageCell ng-scope" ui-scroll="item in movieDataSource">
  <img title="">
</div>

If I console.log("movieDataSource" + $scope.movieDataSource) it shows me movieDataSource[object Object].


Solution

  • You are making this more complex than necessary. The uiScroll directive is a replacement for ngRepeat, which takes a Data Source with 3 properties:

    • index indicates the first data row requested
    • count indicates number of data rows requested
    • success function to call when the data are retrieved. The implementation of the service has to call this function when the data are retrieved and pass it an array of the items retrieved. If no items are retrieved, an empty array has to be passed.

    in your case, you have an array of items. Each time the index or count changes, the success fires, and this function should return a subset of your array from index to index + count. There are multiple ways to accomplish this. The example you posted uses a for loop to iteratively push items into the array. You could also use the Array.slice() method.

    Option 1:

     $scope.movieDataSource = {
    
       get: function(index, count, callback) {
         var i, items = [], item;
    
         for (i = index; i < index + count; i++) {
           item = $scope.userMovies[i];
           items.push(item);
         };
    
         callback(items);
       }
     }
    

    Option 2:

     $scope.movieDataSource = {
    
       get: function(index, count, callback) {
         var items = $scope.userMovies.slice(index, index + count);
         callback(items);
       }
     }
    

    As for your HTML, it should be identical to if you were using ng-repeat:

    <div ui-scroll="item in movieDataSource">
      {{item.title}}
      <img title="{{item.title}}" ng-src="{{item.poster_path}}"></img>
    </div>