math.js 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. /**
  2. * @fileoverview added by tsickle
  3. * @suppress {checkTypes,constantProperty,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
  4. */
  5. import { columnsByPin, columnsTotalWidth } from './column';
  6. /**
  7. * Calculates the Total Flex Grow
  8. * @param {?} columns
  9. * @return {?}
  10. */
  11. export function getTotalFlexGrow(columns) {
  12. /** @type {?} */
  13. let totalFlexGrow = 0;
  14. for (const c of columns) {
  15. totalFlexGrow += c.flexGrow || 0;
  16. }
  17. return totalFlexGrow;
  18. }
  19. /**
  20. * Adjusts the column widths.
  21. * Inspired by: https://github.com/facebook/fixed-data-table/blob/master/src/FixedDataTableWidthHelper.js
  22. * @param {?} allColumns
  23. * @param {?} expectedWidth
  24. * @return {?}
  25. */
  26. export function adjustColumnWidths(allColumns, expectedWidth) {
  27. /** @type {?} */
  28. const columnsWidth = columnsTotalWidth(allColumns);
  29. /** @type {?} */
  30. const totalFlexGrow = getTotalFlexGrow(allColumns);
  31. /** @type {?} */
  32. const colsByGroup = columnsByPin(allColumns);
  33. if (columnsWidth !== expectedWidth) {
  34. scaleColumns(colsByGroup, expectedWidth, totalFlexGrow);
  35. }
  36. }
  37. /**
  38. * Resizes columns based on the flexGrow property, while respecting manually set widths
  39. * @param {?} colsByGroup
  40. * @param {?} maxWidth
  41. * @param {?} totalFlexGrow
  42. * @return {?}
  43. */
  44. function scaleColumns(colsByGroup, maxWidth, totalFlexGrow) {
  45. // calculate total width and flexgrow points for coulumns that can be resized
  46. for (const attr in colsByGroup) {
  47. for (const column of colsByGroup[attr]) {
  48. if (!column.canAutoResize) {
  49. maxWidth -= column.width;
  50. totalFlexGrow -= column.flexGrow ? column.flexGrow : 0;
  51. }
  52. else {
  53. column.width = 0;
  54. }
  55. }
  56. }
  57. /** @type {?} */
  58. const hasMinWidth = {};
  59. /** @type {?} */
  60. let remainingWidth = maxWidth;
  61. // resize columns until no width is left to be distributed
  62. do {
  63. /** @type {?} */
  64. const widthPerFlexPoint = remainingWidth / totalFlexGrow;
  65. remainingWidth = 0;
  66. for (const attr in colsByGroup) {
  67. for (const column of colsByGroup[attr]) {
  68. // if the column can be resize and it hasn't reached its minimum width yet
  69. if (column.canAutoResize && !hasMinWidth[column.prop]) {
  70. /** @type {?} */
  71. const newWidth = column.width + column.flexGrow * widthPerFlexPoint;
  72. if (column.minWidth !== undefined && newWidth < column.minWidth) {
  73. remainingWidth += newWidth - column.minWidth;
  74. column.width = column.minWidth;
  75. hasMinWidth[column.prop] = true;
  76. }
  77. else {
  78. column.width = newWidth;
  79. }
  80. }
  81. }
  82. }
  83. } while (remainingWidth !== 0);
  84. }
  85. /**
  86. * Forces the width of the columns to
  87. * distribute equally but overflowing when necessary
  88. *
  89. * Rules:
  90. *
  91. * - If combined withs are less than the total width of the grid,
  92. * proportion the widths given the min / max / normal widths to fill the width.
  93. *
  94. * - If the combined widths, exceed the total width of the grid,
  95. * use the standard widths.
  96. *
  97. * - If a column is resized, it should always use that width
  98. *
  99. * - The proportional widths should never fall below min size if specified.
  100. *
  101. * - If the grid starts off small but then becomes greater than the size ( + / - )
  102. * the width should use the original width; not the newly proportioned widths.
  103. * @param {?} allColumns
  104. * @param {?} expectedWidth
  105. * @param {?} startIdx
  106. * @param {?} allowBleed
  107. * @param {?=} defaultColWidth
  108. * @return {?}
  109. */
  110. export function forceFillColumnWidths(allColumns, expectedWidth, startIdx, allowBleed, defaultColWidth = 300) {
  111. /** @type {?} */
  112. const columnsToResize = allColumns.slice(startIdx + 1, allColumns.length).filter((/**
  113. * @param {?} c
  114. * @return {?}
  115. */
  116. c => {
  117. return c.canAutoResize !== false;
  118. }));
  119. for (const column of columnsToResize) {
  120. if (!column.$$oldWidth) {
  121. column.$$oldWidth = column.width;
  122. }
  123. }
  124. /** @type {?} */
  125. let additionWidthPerColumn = 0;
  126. /** @type {?} */
  127. let exceedsWindow = false;
  128. /** @type {?} */
  129. let contentWidth = getContentWidth(allColumns, defaultColWidth);
  130. /** @type {?} */
  131. let remainingWidth = expectedWidth - contentWidth;
  132. /** @type {?} */
  133. const columnsProcessed = [];
  134. /** @type {?} */
  135. const remainingWidthLimit = 1;
  136. // This loop takes care of the
  137. do {
  138. additionWidthPerColumn = remainingWidth / columnsToResize.length;
  139. exceedsWindow = contentWidth >= expectedWidth;
  140. for (const column of columnsToResize) {
  141. if (exceedsWindow && allowBleed) {
  142. column.width = column.$$oldWidth || column.width || defaultColWidth;
  143. }
  144. else {
  145. /** @type {?} */
  146. const newSize = (column.width || defaultColWidth) + additionWidthPerColumn;
  147. if (column.minWidth && newSize < column.minWidth) {
  148. column.width = column.minWidth;
  149. columnsProcessed.push(column);
  150. }
  151. else if (column.maxWidth && newSize > column.maxWidth) {
  152. column.width = column.maxWidth;
  153. columnsProcessed.push(column);
  154. }
  155. else {
  156. column.width = newSize;
  157. }
  158. }
  159. column.width = Math.max(0, column.width);
  160. }
  161. contentWidth = getContentWidth(allColumns);
  162. remainingWidth = expectedWidth - contentWidth;
  163. removeProcessedColumns(columnsToResize, columnsProcessed);
  164. } while (remainingWidth > remainingWidthLimit && columnsToResize.length !== 0);
  165. }
  166. /**
  167. * Remove the processed columns from the current active columns.
  168. * @param {?} columnsToResize
  169. * @param {?} columnsProcessed
  170. * @return {?}
  171. */
  172. function removeProcessedColumns(columnsToResize, columnsProcessed) {
  173. for (const column of columnsProcessed) {
  174. /** @type {?} */
  175. const index = columnsToResize.indexOf(column);
  176. columnsToResize.splice(index, 1);
  177. }
  178. }
  179. /**
  180. * Gets the width of the columns
  181. * @param {?} allColumns
  182. * @param {?=} defaultColWidth
  183. * @return {?}
  184. */
  185. function getContentWidth(allColumns, defaultColWidth = 300) {
  186. /** @type {?} */
  187. let contentWidth = 0;
  188. for (const column of allColumns) {
  189. contentWidth += column.width || defaultColWidth;
  190. }
  191. return contentWidth;
  192. }
  193. //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"math.js","sourceRoot":"ng://@swimlane/ngx-datatable/","sources":["lib/utils/math.ts"],"names":[],"mappings":"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","sourcesContent":["import { columnsByPin, columnsTotalWidth } from './column';\n\n/**\n * Calculates the Total Flex Grow\n */\nexport function getTotalFlexGrow(columns: any[]) {\n  let totalFlexGrow = 0;\n\n  for (const c of columns) {\n    totalFlexGrow += c.flexGrow || 0;\n  }\n\n  return totalFlexGrow;\n}\n\n/**\n * Adjusts the column widths.\n * Inspired by: https://github.com/facebook/fixed-data-table/blob/master/src/FixedDataTableWidthHelper.js\n */\nexport function adjustColumnWidths(allColumns: any, expectedWidth: any) {\n  const columnsWidth = columnsTotalWidth(allColumns);\n  const totalFlexGrow = getTotalFlexGrow(allColumns);\n  const colsByGroup = columnsByPin(allColumns);\n\n  if (columnsWidth !== expectedWidth) {\n    scaleColumns(colsByGroup, expectedWidth, totalFlexGrow);\n  }\n}\n\n/**\n * Resizes columns based on the flexGrow property, while respecting manually set widths\n */\nfunction scaleColumns(colsByGroup: any, maxWidth: any, totalFlexGrow: any) {\n  // calculate total width and flexgrow points for coulumns that can be resized\n  for (const attr in colsByGroup) {\n    for (const column of colsByGroup[attr]) {\n      if (!column.canAutoResize) {\n        maxWidth -= column.width;\n        totalFlexGrow -= column.flexGrow ? column.flexGrow : 0;\n      } else {\n        column.width = 0;\n      }\n    }\n  }\n\n  const hasMinWidth = {};\n  let remainingWidth = maxWidth;\n\n  // resize columns until no width is left to be distributed\n  do {\n    const widthPerFlexPoint = remainingWidth / totalFlexGrow;\n    remainingWidth = 0;\n\n    for (const attr in colsByGroup) {\n      for (const column of colsByGroup[attr]) {\n        // if the column can be resize and it hasn't reached its minimum width yet\n        if (column.canAutoResize && !hasMinWidth[column.prop]) {\n          const newWidth = column.width + column.flexGrow * widthPerFlexPoint;\n          if (column.minWidth !== undefined && newWidth < column.minWidth) {\n            remainingWidth += newWidth - column.minWidth;\n            column.width = column.minWidth;\n            hasMinWidth[column.prop] = true;\n          } else {\n            column.width = newWidth;\n          }\n        }\n      }\n    }\n  } while (remainingWidth !== 0);\n}\n\n/**\n * Forces the width of the columns to\n * distribute equally but overflowing when necessary\n *\n * Rules:\n *\n *  - If combined withs are less than the total width of the grid,\n *    proportion the widths given the min / max / normal widths to fill the width.\n *\n *  - If the combined widths, exceed the total width of the grid,\n *    use the standard widths.\n *\n *  - If a column is resized, it should always use that width\n *\n *  - The proportional widths should never fall below min size if specified.\n *\n *  - If the grid starts off small but then becomes greater than the size ( + / - )\n *    the width should use the original width; not the newly proportioned widths.\n */\nexport function forceFillColumnWidths(\n  allColumns: any[],\n  expectedWidth: number,\n  startIdx: number,\n  allowBleed: boolean,\n  defaultColWidth: number = 300\n) {\n  const columnsToResize = allColumns.slice(startIdx + 1, allColumns.length).filter(c => {\n    return c.canAutoResize !== false;\n  });\n\n  for (const column of columnsToResize) {\n    if (!column.$$oldWidth) {\n      column.$$oldWidth = column.width;\n    }\n  }\n\n  let additionWidthPerColumn = 0;\n  let exceedsWindow = false;\n  let contentWidth = getContentWidth(allColumns, defaultColWidth);\n  let remainingWidth = expectedWidth - contentWidth;\n  const columnsProcessed: any[] = [];\n  const remainingWidthLimit = 1; // when to stop\n\n  // This loop takes care of the\n  do {\n    additionWidthPerColumn = remainingWidth / columnsToResize.length;\n    exceedsWindow = contentWidth >= expectedWidth;\n\n    for (const column of columnsToResize) {\n      if (exceedsWindow && allowBleed) {\n        column.width = column.$$oldWidth || column.width || defaultColWidth;\n      } else {\n        const newSize = (column.width || defaultColWidth) + additionWidthPerColumn;\n\n        if (column.minWidth && newSize < column.minWidth) {\n          column.width = column.minWidth;\n          columnsProcessed.push(column);\n        } else if (column.maxWidth && newSize > column.maxWidth) {\n          column.width = column.maxWidth;\n          columnsProcessed.push(column);\n        } else {\n          column.width = newSize;\n        }\n      }\n\n      column.width = Math.max(0, column.width);\n    }\n\n    contentWidth = getContentWidth(allColumns);\n    remainingWidth = expectedWidth - contentWidth;\n    removeProcessedColumns(columnsToResize, columnsProcessed);\n  } while (remainingWidth > remainingWidthLimit && columnsToResize.length !== 0);\n}\n\n/**\n * Remove the processed columns from the current active columns.\n */\nfunction removeProcessedColumns(columnsToResize: any[], columnsProcessed: any[]) {\n  for (const column of columnsProcessed) {\n    const index = columnsToResize.indexOf(column);\n    columnsToResize.splice(index, 1);\n  }\n}\n\n/**\n * Gets the width of the columns\n */\nfunction getContentWidth(allColumns: any, defaultColWidth: number = 300): number {\n  let contentWidth = 0;\n\n  for (const column of allColumns) {\n    contentWidth += column.width || defaultColWidth;\n  }\n\n  return contentWidth;\n}\n"]}