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":"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","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"]}