Source: src/grids/PackingGrid.ts

  1. /**
  2. * egjs-grid
  3. * Copyright (c) 2021-present NAVER Corp.
  4. * MIT license
  5. */
  6. import Grid from "../Grid";
  7. import { PROPERTY_TYPE } from "../consts";
  8. import { GridOptions, Properties, GridOutlines } from "../types";
  9. import { GetterSetter } from "../utils";
  10. import { GridItem } from "../GridItem";
  11. import BoxModel from "./lib/BoxModel";
  12. function getCost(originLength: number, length: number) {
  13. let cost = originLength / length;
  14. if (cost < 1) {
  15. cost = 1 / cost;
  16. }
  17. return cost - 1;
  18. }
  19. function fitArea(
  20. item: BoxModel,
  21. bestFitArea: BoxModel,
  22. itemFitSize: { inlineSize: number, contentSize: number },
  23. containerFitSize: { inlineSize: number, contentSize: number },
  24. isContentDirection: boolean,
  25. ) {
  26. item.contentSize = itemFitSize.contentSize;
  27. item.inlineSize = itemFitSize.inlineSize;
  28. bestFitArea.contentSize = containerFitSize.contentSize;
  29. bestFitArea.inlineSize = containerFitSize.inlineSize;
  30. if (isContentDirection) {
  31. item.contentPos = bestFitArea.contentPos + bestFitArea.contentSize;
  32. item.inlinePos = bestFitArea.inlinePos;
  33. } else {
  34. item.inlinePos = bestFitArea.inlinePos + bestFitArea.inlineSize;
  35. item.contentPos = bestFitArea.contentPos;
  36. }
  37. }
  38. /**
  39. * @typedef
  40. * @memberof Grid.PackingGrid
  41. * @extends Grid.GridOptions
  42. */
  43. export interface PackingGridOptions extends GridOptions {
  44. /**
  45. * The aspect ratio (inlineSize / contentSize) of the container with items.
  46. * <ko>아이템들을 가진 컨테이너의 종횡비(inlineSize / contentSize).</ko>
  47. * @default 1
  48. */
  49. aspectRatio?: number;
  50. /**
  51. * The size weight when placing items.
  52. * <ko>아이템들을 배치하는데 사이즈 가중치.</ko>
  53. * @default 1
  54. */
  55. sizeWeight?: number;
  56. /**
  57. * The weight to keep ratio when placing items.
  58. * <ko>아이템들을 배치하는데 비율을 유지하는 가중치.</ko>
  59. * @default 1
  60. */
  61. ratioWeight?: number;
  62. /**
  63. * The priority that determines the weight of the item. "size" = (sizeWieght: 100, ratioWeight: 1), "ratio" = (sizeWeight: 1, ratioWeight; 100), "custom" = (set sizeWeight, ratioWeight)
  64. * item's weight = item's ratio(inlineSize / contentSize) change * `ratioWeight` + size(inlineSize * contentSize) change * `sizeWeight`.
  65. * <ko> 아이템의 가중치를 결정하는 우선수치. "size" = (sizeWieght: 100, ratioWeight: 1), "ratio" = (sizeWeight: 1, ratioWeight; 100), "custom" = (set sizeWeight, ratioWeight). 아이템의 가중치 = ratio(inlineSize / contentSize)의 변화량 * `ratioWeight` + size(inlineSize * contentSize)의 변화량 * `sizeWeight`.</ko>
  66. * @default "custom"
  67. */
  68. weightPriority?: "size" | "ratio" | "custom";
  69. }
  70. /**
  71. * The PackingGrid is a grid that shows the important items bigger without sacrificing the weight of the items.
  72. * Rows and columns are separated so that items are dynamically placed within the horizontal and vertical space rather than arranged in an orderly fashion.
  73. * If `sizeWeight` is higher than `ratioWeight`, the size of items is preserved as much as possible.
  74. * Conversely, if `ratioWeight` is higher than `sizeWeight`, the ratio of items is preserved as much as possible.
  75. * @ko PackingGrid는 아이템의 본래 크기에 따른 비중을 해치지 않으면서 중요한 카드는 더 크게 보여 주는 레이아웃이다.
  76. * 행과 열이 구분돼 아이템을 정돈되게 배치하는 대신 가로세로 일정 공간 내에서 동적으로 아이템을 배치한다.
  77. * `sizeWeight`가 `ratioWeight`보다 높으면 아이템들의 size가 최대한 보존이 된다.
  78. * 반대로 `ratioWeight`가 `sizeWeight`보다 높으면 아이템들의 비율이 최대한 보존이 된다.
  79. * @memberof Grid
  80. * @param {HTMLElement | string} container - A base element for a module <ko>모듈을 적용할 기준 엘리먼트</ko>
  81. * @param {Grid.PackingGrid.PackingGridOptions} options - The option object of the PackingGrid module <ko>PackingGrid 모듈의 옵션 객체</ko>
  82. */
  83. @GetterSetter
  84. export class PackingGrid extends Grid<PackingGridOptions> {
  85. public static propertyTypes = {
  86. ...Grid.propertyTypes,
  87. aspectRatio: PROPERTY_TYPE.RENDER_PROPERTY,
  88. sizeWeight: PROPERTY_TYPE.RENDER_PROPERTY,
  89. ratioWeight: PROPERTY_TYPE.RENDER_PROPERTY,
  90. weightPriority: PROPERTY_TYPE.RENDER_PROPERTY,
  91. };
  92. public static defaultOptions: Required<PackingGridOptions> = {
  93. ...Grid.defaultOptions,
  94. aspectRatio: 1,
  95. sizeWeight: 1,
  96. ratioWeight: 1,
  97. weightPriority: "custom",
  98. };
  99. public applyGrid(items: GridItem[], direction: "start" | "end", outline: number[]): GridOutlines {
  100. const { aspectRatio } = this.options;
  101. const containerInlineSize = this.getContainerInlineSize();
  102. const containerContentSize = containerInlineSize / aspectRatio;
  103. const inlineGap = this.getInlineGap();
  104. const contentGap = this.getContentGap();
  105. const prevOutline = outline.length ? outline : [0];
  106. const startPoint = direction === "end"
  107. ? Math.max(...prevOutline)
  108. : Math.min(...prevOutline) - containerContentSize - contentGap;
  109. const endPoint = startPoint + containerContentSize + contentGap;
  110. const container = new BoxModel({});
  111. items.forEach((item) => {
  112. const model = new BoxModel({
  113. inlineSize: item.orgInlineSize,
  114. contentSize: item.orgContentSize,
  115. orgInlineSize: item.orgInlineSize,
  116. orgContentSize: item.orgContentSize,
  117. });
  118. this._findBestFitArea(container, model);
  119. container.push(model);
  120. container.scaleTo(containerInlineSize + inlineGap, containerContentSize + contentGap);
  121. });
  122. items.forEach((item, i) => {
  123. const boxItem = container.items[i];
  124. const inlineSize = boxItem.inlineSize - inlineGap;
  125. const contentSize = boxItem.contentSize - contentGap;
  126. const contentPos = startPoint + boxItem.contentPos;
  127. const inlinePos = boxItem.inlinePos;
  128. item.setCSSGridRect({
  129. inlinePos,
  130. contentPos,
  131. inlineSize,
  132. contentSize,
  133. });
  134. });
  135. return {
  136. start: [startPoint],
  137. end: [endPoint],
  138. };
  139. }
  140. private _findBestFitArea(container: BoxModel, item: BoxModel) {
  141. if (container.getRatio() === 0) { // 아이템 최초 삽입시 전체영역 지정
  142. container.orgInlineSize = item.inlineSize;
  143. container.orgContentSize = item.contentSize;
  144. container.inlineSize = item.inlineSize;
  145. container.contentSize = item.contentSize;
  146. return;
  147. }
  148. let bestFitArea!: BoxModel;
  149. let minCost = Infinity;
  150. let isContentDirection = false;
  151. const itemFitSize = {
  152. inlineSize: 0,
  153. contentSize: 0,
  154. };
  155. const containerFitSize = {
  156. inlineSize: 0,
  157. contentSize: 0,
  158. };
  159. const sizeWeight = this._getWeight("size");
  160. const ratioWeight = this._getWeight("ratio");
  161. container.items.forEach((child) => {
  162. const containerSizeCost = getCost(child.getOrgSizeWeight(), child.getSize()) * sizeWeight;
  163. const containerRatioCost = getCost(child.getOrgRatio(), child.getRatio()) * ratioWeight;
  164. const inlineSize = child.inlineSize;
  165. const contentSize = child.contentSize;
  166. for (let i = 0; i < 2; ++i) {
  167. let itemInlineSize;
  168. let itemContentSize;
  169. let containerInlineSize;
  170. let containerContentSize;
  171. if (i === 0) {
  172. // add item to content pos (top, bottom)
  173. itemInlineSize = inlineSize;
  174. itemContentSize = contentSize * (item.contentSize / (child.orgContentSize + item.contentSize));
  175. containerInlineSize = inlineSize;
  176. containerContentSize = contentSize - itemContentSize;
  177. } else {
  178. // add item to inline pos (left, right)
  179. itemContentSize = contentSize;
  180. itemInlineSize = inlineSize * (item.inlineSize / (child.orgInlineSize + item.inlineSize));
  181. containerContentSize = contentSize;
  182. containerInlineSize = inlineSize - itemInlineSize;
  183. }
  184. const itemSize = itemInlineSize * itemContentSize;
  185. const itemRatio = itemInlineSize / itemContentSize;
  186. const containerSize = containerInlineSize * containerContentSize;
  187. const containerRatio = containerContentSize / containerContentSize;
  188. let cost = getCost(item.getSize(), itemSize) * sizeWeight;
  189. cost += getCost(item.getRatio(), itemRatio) * ratioWeight;
  190. cost += getCost(child.getOrgSizeWeight(), containerSize) * sizeWeight - containerSizeCost;
  191. cost += getCost(child.getOrgRatio(), containerRatio) * ratioWeight - containerRatioCost;
  192. if (cost === Math.min(cost, minCost)) {
  193. minCost = cost;
  194. bestFitArea = child;
  195. isContentDirection = (i === 0);
  196. itemFitSize.inlineSize = itemInlineSize;
  197. itemFitSize.contentSize = itemContentSize;
  198. containerFitSize.inlineSize = containerInlineSize;
  199. containerFitSize.contentSize = containerContentSize;
  200. }
  201. }
  202. });
  203. fitArea(item, bestFitArea, itemFitSize, containerFitSize, isContentDirection);
  204. }
  205. public getComputedOutlineLength() {
  206. return 1;
  207. }
  208. public getComputedOutlineSize() {
  209. return this.getContainerInlineSize();
  210. }
  211. private _getWeight(type: "size" | "ratio"): number {
  212. const options = this.options;
  213. const weightPriority = options.weightPriority;
  214. if (weightPriority === type) {
  215. return 100;
  216. } else if (weightPriority === "custom") {
  217. return options[`${type}Weight`];
  218. }
  219. return 1;
  220. }
  221. }
  222. export interface PackingGrid extends Properties<typeof PackingGrid> {
  223. }
  224. /**
  225. * The aspect ratio (inlineSize / contentSize) of the container with items.
  226. * @ko 아이템들을 가진 컨테이너의 종횡비(inlineSize / contentSize).
  227. * @name Grid.PackingGrid#aspectRatio
  228. * @type {$ts:Grid.PackingGrid.PackingGridOptions["aspectRatio"]}
  229. * @default 1
  230. * @example
  231. * ```js
  232. * import { PackingGrid } from "@egjs/grid";
  233. *
  234. * const grid = new PackingGrid(container, {
  235. * aspectRatio: 1,
  236. * });
  237. *
  238. * grid.aspectRatio = 1.5;
  239. * ```
  240. */
  241. /**
  242. * The priority that determines the weight of the item. "size" = (sizeWieght: 2, ratioWeight: 1), "ratio" = (sizeWeight: 1, ratioWeight; 2), "custom" = (set sizeWeight, ratioWeight)
  243. * item's weight = item's ratio(inlineSize / contentSize) change * `ratioWeight` + size(inlineSize * contentSize) change * `sizeWeight`.
  244. * @ko 아이템의 가중치를 결정하는 우선수치. "size" = (sizeWieght: 2, ratioWeight: 1), "ratio" = (sizeWeight: 1, ratioWeight; 2), "custom" = (set sizeWeight, ratioWeight). 아이템의 가중치 = ratio(inlineSize / contentSize)의 변화량 * `ratioWeight` + size(inlineSize * contentSize)의 변화량 * `sizeWeight`.
  245. * @name Grid.PackingGrid#weightPriority
  246. * @type {$ts:Grid.PackingGrid.PackingGridOptions["weightPriority"]}
  247. * @default "custom"
  248. * @example
  249. * ```js
  250. * import { PackingGrid } from "@egjs/grid";
  251. *
  252. * const grid = new PackingGrid(container, {
  253. * weightPriority: "custom",
  254. * sizeWeight: 1,
  255. * ratioWeight: 1,
  256. * });
  257. *
  258. * grid.weightPriority = "size";
  259. * // or
  260. * grid.weightPriority = "ratio";
  261. * ```
  262. */
  263. /**
  264. * The size weight when placing items.
  265. * @ko 아이템들을 배치하는데 사이즈 가중치.
  266. * @name Grid.PackingGrid#sizeWeight
  267. * @type {$ts:Grid.PackingGrid.PackingGridOptions["sizeWeight"]}
  268. * @default 1
  269. * @example
  270. * ```js
  271. * import { PackingGrid } from "@egjs/grid";
  272. *
  273. * const grid = new PackingGrid(container, {
  274. * sizeWeight: 1,
  275. * });
  276. *
  277. * grid.sizeWeight = 10;
  278. * ```
  279. */
  280. /**
  281. * The weight to keep ratio when placing items.
  282. * @ko 아이템들을 배치하는데 비율을 유지하는 가중치.
  283. * @name Grid.PackingGrid#ratioWeight
  284. * @type {$ts:Grid.PackingGrid.PackingGridOptions["ratioWeight"]}
  285. * @default 1
  286. * @example
  287. * ```js
  288. * import { PackingGrid } from "@egjs/grid";
  289. *
  290. * const grid = new PackingGrid(container, {
  291. * ratioWeight: 1,
  292. * });
  293. *
  294. * grid.ratioWeight = 10;
  295. * ```
  296. */
comments powered by Disqus