var app = angular.module('plunker', ['smart-table']);
app.controller('MainCtrl', function($scope) {
$scope.probes = [
{ "type": "temperature", "name": "74 DP22000-15060", "value": "+12.7" },
{ "type": "temperature", "name": "Ambiance C8", "value": "+31.6°C" },
{ "type": "temperature", "name": "Ambiance ST", "value": "+33.6°C" },
{ "type": "temperature", "name": "ambiance C12", "value": "+23.6°C" },
{ "type": "temperature", "name": "Ambiance C10", "value": "+23.6°C" },
{ "type": "hygrometry", "name": "Congélateur", "value": "25%" },
{ "type": "hygrometry", "name": "Congélateur 2", "value": "53%" },
{ "type": "hygrometry", "name": "Congélateur 6", "value": "18%" },
{ "type": "hygrometry", "name": "Hygrometry 6", "value": "5%" },
{ "type": "temperature", "name": "Frigo 1", "value": "4°C" },
{ "type": "", "name": "mobile 05", "value": "#######" },
{ "type": "particle", "name": "Particule 420", "value": "555 p/m" },
{ "type": "particle", "name": "Ambiance C9", "value": "#######" },
{ "type": "particle", "name": "Salle blanche C1", "value": "#######" },
{ "type": "particle", "name": "salle blanche C8", "value": "-######" },
{ "type": "pressure", "name": "Salle blanche C7", "value": "-######" },
{ "type": "tor", "name": "TOR 75", "value": "ON" },
{ "type": "tor", "name": "TOR PRE", "value": "ON" }
];
$scope.sortByName = function (row) {
return row.name.toLowerCase();
};
$scope.rowCollection = [
{firstName: 'Laurent', lastName: 'Renard', birthDate: new Date('1987-05-21'), balance: 102, email: 'whatever@gmail.com'},
{firstName: 'Blandine', lastName: 'Faivre', birthDate: new Date('1987-04-25'), balance: -2323.22, email: 'oufblandou@gmail.com'},
{firstName: 'Francoise', lastName: 'Frere', birthDate: new Date('1955-08-27'), balance: 42343, email: 'raymondef@gmail.com'}
];
$scope.getters = {
firstName: function (value) {
//this will sort by the length of the first name string
return value.firstName.length;
}
};
});
<!DOCTYPE html>
<html ng-app="plunker">
<head>
<meta charset="utf-8" />
<title>AngularJS Plunker</title>
<script>
document.write('<base href="' + document.location + '" />');
</script>
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css" integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
<link rel="stylesheet" href="style.css" />
<link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap-theme.min.css" integrity="sha384-rHyoN1iRsVXV4nD0JutlnGaslCJuC7uwjduW9SVrLvRYooPp2bWYgmgJQIXwl/Sp" crossorigin="anonymous">
<script data-require="angular.js@1.6.x" src="https://code.angularjs.org/1.6.8/angular.js" data-semver="1.6.8"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/angular-smart-table/2.1.11/smart-table.min.js"></script>
<script src="st-multi-sort.js"></script>
<script src="app.js"></script>
</head>
<body ng-controller="MainCtrl">
<div class="container-fluid">
<h1>Probes</h1>
<table st-safe-src="probes" st-table="displayProbes" class="table table-striped">
<thead>
<tr>
<th st-multi-sort="type">Type</th>
<th st-multi-sort="sortByName" st-sort-default="true">Name</th>
<th st-multi-sort="value">Value</th>
</tr>
</thead>
<tbody>
<tr ng-repeat="probe in displayProbes track by $index">
<td><img src="./img/{{probe.type}}" alt="{{probe.type}}" /></td>
<td>{{probe.name}}</td>
<td>{{probe.value}}</td>
</tr>
</tbody>
</table>
<h2>Second exemple</h2>
<table st-table="rowCollection" class="table table-striped">
<thead>
<tr>
<th st-sort="getters.firstName" st-sort-default="true">first name</th>
<th st-sort="lastName">last name</th>
<th st-sort="birthDate">birth date</th>
<th st-sort="balance">balance</th>
<th>email</th>
</tr>
</thead>
<tbody>
<tr ng-repeat="row in rowCollection">
<td>{{row.firstName | uppercase}}</td>
<td>{{row.lastName}}</td>
<td>{{row.birthDate | date}}</td>
<td>{{row.balance | currency}}</td>
<td><a ng-href="mailto:{{row.email}}">email</a></td>
</tr>
</tbody>
</table>
</div>
</body>
</html>
/* Styles go here */
.st-sort-ascent:before {
content: '\25B2';
}
.st-sort-descent:before {
content: '\25BC';
}
(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){
// Generated by CoffeeScript 1.9.2
var compareProperty, dot;
module.exports = function(collection, expressions) {
return collection.sort(function(a, b) {
var expression, i, len, predicate, reverse, value;
for (i = 0, len = expressions.length; i < len; i++) {
expression = expressions[i];
if (typeof expression === 'object') {
predicate = expression.predicate;
reverse = expression.reverse;
} else {
predicate = expression;
}
value = compareProperty(predicate, reverse)(a, b);
if (value !== 0) {
return value;
}
}
});
};
dot = {
get: function(obj, field) {
var i, key, keys, len, value;
keys = field.split('.');
value = obj;
for (i = 0, len = keys.length; i < len; i++) {
key = keys[i];
value = value[key];
}
return value;
},
set: function(obj, field, setValue) {
var allButLastKey, i, key, keys, lastKey, len, value;
keys = field.split('.');
allButLastKey = keys.slice(0, -1);
lastKey = keys[keys.length - 1];
value = obj;
for (i = 0, len = allButLastKey.length; i < len; i++) {
key = allButLastKey[i];
value = value[key] != null ? value[key] : value[key] = {};
}
return value[lastKey] = setValue;
}
};
compareProperty = function(predicate, reverse) {
var getter;
getter = typeof predicate === 'function' ? function(obj) {
return predicate(obj);
} : function(obj) {
return dot.get(obj, predicate);
};
getter;
if (!reverse) {
return function(a, b) {
if (getter(a) < getter(b)) {
return -1;
} else if (getter(a) > getter(b)) {
return 1;
} else {
return 0;
}
};
} else {
return function(a, b) {
if (getter(a) > getter(b)) {
return -1;
} else if (getter(a) < getter(b)) {
return 1;
} else {
return 0;
}
};
}
};
},{}],2:[function(require,module,exports){
var ng;
require('./multi_order_by');
require('./st_element_id');
ng = angular;
angular.module('smart-table').directive('stMultiSort', [
'stConfig', '$parse', '$rootScope', 'stUniqueId', function(stConfig, $parse, $rootScope, stUniqueId) {
return {
restrict: 'A',
require: '^stTable',
link: function(scope, element, attr, ctrl) {
var classAscent, classDescent, elementId, getter, index, predicate, sort, sortDefault, stateClasses;
predicate = attr.stMultiSort;
getter = $parse(predicate);
index = 0;
classAscent = attr.stClassAscent || stConfig.sort.ascentClass;
classDescent = attr.stClassDescent || stConfig.sort.descentClass;
stateClasses = [classAscent, classDescent];
sortDefault = void 0;
elementId = stUniqueId.generate();
/*
Use our custom orderBy filter, which supports reversing rows independently
*/
ctrl.setSortFunction('multiOrderBy');
/*
Sort the rows.
@param {Boolean} holdingShiftKey
*/
sort = function(holdingShiftKey) {
var base, reverse, tableState;
index++;
tableState = ctrl.tableState();
if ((base = tableState.sort).predicate == null) {
base.predicate = [];
}
reverse = index % 2 === 0;
predicate = ng.isFunction(getter(scope)) ? getter(scope) : attr.stMultiSort;
(function() {
var indexOfExistingSort;
indexOfExistingSort = (function() {
var i, ref, sortConfig;
ref = ctrl.tableState().sort.predicate;
for (i in ref) {
sortConfig = ref[i];
if (sortConfig.elementId === elementId) {
return i;
}
}
return -1;
})();
if (indexOfExistingSort !== -1) {
return tableState.sort.predicate.splice(indexOfExistingSort, 1);
}
})();
(function() {
index = index % 2 === 0 ? 2 : 1;
element.removeClass(stateClasses[index % 2]).addClass(stateClasses[index - 1]);
if (!holdingShiftKey) {
return $rootScope.$broadcast('clearOtherSortClasses', elementId);
}
})();
(function() {
if (!holdingShiftKey) {
tableState.sort.predicate.length = 0;
}
return tableState.sort.predicate.push({
elementId: elementId,
predicate: predicate,
reverse: reverse === true
});
})();
tableState.pagination.start = 0;
return ctrl.pipe();
};
if (attr.stSortDefault) {
sortDefault = scope.$eval(attr.stSortDefault) != null ? scope.$eval(attr.stSortDefault) : attr.stSortDefault;
}
if (sortDefault) {
index = sortDefault === 'reverse' ? 1 : 0;
sort();
}
element.bind('click', function(e) {
if (!predicate) {
return;
}
return scope.$apply(function() {
return sort(e.shiftKey || e.ctrlKey);
});
});
return scope.$on('clearOtherSortClasses', function(e, sortedElementId) {
if (sortedElementId !== elementId) {
index = 0;
return element.removeClass(classAscent).removeClass(classDescent);
}
});
}
};
}
]);
},{"./multi_order_by":3,"./st_element_id":4}],3:[function(require,module,exports){
/*
Like angular orderBy filter, but allows reversing each parameter individually (even getters)
*/
angular.module('smart-table').filter('multiOrderBy', function() {
return require('orderby');
});
},{"orderby":1}],4:[function(require,module,exports){
/*
Generate unique ids to identify each sortable element on the page
*/
angular.module('smart-table').factory('stUniqueId', function() {
var id;
id = 0;
return {
generate: function() {
return id++;
}
};
});
},{}]},{},[2])
//# sourceMappingURL=data:application/json;charset:utf-8;base64,{"version":3,"sources":["node_modules/browser-pack/_prelude.js","node_modules/orderby/lib/index.js","/Users/dannynelson/Projects/st-multi-sort/src/index.coffee","/Users/dannynelson/Projects/st-multi-sort/src/multi_order_by.coffee","/Users/dannynelson/Projects/st-multi-sort/src/st_element_id.coffee"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7EA,IAAA;;AAAA,OAAA,CAAQ,kBAAR;;AACA,OAAA,CAAQ,iBAAR;;AAEA,EAAA,GAAK;;AAEL,OAAO,CAAC,MAAR,CAAe,aAAf,CAA6B,CAAC,SAA9B,CAAwC,aAAxC,EAAuD;EACrD,UADqD,EAErD,QAFqD,EAGrD,YAHqD,EAIrD,YAJqD,EAKrD,SAAC,QAAD,EAAW,MAAX,EAAmB,UAAnB,EAA+B,UAA/B;WACE;MAAA,QAAA,EAAU,GAAV;MACA,OAAA,EAAS,UADT;MAEA,IAAA,EAAM,SAAC,KAAD,EAAQ,OAAR,EAAiB,IAAjB,EAAuB,IAAvB;AACJ,YAAA;QAAA,SAAA,GAAY,IAAI,CAAC;QACjB,MAAA,GAAS,MAAA,CAAO,SAAP;QACT,KAAA,GAAQ;QACR,WAAA,GAAc,IAAI,CAAC,aAAL,IAAsB,QAAQ,CAAC,IAAI,CAAC;QAClD,YAAA,GAAe,IAAI,CAAC,cAAL,IAAuB,QAAQ,CAAC,IAAI,CAAC;QACpD,YAAA,GAAe,CACb,WADa,EAEb,YAFa;QAIf,WAAA,GAAc;QACd,SAAA,GAAY,UAAU,CAAC,QAAX,CAAA;;AAEZ;;;QAGA,IAAI,CAAC,eAAL,CAAqB,cAArB;;AAEA;;;;QAIA,IAAA,GAAO,SAAC,eAAD;AACL,cAAA;UAAA,KAAA;UACA,UAAA,GAAa,IAAI,CAAC,UAAL,CAAA;;gBACE,CAAC,YAAa;;UAC7B,OAAA,GAAU,KAAA,GAAQ,CAAR,KAAa;UACvB,SAAA,GAAe,EAAE,CAAC,UAAH,CAAc,MAAA,CAAO,KAAP,CAAd,CAAH,GAAqC,MAAA,CAAO,KAAP,CAArC,GAAwD,IAAI,CAAC;UAEtE,CAAA,SAAA;AACD,gBAAA;YAAA,mBAAA,GAAyB,CAAA,SAAA;AACvB,kBAAA;AAAA;AAAA,mBAAA,QAAA;;gBACE,IAAG,UAAU,CAAC,SAAX,KAAwB,SAA3B;AACE,yBAAO,EADT;;AADF;AAGA,qBAAO,CAAC;YAJe,CAAA,CAAH,CAAA;YAKtB,IAAG,mBAAA,KAAyB,CAAC,CAA7B;qBACE,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,MAA1B,CAAiC,mBAAjC,EAAsD,CAAtD,EADF;;UANC,CAAA,CAAH,CAAA;UASG,CAAA,SAAA;YACD,KAAA,GAAW,KAAA,GAAQ,CAAR,KAAa,CAAhB,GAAuB,CAAvB,GAA8B;YACtC,OAAO,CAAC,WAAR,CAAoB,YAAa,CAAA,KAAA,GAAQ,CAAR,CAAjC,CAA4C,CAAC,QAA7C,CAAsD,YAAa,CAAA,KAAA,GAAQ,CAAR,CAAnE;YACA,IAAG,CAAC,eAAJ;qBACE,UAAU,CAAC,UAAX,CAAsB,uBAAtB,EAA+C,SAA/C,EADF;;UAHC,CAAA,CAAH,CAAA;UAMG,CAAA,SAAA;YACD,IAAG,CAAC,eAAJ;cACE,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,MAA1B,GAAmC,EADrC;;mBAEA,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,IAA1B,CACE;cAAA,SAAA,EAAW,SAAX;cACA,SAAA,EAAW,SADX;cAEA,OAAA,EAAS,OAAA,KAAW,IAFpB;aADF;UAHC,CAAA,CAAH,CAAA;UAQA,UAAU,CAAC,UAAU,CAAC,KAAtB,GAA8B;iBAC9B,IAAI,CAAC,IAAL,CAAA;QA/BK;QAiCP,IAAG,IAAI,CAAC,aAAR;UACE,WAAA,GAAiB,uCAAH,GAAyC,KAAK,CAAC,KAAN,CAAY,IAAI,CAAC,aAAjB,CAAzC,GAA8E,IAAI,CAAC,cADnG;;QAGA,IAAG,WAAH;UACE,KAAA,GAAW,WAAA,KAAe,SAAlB,GAAiC,CAAjC,GAAwC;UAChD,IAAA,CAAA,EAFF;;QAIA,OAAO,CAAC,IAAR,CAAa,OAAb,EAAsB,SAAC,CAAD;UACpB,IAAA,CAAc,SAAd;AAAA,mBAAA;;iBACA,KAAK,CAAC,MAAN,CAAa,SAAA;mBAAG,IAAA,CAAK,CAAC,CAAC,QAAF,IAAc,CAAC,CAAC,OAArB;UAAH,CAAb;QAFoB,CAAtB;eAIA,KAAK,CAAC,GAAN,CAAU,uBAAV,EAAmC,SAAC,CAAD,EAAI,eAAJ;UACjC,IAAG,eAAA,KAAqB,SAAxB;YACE,KAAA,GAAQ;mBACR,OAAO,CAAC,WAAR,CAAoB,WAApB,CAAgC,CAAC,WAAjC,CAA6C,YAA7C,EAFF;;QADiC,CAAnC;MAlEI,CAFN;;EADF,CALqD;CAAvD;;;;;ACLA;;;AAGA,OAAO,CAAC,MAAR,CAAe,aAAf,CAA6B,CAAC,MAA9B,CAAqC,cAArC,EAAqD,SAAA;SACnD,OAAA,CAAQ,SAAR;AADmD,CAArD;;;;;ACHA;;;AAGA,OAAO,CAAC,MAAR,CAAe,aAAf,CAEA,CAAC,OAFD,CAES,YAFT,EAEuB,SAAA;AACrB,MAAA;EAAA,EAAA,GAAK;SAEL;IACE,QAAA,EAAU,SAAA;aAAG,EAAA;IAAH,CADZ;;AAHqB,CAFvB","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","// Generated by CoffeeScript 1.9.2\nvar compareProperty, dot;\n\nmodule.exports = function(collection, expressions) {\n  return collection.sort(function(a, b) {\n    var expression, i, len, predicate, reverse, value;\n    for (i = 0, len = expressions.length; i < len; i++) {\n      expression = expressions[i];\n      if (typeof expression === 'object') {\n        predicate = expression.predicate;\n        reverse = expression.reverse;\n      } else {\n        predicate = expression;\n      }\n      value = compareProperty(predicate, reverse)(a, b);\n      if (value !== 0) {\n        return value;\n      }\n    }\n  });\n};\n\ndot = {\n  get: function(obj, field) {\n    var i, key, keys, len, value;\n    keys = field.split('.');\n    value = obj;\n    for (i = 0, len = keys.length; i < len; i++) {\n      key = keys[i];\n      value = value[key];\n    }\n    return value;\n  },\n  set: function(obj, field, setValue) {\n    var allButLastKey, i, key, keys, lastKey, len, value;\n    keys = field.split('.');\n    allButLastKey = keys.slice(0, -1);\n    lastKey = keys[keys.length - 1];\n    value = obj;\n    for (i = 0, len = allButLastKey.length; i < len; i++) {\n      key = allButLastKey[i];\n      value = value[key] != null ? value[key] : value[key] = {};\n    }\n    return value[lastKey] = setValue;\n  }\n};\n\ncompareProperty = function(predicate, reverse) {\n  var getter;\n  getter = typeof predicate === 'function' ? function(obj) {\n    return predicate(obj);\n  } : function(obj) {\n    return dot.get(obj, predicate);\n  };\n  getter;\n  if (!reverse) {\n    return function(a, b) {\n      if (getter(a) < getter(b)) {\n        return -1;\n      } else if (getter(a) > getter(b)) {\n        return 1;\n      } else {\n        return 0;\n      }\n    };\n  } else {\n    return function(a, b) {\n      if (getter(a) > getter(b)) {\n        return -1;\n      } else if (getter(a) < getter(b)) {\n        return 1;\n      } else {\n        return 0;\n      }\n    };\n  }\n};\n","require './multi_order_by'\nrequire './st_element_id'\n\nng = angular\n\nangular.module('smart-table').directive 'stMultiSort', [\n  'stConfig'\n  '$parse'\n  '$rootScope'\n  'stUniqueId'\n  (stConfig, $parse, $rootScope, stUniqueId) ->\n    restrict: 'A'\n    require: '^stTable'\n    link: (scope, element, attr, ctrl) ->\n      predicate = attr.stMultiSort\n      getter = $parse(predicate)\n      index = 0\n      classAscent = attr.stClassAscent or stConfig.sort.ascentClass\n      classDescent = attr.stClassDescent or stConfig.sort.descentClass\n      stateClasses = [\n        classAscent\n        classDescent\n      ]\n      sortDefault = undefined\n      elementId = stUniqueId.generate()\n\n      ###\n      Use our custom orderBy filter, which supports reversing rows independently\n      ###\n      ctrl.setSortFunction 'multiOrderBy'\n\n      ###\n      Sort the rows.\n      @param {Boolean} holdingShiftKey\n      ###\n      sort = (holdingShiftKey) ->\n        index++\n        tableState = ctrl.tableState()\n        tableState.sort.predicate ?= []\n        reverse = index % 2 is 0\n        predicate = if ng.isFunction(getter(scope)) then getter(scope) else attr.stMultiSort\n\n        do -> # clear existing sort\n          indexOfExistingSort = do ->\n            for i, sortConfig of ctrl.tableState().sort.predicate\n              if sortConfig.elementId is elementId\n                return i\n            return -1\n          if indexOfExistingSort isnt -1\n            tableState.sort.predicate.splice indexOfExistingSort, 1\n\n        do -> # update sort classes\n          index = if index % 2 == 0 then 2 else 1\n          element.removeClass(stateClasses[index % 2]).addClass stateClasses[index - 1]\n          if !holdingShiftKey\n            $rootScope.$broadcast 'clearOtherSortClasses', elementId\n\n        do -> # update sort\n          if !holdingShiftKey\n            tableState.sort.predicate.length = 0;\n          tableState.sort.predicate.push\n            elementId: elementId\n            predicate: predicate\n            reverse: reverse == true\n\n        tableState.pagination.start = 0\n        ctrl.pipe()\n\n      if attr.stSortDefault\n        sortDefault = if scope.$eval(attr.stSortDefault)? then scope.$eval(attr.stSortDefault) else attr.stSortDefault\n\n      if sortDefault\n        index = if sortDefault == 'reverse' then 1 else 0\n        sort()\n\n      element.bind 'click', (e) ->\n        return unless predicate\n        scope.$apply -> sort(e.shiftKey or e.ctrlKey)\n\n      scope.$on 'clearOtherSortClasses', (e, sortedElementId) ->\n        if sortedElementId isnt elementId\n          index = 0\n          element.removeClass(classAscent).removeClass(classDescent)\n]\n","###\nLike angular orderBy filter, but allows reversing each parameter individually (even getters)\n###\nangular.module('smart-table').filter 'multiOrderBy', ->\n  require 'orderby'\n","###\nGenerate unique ids to identify each sortable element on the page\n###\nangular.module('smart-table')\n\n.factory 'stUniqueId', ->\n  id = 0\n\n  {\n    generate: -> id++\n  }\n"]}