body.component.js 81 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { Component, Output, EventEmitter, Input, HostBinding, ChangeDetectorRef, ViewChild, ChangeDetectionStrategy } from '@angular/core';
  6. import { ScrollerComponent } from './scroller.component';
  7. import { SelectionType } from '../../types/selection.type';
  8. import { columnsByPin, columnGroupWidths } from '../../utils/column';
  9. import { RowHeightCache } from '../../utils/row-height-cache';
  10. import { translateXY } from '../../utils/translate';
  11. export class DataTableBodyComponent {
  12. /**
  13. * Creates an instance of DataTableBodyComponent.
  14. * @param {?} cd
  15. */
  16. constructor(cd) {
  17. this.cd = cd;
  18. this.selected = [];
  19. this.scroll = new EventEmitter();
  20. this.page = new EventEmitter();
  21. this.activate = new EventEmitter();
  22. this.select = new EventEmitter();
  23. this.detailToggle = new EventEmitter();
  24. this.rowContextmenu = new EventEmitter(false);
  25. this.treeAction = new EventEmitter();
  26. this.rowHeightsCache = new RowHeightCache();
  27. this.temp = [];
  28. this.offsetY = 0;
  29. this.indexes = {};
  30. this.rowIndexes = new Map();
  31. this.rowExpansions = [];
  32. /**
  33. * Get the height of the detail row.
  34. */
  35. this.getDetailRowHeight = (/**
  36. * @param {?=} row
  37. * @param {?=} index
  38. * @return {?}
  39. */
  40. (row, index) => {
  41. if (!this.rowDetail) {
  42. return 0;
  43. }
  44. /** @type {?} */
  45. const rowHeight = this.rowDetail.rowHeight;
  46. return typeof rowHeight === 'function' ? rowHeight(row, index) : ((/** @type {?} */ (rowHeight)));
  47. });
  48. // declare fn here so we can get access to the `this` property
  49. this.rowTrackingFn = (/**
  50. * @param {?} index
  51. * @param {?} row
  52. * @return {?}
  53. */
  54. (index, row) => {
  55. /** @type {?} */
  56. const idx = this.getRowIndex(row);
  57. if (this.trackByProp) {
  58. return row[this.trackByProp];
  59. }
  60. else {
  61. return idx;
  62. }
  63. });
  64. }
  65. /**
  66. * @param {?} val
  67. * @return {?}
  68. */
  69. set pageSize(val) {
  70. this._pageSize = val;
  71. this.recalcLayout();
  72. }
  73. /**
  74. * @return {?}
  75. */
  76. get pageSize() {
  77. return this._pageSize;
  78. }
  79. /**
  80. * @param {?} val
  81. * @return {?}
  82. */
  83. set rows(val) {
  84. this._rows = val;
  85. this.recalcLayout();
  86. }
  87. /**
  88. * @return {?}
  89. */
  90. get rows() {
  91. return this._rows;
  92. }
  93. /**
  94. * @param {?} val
  95. * @return {?}
  96. */
  97. set columns(val) {
  98. this._columns = val;
  99. /** @type {?} */
  100. const colsByPin = columnsByPin(val);
  101. this.columnGroupWidths = columnGroupWidths(colsByPin, val);
  102. }
  103. /**
  104. * @return {?}
  105. */
  106. get columns() {
  107. return this._columns;
  108. }
  109. /**
  110. * @param {?} val
  111. * @return {?}
  112. */
  113. set offset(val) {
  114. this._offset = val;
  115. if (!this.scrollbarV || (this.scrollbarV && !this.virtualization))
  116. this.recalcLayout();
  117. }
  118. /**
  119. * @return {?}
  120. */
  121. get offset() {
  122. return this._offset;
  123. }
  124. /**
  125. * @param {?} val
  126. * @return {?}
  127. */
  128. set rowCount(val) {
  129. this._rowCount = val;
  130. this.recalcLayout();
  131. }
  132. /**
  133. * @return {?}
  134. */
  135. get rowCount() {
  136. return this._rowCount;
  137. }
  138. /**
  139. * @return {?}
  140. */
  141. get bodyWidth() {
  142. if (this.scrollbarH) {
  143. return this.innerWidth + 'px';
  144. }
  145. else {
  146. return '100%';
  147. }
  148. }
  149. /**
  150. * @param {?} val
  151. * @return {?}
  152. */
  153. set bodyHeight(val) {
  154. if (this.scrollbarV) {
  155. this._bodyHeight = val + 'px';
  156. }
  157. else {
  158. this._bodyHeight = 'auto';
  159. }
  160. this.recalcLayout();
  161. }
  162. /**
  163. * @return {?}
  164. */
  165. get bodyHeight() {
  166. return this._bodyHeight;
  167. }
  168. /**
  169. * Returns if selection is enabled.
  170. * @return {?}
  171. */
  172. get selectEnabled() {
  173. return !!this.selectionType;
  174. }
  175. /**
  176. * Property that would calculate the height of scroll bar
  177. * based on the row heights cache for virtual scroll and virtualization. Other scenarios
  178. * calculate scroll height automatically (as height will be undefined).
  179. * @return {?}
  180. */
  181. get scrollHeight() {
  182. if (this.scrollbarV && this.virtualization && this.rowCount) {
  183. return this.rowHeightsCache.query(this.rowCount - 1);
  184. }
  185. // avoid TS7030: Not all code paths return a value.
  186. return undefined;
  187. }
  188. /**
  189. * Called after the constructor, initializing input properties
  190. * @return {?}
  191. */
  192. ngOnInit() {
  193. if (this.rowDetail) {
  194. this.listener = this.rowDetail.toggle.subscribe((/**
  195. * @param {?} __0
  196. * @return {?}
  197. */
  198. ({ type, value }) => {
  199. if (type === 'row') {
  200. this.toggleRowExpansion(value);
  201. }
  202. if (type === 'all') {
  203. this.toggleAllRows(value);
  204. }
  205. // Refresh rows after toggle
  206. // Fixes #883
  207. this.updateIndexes();
  208. this.updateRows();
  209. this.cd.markForCheck();
  210. }));
  211. }
  212. if (this.groupHeader) {
  213. this.listener = this.groupHeader.toggle.subscribe((/**
  214. * @param {?} __0
  215. * @return {?}
  216. */
  217. ({ type, value }) => {
  218. if (type === 'group') {
  219. this.toggleRowExpansion(value);
  220. }
  221. if (type === 'all') {
  222. this.toggleAllRows(value);
  223. }
  224. // Refresh rows after toggle
  225. // Fixes #883
  226. this.updateIndexes();
  227. this.updateRows();
  228. this.cd.markForCheck();
  229. }));
  230. }
  231. }
  232. /**
  233. * Called once, before the instance is destroyed.
  234. * @return {?}
  235. */
  236. ngOnDestroy() {
  237. if (this.rowDetail || this.groupHeader) {
  238. this.listener.unsubscribe();
  239. }
  240. }
  241. /**
  242. * Updates the Y offset given a new offset.
  243. * @param {?=} offset
  244. * @return {?}
  245. */
  246. updateOffsetY(offset) {
  247. // scroller is missing on empty table
  248. if (!this.scroller) {
  249. return;
  250. }
  251. if (this.scrollbarV && this.virtualization && offset) {
  252. // First get the row Index that we need to move to.
  253. /** @type {?} */
  254. const rowIndex = this.pageSize * offset;
  255. offset = this.rowHeightsCache.query(rowIndex - 1);
  256. }
  257. else if (this.scrollbarV && !this.virtualization) {
  258. offset = 0;
  259. }
  260. this.scroller.setOffset(offset || 0);
  261. }
  262. /**
  263. * Body was scrolled, this is mainly useful for
  264. * when a user is server-side pagination via virtual scroll.
  265. * @param {?} event
  266. * @return {?}
  267. */
  268. onBodyScroll(event) {
  269. /** @type {?} */
  270. const scrollYPos = event.scrollYPos;
  271. /** @type {?} */
  272. const scrollXPos = event.scrollXPos;
  273. // if scroll change, trigger update
  274. // this is mainly used for header cell positions
  275. if (this.offsetY !== scrollYPos || this.offsetX !== scrollXPos) {
  276. this.scroll.emit({
  277. offsetY: scrollYPos,
  278. offsetX: scrollXPos
  279. });
  280. }
  281. this.offsetY = scrollYPos;
  282. this.offsetX = scrollXPos;
  283. this.updateIndexes();
  284. this.updatePage(event.direction);
  285. this.updateRows();
  286. }
  287. /**
  288. * Updates the page given a direction.
  289. * @param {?} direction
  290. * @return {?}
  291. */
  292. updatePage(direction) {
  293. /** @type {?} */
  294. let offset = this.indexes.first / this.pageSize;
  295. if (direction === 'up') {
  296. offset = Math.ceil(offset);
  297. }
  298. else if (direction === 'down') {
  299. offset = Math.floor(offset);
  300. }
  301. if (direction !== undefined && !isNaN(offset)) {
  302. this.page.emit({ offset });
  303. }
  304. }
  305. /**
  306. * Updates the rows in the view port
  307. * @return {?}
  308. */
  309. updateRows() {
  310. const { first, last } = this.indexes;
  311. /** @type {?} */
  312. let rowIndex = first;
  313. /** @type {?} */
  314. let idx = 0;
  315. /** @type {?} */
  316. const temp = [];
  317. this.rowIndexes.clear();
  318. // if grouprowsby has been specified treat row paging
  319. // parameters as group paging parameters ie if limit 10 has been
  320. // specified treat it as 10 groups rather than 10 rows
  321. if (this.groupedRows) {
  322. /** @type {?} */
  323. let maxRowsPerGroup = 3;
  324. // if there is only one group set the maximum number of
  325. // rows per group the same as the total number of rows
  326. if (this.groupedRows.length === 1) {
  327. maxRowsPerGroup = this.groupedRows[0].value.length;
  328. }
  329. while (rowIndex < last && rowIndex < this.groupedRows.length) {
  330. // Add the groups into this page
  331. /** @type {?} */
  332. const group = this.groupedRows[rowIndex];
  333. temp[idx] = group;
  334. idx++;
  335. // Group index in this context
  336. rowIndex++;
  337. }
  338. }
  339. else {
  340. while (rowIndex < last && rowIndex < this.rowCount) {
  341. /** @type {?} */
  342. const row = this.rows[rowIndex];
  343. if (row) {
  344. this.rowIndexes.set(row, rowIndex);
  345. temp[idx] = row;
  346. }
  347. idx++;
  348. rowIndex++;
  349. }
  350. }
  351. this.temp = temp;
  352. }
  353. /**
  354. * Get the row height
  355. * @param {?} row
  356. * @return {?}
  357. */
  358. getRowHeight(row) {
  359. // if its a function return it
  360. if (typeof this.rowHeight === 'function') {
  361. return this.rowHeight(row);
  362. }
  363. return (/** @type {?} */ (this.rowHeight));
  364. }
  365. /**
  366. * @param {?} group the group with all rows
  367. * @return {?}
  368. */
  369. getGroupHeight(group) {
  370. /** @type {?} */
  371. let rowHeight = 0;
  372. if (group.value) {
  373. for (let index = 0; index < group.value.length; index++) {
  374. rowHeight += this.getRowAndDetailHeight(group.value[index]);
  375. }
  376. }
  377. return rowHeight;
  378. }
  379. /**
  380. * Calculate row height based on the expanded state of the row.
  381. * @param {?} row
  382. * @return {?}
  383. */
  384. getRowAndDetailHeight(row) {
  385. /** @type {?} */
  386. let rowHeight = this.getRowHeight(row);
  387. /** @type {?} */
  388. const expanded = this.getRowExpanded(row);
  389. // Adding detail row height if its expanded.
  390. if (expanded) {
  391. rowHeight += this.getDetailRowHeight(row);
  392. }
  393. return rowHeight;
  394. }
  395. /**
  396. * Calculates the styles for the row so that the rows can be moved in 2D space
  397. * during virtual scroll inside the DOM. In the below case the Y position is
  398. * manipulated. As an example, if the height of row 0 is 30 px and row 1 is
  399. * 100 px then following styles are generated:
  400. *
  401. * transform: translate3d(0px, 0px, 0px); -> row0
  402. * transform: translate3d(0px, 30px, 0px); -> row1
  403. * transform: translate3d(0px, 130px, 0px); -> row2
  404. *
  405. * Row heights have to be calculated based on the row heights cache as we wont
  406. * be able to determine which row is of what height before hand. In the above
  407. * case the positionY of the translate3d for row2 would be the sum of all the
  408. * heights of the rows before it (i.e. row0 and row1).
  409. *
  410. * \@memberOf DataTableBodyComponent
  411. * @param {?} rows the row that needs to be placed in the 2D space.
  412. * @return {?} the CSS3 style to be applied
  413. *
  414. */
  415. getRowsStyles(rows) {
  416. /** @type {?} */
  417. const styles = {};
  418. // only add styles for the group if there is a group
  419. if (this.groupedRows) {
  420. styles.width = this.columnGroupWidths.total;
  421. }
  422. if (this.scrollbarV && this.virtualization) {
  423. /** @type {?} */
  424. let idx = 0;
  425. if (this.groupedRows) {
  426. // Get the latest row rowindex in a group
  427. /** @type {?} */
  428. const row = rows[rows.length - 1];
  429. idx = row ? this.getRowIndex(row) : 0;
  430. }
  431. else {
  432. idx = this.getRowIndex(rows);
  433. }
  434. // const pos = idx * rowHeight;
  435. // The position of this row would be the sum of all row heights
  436. // until the previous row position.
  437. /** @type {?} */
  438. const pos = this.rowHeightsCache.query(idx - 1);
  439. translateXY(styles, 0, pos);
  440. }
  441. return styles;
  442. }
  443. /**
  444. * Calculate bottom summary row offset for scrollbar mode.
  445. * For more information about cache and offset calculation
  446. * see description for `getRowsStyles` method
  447. *
  448. * \@memberOf DataTableBodyComponent
  449. * @return {?} the CSS3 style to be applied
  450. *
  451. */
  452. getBottomSummaryRowStyles() {
  453. if (!this.scrollbarV || !this.rows || !this.rows.length) {
  454. return null;
  455. }
  456. /** @type {?} */
  457. const styles = { position: 'absolute' };
  458. /** @type {?} */
  459. const pos = this.rowHeightsCache.query(this.rows.length - 1);
  460. translateXY(styles, 0, pos);
  461. return styles;
  462. }
  463. /**
  464. * Hides the loading indicator
  465. * @return {?}
  466. */
  467. hideIndicator() {
  468. setTimeout((/**
  469. * @return {?}
  470. */
  471. () => (this.loadingIndicator = false)), 500);
  472. }
  473. /**
  474. * Updates the index of the rows in the viewport
  475. * @return {?}
  476. */
  477. updateIndexes() {
  478. /** @type {?} */
  479. let first = 0;
  480. /** @type {?} */
  481. let last = 0;
  482. if (this.scrollbarV) {
  483. if (this.virtualization) {
  484. // Calculation of the first and last indexes will be based on where the
  485. // scrollY position would be at. The last index would be the one
  486. // that shows up inside the view port the last.
  487. /** @type {?} */
  488. const height = parseInt(this.bodyHeight, 0);
  489. first = this.rowHeightsCache.getRowIndex(this.offsetY);
  490. last = this.rowHeightsCache.getRowIndex(height + this.offsetY) + 1;
  491. }
  492. else {
  493. // If virtual rows are not needed
  494. // We render all in one go
  495. first = 0;
  496. last = this.rowCount;
  497. }
  498. }
  499. else {
  500. // The server is handling paging and will pass an array that begins with the
  501. // element at a specified offset. first should always be 0 with external paging.
  502. if (!this.externalPaging) {
  503. first = Math.max(this.offset * this.pageSize, 0);
  504. }
  505. last = Math.min(first + this.pageSize, this.rowCount);
  506. }
  507. this.indexes = { first, last };
  508. }
  509. /**
  510. * Refreshes the full Row Height cache. Should be used
  511. * when the entire row array state has changed.
  512. * @return {?}
  513. */
  514. refreshRowHeightCache() {
  515. if (!this.scrollbarV || (this.scrollbarV && !this.virtualization)) {
  516. return;
  517. }
  518. // clear the previous row height cache if already present.
  519. // this is useful during sorts, filters where the state of the
  520. // rows array is changed.
  521. this.rowHeightsCache.clearCache();
  522. // Initialize the tree only if there are rows inside the tree.
  523. if (this.rows && this.rows.length) {
  524. /** @type {?} */
  525. const rowExpansions = new Set();
  526. for (const row of this.rows) {
  527. if (this.getRowExpanded(row)) {
  528. rowExpansions.add(row);
  529. }
  530. }
  531. this.rowHeightsCache.initCache({
  532. rows: this.rows,
  533. rowHeight: this.rowHeight,
  534. detailRowHeight: this.getDetailRowHeight,
  535. externalVirtual: this.scrollbarV && this.externalPaging,
  536. rowCount: this.rowCount,
  537. rowIndexes: this.rowIndexes,
  538. rowExpansions
  539. });
  540. }
  541. }
  542. /**
  543. * Gets the index for the view port
  544. * @return {?}
  545. */
  546. getAdjustedViewPortIndex() {
  547. // Capture the row index of the first row that is visible on the viewport.
  548. // If the scroll bar is just below the row which is highlighted then make that as the
  549. // first index.
  550. /** @type {?} */
  551. const viewPortFirstRowIndex = this.indexes.first;
  552. if (this.scrollbarV && this.virtualization) {
  553. /** @type {?} */
  554. const offsetScroll = this.rowHeightsCache.query(viewPortFirstRowIndex - 1);
  555. return offsetScroll <= this.offsetY ? viewPortFirstRowIndex - 1 : viewPortFirstRowIndex;
  556. }
  557. return viewPortFirstRowIndex;
  558. }
  559. /**
  560. * Toggle the Expansion of the row i.e. if the row is expanded then it will
  561. * collapse and vice versa. Note that the expanded status is stored as
  562. * a part of the row object itself as we have to preserve the expanded row
  563. * status in case of sorting and filtering of the row set.
  564. * @param {?} row
  565. * @return {?}
  566. */
  567. toggleRowExpansion(row) {
  568. // Capture the row index of the first row that is visible on the viewport.
  569. /** @type {?} */
  570. const viewPortFirstRowIndex = this.getAdjustedViewPortIndex();
  571. /** @type {?} */
  572. const rowExpandedIdx = this.getRowExpandedIdx(row, this.rowExpansions);
  573. /** @type {?} */
  574. const expanded = rowExpandedIdx > -1;
  575. // If the detailRowHeight is auto --> only in case of non-virtualized scroll
  576. if (this.scrollbarV && this.virtualization) {
  577. /** @type {?} */
  578. const detailRowHeight = this.getDetailRowHeight(row) * (expanded ? -1 : 1);
  579. // const idx = this.rowIndexes.get(row) || 0;
  580. /** @type {?} */
  581. const idx = this.getRowIndex(row);
  582. this.rowHeightsCache.update(idx, detailRowHeight);
  583. }
  584. // Update the toggled row and update thive nevere heights in the cache.
  585. if (expanded) {
  586. this.rowExpansions.splice(rowExpandedIdx, 1);
  587. }
  588. else {
  589. this.rowExpansions.push(row);
  590. }
  591. this.detailToggle.emit({
  592. rows: [row],
  593. currentIndex: viewPortFirstRowIndex
  594. });
  595. }
  596. /**
  597. * Expand/Collapse all the rows no matter what their state is.
  598. * @param {?} expanded
  599. * @return {?}
  600. */
  601. toggleAllRows(expanded) {
  602. // clear prev expansions
  603. this.rowExpansions = [];
  604. // Capture the row index of the first row that is visible on the viewport.
  605. /** @type {?} */
  606. const viewPortFirstRowIndex = this.getAdjustedViewPortIndex();
  607. if (expanded) {
  608. for (const row of this.rows) {
  609. this.rowExpansions.push(row);
  610. }
  611. }
  612. if (this.scrollbarV) {
  613. // Refresh the full row heights cache since every row was affected.
  614. this.recalcLayout();
  615. }
  616. // Emit all rows that have been expanded.
  617. this.detailToggle.emit({
  618. rows: this.rows,
  619. currentIndex: viewPortFirstRowIndex
  620. });
  621. }
  622. /**
  623. * Recalculates the table
  624. * @return {?}
  625. */
  626. recalcLayout() {
  627. this.refreshRowHeightCache();
  628. this.updateIndexes();
  629. this.updateRows();
  630. }
  631. /**
  632. * Tracks the column
  633. * @param {?} index
  634. * @param {?} column
  635. * @return {?}
  636. */
  637. columnTrackingFn(index, column) {
  638. return column.$$id;
  639. }
  640. /**
  641. * Gets the row pinning group styles
  642. * @param {?} group
  643. * @return {?}
  644. */
  645. stylesByGroup(group) {
  646. /** @type {?} */
  647. const widths = this.columnGroupWidths;
  648. /** @type {?} */
  649. const offsetX = this.offsetX;
  650. /** @type {?} */
  651. const styles = {
  652. width: `${widths[group]}px`
  653. };
  654. if (group === 'left') {
  655. translateXY(styles, offsetX, 0);
  656. }
  657. else if (group === 'right') {
  658. /** @type {?} */
  659. const bodyWidth = parseInt(this.innerWidth + '', 0);
  660. /** @type {?} */
  661. const totalDiff = widths.total - bodyWidth;
  662. /** @type {?} */
  663. const offsetDiff = totalDiff - offsetX;
  664. /** @type {?} */
  665. const offset = offsetDiff * -1;
  666. translateXY(styles, offset, 0);
  667. }
  668. return styles;
  669. }
  670. /**
  671. * Returns if the row was expanded and set default row expansion when row expansion is empty
  672. * @param {?} row
  673. * @return {?}
  674. */
  675. getRowExpanded(row) {
  676. if (this.rowExpansions.length === 0 && this.groupExpansionDefault) {
  677. for (const group of this.groupedRows) {
  678. this.rowExpansions.push(group);
  679. }
  680. }
  681. return this.getRowExpandedIdx(row, this.rowExpansions) > -1;
  682. }
  683. /**
  684. * @param {?} row
  685. * @param {?} expanded
  686. * @return {?}
  687. */
  688. getRowExpandedIdx(row, expanded) {
  689. if (!expanded || !expanded.length)
  690. return -1;
  691. /** @type {?} */
  692. const rowId = this.rowIdentity(row);
  693. return expanded.findIndex((/**
  694. * @param {?} r
  695. * @return {?}
  696. */
  697. (r) => {
  698. /** @type {?} */
  699. const id = this.rowIdentity(r);
  700. return id === rowId;
  701. }));
  702. }
  703. /**
  704. * Gets the row index given a row
  705. * @param {?} row
  706. * @return {?}
  707. */
  708. getRowIndex(row) {
  709. return this.rowIndexes.get(row) || 0;
  710. }
  711. /**
  712. * @param {?} row
  713. * @return {?}
  714. */
  715. onTreeAction(row) {
  716. this.treeAction.emit({ row });
  717. }
  718. }
  719. DataTableBodyComponent.decorators = [
  720. { type: Component, args: [{
  721. selector: 'datatable-body',
  722. template: `
  723. <datatable-selection
  724. #selector
  725. [selected]="selected"
  726. [rows]="rows"
  727. [selectCheck]="selectCheck"
  728. [selectEnabled]="selectEnabled"
  729. [selectionType]="selectionType"
  730. [rowIdentity]="rowIdentity"
  731. (select)="select.emit($event)"
  732. (activate)="activate.emit($event)"
  733. >
  734. <datatable-progress *ngIf="loadingIndicator"> </datatable-progress>
  735. <datatable-scroller
  736. *ngIf="rows?.length"
  737. [scrollbarV]="scrollbarV"
  738. [scrollbarH]="scrollbarH"
  739. [scrollHeight]="scrollHeight"
  740. [scrollWidth]="columnGroupWidths?.total"
  741. (scroll)="onBodyScroll($event)"
  742. >
  743. <datatable-summary-row
  744. *ngIf="summaryRow && summaryPosition === 'top'"
  745. [rowHeight]="summaryHeight"
  746. [offsetX]="offsetX"
  747. [innerWidth]="innerWidth"
  748. [rows]="rows"
  749. [columns]="columns"
  750. >
  751. </datatable-summary-row>
  752. <datatable-row-wrapper
  753. [groupedRows]="groupedRows"
  754. *ngFor="let group of temp; let i = index; trackBy: rowTrackingFn"
  755. [innerWidth]="innerWidth"
  756. [ngStyle]="getRowsStyles(group)"
  757. [rowDetail]="rowDetail"
  758. [groupHeader]="groupHeader"
  759. [offsetX]="offsetX"
  760. [detailRowHeight]="getDetailRowHeight(group[i], i)"
  761. [row]="group"
  762. [expanded]="getRowExpanded(group)"
  763. [rowIndex]="getRowIndex(group[i])"
  764. (rowContextmenu)="rowContextmenu.emit($event)"
  765. >
  766. <datatable-body-row
  767. *ngIf="!groupedRows; else groupedRowsTemplate"
  768. tabindex="-1"
  769. [isSelected]="selector.getRowSelected(group)"
  770. [innerWidth]="innerWidth"
  771. [offsetX]="offsetX"
  772. [columns]="columns"
  773. [rowHeight]="getRowHeight(group)"
  774. [row]="group"
  775. [rowIndex]="getRowIndex(group)"
  776. [expanded]="getRowExpanded(group)"
  777. [rowClass]="rowClass"
  778. [displayCheck]="displayCheck"
  779. [treeStatus]="group.treeStatus"
  780. (treeAction)="onTreeAction(group)"
  781. (activate)="selector.onActivate($event, indexes.first + i)"
  782. >
  783. </datatable-body-row>
  784. <ng-template #groupedRowsTemplate>
  785. <datatable-body-row
  786. *ngFor="let row of group.value; let i = index; trackBy: rowTrackingFn"
  787. tabindex="-1"
  788. [isSelected]="selector.getRowSelected(row)"
  789. [innerWidth]="innerWidth"
  790. [offsetX]="offsetX"
  791. [columns]="columns"
  792. [rowHeight]="getRowHeight(row)"
  793. [row]="row"
  794. [group]="group.value"
  795. [rowIndex]="getRowIndex(row)"
  796. [expanded]="getRowExpanded(row)"
  797. [rowClass]="rowClass"
  798. (activate)="selector.onActivate($event, i)"
  799. >
  800. </datatable-body-row>
  801. </ng-template>
  802. </datatable-row-wrapper>
  803. <datatable-summary-row
  804. *ngIf="summaryRow && summaryPosition === 'bottom'"
  805. [ngStyle]="getBottomSummaryRowStyles()"
  806. [rowHeight]="summaryHeight"
  807. [offsetX]="offsetX"
  808. [innerWidth]="innerWidth"
  809. [rows]="rows"
  810. [columns]="columns"
  811. >
  812. </datatable-summary-row>
  813. </datatable-scroller>
  814. <div class="empty-row" *ngIf="!rows?.length && !loadingIndicator" [innerHTML]="emptyMessage"></div>
  815. </datatable-selection>
  816. `,
  817. changeDetection: ChangeDetectionStrategy.OnPush,
  818. host: {
  819. class: 'datatable-body'
  820. }
  821. }] }
  822. ];
  823. /** @nocollapse */
  824. DataTableBodyComponent.ctorParameters = () => [
  825. { type: ChangeDetectorRef }
  826. ];
  827. DataTableBodyComponent.propDecorators = {
  828. scrollbarV: [{ type: Input }],
  829. scrollbarH: [{ type: Input }],
  830. loadingIndicator: [{ type: Input }],
  831. externalPaging: [{ type: Input }],
  832. rowHeight: [{ type: Input }],
  833. offsetX: [{ type: Input }],
  834. emptyMessage: [{ type: Input }],
  835. selectionType: [{ type: Input }],
  836. selected: [{ type: Input }],
  837. rowIdentity: [{ type: Input }],
  838. rowDetail: [{ type: Input }],
  839. groupHeader: [{ type: Input }],
  840. selectCheck: [{ type: Input }],
  841. displayCheck: [{ type: Input }],
  842. trackByProp: [{ type: Input }],
  843. rowClass: [{ type: Input }],
  844. groupedRows: [{ type: Input }],
  845. groupExpansionDefault: [{ type: Input }],
  846. innerWidth: [{ type: Input }],
  847. groupRowsBy: [{ type: Input }],
  848. virtualization: [{ type: Input }],
  849. summaryRow: [{ type: Input }],
  850. summaryPosition: [{ type: Input }],
  851. summaryHeight: [{ type: Input }],
  852. pageSize: [{ type: Input }],
  853. rows: [{ type: Input }],
  854. columns: [{ type: Input }],
  855. offset: [{ type: Input }],
  856. rowCount: [{ type: Input }],
  857. bodyWidth: [{ type: HostBinding, args: ['style.width',] }],
  858. bodyHeight: [{ type: Input }, { type: HostBinding, args: ['style.height',] }],
  859. scroll: [{ type: Output }],
  860. page: [{ type: Output }],
  861. activate: [{ type: Output }],
  862. select: [{ type: Output }],
  863. detailToggle: [{ type: Output }],
  864. rowContextmenu: [{ type: Output }],
  865. treeAction: [{ type: Output }],
  866. scroller: [{ type: ViewChild, args: [ScrollerComponent, { static: false },] }]
  867. };
  868. if (false) {
  869. /** @type {?} */
  870. DataTableBodyComponent.prototype.scrollbarV;
  871. /** @type {?} */
  872. DataTableBodyComponent.prototype.scrollbarH;
  873. /** @type {?} */
  874. DataTableBodyComponent.prototype.loadingIndicator;
  875. /** @type {?} */
  876. DataTableBodyComponent.prototype.externalPaging;
  877. /** @type {?} */
  878. DataTableBodyComponent.prototype.rowHeight;
  879. /** @type {?} */
  880. DataTableBodyComponent.prototype.offsetX;
  881. /** @type {?} */
  882. DataTableBodyComponent.prototype.emptyMessage;
  883. /** @type {?} */
  884. DataTableBodyComponent.prototype.selectionType;
  885. /** @type {?} */
  886. DataTableBodyComponent.prototype.selected;
  887. /** @type {?} */
  888. DataTableBodyComponent.prototype.rowIdentity;
  889. /** @type {?} */
  890. DataTableBodyComponent.prototype.rowDetail;
  891. /** @type {?} */
  892. DataTableBodyComponent.prototype.groupHeader;
  893. /** @type {?} */
  894. DataTableBodyComponent.prototype.selectCheck;
  895. /** @type {?} */
  896. DataTableBodyComponent.prototype.displayCheck;
  897. /** @type {?} */
  898. DataTableBodyComponent.prototype.trackByProp;
  899. /** @type {?} */
  900. DataTableBodyComponent.prototype.rowClass;
  901. /** @type {?} */
  902. DataTableBodyComponent.prototype.groupedRows;
  903. /** @type {?} */
  904. DataTableBodyComponent.prototype.groupExpansionDefault;
  905. /** @type {?} */
  906. DataTableBodyComponent.prototype.innerWidth;
  907. /** @type {?} */
  908. DataTableBodyComponent.prototype.groupRowsBy;
  909. /** @type {?} */
  910. DataTableBodyComponent.prototype.virtualization;
  911. /** @type {?} */
  912. DataTableBodyComponent.prototype.summaryRow;
  913. /** @type {?} */
  914. DataTableBodyComponent.prototype.summaryPosition;
  915. /** @type {?} */
  916. DataTableBodyComponent.prototype.summaryHeight;
  917. /** @type {?} */
  918. DataTableBodyComponent.prototype.scroll;
  919. /** @type {?} */
  920. DataTableBodyComponent.prototype.page;
  921. /** @type {?} */
  922. DataTableBodyComponent.prototype.activate;
  923. /** @type {?} */
  924. DataTableBodyComponent.prototype.select;
  925. /** @type {?} */
  926. DataTableBodyComponent.prototype.detailToggle;
  927. /** @type {?} */
  928. DataTableBodyComponent.prototype.rowContextmenu;
  929. /** @type {?} */
  930. DataTableBodyComponent.prototype.treeAction;
  931. /** @type {?} */
  932. DataTableBodyComponent.prototype.scroller;
  933. /** @type {?} */
  934. DataTableBodyComponent.prototype.rowHeightsCache;
  935. /** @type {?} */
  936. DataTableBodyComponent.prototype.temp;
  937. /** @type {?} */
  938. DataTableBodyComponent.prototype.offsetY;
  939. /** @type {?} */
  940. DataTableBodyComponent.prototype.indexes;
  941. /** @type {?} */
  942. DataTableBodyComponent.prototype.columnGroupWidths;
  943. /** @type {?} */
  944. DataTableBodyComponent.prototype.columnGroupWidthsWithoutGroup;
  945. /** @type {?} */
  946. DataTableBodyComponent.prototype.rowTrackingFn;
  947. /** @type {?} */
  948. DataTableBodyComponent.prototype.listener;
  949. /** @type {?} */
  950. DataTableBodyComponent.prototype.rowIndexes;
  951. /** @type {?} */
  952. DataTableBodyComponent.prototype.rowExpansions;
  953. /** @type {?} */
  954. DataTableBodyComponent.prototype._rows;
  955. /** @type {?} */
  956. DataTableBodyComponent.prototype._bodyHeight;
  957. /** @type {?} */
  958. DataTableBodyComponent.prototype._columns;
  959. /** @type {?} */
  960. DataTableBodyComponent.prototype._rowCount;
  961. /** @type {?} */
  962. DataTableBodyComponent.prototype._offset;
  963. /** @type {?} */
  964. DataTableBodyComponent.prototype._pageSize;
  965. /**
  966. * Get the height of the detail row.
  967. * @type {?}
  968. */
  969. DataTableBodyComponent.prototype.getDetailRowHeight;
  970. /**
  971. * @type {?}
  972. * @private
  973. */
  974. DataTableBodyComponent.prototype.cd;
  975. }
  976. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"body.component.js","sourceRoot":"ng://@swimlane/ngx-datatable/","sources":["lib/components/body/body.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Output,\n  EventEmitter,\n  Input,\n  HostBinding,\n  ChangeDetectorRef,\n  ViewChild,\n  OnInit,\n  OnDestroy,\n  ChangeDetectionStrategy\n} from '@angular/core';\nimport { ScrollerComponent } from './scroller.component';\nimport { MouseEvent } from '../../events';\nimport { SelectionType } from '../../types/selection.type';\nimport { columnsByPin, columnGroupWidths } from '../../utils/column';\nimport { RowHeightCache } from '../../utils/row-height-cache';\nimport { translateXY } from '../../utils/translate';\n\n@Component({\n  selector: 'datatable-body',\n  template: `\n    <datatable-selection\n      #selector\n      [selected]=\"selected\"\n      [rows]=\"rows\"\n      [selectCheck]=\"selectCheck\"\n      [selectEnabled]=\"selectEnabled\"\n      [selectionType]=\"selectionType\"\n      [rowIdentity]=\"rowIdentity\"\n      (select)=\"select.emit($event)\"\n      (activate)=\"activate.emit($event)\"\n    >\n      <datatable-progress *ngIf=\"loadingIndicator\"> </datatable-progress>\n      <datatable-scroller\n        *ngIf=\"rows?.length\"\n        [scrollbarV]=\"scrollbarV\"\n        [scrollbarH]=\"scrollbarH\"\n        [scrollHeight]=\"scrollHeight\"\n        [scrollWidth]=\"columnGroupWidths?.total\"\n        (scroll)=\"onBodyScroll($event)\"\n      >\n        <datatable-summary-row\n          *ngIf=\"summaryRow && summaryPosition === 'top'\"\n          [rowHeight]=\"summaryHeight\"\n          [offsetX]=\"offsetX\"\n          [innerWidth]=\"innerWidth\"\n          [rows]=\"rows\"\n          [columns]=\"columns\"\n        >\n        </datatable-summary-row>\n        <datatable-row-wrapper\n          [groupedRows]=\"groupedRows\"\n          *ngFor=\"let group of temp; let i = index; trackBy: rowTrackingFn\"\n          [innerWidth]=\"innerWidth\"\n          [ngStyle]=\"getRowsStyles(group)\"\n          [rowDetail]=\"rowDetail\"\n          [groupHeader]=\"groupHeader\"\n          [offsetX]=\"offsetX\"\n          [detailRowHeight]=\"getDetailRowHeight(group[i], i)\"\n          [row]=\"group\"\n          [expanded]=\"getRowExpanded(group)\"\n          [rowIndex]=\"getRowIndex(group[i])\"\n          (rowContextmenu)=\"rowContextmenu.emit($event)\"\n        >\n          <datatable-body-row\n            *ngIf=\"!groupedRows; else groupedRowsTemplate\"\n            tabindex=\"-1\"\n            [isSelected]=\"selector.getRowSelected(group)\"\n            [innerWidth]=\"innerWidth\"\n            [offsetX]=\"offsetX\"\n            [columns]=\"columns\"\n            [rowHeight]=\"getRowHeight(group)\"\n            [row]=\"group\"\n            [rowIndex]=\"getRowIndex(group)\"\n            [expanded]=\"getRowExpanded(group)\"\n            [rowClass]=\"rowClass\"\n            [displayCheck]=\"displayCheck\"\n            [treeStatus]=\"group.treeStatus\"\n            (treeAction)=\"onTreeAction(group)\"\n            (activate)=\"selector.onActivate($event, indexes.first + i)\"\n          >\n          </datatable-body-row>\n          <ng-template #groupedRowsTemplate>\n            <datatable-body-row\n              *ngFor=\"let row of group.value; let i = index; trackBy: rowTrackingFn\"\n              tabindex=\"-1\"\n              [isSelected]=\"selector.getRowSelected(row)\"\n              [innerWidth]=\"innerWidth\"\n              [offsetX]=\"offsetX\"\n              [columns]=\"columns\"\n              [rowHeight]=\"getRowHeight(row)\"\n              [row]=\"row\"\n              [group]=\"group.value\"\n              [rowIndex]=\"getRowIndex(row)\"\n              [expanded]=\"getRowExpanded(row)\"\n              [rowClass]=\"rowClass\"\n              (activate)=\"selector.onActivate($event, i)\"\n            >\n            </datatable-body-row>\n          </ng-template>\n        </datatable-row-wrapper>\n        <datatable-summary-row\n          *ngIf=\"summaryRow && summaryPosition === 'bottom'\"\n          [ngStyle]=\"getBottomSummaryRowStyles()\"\n          [rowHeight]=\"summaryHeight\"\n          [offsetX]=\"offsetX\"\n          [innerWidth]=\"innerWidth\"\n          [rows]=\"rows\"\n          [columns]=\"columns\"\n        >\n        </datatable-summary-row>\n      </datatable-scroller>\n      <div class=\"empty-row\" *ngIf=\"!rows?.length && !loadingIndicator\" [innerHTML]=\"emptyMessage\"></div>\n    </datatable-selection>\n  `,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  host: {\n    class: 'datatable-body'\n  }\n})\nexport class DataTableBodyComponent implements OnInit, OnDestroy {\n  @Input() scrollbarV: boolean;\n  @Input() scrollbarH: boolean;\n  @Input() loadingIndicator: boolean;\n  @Input() externalPaging: boolean;\n  @Input() rowHeight: number | 'auto' | ((row?: any) => number);\n  @Input() offsetX: number;\n  @Input() emptyMessage: string;\n  @Input() selectionType: SelectionType;\n  @Input() selected: any[] = [];\n  @Input() rowIdentity: any;\n  @Input() rowDetail: any;\n  @Input() groupHeader: any;\n  @Input() selectCheck: any;\n  @Input() displayCheck: any;\n  @Input() trackByProp: string;\n  @Input() rowClass: any;\n  @Input() groupedRows: any;\n  @Input() groupExpansionDefault: boolean;\n  @Input() innerWidth: number;\n  @Input() groupRowsBy: string;\n  @Input() virtualization: boolean;\n  @Input() summaryRow: boolean;\n  @Input() summaryPosition: string;\n  @Input() summaryHeight: number;\n\n  @Input() set pageSize(val: number) {\n    this._pageSize = val;\n    this.recalcLayout();\n  }\n\n  get pageSize(): number {\n    return this._pageSize;\n  }\n\n  @Input() set rows(val: any[]) {\n    this._rows = val;\n    this.recalcLayout();\n  }\n\n  get rows(): any[] {\n    return this._rows;\n  }\n\n  @Input() set columns(val: any[]) {\n    this._columns = val;\n    const colsByPin = columnsByPin(val);\n    this.columnGroupWidths = columnGroupWidths(colsByPin, val);\n  }\n\n  get columns(): any[] {\n    return this._columns;\n  }\n\n  @Input() set offset(val: number) {\n    this._offset = val;\n    if (!this.scrollbarV || (this.scrollbarV && !this.virtualization))\n      this.recalcLayout();\n  }\n\n  get offset(): number {\n    return this._offset;\n  }\n\n  @Input() set rowCount(val: number) {\n    this._rowCount = val;\n    this.recalcLayout();\n  }\n\n  get rowCount(): number {\n    return this._rowCount;\n  }\n\n  @HostBinding('style.width')\n  get bodyWidth(): string {\n    if (this.scrollbarH) {\n      return this.innerWidth + 'px';\n    } else {\n      return '100%';\n    }\n  }\n\n  @Input()\n  @HostBinding('style.height')\n  set bodyHeight(val) {\n    if (this.scrollbarV) {\n      this._bodyHeight = val + 'px';\n    } else {\n      this._bodyHeight = 'auto';\n    }\n\n    this.recalcLayout();\n  }\n\n  get bodyHeight() {\n    return this._bodyHeight;\n  }\n\n  @Output() scroll: EventEmitter<any> = new EventEmitter();\n  @Output() page: EventEmitter<any> = new EventEmitter();\n  @Output() activate: EventEmitter<any> = new EventEmitter();\n  @Output() select: EventEmitter<any> = new EventEmitter();\n  @Output() detailToggle: EventEmitter<any> = new EventEmitter();\n  @Output() rowContextmenu = new EventEmitter<{ event: MouseEvent; row: any }>(false);\n  @Output() treeAction: EventEmitter<any> = new EventEmitter();\n\n  @ViewChild(ScrollerComponent, { static: false }) scroller: ScrollerComponent;\n\n  /**\n   * Returns if selection is enabled.\n   */\n  get selectEnabled(): boolean {\n    return !!this.selectionType;\n  }\n\n  /**\n   * Property that would calculate the height of scroll bar\n   * based on the row heights cache for virtual scroll and virtualization. Other scenarios\n   * calculate scroll height automatically (as height will be undefined).\n   */\n  get scrollHeight(): number | undefined {\n    if (this.scrollbarV && this.virtualization && this.rowCount) {\n      return this.rowHeightsCache.query(this.rowCount - 1);\n    }\n    // avoid TS7030: Not all code paths return a value.\n    return undefined;\n  }\n\n  rowHeightsCache: RowHeightCache = new RowHeightCache();\n  temp: any[] = [];\n  offsetY = 0;\n  indexes: any = {};\n  columnGroupWidths: any;\n  columnGroupWidthsWithoutGroup: any;\n  rowTrackingFn: any;\n  listener: any;\n  rowIndexes: any = new Map();\n  rowExpansions: any[] = [];\n\n  _rows: any[];\n  _bodyHeight: any;\n  _columns: any[];\n  _rowCount: number;\n  _offset: number;\n  _pageSize: number;\n\n  /**\n   * Creates an instance of DataTableBodyComponent.\n   */\n  constructor(private cd: ChangeDetectorRef) {\n    // declare fn here so we can get access to the `this` property\n    this.rowTrackingFn = (index: number, row: any): any => {\n      const idx = this.getRowIndex(row);\n      if (this.trackByProp) {\n        return row[this.trackByProp];\n      } else {\n        return idx;\n      }\n    };\n  }\n\n  /**\n   * Called after the constructor, initializing input properties\n   */\n  ngOnInit(): void {\n    if (this.rowDetail) {\n      this.listener = this.rowDetail.toggle.subscribe(({ type, value }: { type: string; value: any }) => {\n        if (type === 'row') {\n          this.toggleRowExpansion(value);\n        }\n        if (type === 'all') {\n          this.toggleAllRows(value);\n        }\n\n        // Refresh rows after toggle\n        // Fixes #883\n        this.updateIndexes();\n        this.updateRows();\n        this.cd.markForCheck();\n      });\n    }\n\n    if (this.groupHeader) {\n      this.listener = this.groupHeader.toggle.subscribe(({ type, value }: { type: string; value: any }) => {\n        if (type === 'group') {\n          this.toggleRowExpansion(value);\n        }\n        if (type === 'all') {\n          this.toggleAllRows(value);\n        }\n\n        // Refresh rows after toggle\n        // Fixes #883\n        this.updateIndexes();\n        this.updateRows();\n        this.cd.markForCheck();\n      });\n    }\n  }\n\n  /**\n   * Called once, before the instance is destroyed.\n   */\n  ngOnDestroy(): void {\n    if (this.rowDetail || this.groupHeader) {\n      this.listener.unsubscribe();\n    }\n  }\n\n  /**\n   * Updates the Y offset given a new offset.\n   */\n  updateOffsetY(offset?: number): void {\n    // scroller is missing on empty table\n    if (!this.scroller) {\n      return;\n    }\n\n    if (this.scrollbarV && this.virtualization && offset) {\n      // First get the row Index that we need to move to.\n      const rowIndex = this.pageSize * offset;\n      offset = this.rowHeightsCache.query(rowIndex - 1);\n    } else if (this.scrollbarV && !this.virtualization) {\n      offset = 0;\n    }\n\n    this.scroller.setOffset(offset || 0);\n  }\n\n  /**\n   * Body was scrolled, this is mainly useful for\n   * when a user is server-side pagination via virtual scroll.\n   */\n  onBodyScroll(event: any): void {\n    const scrollYPos: number = event.scrollYPos;\n    const scrollXPos: number = event.scrollXPos;\n\n    // if scroll change, trigger update\n    // this is mainly used for header cell positions\n    if (this.offsetY !== scrollYPos || this.offsetX !== scrollXPos) {\n      this.scroll.emit({\n        offsetY: scrollYPos,\n        offsetX: scrollXPos\n      });\n    }\n\n    this.offsetY = scrollYPos;\n    this.offsetX = scrollXPos;\n\n    this.updateIndexes();\n    this.updatePage(event.direction);\n    this.updateRows();\n  }\n\n  /**\n   * Updates the page given a direction.\n   */\n  updatePage(direction: string): void {\n    let offset = this.indexes.first / this.pageSize;\n\n    if (direction === 'up') {\n      offset = Math.ceil(offset);\n    } else if (direction === 'down') {\n      offset = Math.floor(offset);\n    }\n\n    if (direction !== undefined && !isNaN(offset)) {\n      this.page.emit({ offset });\n    }\n  }\n\n  /**\n   * Updates the rows in the view port\n   */\n  updateRows(): void {\n    const { first, last } = this.indexes;\n    let rowIndex = first;\n    let idx = 0;\n    const temp: any[] = [];\n\n    this.rowIndexes.clear();\n\n    // if grouprowsby has been specified treat row paging\n    // parameters as group paging parameters ie if limit 10 has been\n    // specified treat it as 10 groups rather than 10 rows\n    if (this.groupedRows) {\n      let maxRowsPerGroup = 3;\n      // if there is only one group set the maximum number of\n      // rows per group the same as the total number of rows\n      if (this.groupedRows.length === 1) {\n        maxRowsPerGroup = this.groupedRows[0].value.length;\n      }\n\n      while (rowIndex < last && rowIndex < this.groupedRows.length) {\n        // Add the groups into this page\n        const group = this.groupedRows[rowIndex];\n        temp[idx] = group;\n        idx++;\n\n        // Group index in this context\n        rowIndex++;\n      }\n    } else {\n      while (rowIndex < last && rowIndex < this.rowCount) {\n        const row = this.rows[rowIndex];\n\n        if (row) {\n          this.rowIndexes.set(row, rowIndex);\n          temp[idx] = row;\n        }\n\n        idx++;\n        rowIndex++;\n      }\n    }\n\n    this.temp = temp;\n  }\n\n  /**\n   * Get the row height\n   */\n  getRowHeight(row: any): number {\n    // if its a function return it\n    if (typeof this.rowHeight === 'function') {\n      return this.rowHeight(row);\n    }\n\n    return this.rowHeight as number;\n  }\n\n  /**\n   * @param group the group with all rows\n   */\n  getGroupHeight(group: any): number {\n    let rowHeight = 0;\n\n    if (group.value) {\n      for (let index = 0; index < group.value.length; index++) {\n        rowHeight += this.getRowAndDetailHeight(group.value[index]);\n      }\n    }\n\n    return rowHeight;\n  }\n\n  /**\n   * Calculate row height based on the expanded state of the row.\n   */\n  getRowAndDetailHeight(row: any): number {\n    let rowHeight = this.getRowHeight(row);\n    const expanded = this.getRowExpanded(row);\n\n    // Adding detail row height if its expanded.\n    if (expanded) {\n      rowHeight += this.getDetailRowHeight(row);\n    }\n\n    return rowHeight;\n  }\n\n  /**\n   * Get the height of the detail row.\n   */\n  getDetailRowHeight = (row?: any, index?: any): number => {\n    if (!this.rowDetail) {\n      return 0;\n    }\n    const rowHeight = this.rowDetail.rowHeight;\n    return typeof rowHeight === 'function' ? rowHeight(row, index) : (rowHeight as number);\n  };\n\n  /**\n   * Calculates the styles for the row so that the rows can be moved in 2D space\n   * during virtual scroll inside the DOM.   In the below case the Y position is\n   * manipulated.   As an example, if the height of row 0 is 30 px and row 1 is\n   * 100 px then following styles are generated:\n   *\n   * transform: translate3d(0px, 0px, 0px);    ->  row0\n   * transform: translate3d(0px, 30px, 0px);   ->  row1\n   * transform: translate3d(0px, 130px, 0px);  ->  row2\n   *\n   * Row heights have to be calculated based on the row heights cache as we wont\n   * be able to determine which row is of what height before hand.  In the above\n   * case the positionY of the translate3d for row2 would be the sum of all the\n   * heights of the rows before it (i.e. row0 and row1).\n   *\n   * @param rows the row that needs to be placed in the 2D space.\n   * @returns the CSS3 style to be applied\n   *\n   * @memberOf DataTableBodyComponent\n   */\n  getRowsStyles(rows: any): any {\n    const styles: any = {};\n\n    // only add styles for the group if there is a group\n    if (this.groupedRows) {\n      styles.width = this.columnGroupWidths.total;\n    }\n\n    if (this.scrollbarV && this.virtualization) {\n      let idx = 0;\n\n      if (this.groupedRows) {\n        // Get the latest row rowindex in a group\n        const row = rows[rows.length - 1];\n        idx = row ? this.getRowIndex(row) : 0;\n      } else {\n        idx = this.getRowIndex(rows);\n      }\n\n      // const pos = idx * rowHeight;\n      // The position of this row would be the sum of all row heights\n      // until the previous row position.\n      const pos = this.rowHeightsCache.query(idx - 1);\n\n      translateXY(styles, 0, pos);\n    }\n\n    return styles;\n  }\n\n  /**\n   * Calculate bottom summary row offset for scrollbar mode.\n   * For more information about cache and offset calculation\n   * see description for `getRowsStyles` method\n   *\n   * @returns the CSS3 style to be applied\n   *\n   * @memberOf DataTableBodyComponent\n   */\n  getBottomSummaryRowStyles(): any {\n    if (!this.scrollbarV || !this.rows || !this.rows.length) {\n      return null;\n    }\n\n    const styles = { position: 'absolute' };\n    const pos = this.rowHeightsCache.query(this.rows.length - 1);\n\n    translateXY(styles, 0, pos);\n\n    return styles;\n  }\n\n  /**\n   * Hides the loading indicator\n   */\n  hideIndicator(): void {\n    setTimeout(() => (this.loadingIndicator = false), 500);\n  }\n\n  /**\n   * Updates the index of the rows in the viewport\n   */\n  updateIndexes(): void {\n    let first = 0;\n    let last = 0;\n\n    if (this.scrollbarV) {\n      if (this.virtualization) {\n        // Calculation of the first and last indexes will be based on where the\n        // scrollY position would be at.  The last index would be the one\n        // that shows up inside the view port the last.\n        const height = parseInt(this.bodyHeight, 0);\n        first = this.rowHeightsCache.getRowIndex(this.offsetY);\n        last = this.rowHeightsCache.getRowIndex(height + this.offsetY) + 1;\n      } else {\n        // If virtual rows are not needed\n        // We render all in one go\n        first = 0;\n        last = this.rowCount;\n      }\n    } else {\n      // The server is handling paging and will pass an array that begins with the\n      // element at a specified offset.  first should always be 0 with external paging.\n      if (!this.externalPaging) {\n        first = Math.max(this.offset * this.pageSize, 0);\n      }\n      last = Math.min(first + this.pageSize, this.rowCount);\n    }\n\n    this.indexes = { first, last };\n  }\n\n  /**\n   * Refreshes the full Row Height cache.  Should be used\n   * when the entire row array state has changed.\n   */\n  refreshRowHeightCache(): void {\n    if (!this.scrollbarV || (this.scrollbarV && !this.virtualization)) {\n      return;\n    }\n\n    // clear the previous row height cache if already present.\n    // this is useful during sorts, filters where the state of the\n    // rows array is changed.\n    this.rowHeightsCache.clearCache();\n\n    // Initialize the tree only if there are rows inside the tree.\n    if (this.rows && this.rows.length) {\n      const rowExpansions = new Set();\n      for (const row of this.rows) {\n        if (this.getRowExpanded(row)) {\n          rowExpansions.add(row);\n        }\n      }\n\n      this.rowHeightsCache.initCache({\n        rows: this.rows,\n        rowHeight: this.rowHeight,\n        detailRowHeight: this.getDetailRowHeight,\n        externalVirtual: this.scrollbarV && this.externalPaging,\n        rowCount: this.rowCount,\n        rowIndexes: this.rowIndexes,\n        rowExpansions\n      });\n    }\n  }\n\n  /**\n   * Gets the index for the view port\n   */\n  getAdjustedViewPortIndex(): number {\n    // Capture the row index of the first row that is visible on the viewport.\n    // If the scroll bar is just below the row which is highlighted then make that as the\n    // first index.\n    const viewPortFirstRowIndex = this.indexes.first;\n\n    if (this.scrollbarV && this.virtualization) {\n      const offsetScroll = this.rowHeightsCache.query(viewPortFirstRowIndex - 1);\n      return offsetScroll <= this.offsetY ? viewPortFirstRowIndex - 1 : viewPortFirstRowIndex;\n    }\n\n    return viewPortFirstRowIndex;\n  }\n\n  /**\n   * Toggle the Expansion of the row i.e. if the row is expanded then it will\n   * collapse and vice versa.   Note that the expanded status is stored as\n   * a part of the row object itself as we have to preserve the expanded row\n   * status in case of sorting and filtering of the row set.\n   */\n  toggleRowExpansion(row: any): void {\n    // Capture the row index of the first row that is visible on the viewport.\n    const viewPortFirstRowIndex = this.getAdjustedViewPortIndex();\n    const rowExpandedIdx = this.getRowExpandedIdx(row, this.rowExpansions);\n    const expanded = rowExpandedIdx > -1;\n\n    // If the detailRowHeight is auto --> only in case of non-virtualized scroll\n    if (this.scrollbarV && this.virtualization) {\n      const detailRowHeight = this.getDetailRowHeight(row) * (expanded ? -1 : 1);\n      // const idx = this.rowIndexes.get(row) || 0;\n      const idx = this.getRowIndex(row);\n      this.rowHeightsCache.update(idx, detailRowHeight);\n    }\n\n    // Update the toggled row and update thive nevere heights in the cache.\n    if (expanded) {\n      this.rowExpansions.splice(rowExpandedIdx, 1);\n    } else {\n      this.rowExpansions.push(row);\n    }\n\n    this.detailToggle.emit({\n      rows: [row],\n      currentIndex: viewPortFirstRowIndex\n    });\n  }\n\n  /**\n   * Expand/Collapse all the rows no matter what their state is.\n   */\n  toggleAllRows(expanded: boolean): void {\n    // clear prev expansions\n    this.rowExpansions = [];\n\n    // Capture the row index of the first row that is visible on the viewport.\n    const viewPortFirstRowIndex = this.getAdjustedViewPortIndex();\n\n    if (expanded) {\n      for (const row of this.rows) {\n        this.rowExpansions.push(row);\n      }\n    }\n\n    if (this.scrollbarV) {\n      // Refresh the full row heights cache since every row was affected.\n      this.recalcLayout();\n    }\n\n    // Emit all rows that have been expanded.\n    this.detailToggle.emit({\n      rows: this.rows,\n      currentIndex: viewPortFirstRowIndex\n    });\n  }\n\n  /**\n   * Recalculates the table\n   */\n  recalcLayout(): void {\n    this.refreshRowHeightCache();\n    this.updateIndexes();\n    this.updateRows();\n  }\n\n  /**\n   * Tracks the column\n   */\n  columnTrackingFn(index: number, column: any): any {\n    return column.$$id;\n  }\n\n  /**\n   * Gets the row pinning group styles\n   */\n  stylesByGroup(group: string) {\n    const widths = this.columnGroupWidths;\n    const offsetX = this.offsetX;\n\n    const styles = {\n      width: `${widths[group]}px`\n    };\n\n    if (group === 'left') {\n      translateXY(styles, offsetX, 0);\n    } else if (group === 'right') {\n      const bodyWidth = parseInt(this.innerWidth + '', 0);\n      const totalDiff = widths.total - bodyWidth;\n      const offsetDiff = totalDiff - offsetX;\n      const offset = offsetDiff * -1;\n      translateXY(styles, offset, 0);\n    }\n\n    return styles;\n  }\n\n  /**\n   * Returns if the row was expanded and set default row expansion when row expansion is empty\n   */\n  getRowExpanded(row: any): boolean {\n    if (this.rowExpansions.length === 0 && this.groupExpansionDefault) {\n      for (const group of this.groupedRows) {\n        this.rowExpansions.push(group);\n      }\n    }\n\n    return this.getRowExpandedIdx(row, this.rowExpansions) > -1;\n  }\n\n  getRowExpandedIdx(row: any, expanded: any[]): number {\n    if (!expanded || !expanded.length) return -1;\n\n    const rowId = this.rowIdentity(row);\n    return expanded.findIndex((r) => {\n      const id = this.rowIdentity(r);\n      return id === rowId;\n    });\n  }\n\n  /**\n   * Gets the row index given a row\n   */\n  getRowIndex(row: any): number {\n    return this.rowIndexes.get(row) || 0;\n  }\n\n  onTreeAction(row: any) {\n    this.treeAction.emit({ row });\n  }\n}\n"]}