row-height-cache.js 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. /**
  6. * This object contains the cache of the various row heights that are present inside
  7. * the data table. Its based on Fenwick tree data structure that helps with
  8. * querying sums that have time complexity of log n.
  9. *
  10. * Fenwick Tree Credits: http://petr-mitrichev.blogspot.com/2013/05/fenwick-tree-range-updates.html
  11. * https://github.com/mikolalysenko/fenwick-tree
  12. *
  13. */
  14. var /**
  15. * This object contains the cache of the various row heights that are present inside
  16. * the data table. Its based on Fenwick tree data structure that helps with
  17. * querying sums that have time complexity of log n.
  18. *
  19. * Fenwick Tree Credits: http://petr-mitrichev.blogspot.com/2013/05/fenwick-tree-range-updates.html
  20. * https://github.com/mikolalysenko/fenwick-tree
  21. *
  22. */
  23. RowHeightCache = /** @class */ (function () {
  24. function RowHeightCache() {
  25. /**
  26. * Tree Array stores the cumulative information of the row heights to perform efficient
  27. * range queries and updates. Currently the tree is initialized to the base row
  28. * height instead of the detail row height.
  29. */
  30. this.treeArray = [];
  31. }
  32. /**
  33. * Clear the Tree array.
  34. */
  35. /**
  36. * Clear the Tree array.
  37. * @return {?}
  38. */
  39. RowHeightCache.prototype.clearCache = /**
  40. * Clear the Tree array.
  41. * @return {?}
  42. */
  43. function () {
  44. this.treeArray = [];
  45. };
  46. /**
  47. * Initialize the Fenwick tree with row Heights.
  48. *
  49. * @param rows The array of rows which contain the expanded status.
  50. * @param rowHeight The row height.
  51. * @param detailRowHeight The detail row height.
  52. */
  53. /**
  54. * Initialize the Fenwick tree with row Heights.
  55. *
  56. * @param {?} details
  57. * @return {?}
  58. */
  59. RowHeightCache.prototype.initCache = /**
  60. * Initialize the Fenwick tree with row Heights.
  61. *
  62. * @param {?} details
  63. * @return {?}
  64. */
  65. function (details) {
  66. var rows = details.rows, rowHeight = details.rowHeight, detailRowHeight = details.detailRowHeight, externalVirtual = details.externalVirtual, rowCount = details.rowCount, rowIndexes = details.rowIndexes, rowExpansions = details.rowExpansions;
  67. /** @type {?} */
  68. var isFn = typeof rowHeight === 'function';
  69. /** @type {?} */
  70. var isDetailFn = typeof detailRowHeight === 'function';
  71. if (!isFn && isNaN(rowHeight)) {
  72. throw new Error("Row Height cache initialization failed. Please ensure that 'rowHeight' is a\n valid number or function value: (" + rowHeight + ") when 'scrollbarV' is enabled.");
  73. }
  74. // Add this additional guard in case detailRowHeight is set to 'auto' as it wont work.
  75. if (!isDetailFn && isNaN(detailRowHeight)) {
  76. throw new Error("Row Height cache initialization failed. Please ensure that 'detailRowHeight' is a\n valid number or function value: (" + detailRowHeight + ") when 'scrollbarV' is enabled.");
  77. }
  78. /** @type {?} */
  79. var n = externalVirtual ? rowCount : rows.length;
  80. this.treeArray = new Array(n);
  81. for (var i = 0; i < n; ++i) {
  82. this.treeArray[i] = 0;
  83. }
  84. for (var i = 0; i < n; ++i) {
  85. /** @type {?} */
  86. var row = rows[i];
  87. /** @type {?} */
  88. var currentRowHeight = rowHeight;
  89. if (isFn) {
  90. currentRowHeight = rowHeight(row);
  91. }
  92. // Add the detail row height to the already expanded rows.
  93. // This is useful for the table that goes through a filter or sort.
  94. /** @type {?} */
  95. var expanded = rowExpansions.has(row);
  96. if (row && expanded) {
  97. if (isDetailFn) {
  98. /** @type {?} */
  99. var index = rowIndexes.get(row);
  100. currentRowHeight += detailRowHeight(row, index);
  101. }
  102. else {
  103. currentRowHeight += detailRowHeight;
  104. }
  105. }
  106. this.update(i, currentRowHeight);
  107. }
  108. };
  109. /**
  110. * Given the ScrollY position i.e. sum, provide the rowIndex
  111. * that is present in the current view port. Below handles edge cases.
  112. */
  113. /**
  114. * Given the ScrollY position i.e. sum, provide the rowIndex
  115. * that is present in the current view port. Below handles edge cases.
  116. * @param {?} scrollY
  117. * @return {?}
  118. */
  119. RowHeightCache.prototype.getRowIndex = /**
  120. * Given the ScrollY position i.e. sum, provide the rowIndex
  121. * that is present in the current view port. Below handles edge cases.
  122. * @param {?} scrollY
  123. * @return {?}
  124. */
  125. function (scrollY) {
  126. if (scrollY === 0)
  127. return 0;
  128. return this.calcRowIndex(scrollY);
  129. };
  130. /**
  131. * When a row is expanded or rowHeight is changed, update the height. This can
  132. * be utilized in future when Angular Data table supports dynamic row heights.
  133. */
  134. /**
  135. * When a row is expanded or rowHeight is changed, update the height. This can
  136. * be utilized in future when Angular Data table supports dynamic row heights.
  137. * @param {?} atRowIndex
  138. * @param {?} byRowHeight
  139. * @return {?}
  140. */
  141. RowHeightCache.prototype.update = /**
  142. * When a row is expanded or rowHeight is changed, update the height. This can
  143. * be utilized in future when Angular Data table supports dynamic row heights.
  144. * @param {?} atRowIndex
  145. * @param {?} byRowHeight
  146. * @return {?}
  147. */
  148. function (atRowIndex, byRowHeight) {
  149. if (!this.treeArray.length) {
  150. throw new Error("Update at index " + atRowIndex + " with value " + byRowHeight + " failed:\n Row Height cache not initialized.");
  151. }
  152. /** @type {?} */
  153. var n = this.treeArray.length;
  154. atRowIndex |= 0;
  155. while (atRowIndex < n) {
  156. this.treeArray[atRowIndex] += byRowHeight;
  157. atRowIndex |= atRowIndex + 1;
  158. }
  159. };
  160. /**
  161. * Range Sum query from 1 to the rowIndex
  162. */
  163. /**
  164. * Range Sum query from 1 to the rowIndex
  165. * @param {?} atIndex
  166. * @return {?}
  167. */
  168. RowHeightCache.prototype.query = /**
  169. * Range Sum query from 1 to the rowIndex
  170. * @param {?} atIndex
  171. * @return {?}
  172. */
  173. function (atIndex) {
  174. if (!this.treeArray.length) {
  175. throw new Error("query at index " + atIndex + " failed: Fenwick tree array not initialized.");
  176. }
  177. /** @type {?} */
  178. var sum = 0;
  179. atIndex |= 0;
  180. while (atIndex >= 0) {
  181. sum += this.treeArray[atIndex];
  182. atIndex = (atIndex & (atIndex + 1)) - 1;
  183. }
  184. return sum;
  185. };
  186. /**
  187. * Find the total height between 2 row indexes
  188. */
  189. /**
  190. * Find the total height between 2 row indexes
  191. * @param {?} atIndexA
  192. * @param {?} atIndexB
  193. * @return {?}
  194. */
  195. RowHeightCache.prototype.queryBetween = /**
  196. * Find the total height between 2 row indexes
  197. * @param {?} atIndexA
  198. * @param {?} atIndexB
  199. * @return {?}
  200. */
  201. function (atIndexA, atIndexB) {
  202. return this.query(atIndexB) - this.query(atIndexA - 1);
  203. };
  204. /**
  205. * Given the ScrollY position i.e. sum, provide the rowIndex
  206. * that is present in the current view port.
  207. */
  208. /**
  209. * Given the ScrollY position i.e. sum, provide the rowIndex
  210. * that is present in the current view port.
  211. * @private
  212. * @param {?} sum
  213. * @return {?}
  214. */
  215. RowHeightCache.prototype.calcRowIndex = /**
  216. * Given the ScrollY position i.e. sum, provide the rowIndex
  217. * that is present in the current view port.
  218. * @private
  219. * @param {?} sum
  220. * @return {?}
  221. */
  222. function (sum) {
  223. if (!this.treeArray.length)
  224. return 0;
  225. /** @type {?} */
  226. var pos = -1;
  227. /** @type {?} */
  228. var dataLength = this.treeArray.length;
  229. // Get the highest bit for the block size.
  230. /** @type {?} */
  231. var highestBit = Math.pow(2, dataLength.toString(2).length - 1);
  232. for (var blockSize = highestBit; blockSize !== 0; blockSize >>= 1) {
  233. /** @type {?} */
  234. var nextPos = pos + blockSize;
  235. if (nextPos < dataLength && sum >= this.treeArray[nextPos]) {
  236. sum -= this.treeArray[nextPos];
  237. pos = nextPos;
  238. }
  239. }
  240. return pos + 1;
  241. };
  242. return RowHeightCache;
  243. }());
  244. /**
  245. * This object contains the cache of the various row heights that are present inside
  246. * the data table. Its based on Fenwick tree data structure that helps with
  247. * querying sums that have time complexity of log n.
  248. *
  249. * Fenwick Tree Credits: http://petr-mitrichev.blogspot.com/2013/05/fenwick-tree-range-updates.html
  250. * https://github.com/mikolalysenko/fenwick-tree
  251. *
  252. */
  253. export { RowHeightCache };
  254. if (false) {
  255. /**
  256. * Tree Array stores the cumulative information of the row heights to perform efficient
  257. * range queries and updates. Currently the tree is initialized to the base row
  258. * height instead of the detail row height.
  259. * @type {?}
  260. * @private
  261. */
  262. RowHeightCache.prototype.treeArray;
  263. }
  264. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"row-height-cache.js","sourceRoot":"ng://@swimlane/ngx-datatable/","sources":["lib/utils/row-height-cache.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * This object contains the cache of the various row heights that are present inside\n * the data table.   Its based on Fenwick tree data structure that helps with\n * querying sums that have time complexity of log n.\n *\n * Fenwick Tree Credits: http://petr-mitrichev.blogspot.com/2013/05/fenwick-tree-range-updates.html\n * https://github.com/mikolalysenko/fenwick-tree\n *\n */\nexport class RowHeightCache {\n  /**\n   * Tree Array stores the cumulative information of the row heights to perform efficient\n   * range queries and updates.  Currently the tree is initialized to the base row\n   * height instead of the detail row height.\n   */\n  private treeArray: number[] = [];\n\n  /**\n   * Clear the Tree array.\n   */\n  clearCache(): void {\n    this.treeArray = [];\n  }\n\n  /**\n   * Initialize the Fenwick tree with row Heights.\n   *\n   * @param rows The array of rows which contain the expanded status.\n   * @param rowHeight The row height.\n   * @param detailRowHeight The detail row height.\n   */\n  initCache(details: any): void {\n    const { rows, rowHeight, detailRowHeight, externalVirtual, rowCount, rowIndexes, rowExpansions } = details;\n    const isFn = typeof rowHeight === 'function';\n    const isDetailFn = typeof detailRowHeight === 'function';\n\n    if (!isFn && isNaN(rowHeight)) {\n      throw new Error(`Row Height cache initialization failed. Please ensure that 'rowHeight' is a\n        valid number or function value: (${rowHeight}) when 'scrollbarV' is enabled.`);\n    }\n\n    // Add this additional guard in case detailRowHeight is set to 'auto' as it wont work.\n    if (!isDetailFn && isNaN(detailRowHeight)) {\n      throw new Error(`Row Height cache initialization failed. Please ensure that 'detailRowHeight' is a\n        valid number or function value: (${detailRowHeight}) when 'scrollbarV' is enabled.`);\n    }\n\n    const n = externalVirtual ? rowCount : rows.length;\n    this.treeArray = new Array(n);\n\n    for (let i = 0; i < n; ++i) {\n      this.treeArray[i] = 0;\n    }\n\n    for (let i = 0; i < n; ++i) {\n      const row = rows[i];\n      let currentRowHeight = rowHeight;\n      if (isFn) {\n        currentRowHeight = rowHeight(row);\n      }\n\n      // Add the detail row height to the already expanded rows.\n      // This is useful for the table that goes through a filter or sort.\n      const expanded = rowExpansions.has(row);\n      if (row && expanded) {\n        if (isDetailFn) {\n          const index = rowIndexes.get(row);\n          currentRowHeight += detailRowHeight(row, index);\n        } else {\n          currentRowHeight += detailRowHeight;\n        }\n      }\n\n      this.update(i, currentRowHeight);\n    }\n  }\n\n  /**\n   * Given the ScrollY position i.e. sum, provide the rowIndex\n   * that is present in the current view port.  Below handles edge cases.\n   */\n  getRowIndex(scrollY: number): number {\n    if (scrollY === 0) return 0;\n    return this.calcRowIndex(scrollY);\n  }\n\n  /**\n   * When a row is expanded or rowHeight is changed, update the height.  This can\n   * be utilized in future when Angular Data table supports dynamic row heights.\n   */\n  update(atRowIndex: number, byRowHeight: number): void {\n    if (!this.treeArray.length) {\n      throw new Error(`Update at index ${atRowIndex} with value ${byRowHeight} failed:\n        Row Height cache not initialized.`);\n    }\n\n    const n = this.treeArray.length;\n    atRowIndex |= 0;\n\n    while (atRowIndex < n) {\n      this.treeArray[atRowIndex] += byRowHeight;\n      atRowIndex |= atRowIndex + 1;\n    }\n  }\n\n  /**\n   * Range Sum query from 1 to the rowIndex\n   */\n  query(atIndex: number): number {\n    if (!this.treeArray.length) {\n      throw new Error(`query at index ${atIndex} failed: Fenwick tree array not initialized.`);\n    }\n\n    let sum = 0;\n    atIndex |= 0;\n\n    while (atIndex >= 0) {\n      sum += this.treeArray[atIndex];\n      atIndex = (atIndex & (atIndex + 1)) - 1;\n    }\n\n    return sum;\n  }\n\n  /**\n   * Find the total height between 2 row indexes\n   */\n  queryBetween(atIndexA: number, atIndexB: number): number {\n    return this.query(atIndexB) - this.query(atIndexA - 1);\n  }\n\n  /**\n   * Given the ScrollY position i.e. sum, provide the rowIndex\n   * that is present in the current view port.\n   */\n  private calcRowIndex(sum: number): number {\n    if (!this.treeArray.length) return 0;\n\n    let pos = -1;\n    const dataLength = this.treeArray.length;\n\n    // Get the highest bit for the block size.\n    const highestBit = Math.pow(2, dataLength.toString(2).length - 1);\n\n    for (let blockSize = highestBit; blockSize !== 0; blockSize >>= 1) {\n      const nextPos = pos + blockSize;\n      if (nextPos < dataLength && sum >= this.treeArray[nextPos]) {\n        sum -= this.treeArray[nextPos];\n        pos = nextPos;\n      }\n    }\n\n    return pos + 1;\n  }\n}\n"]}