Highly performant list and datagrid for AngularJS designed for mobile devices which just makes it all the better on a desktop.
Most lists work with recycled or just in time created rows for long lists in angular. Datagrid doesn't work that way because in order to get a smooth scrolling experience on a mobile device you need to allow the browser to handle the scrolling as much as possible. So to accomplish this Datagrid works on a chunking algorithm to create pieces as they come into view, but never destroy them until the whole grid is destroyed. This allows the scrolling experience to be smooth because GPU is leveraged from the browser as the dom is considered to be static until in view.
Of course since this is angular there is then the issue with a digest being too expensive. This is not a problem because all rows are disabled from the digest when out of view and enabled when in view. So they don't loose their scopes or have to create new ones constantly, but only once. Then they are just enabled or disabled as they come into view.
Chunking is the concept of grouping the dom elements in hierarchies so that the browser does not calculate the position of every element when one changes, but only those in that container, it's parents in it's container, and so on up the chain. The smaller the groups the less the browser has to redraw, size, and position elements because it can just move the parent of necessary. This works regardless of rows all having the same heights or dynamic heights as long as there is a template for each different height.
Using this concept datagrid is able to create many more rows that are listed natively like static html and allows the browser to use GPU snapshots to keep scrolling smooth. Dynamic changes cause repaints that are expensive, so this avoids that.
Chunking doesn't have to create them all though. To make sure the datagrid could start up and jump to any section quickly it needed to be able to only put in chunks that are in view. So once a chunk is requested it will be created if it was not already. This makes it so that a list of 50,000 items can still start up quickly and jump to any location and scroll as smooth as glass on your mobile device.
Datagrid uses the concept of templates for your rows. You can have as many templates as you like. You can even overwrite the templateModel for one of your own as long as you implement the same API. You just need to assign it as an addon. (We will discuss those later).
First we will create the directive and assign it an array so it has something to display.
<div ux-datagrid="items"></div>
These items are read from the scope that the datagrid exists in. It will then take that data and convert it into a normalized array so it can display every item. This does nothing for single dimentional arrays, but for arrays that are multi-dimentional it creates a single array of those based on the grouped property.
Grouped property is for reading a hiearchy. If you have an object with the property children that is an array, then "children" would be your grouped property for the datagrid to read it. See this example for use of grouped data. Grouped Data Example
In most cases your datagrid will look similar to this one below.
<div ux-datagrid="items">
<script type="template/html" data-template-name="default" data-template-item="item">
<div class="row">{{item}}</div>
</script>
</div>
What is really happening here is that the directive is setup with a script template inside of it. The script template allows the contents of it to be read as a string rather than be interpreted as dom by the browser. So it makes a very convenient way of writing out your row templates.
The script template has 3 properties.
- type="template/html" it uses this to match on. So if you don't get it right might not work. This is required.
- template-name if this is not defined it will be changed to 'default'. So this one is not required if you only have one template, but if you have more than one the second will overwrite the first without this property.
- template-item this is what you want the data to be referenced by on the row scope. Such as in angular if you do a repeat with "item in items" you then reference item in your template. This does the same thing with this property.
Addons are like a plugin to the datagrid. It actually becomes a behavior modifier of the datagrid object. Addons are created as factories and applied directly to the datagrid instance. The internals of the datagrid are constructed with addons as well.
Because these addons assign themselves as behaviors to the instance behaviors can be overwritten. So if you need to modify core behavior then go right ahead and add an addon that overwrites those methods.
How to create an addon Addons are factories that modify the behavior. To create an addon you just need to create the factory and then add it to the dom.
angular.module('ux').factory('datagridVersion', function () {
return function (datagrid) {
datagrid.version = "0.1.0";
};
});
Now we need to add it to the dom.
<div ux-datagrid="items" data-addons="datagridVersion">...</div>
Now when this datagrid is created it will call the "datagridVersion" factory and pass to it the datagrid instance so that it can be modified.
Any public methods can be overwritten allowing you to change internal behavior. Not all methods are public, but hopefully the ones you need.
It is also handy to have addons that will update based on events. Here is an example of an addon that updates based off of some events that the datagrid will throw. For this example we will assume we are waiting for some data to load asynchronously through a service. So we want a spinner until the data is loaded.
angular.module('ux').factory('listLoader', function () {
return function (datagrid) {
datagrid.unwatchers.push(datagrid.scope.$on(ux.datagrid.events.AFTER_UPDATE_WATCHERS, function () {
if (datagrid.data.length) {
alert('loading is complete');
}
}))
};
})