legend.js

  1. 'use strict';
  2. const TOO_MANY_LAYERS = 15;
  3. // This file relates to legends on an exported map, not legends in the layer selector
  4. /**
  5. * Generate all permutations of length M, with exactly N `true` values.
  6. *
  7. * @function
  8. * @param {int} M the size of the array (must be greater than 0)
  9. * @param {int} N the number of entries which should be true (must not be greater than M)
  10. * @return an array containing all possible size M arrays of boolean values with N true entries
  11. */
  12. function allComb(M, N) {
  13. const maxTrue = N;
  14. const maxFalse = M - N;
  15. const C = [[[[]]]]; // C[m][n] is the solution to all_comb(m,n), C[0][0] starts with an empty array
  16. for (let m = 1; m <= M; ++m) {
  17. C[m] = [];
  18. for (let n = 0; n <= m; ++n) {
  19. // if this would place more than the max number of true or false values we don't need this part of the array
  20. if (n > maxTrue || (m - n) > maxFalse) { continue; }
  21. const a = n > 0 ? C[m - 1][n - 1].map(x => x.concat(true)) : [];
  22. const b = m > n ? C[m - 1][n].map(x => x.concat(false)) : [];
  23. C[m][n] = a.concat(b);
  24. }
  25. }
  26. return C[M][N];
  27. }
  28. /**
  29. * Convenience function for assigning the `splitBefore` property on layers at specified points.
  30. * NOTE: this function modifies data in place
  31. * @function
  32. * @private
  33. * @param {Array} layers a list of layers to be updated (modified in place)
  34. * @param {Array} splitPoints an array of boolean values indicating if the layer list should be split at that point (must be layers.length-1 in size)
  35. * @return layers the same array as passed in
  36. */
  37. function assignLayerSplits(layers, splitPoints) {
  38. layers[0].splitBefore = false;
  39. splitPoints.forEach((split, i) => layers[i + 1].splitBefore = split);
  40. return layers;
  41. }
  42. /**
  43. * Groups multiple layers into each section while attempting to minimize the legend height.
  44. * Allocates to the exact number specified in the `sections` argument.
  45. * NOTE: don't call this with too many layers as it tests all possible groupings and can be
  46. * computationally expensive (< 15 layers should be fine)
  47. * @function
  48. * @private
  49. * @param {Array} layers a list of layers to be fitted
  50. * @param {int} sections the number of sections to use
  51. * @return {Object} an object in the form { layers, sectionsUsed, bestPerm, bestHeight }
  52. */
  53. function packLayersIntoExactSections(layers, sections) {
  54. const potentialSplits = layers.length - 1;
  55. const requiredSplits = sections - 1;
  56. const permutations = allComb(potentialSplits, requiredSplits);
  57. let bestHeight = Number.MAX_VALUE;
  58. let bestPerm = null;
  59. const heights = Array(sections);
  60. permutations.forEach(perm => {
  61. heights.fill(0);
  62. let curSec = 0;
  63. layers.forEach((l, i) => {
  64. heights[curSec] += l.height;
  65. if (perm[i]) {
  66. ++curSec;
  67. }
  68. });
  69. const h = Math.max(...heights);
  70. if (h <= bestHeight) {
  71. bestHeight = h;
  72. bestPerm = perm;
  73. }
  74. });
  75. return { layers, sectionsUsed: sections, bestPerm, bestHeight };
  76. }
  77. /**
  78. * Groups multiple layers into each section while attempting to minimize the legend height.
  79. * Repeats as necessary to use the least number of sections while still keeping the resulting
  80. * legend height within 20% of optimal.
  81. * NOTE: don't call this with too many layers as it tests all possible groupings and can be
  82. * computationally expensive (< 15 layers should be fine)
  83. * @function
  84. * @private
  85. * @param {Array} layers a list of layers to be updated (modified in place)
  86. * @param {int} sections the number of sections to use
  87. * @return {Object} an object in the form { layers, sectionsUsed }
  88. */
  89. function packLayersIntoOptimalSections(layers, sections) {
  90. let bestHeight = Number.MAX_VALUE;
  91. let bestPerm = null;
  92. let sectionsUsed = -1;
  93. for (let n = sections; n > 1; --n) {
  94. const { bestPerm: perm, bestHeight: height } = packLayersIntoExactSections(layers, n);
  95. if (height * 0.8 > bestHeight) {
  96. break;
  97. } else if (height <= bestHeight) {
  98. [bestHeight, bestPerm, sectionsUsed] = [height, perm, n];
  99. }
  100. }
  101. assignLayerSplits(layers, bestPerm);
  102. return { layers, sectionsUsed };
  103. }
  104. /**
  105. * Split a layer into `splitCount` parts of roughly equal size.
  106. * @function
  107. * @private
  108. * @param {Object} layer a layer object to be split into `splitCount` parts
  109. * @param {int} chunkSize the maximum height in pixels of the legend sections
  110. * @param {int} splitCount the number of pieces which the layer should be broken into
  111. * @return an object with properties whiteSpace: <int>, splits: [ <layerItems> ]
  112. */
  113. function splitLayer(layer, chunkSize, splitCount) {
  114. let itemYOffset = layer.y;
  115. let itemYMax = 0;
  116. const splits = [];
  117. const splitSizes = Array(splitCount).fill(0);
  118. function traverse(items) {
  119. items.forEach(item => {
  120. if (splitCount === 1) {
  121. return;
  122. }
  123. splitSizes[splitCount - 1] = itemYMax - itemYOffset; // bottom of current item - offset at current section start
  124. // this is the y coordinate of the item's bottom boundary
  125. itemYMax = item.y + (item.type === 'group' ? item.headerHeight : item.height);
  126. if (itemYMax - itemYOffset >= chunkSize) {
  127. splitCount--;
  128. // whitespace is created when an item sitting on the boundary pulled into the next chunk, the space
  129. // it would have occupied is wasted; the waste doubles as the entire item is moved to the next legend chunk
  130. itemYOffset = item.y;
  131. splits.push(item);
  132. }
  133. if (item.type === 'group') {
  134. traverse(item.items);
  135. }
  136. });
  137. }
  138. traverse(layer.items);
  139. splitSizes[splitCount - 1] = layer.height - (itemYOffset - layer.y); // bottom of layer - start of last section; start of last section = total offset of last section - offset at start of layer
  140. // with whiteSpace we want to find the difference between the chunkSize and used space
  141. // for each section used (whiteSpace may be negative indicating that a section is
  142. // spilling past the target size); the total amount of whiteSpace is a measure of how
  143. // bad the layer allocation was
  144. return { whiteSpace: splitSizes.reduce((a, b) => a + Math.abs(chunkSize - b), 0), splits };
  145. }
  146. /**
  147. * Find the optimal split points for the given layer.
  148. * @function
  149. * @private
  150. * @param {Object} layer a layer object to be split into `splitCount` parts
  151. * @param {int} splitCount the number of pieces which the layer should be broken into
  152. * @return a reference to the layer passed in
  153. */
  154. function findOptimalSplit(layer, splitCount) {
  155. if (splitCount === 1) {
  156. return layer;
  157. }
  158. let chunkSize = layer.height / splitCount; // get initial chunk size for starters
  159. // get initial splits and whitespace with initial chunk size; this will serve to determine the steps at which the chunk size will be increased
  160. let { splits: minSplits, whiteSpace: minWhiteSpace } = splitLayer(layer, chunkSize, splitCount);
  161. const stepCount = 8; // number of attempts
  162. const step = minWhiteSpace / stepCount;
  163. // calculate splits while increasing the chunk size
  164. for (let i = 1; i <= stepCount; i++) {
  165. chunkSize += step;
  166. let { splits, whiteSpace } = splitLayer(layer, chunkSize, splitCount);
  167. // store splits corresponding to the minimum whitespace
  168. if (whiteSpace < minWhiteSpace) {
  169. minWhiteSpace = whiteSpace;
  170. minSplits = splits;
  171. }
  172. }
  173. // apply split to the splits that result in the minimum of whitespace
  174. minSplits.forEach(split => split.splitBefore = true);
  175. return layer;
  176. }
  177. /**
  178. * @function
  179. * @private
  180. * @param {Array} layers a list of layers to be updated (modified in place)
  181. * @param {int} sectionsAvailable the maximum number of sections to use
  182. * @param {int} mapHeight the rendered height of the map image
  183. * @return the same layers array as passed in
  184. */
  185. function allocateLayersToSections(layers, sectionsAvailable, mapHeight) {
  186. assignLayerSplits(layers, Array(layers.length - 1).fill(true));
  187. const bestSectionUsage = {}; // maps number of sections used to best height achieved
  188. bestSectionUsage[layers.length] = {
  189. height: Math.max(...layers.map(l => l.height)),
  190. segments: Array(layers.length)
  191. };
  192. bestSectionUsage[layers.length].segments.fill(1);
  193. let curSectionsUsed = layers.length;
  194. while (curSectionsUsed < sectionsAvailable && bestSectionUsage[curSectionsUsed].height > mapHeight * 2) {
  195. const oldSegments = bestSectionUsage[curSectionsUsed].segments;
  196. const normalizedLayers = oldSegments.map((seg, i) => layers[i].height / seg);
  197. const worstLayerIndex = normalizedLayers.indexOf(Math.max(...normalizedLayers));
  198. const newSegments = oldSegments.map((seg, i) => i === worstLayerIndex ? seg + 1 : seg);
  199. ++curSectionsUsed;
  200. bestSectionUsage[curSectionsUsed] = {
  201. height: Math.max(...newSegments.map((seg, i) => layers[i].height / seg)),
  202. segments: newSegments
  203. };
  204. }
  205. while (curSectionsUsed > layers.length) {
  206. if (bestSectionUsage[curSectionsUsed].height < 0.9 * bestSectionUsage[curSectionsUsed - 1].height) {
  207. break;
  208. }
  209. --curSectionsUsed;
  210. }
  211. layers.forEach((l, i) => findOptimalSplit(l, bestSectionUsage[curSectionsUsed].segments[i]));
  212. return { layers, sectionsUsed: curSectionsUsed };
  213. }
  214. /**
  215. * Generate the structure for a legend given a set of layers.
  216. * @function
  217. * @param {Array} layerList a list of layers to be updated (modified in place)
  218. * @param {int} sectionsAvailable the maximum number of sections to use
  219. * @param {int} mapHeight the rendered height of the map image
  220. * @return an object with properties layers, sectionsUsed. (layerList is modified in place)
  221. */
  222. function makeLegend(layerList, sectionsAvailable, mapHeight) {
  223. if (layerList.length > TOO_MANY_LAYERS) {
  224. const layersPerSection = Math.ceil(layerList.length / sectionsAvailable);
  225. const splitPoints = Array(layerList.length - 1).fill(0).map((v, i) => (i + 1) % layersPerSection === 0); // I don't know why the useless fill is necessary
  226. assignLayerSplits(layerList, splitPoints);
  227. return { layers: layerList, sectionsUsed: sectionsAvailable };
  228. }
  229. if (layerList.length <= sectionsAvailable) {
  230. return allocateLayersToSections(layerList, sectionsAvailable, mapHeight);
  231. } else {
  232. return packLayersIntoOptimalSections(layerList, sectionsAvailable);
  233. }
  234. }
  235. module.exports = () => ({ makeLegend, allComb, splitLayer, findOptimalSplit });