#ngDynamicColumns
License: The MIT License.
ngDynamicColumns is providing two angularJS directives to render a datatable with full dynamic columns. That means, that the order of the columns in a datatable is not hard-coded but can be freely changed. A column can be toggled on and off too.
The cell contents and the contents of the headers are defined with their own directives so that they can be as complex and flexible as angularJS allows. You can use every angularJS feature in them as you like.
See it in action on the Demo site
##Installation
- with bower:
bower install ngDynamicColumns
(not yet!) - manual: download
ngDynamicColumns.js
Inside your angularJS app define a dependency to 'ngDynamicColumns' like this: angular.module("app", ["ngDynamicColumns"])
The current version is only tested with angularJS 1.3.x, but it should run on older version too. Even 1.0.x versions should be working without problems.
An ES5 browser is needed when using this module. If you have to support IE 8 and below, use es5-shim.
##Usage
ngDynamicColumns currently is specifically designed to render the <td> and <th> tags in complex and/or huge html tables. If you only have small and simple tables, this module is probably not for you.
This is an example usage:
<table>
<thead>
<tr column-header="columns"/>
</thead>
<tbody>
<tr ng-repeat="item in items" dynamic-row="columns" />
</tbody>
</table>
###dynamic-row directive
This directive should be used in conjunction with a ng-repeat on a tr element and renders for every repeat a set of td elements in the desired order. This directive needs a column configurstion object to get the neccessary information how to render the columns/td tags. See below for info about this configuration object.
###column-header directive
This directive should be used on a tr element in a thead section and renders th elements according to the column configuration object.
##Configuration
Both directives need the same column configuration to render the columns. The configuration should be an array of objects which need to have these mandatory properties:
{String} id
A unique id for the column
{String} rowDirective
The directive name used to render the cells (<td> tags). This directive is used to render your table data and you need to implement this directive yourself.
{String} columnDirective (optional when not using column-header directive)
The directive name used to render the header (<th> tags). This directive is used when you want to have a header in your table and you have to implement the directive yourself. If you don't use the column-header directive you can omit this property.
{boolean} visible
Indicates if this column is visible or not. Note, that the DOM elements for the column will be created even when currently not visible. This is done to speed up performance when the column will be displayed later. The class 'ng-hide' of angularJS is used to hide the column when not visible.
{String} clazz (optional)
If set, this class will be set on every element in your column (<td> and <th> tags), so you can style your columns individually.
{Object} scopedAttrs (optional)
If set, they key/value of the given object will be rendered as attributes on the element. This can be used in conjunction with directives which need an isolated scope and therefore cannot access their parent scope. You must reference the attributes in your directive, so angularJS will inject them. Look into the demo application for an example ('isoolated scope' column)
Example column configuration:
$scope.columns = [
{"id": "column1", rowDirective: "column1cell", columnDirective: 'column1header', visible: true, clazz: 'column1class'},
{"id": "column2", rowDirective: "column2cell", columnDirective: 'column2header', visible: false, clazz: 'column2class', scopedAttrs: {'my-attr': 'myObject'}},
];
##Scope and column directives (templates)
Both directives don't use an isolated scope which is normally a good practive to avoid conflicts with surrounding scopes.
But as you are writing your own directives which will be used by this module to render your datatable, isolated scope would mean, that you would have no access whatsoever to your application controller. An alternative would be to only give access to some to-be-defined properties, which would complicate usage quite a lot.
Therefore, ngDynamicColumns is using its parent scope (=your application controller), so you can access your controller properties and methods in your column templates (aka directives).
If you don't use isolated scope in your own directives you can directly access the controller scope in which you
rendering the table. This is the quick and easy way to write your column directives.
If you need isolated scope in your directives however, you have to use the 'scopedAttrs'
configuration in your column configuration, to make use of normal angularJS databinding features (one-way, two-way,
function calls). Please have a look into the demo application to see a working example.
To minimize its footprint and avoid conflicts, this module is adding no properties whatsoever on the scope.
##Events
As ngDynamicColumns was designed to be used with complex and huge tables in mind, it does not use any watchers by itself to minimize impact on performance (though your column and row directives can use watchers of course). It therefore doesn not do anything by itself to avoid unwanted DOM operations.
So, to rerender the table after changes in the column configuration you have to emit events on the $rootScope. ngDynamicColumns acts on these events:
- columnToggled - this event expects a column ID as transmitted parameter and toggles visibility of said column ID. To toggle the visibility it sets and removes the ng-hide class on <td> and <th> elements
- recreateColumns - this rerenders the whole <table> with all elements. Warning: this can have huge impact on rendering performance when your table has a lot of DOM elements, as all elements are deleted and recreated again. You can use this event, when column order has changed for instance. It does not need any parameters
- columnOrderChanged - this event moves one column to the place of another column. It expects two paramters: the
source column ID and the destination column ID. This can be used for instance after a drag and drop action by the user
who moves columns around. As this does not delete elements from the DOM but moves existing elements to another place, its
very fast even in huge and complex tables.
The moved column will be inserted after the destination column depending on the move direction. If you move a column to the right, the moved column will be inserted to the right of the destination column. If you move a column to the left, the moved column will be inserted to the left of the destination column. This is done, so a user can actually move a column as first column with only one drag&drop action.
Warning: This also changes the column configuration array in your scope so you don't have to implement the same sorting behaviour as this event does, to also change the column configuration. If you don't want this, you can specify a third event parameter:$rootScope.$emit("columnOrderChanged", "sourceColumn", "destColumn", {skipUpdatingColumnConfiguration: true});
###Content directives
insert more doc here