/sdk/src/mixin/Sortable.js

https://bitbucket.org/mobile_venkys/mobisale1 · JavaScript · 341 lines · 183 code · 44 blank · 114 comment · 39 complexity · 1cd53e5788fcf0f6106c06dec7e434e7 MD5 · raw file

  1. /**
  2. * @private
  3. */
  4. Ext.define('Ext.mixin.Sortable', {
  5. extend: 'Ext.mixin.Mixin',
  6. requires: [
  7. 'Ext.util.Sorter'
  8. ],
  9. mixinConfig: {
  10. id: 'sortable'
  11. },
  12. config: {
  13. /**
  14. * @cfg {Array} sorters
  15. * An array with sorters. A sorter can be an instance of Ext.util.Sorter, a string
  16. * indicating a property name, an object representing an Ext.util.Sorter configuration,
  17. * or a sort function.
  18. */
  19. sorters: null,
  20. /**
  21. * @cfg {String} defaultSortDirection
  22. * The default sort direction to use if one is not specified (defaults to "ASC")
  23. */
  24. defaultSortDirection: "ASC",
  25. /**
  26. * @cfg {String} sortRoot
  27. * The root inside each item in which the properties exist that we want to sort on.
  28. * This is useful for sorting records in which the data exists inside a 'data' property.
  29. */
  30. sortRoot: null
  31. },
  32. /**
  33. * @property {Boolean} dirtySortFn
  34. * A flag indicating wether the currently cashed sort function is still valid. Read-only.
  35. */
  36. dirtySortFn: false,
  37. /**
  38. * @property currentSortFn
  39. * This is the cached sorting function which is a generated function that calls all the
  40. * configured sorters in the correct order. This is a read-only property.
  41. */
  42. sortFn: null,
  43. /**
  44. * @property {Boolean} sorted
  45. * A read-only flag indicating if this object is sorted
  46. */
  47. sorted: false,
  48. applySorters: function(sorters, collection) {
  49. if (!collection) {
  50. collection = this.createSortersCollection();
  51. }
  52. collection.clear();
  53. this.sorted = false;
  54. if (sorters) {
  55. this.addSorters(sorters);
  56. }
  57. return collection;
  58. },
  59. createSortersCollection: function() {
  60. this._sorters = Ext.create('Ext.util.Collection', function(sorter) {
  61. return sorter.getId();
  62. });
  63. return this._sorters;
  64. },
  65. /**
  66. * This method adds a sorter.
  67. * @param {Ext.util.Sorter/String/Function/Object} sorter Can be an instance of
  68. * Ext.util.Sorter, a string indicating a property name, an object representing an Ext.util.Sorter
  69. * configuration, or a sort function.
  70. * @param {String} defaultDirection The default direction for each sorter in the array. Defaults
  71. * to the value of {@link #defaultSortDirection}. Can be either 'ASC' or 'DESC'.
  72. */
  73. addSorter: function(sorter, defaultDirection) {
  74. this.addSorters([sorter], defaultDirection);
  75. },
  76. /**
  77. * This method adds all the sorters in a passed array.
  78. * @param {Array} sorters An array with sorters. A sorter can be an instance of Ext.util.Sorter, a string
  79. * indicating a property name, an object representing an Ext.util.Sorter configuration,
  80. * or a sort function.
  81. * @param {String} defaultDirection The default direction for each sorter in the array. Defaults
  82. * to the value of {@link #defaultSortDirection}. Can be either 'ASC' or 'DESC'.
  83. */
  84. addSorters: function(sorters, defaultDirection) {
  85. var currentSorters = this.getSorters();
  86. return this.insertSorters(currentSorters ? currentSorters.length : 0, sorters, defaultDirection);
  87. },
  88. /**
  89. * This method adds a sorter at a given index.
  90. * @param {Number} index The index at which to insert the sorter.
  91. * @param {Ext.util.Sorter/String/Function/Object} sorter Can be an instance of Ext.util.Sorter,
  92. * a string indicating a property name, an object representing an Ext.util.Sorter configuration,
  93. * or a sort function.
  94. * @param {String} defaultDirection The default direction for each sorter in the array. Defaults
  95. * to the value of {@link #defaultSortDirection}. Can be either 'ASC' or 'DESC'.
  96. */
  97. insertSorter: function(index, sorter, defaultDirection) {
  98. return this.insertSorters(index, [sorter], defaultDirection);
  99. },
  100. /**
  101. * This method inserts all the sorters in the passed array at the given index.
  102. * @param {Number} index The index at which to insert the sorters.
  103. * @param {Array} sorters Can be an instance of Ext.util.Sorter, a string indicating a property name,
  104. * an object representing an Ext.util.Sorter configuration, or a sort function.
  105. * @param {String} defaultDirection The default direction for each sorter in the array. Defaults
  106. * to the value of {@link #defaultSortDirection}. Can be either 'ASC' or 'DESC'.
  107. */
  108. insertSorters: function(index, sorters, defaultDirection) {
  109. // We begin by making sure we are dealing with an array of sorters
  110. if (!Ext.isArray(sorters)) {
  111. sorters = [sorters];
  112. }
  113. var ln = sorters.length,
  114. direction = defaultDirection || this.getDefaultSortDirection(),
  115. sortRoot = this.getSortRoot(),
  116. currentSorters = this.getSorters(),
  117. newSorters = [],
  118. sorterConfig, i, sorter, currentSorter;
  119. if (!currentSorters) {
  120. // This will guarantee that we get the collection
  121. currentSorters = this.createSortersCollection();
  122. }
  123. // We first have to convert every sorter into a proper Sorter instance
  124. for (i = 0; i < ln; i++) {
  125. sorter = sorters[i];
  126. sorterConfig = {
  127. direction: direction,
  128. root: sortRoot
  129. };
  130. // If we are dealing with a string we assume it is a property they want to sort on.
  131. if (typeof sorter === 'string') {
  132. currentSorter = currentSorters.get(sorter);
  133. if (!currentSorter) {
  134. sorterConfig.property = sorter;
  135. } else {
  136. if (defaultDirection) {
  137. currentSorter.setDirection(defaultDirection);
  138. } else {
  139. // If we already have a sorter for this property we just toggle its direction.
  140. currentSorter.toggle();
  141. }
  142. continue;
  143. }
  144. }
  145. // If it is a function, we assume its a sorting function.
  146. else if (Ext.isFunction(sorter)) {
  147. sorterConfig.sorterFn = sorter;
  148. }
  149. // If we are dealing with an object, we assume its a Sorter configuration. In this case
  150. // we create an instance of Sorter passing this configuration.
  151. else if (Ext.isObject(sorter)) {
  152. if (!sorter.isSorter) {
  153. if (sorter.fn) {
  154. sorter.sorterFn = sorter.fn;
  155. delete sorter.fn;
  156. }
  157. sorterConfig = Ext.apply(sorterConfig, sorter);
  158. }
  159. else {
  160. newSorters.push(sorter);
  161. if (!sorter.getRoot()) {
  162. sorter.setRoot(sortRoot);
  163. }
  164. continue;
  165. }
  166. }
  167. // Finally we get to the point where it has to be invalid
  168. // <debug>
  169. else {
  170. Ext.Logger.warn('Invalid sorter specified:', sorter);
  171. }
  172. // </debug>
  173. // If a sorter config was created, make it an instance
  174. sorter = Ext.create('Ext.util.Sorter', sorterConfig);
  175. newSorters.push(sorter);
  176. }
  177. // Now lets add the newly created sorters.
  178. for (i = 0, ln = newSorters.length; i < ln; i++) {
  179. currentSorters.insert(index + i, newSorters[i]);
  180. }
  181. this.dirtySortFn = true;
  182. if (currentSorters.length) {
  183. this.sorted = true;
  184. }
  185. return currentSorters;
  186. },
  187. /**
  188. * This method removes a sorter.
  189. * @param {Ext.util.Sorter/String/Function/Object} sorter Can be an instance of Ext.util.Sorter,
  190. * a string indicating a property name, an object representing an Ext.util.Sorter configuration,
  191. * or a sort function.
  192. */
  193. removeSorter: function(sorter) {
  194. return this.removeSorters([sorter]);
  195. },
  196. /**
  197. * This method removes all the sorters in a passed array.
  198. * @param {Array} sorters Each value in the array can be a string (property name),
  199. * function (sorterFn) or {@link Ext.util.Sorter Sorter} instance.
  200. */
  201. removeSorters: function(sorters) {
  202. // We begin by making sure we are dealing with an array of sorters
  203. if (!Ext.isArray(sorters)) {
  204. sorters = [sorters];
  205. }
  206. var ln = sorters.length,
  207. currentSorters = this.getSorters(),
  208. i, sorter;
  209. for (i = 0; i < ln; i++) {
  210. sorter = sorters[i];
  211. if (typeof sorter === 'string') {
  212. currentSorters.removeAtKey(sorter);
  213. }
  214. else if (typeof sorter === 'function') {
  215. currentSorters.each(function(item) {
  216. if (item.getSorterFn() === sorter) {
  217. currentSorters.remove(item);
  218. }
  219. });
  220. }
  221. else if (sorter.isSorter) {
  222. currentSorters.remove(sorter);
  223. }
  224. }
  225. if (!currentSorters.length) {
  226. this.sorted = false;
  227. }
  228. },
  229. /**
  230. * This updates the cached sortFn based on the current sorters.
  231. * @return {Function} sortFn The generated sort function.
  232. * @private
  233. */
  234. updateSortFn: function() {
  235. var sorters = this.getSorters().items;
  236. this.sortFn = function(r1, r2) {
  237. var ln = sorters.length,
  238. result, i;
  239. // We loop over each sorter and check if r1 should be before or after r2
  240. for (i = 0; i < ln; i++) {
  241. result = sorters[i].sort.call(this, r1, r2);
  242. // If the result is -1 or 1 at this point it means that the sort is done.
  243. // Only if they are equal (0) we continue to see if a next sort function
  244. // actually might find a winner.
  245. if (result !== 0) {
  246. break;
  247. }
  248. }
  249. return result;
  250. };
  251. this.dirtySortFn = false;
  252. return this.sortFn;
  253. },
  254. /**
  255. * Returns an up to date sort function.
  256. * @return {Function} sortFn The sort function.
  257. */
  258. getSortFn: function() {
  259. if (this.dirtySortFn) {
  260. return this.updateSortFn();
  261. }
  262. return this.sortFn;
  263. },
  264. /**
  265. * This method will sort an array based on the currently configured {@link #sorters}.
  266. * @param {Array} data The array you want to have sorted
  267. * @return {Array} data The array you passed after it is sorted
  268. */
  269. sort: function(data) {
  270. Ext.Array.sort(data, this.getSortFn());
  271. return data;
  272. },
  273. /**
  274. * This method returns the index that a given item would be inserted into a given array based
  275. * on the current sorters.
  276. * @param {Array} items The array that you want to insert the item into.
  277. * @param {Mixed} item The item that you want to insert into the items array.
  278. * @returns {Number} index The index for the given item in the given array based on the current sorters.
  279. */
  280. findInsertionIndex: function(items, item, sortFn) {
  281. var start = 0,
  282. end = items.length - 1,
  283. sorterFn = sortFn || this.getSortFn(),
  284. middle,
  285. comparison;
  286. while (start <= end) {
  287. middle = (start + end) >> 1;
  288. comparison = sorterFn(item, items[middle]);
  289. if (comparison >= 0) {
  290. start = middle + 1;
  291. } else if (comparison < 0) {
  292. end = middle - 1;
  293. }
  294. }
  295. return start;
  296. }
  297. });