Source: src/grids/JustifiedGrid.ts

  1. /**
  2. * egjs-grid
  3. * Copyright (c) 2021-present NAVER Corp.
  4. * MIT license
  5. */
  6. import Grid from "../Grid";
  7. import { MOUNT_STATE, PROPERTY_TYPE } from "../consts";
  8. import { GridOptions, GridOutlines, Properties } from "../types";
  9. import { between, getRangeCost, GetterSetter, isNumber, isObject, sum, throttle } from "../utils";
  10. import { find_path } from "./lib/dijkstra";
  11. import { GridItem } from "../GridItem";
  12. interface Link {
  13. path: number[];
  14. cost: number;
  15. length: number;
  16. currentNode: number;
  17. isOver?: boolean;
  18. }
  19. function splitItems(items: GridItem[], path: string[]) {
  20. const length = path.length;
  21. const groups: GridItem[][] = [];
  22. for (let i = 0; i < length - 1; ++i) {
  23. const path1 = parseInt(path[i], 10);
  24. const path2 = parseInt(path[i + 1], 10);
  25. groups.push(items.slice(path1, path2));
  26. }
  27. return groups;
  28. }
  29. function parseStretchSize(inlineSize: number, size: number | string) {
  30. if (isNumber(size)) {
  31. return size;
  32. }
  33. const signText = size.charAt(0);
  34. const sign = signText === "+" ? 1 : (signText === "-" ? -1 : 0);
  35. let nextSize = parseFloat(size);
  36. if (size.match(/%$/g)) {
  37. nextSize *= inlineSize / 100;
  38. }
  39. if (sign) {
  40. return inlineSize + nextSize;
  41. }
  42. return nextSize;
  43. }
  44. function getExpectedItemInlineSize(item: GridItem, rowSize: number) {
  45. const inlineSize = item.orgInlineSize;
  46. const contentSize = item.orgContentSize;
  47. const inlineOffset = item.gridData.inlineOffset || 0;
  48. const contentOffset = item.gridData.contentOffset || 0;
  49. if (!inlineSize || !contentSize) {
  50. return rowSize;
  51. }
  52. const ratio = contentSize <= contentOffset ? 1 : (inlineSize - inlineOffset) / (contentSize - contentOffset);
  53. return ratio * (rowSize - contentOffset) + inlineOffset;
  54. }
  55. /**
  56. * @typedef
  57. * @memberof Grid.JustifiedGrid
  58. * @extends Grid.GridOptions
  59. */
  60. export interface JustifiedGridOptions extends GridOptions {
  61. /**
  62. * The minimum and maximum number of items per line.
  63. * <ko> 한 줄에 들어가는 아이템의 최소, 최대 개수.</ko>
  64. * @default [1, 8]
  65. */
  66. columnRange?: number | number[];
  67. /**
  68. * The minimum and maximum number of rows in a group, 0 is not set.
  69. * <ko> 한 그룹에 들어가는 행의 최소, 최대 개수, 0은 미설정이다.</ko>
  70. * @default 0
  71. */
  72. rowRange?: number | number[];
  73. /**
  74. * The minimum and maximum size by which the item is adjusted. If it is not calculated, it may deviate from the minimum and maximum sizes.
  75. * <ko>아이템이 조정되는 최소, 최대 사이즈. 계산이 되지 않는 경우 최소, 최대 사이즈를 벗어날 수 있다.</ko>
  76. * @default [0, Infinity]
  77. */
  78. sizeRange?: number | number[];
  79. /**
  80. * Maximum number of rows to be counted for container size. You can hide it on the screen by setting overflow: hidden. -1 is not set.
  81. * <ko>컨테이너 크기에 계산될 최대 row 개수. overflow: hidden을 설정하면 화면에 가릴 수 있다. -1은 미설정이다.</ko>
  82. * @default -1
  83. */
  84. displayedRow?: number;
  85. /**
  86. * Whether to crop when the row size is out of sizeRange. If set to true, this ratio can be broken.
  87. * <ko>row사이즈가 sizeRange에 벗어나면 크롭할지 여부. true로 설정하면 비율이 깨질 수 있다.</ko>
  88. * @default false
  89. */
  90. isCroppedSize?: boolean;
  91. /**
  92. * The ratio is maintained except for the offset value in the inline direction. If 'data-grid-inline-offset' is set in the element of each item, it will be applied first.
  93. * <ko>inline 방향의 offset 수치 만큼 제외하고 비율을 유지한다. 각 아이템의 element에 'data-grid-inline-offset' 을 설정하면 우선 적용한다.</ko>
  94. * @default 0
  95. */
  96. inlineOffset?: number;
  97. /**
  98. * The ratio is maintained except for the offset value in the content direction. If 'data-grid-content-offset' is set in the element or JSX of each item, it will be applied first.
  99. * <ko>content 방향의 offset 수치 만큼 제외하고 비율을 유지한다. 각 아이템의 Element 또는 JSX에 'data-grid-content-offset' 을 설정하면 우선 적용한다.</ko>
  100. * @default 0
  101. */
  102. contentOffset?: number;
  103. /**
  104. * it is possible to basically break the proportion of the item and stretch the inline size to fill the container.
  105. * If you set the `sizeRange` range narrowly, you can stretch well.
  106. * <ko>기본적으로 아이템의 비율을 깨서 inline size를 stretch하여 container를 꽉 채우게 가능하다. sizeRange의 범위를 좁게 설정하면 stretch가 잘 될 수 있다. </ko>
  107. * @default false
  108. */
  109. stretch?: boolean;
  110. /**
  111. * If `-`, `+`, or `%` are added as a string value, it is a relative value to the original size. If it is a number value, the stretch range can be set as an absolute value.
  112. * If `data-grid-min-stretch` and `data-grid-max-stretch` are set in the Element or JSX of each item, they will be applied first.
  113. * <ko>string 값으로 `-`, `+`, `%`이 붙으면 원본 크기에 대한 상대값이며 number 값으로 들어오면 절대 값으로 stretch 범위를 설정할 수 있습니다.
  114. * 각 아이템의 Element 또는 JSX에 `data-grid-min-stretch`, `data-grid-max-stretch`을 설정하면 우선 적용한다.</ko>
  115. * @
  116. * @default ["-10%", "+10%"]
  117. */
  118. stretchRange?: Array<string | number>;
  119. /**
  120. * Items placed in the last row are not stretched and are drawn maintaining their proportions. When using InfiniteGrid, it is calculated and re-rendered as follows:
  121. * <ko>마지막 row에 배치되는 아이템들 경우 stretch되지 않고 비율유지한채로 그려진다. InfiniteGrid를 사용하는 경우 다음 그룹과 같이 계산되어 재렌더링한다.</ko>
  122. */
  123. passUnstretchRow?: boolean;
  124. }
  125. /**
  126. * 'justified' is a printing term with the meaning that 'it fits in one row wide'. JustifiedGrid is a grid that the item is filled up on the basis of a line given a size.
  127. * If 'data-grid-inline-offset' or 'data-grid-content-offset' are set for item element, the ratio is maintained except for the offset value.
  128. * If 'data-grid-maintained-target' is set for an element whose ratio is to be maintained, the item is rendered while maintaining the ratio of the element.
  129. * @ko 'justified'는 '1행의 너비에 맞게 꼭 들어찬'이라는 의미를 가진 인쇄 용어다. JustifiedGrid는 용어의 의미대로 너비가 주어진 사이즈를 기준으로 아이템가 가득 차도록 배치하는 Grid다.
  130. * 아이템 엘리먼트에 'data-grid-inline-offset' 또는 'data-grid-content-offset'를 설정하면 offset 값을 제외하고 비율을 유지한다.
  131. * 비율을 유지하고 싶은 엘리먼트에 'data-grid-maintained-target'을 설정한다면 해당 엘리먼트의 비율을 유지하면서 아이템이 렌더링이 된다.
  132. * @memberof Grid
  133. * @param {HTMLElement | string} container - A base element for a module <ko>모듈을 적용할 기준 엘리먼트</ko>
  134. * @param {Grid.JustifiedGrid.JustifiedGridOptions} options - The option object of the JustifiedGrid module <ko>JustifiedGrid 모듈의 옵션 객체</ko>
  135. */
  136. @GetterSetter
  137. export class JustifiedGrid extends Grid<JustifiedGridOptions> {
  138. public static propertyTypes = {
  139. ...Grid.propertyTypes,
  140. columnRange: PROPERTY_TYPE.RENDER_PROPERTY,
  141. rowRange: PROPERTY_TYPE.RENDER_PROPERTY,
  142. sizeRange: PROPERTY_TYPE.RENDER_PROPERTY,
  143. isCroppedSize: PROPERTY_TYPE.RENDER_PROPERTY,
  144. displayedRow: PROPERTY_TYPE.RENDER_PROPERTY,
  145. stretch: PROPERTY_TYPE.RENDER_PROPERTY,
  146. stretchRange: PROPERTY_TYPE.RENDER_PROPERTY,
  147. passUnstretchRow: PROPERTY_TYPE.RENDER_PROPERTY,
  148. inlineMargin: PROPERTY_TYPE.RENDER_PROPERTY,
  149. contentMargin: PROPERTY_TYPE.RENDER_PROPERTY,
  150. inlineOffset: PROPERTY_TYPE.RENDER_PROPERTY,
  151. contentOffset: PROPERTY_TYPE.RENDER_PROPERTY,
  152. };
  153. public static defaultOptions: Required<JustifiedGridOptions> = {
  154. ...Grid.defaultOptions,
  155. columnRange: [1, 8],
  156. rowRange: 0,
  157. sizeRange: [0, Infinity],
  158. displayedRow: -1,
  159. isCroppedSize: false,
  160. stretch: false,
  161. passUnstretchRow: true,
  162. stretchRange: ["-20%", "+20%"],
  163. inlineOffset: 0,
  164. contentOffset: 0,
  165. };
  166. public applyGrid(items: GridItem[], direction: "start" | "end", outline: number[]): GridOutlines {
  167. const {
  168. attributePrefix,
  169. horizontal,
  170. } = this.options;
  171. items.forEach((item) => {
  172. if (!item.isUpdating) {
  173. return;
  174. }
  175. const element = item.element;
  176. const attributes = item.attributes;
  177. const gridData = item.gridData;
  178. let inlineOffset = parseFloat(attributes.inlineOffset);
  179. let contentOffset = parseFloat(attributes.contentOffset);
  180. // let contentMargin = parseFloat(attributes.contentMargin);
  181. if (isNaN(inlineOffset)) {
  182. inlineOffset = this.inlineOffset || gridData.inlineOffset || 0;
  183. }
  184. if (isNaN(contentOffset)) {
  185. contentOffset = this.contentOffset || gridData.contentOffset | 0;
  186. }
  187. // if (isNaN(contentMargin)) {
  188. // contentMargin = this.contentMargin || gridData.contentMargin | 0;
  189. // }
  190. if (
  191. element && !("inlineOffset" in attributes) && !("contentOffset" in attributes)
  192. && item.mountState === MOUNT_STATE.MOUNTED
  193. ) {
  194. const maintainedTarget = element.querySelector<HTMLImageElement>(`[${attributePrefix}maintained-target]`);
  195. if (maintainedTarget) {
  196. const widthOffset = element.offsetWidth - element.clientWidth
  197. + element.scrollWidth - maintainedTarget.clientWidth;
  198. const heightOffset = element.offsetHeight - element.clientHeight
  199. + element.scrollHeight - maintainedTarget.clientHeight;
  200. if (horizontal) {
  201. inlineOffset = heightOffset;
  202. contentOffset = widthOffset;
  203. } else {
  204. inlineOffset = widthOffset;
  205. contentOffset = heightOffset;
  206. }
  207. }
  208. }
  209. gridData.inlineOffset = inlineOffset;
  210. gridData.contentOffset = contentOffset;
  211. // gridData.contentMargin = contentMargin;
  212. });
  213. const rowRange = this.options.rowRange;
  214. let path: string[] = [];
  215. const isEndDirection = direction === "end";
  216. if (items.length) {
  217. path = rowRange ? this._getRowPath(items, isEndDirection) : this._getPath(items, isEndDirection);
  218. }
  219. return this._setStyle(items, path, outline, direction === "end");
  220. }
  221. private _getRowPath(items: GridItem[], isEndDirection: boolean) {
  222. const columnRange = this._getColumnRange();
  223. const rowRange = this._getRowRange();
  224. const pathLink = this._getRowLink(items, {
  225. path: [0],
  226. cost: 0,
  227. length: 0,
  228. currentNode: 0,
  229. }, columnRange, rowRange, isEndDirection);
  230. return pathLink?.path.map((node) => `${node}`) ?? [];
  231. }
  232. private _getRowLink(
  233. items: GridItem[],
  234. currentLink: Link,
  235. columnRange: number[],
  236. rowRange: number[],
  237. isEndDirection: boolean,
  238. ): Link {
  239. const [minColumn] = columnRange;
  240. const [minRow, maxRow] = rowRange;
  241. const lastNode = items.length;
  242. const {
  243. path,
  244. length: pathLength,
  245. cost,
  246. currentNode,
  247. } = currentLink;
  248. // not reached lastNode but path is exceed or the number of remaining nodes is less than minColumn.
  249. if (currentNode < lastNode && (maxRow <= pathLength || currentNode + minColumn > lastNode)) {
  250. const rangeCost = getRangeCost(lastNode - currentNode, columnRange);
  251. const lastCost = rangeCost * Math.abs(this._getCost(items, currentNode, lastNode, isEndDirection));
  252. return {
  253. ...currentLink,
  254. length: pathLength + 1,
  255. path: [...path, lastNode],
  256. currentNode: lastNode,
  257. cost: cost + lastCost,
  258. isOver: true,
  259. };
  260. } else if (currentNode >= lastNode) {
  261. return {
  262. ...currentLink,
  263. currentNode: lastNode,
  264. isOver: minRow > pathLength || maxRow < pathLength,
  265. };
  266. } else {
  267. return this._searchRowLink(items, currentLink, lastNode, columnRange, rowRange, isEndDirection);
  268. }
  269. }
  270. private _searchRowLink(
  271. items: GridItem[],
  272. currentLink: Link,
  273. lastNode: number,
  274. columnRange: number[],
  275. rowRange: number[],
  276. isEndDirection: boolean,
  277. ) {
  278. const [minColumn, maxColumn] = columnRange;
  279. const {
  280. currentNode,
  281. path,
  282. length: pathLength,
  283. cost,
  284. } = currentLink;
  285. const length = Math.min(lastNode, currentNode + maxColumn);
  286. const links: Link[] = [];
  287. for (let nextNode = currentNode + minColumn; nextNode <= length; ++nextNode) {
  288. if (nextNode === currentNode) {
  289. continue;
  290. }
  291. const nextCost = Math.abs(this._getCost(items, currentNode, nextNode, isEndDirection));
  292. const nextLink = this._getRowLink(items, {
  293. path: [...path, nextNode],
  294. length: pathLength + 1,
  295. cost: cost + nextCost,
  296. currentNode: nextNode,
  297. }, columnRange, rowRange, isEndDirection);
  298. if (nextLink) {
  299. links.push(nextLink);
  300. }
  301. }
  302. links.sort((a, b) => {
  303. const aIsOver = a.isOver;
  304. const bIsOver = b.isOver;
  305. if (aIsOver !== bIsOver) {
  306. // If it is over, the cost is high.
  307. return aIsOver ? 1 : -1;
  308. }
  309. const aRangeCost = getRangeCost(a.length, rowRange);
  310. const bRangeCost = getRangeCost(b.length, rowRange);
  311. return aRangeCost - bRangeCost || a.cost - b.cost;
  312. });
  313. // It returns the lowest cost link.
  314. return links[0];
  315. }
  316. private _getExpectedRowSize(items: GridItem[], forceStretch?: boolean) {
  317. const containerInlineSize = this.getContainerInlineSize()! - this.getInlineGap() * (items.length - 1);
  318. let fixedContainerInsize = containerInlineSize;
  319. let ratioSum = 0;
  320. let inlineSum = 0;
  321. items.forEach((item) => {
  322. const inlineSize = item.orgInlineSize;
  323. const contentSize = item.orgContentSize;
  324. if (!inlineSize || !contentSize) {
  325. ratioSum += 1;
  326. return;
  327. }
  328. // sum((expect - offset) * ratio) = container inline size
  329. const inlineOffset = item.gridData.inlineOffset || 0;
  330. const contentOffset = item.gridData.contentOffset || 0;
  331. // const contentMargin = item.gridData.contentMargin || 0;
  332. const maintainedRatio = contentSize <= contentOffset ? 1
  333. : (inlineSize - inlineOffset) / (contentSize - contentOffset);
  334. ratioSum += maintainedRatio;
  335. // inlineSum += (contentOffset + contentMargin) * maintainedRatio;
  336. inlineSum += contentOffset * maintainedRatio;
  337. fixedContainerInsize -= inlineOffset;
  338. });
  339. if (ratioSum) {
  340. const nextRowSize = (fixedContainerInsize + inlineSum) / ratioSum;
  341. if (this.stretch) {
  342. const [minRowSize, maxRowSize] = this._getSizeRange();
  343. const stretchRowSize = between(nextRowSize, minRowSize, maxRowSize);
  344. if (forceStretch) {
  345. return stretchRowSize;
  346. }
  347. const stretchRange = this.stretchRange;
  348. const inlineSizes = items.map((item) => {
  349. return getExpectedItemInlineSize(item, stretchRowSize);
  350. });
  351. const minInlineSize = inlineSizes.reduce((prev, itemInlineSize, i) => {
  352. return prev + parseStretchSize(itemInlineSize, items[i].attributes.minStretch || stretchRange[0]);
  353. }, 0);
  354. const maxInlineSize = inlineSizes.reduce((prev, itemInlineSize, i) => {
  355. return prev + parseStretchSize(itemInlineSize, items[i].attributes.maxStretch || stretchRange[1]);
  356. }, 0);
  357. // for stretch
  358. if (minInlineSize <= containerInlineSize && containerInlineSize <= maxInlineSize) {
  359. return stretchRowSize;
  360. }
  361. }
  362. return nextRowSize;
  363. }
  364. return 0;
  365. }
  366. private _getExpectedInlineSizes(items: GridItem[], rowSize: number) {
  367. const {
  368. stretch,
  369. stretchRange,
  370. } = this.options;
  371. return items.map((item) => {
  372. const minInlineSize = stretch
  373. ? parseStretchSize(item.orgInlineSize, item.attributes.minStretch || stretchRange[0])
  374. : -Infinity;
  375. const maxInlineSize = stretch
  376. ? parseStretchSize(item.orgInlineSize, item.attributes.maxStretch || stretchRange[1])
  377. : Infinity;
  378. const itemInlineSize = getExpectedItemInlineSize(item, rowSize);
  379. let isMax = false;
  380. let isMin = false;
  381. if (itemInlineSize >= maxInlineSize) {
  382. isMax = true;
  383. } else if (itemInlineSize <= minInlineSize) {
  384. isMin = true;
  385. }
  386. return {
  387. minSize: minInlineSize,
  388. maxSize: maxInlineSize,
  389. size: between(itemInlineSize, minInlineSize, maxInlineSize),
  390. originalSize: itemInlineSize,
  391. isMax,
  392. isMin,
  393. };
  394. });
  395. }
  396. private _getStretchItemInfos(items: GridItem[], rowSize: number) {
  397. const itemsLength = items.length;
  398. const containerInlineSize = this.getContainerInlineSize() - this.getInlineGap() * (Math.max(1, itemsLength) - 1);
  399. const itemInfos = this._getExpectedInlineSizes(items, rowSize);
  400. const firstItemsSize = sum(itemInfos.map((info) => info.size));
  401. const distSize = containerInlineSize - firstItemsSize;
  402. const firstScale = containerInlineSize / sum(itemInfos.map((info) => info.originalSize));
  403. const costInfos = itemInfos.map((info) => {
  404. return {
  405. ...info,
  406. passed: false,
  407. size: info.originalSize * firstScale,
  408. };
  409. });
  410. if (distSize === 0) {
  411. return {
  412. infos: costInfos,
  413. cost: 0,
  414. };
  415. }
  416. // increase
  417. const isIncrease = distSize > 0;
  418. const costInfosLength = costInfos.length;
  419. for (let i = 0; i < costInfosLength; ++i) {
  420. const passedItemsSize = sum(costInfos.map((info) => info.passed ? info.size : 0));
  421. const restItemsSize = sum(costInfos.map((info) => info.passed ? 0 : info.originalSize));
  422. let distScale = (containerInlineSize - passedItemsSize) / restItemsSize;
  423. // minimize or maximize
  424. costInfos.forEach((info) => {
  425. if (info.passed) {
  426. return;
  427. }
  428. if (isIncrease) {
  429. if (info.size > info.maxSize) {
  430. distScale = Math.min(distScale, info.maxSize / info.originalSize);
  431. }
  432. } else {
  433. if (info.size < info.minSize) {
  434. distScale = Math.max(distScale, info.minSize / info.originalSize);
  435. }
  436. }
  437. });
  438. costInfos.forEach((info) => {
  439. if (!info.passed) {
  440. info.size = between(info.originalSize * distScale, info.minSize, info.maxSize);
  441. if (
  442. (isIncrease && !throttle(info.size - info.maxSize, 0.001))
  443. || (!isIncrease && !throttle(info.size - info.minSize, 0.001))
  444. ) {
  445. info.passed = true;
  446. }
  447. }
  448. });
  449. if (costInfos.every((info) => info.passed)) {
  450. break;
  451. }
  452. }
  453. const lastDistScale = containerInlineSize / sum(costInfos.map((info) => info.size));
  454. // last
  455. if (throttle(lastDistScale - 1, 0.001)) {
  456. costInfos.forEach((info) => {
  457. info.size *= lastDistScale;
  458. });
  459. }
  460. return {
  461. infos: costInfos,
  462. cost: sum(costInfos.map((info) => {
  463. let costRatio = 1;
  464. if (info.size > info.maxSize || info.size < info.minSize) {
  465. costRatio = 2;
  466. }
  467. let originalSize = info.originalSize;
  468. if (isIncrease) {
  469. originalSize = Math.max(originalSize, info.minSize);
  470. } else {
  471. originalSize = Math.min(originalSize, info.maxSize);
  472. }
  473. return Math.abs(info.size - originalSize) * costRatio;
  474. })),
  475. };
  476. }
  477. private _getExpectedInlineSize(items: GridItem[], rowSize: number) {
  478. const inlineGap = this.getInlineGap();
  479. const itemInfos = this._getExpectedInlineSizes(items, rowSize);
  480. return itemInfos.length ? sum(itemInfos.map((info) => info.size)) + inlineGap * (items.length - 1) : 0;
  481. }
  482. private _getCost(
  483. items: GridItem[],
  484. i: number,
  485. j: number,
  486. isEndDirection: boolean,
  487. ) {
  488. const lineItems = items.slice(i, j);
  489. const containerInlineSize = this.getContainerInlineSize();
  490. let rowSize = this._getExpectedRowSize(lineItems);
  491. const [minSize, maxSize] = this._getSizeRange();
  492. if (this.isCroppedSize) {
  493. if (minSize <= rowSize && rowSize <= maxSize) {
  494. return 0;
  495. }
  496. const expectedInlineSize = this._getExpectedInlineSize(
  497. lineItems,
  498. rowSize < minSize ? minSize : maxSize,
  499. );
  500. return Math.pow(expectedInlineSize - containerInlineSize, 2);
  501. }
  502. let extraCost = 0;
  503. if (this.stretch) {
  504. if (rowSize < minSize) {
  505. rowSize = minSize;
  506. } else if (rowSize > maxSize) {
  507. rowSize = maxSize;
  508. }
  509. const sizeCost = Math.abs(rowSize - minSize);
  510. const expectedInlineSize = this._getExpectedInlineSize(
  511. lineItems,
  512. rowSize,
  513. );
  514. if (
  515. !this.passUnstretchRow
  516. || (isEndDirection ? j !== items.length : i !== 0)
  517. || expectedInlineSize >= containerInlineSize
  518. ) {
  519. const res = this._getStretchItemInfos(lineItems, rowSize);
  520. extraCost = res.cost;
  521. }
  522. return extraCost + sizeCost;
  523. }
  524. if (isFinite(maxSize)) {
  525. // if this size is not in range, the cost increases sharply.
  526. if (rowSize < minSize) {
  527. return Math.pow(rowSize - minSize, 2) + Math.pow(maxSize, 2) + extraCost;
  528. } else if (rowSize > maxSize) {
  529. return Math.pow(rowSize - maxSize, 2) + Math.pow(maxSize, 2) + extraCost;
  530. }
  531. } else if (rowSize < minSize) {
  532. return Math.max(Math.pow(minSize, 2), Math.pow(rowSize, 2)) + Math.pow(maxSize, 2) + extraCost;
  533. }
  534. // if this size in range, the cost is row
  535. return rowSize - minSize + extraCost;
  536. }
  537. private _getPath(items: GridItem[], isEndDirection: boolean) {
  538. const lastNode = items.length;
  539. const columnRangeOption = this.options.columnRange;
  540. const [minColumn, maxColumn]: number[] = isObject(columnRangeOption)
  541. ? columnRangeOption
  542. : [columnRangeOption, columnRangeOption];
  543. const graph = (nodeKey: string) => {
  544. const results: { [key: string]: number } = {};
  545. const currentNode = parseInt(nodeKey, 10);
  546. for (let nextNode = Math.min(currentNode + minColumn, lastNode); nextNode <= lastNode; ++nextNode) {
  547. if (nextNode - currentNode > maxColumn) {
  548. break;
  549. }
  550. let cost = this._getCost(
  551. items,
  552. currentNode,
  553. nextNode,
  554. isEndDirection,
  555. );
  556. if (cost < 0 && nextNode === lastNode) {
  557. cost = 0;
  558. }
  559. results[`${nextNode}`] = Math.pow(cost, 2);
  560. }
  561. return results;
  562. };
  563. // shortest path for items' total height.
  564. return find_path(graph, "0", `${lastNode}`);
  565. }
  566. private _setStyle(
  567. items: GridItem[],
  568. path: string[],
  569. outline: number[] = [],
  570. isEndDirection: boolean,
  571. ) {
  572. const {
  573. isCroppedSize,
  574. displayedRow,
  575. stretch,
  576. passUnstretchRow,
  577. } = this.options;
  578. const itemsLength = items.length;
  579. const sizeRange = this._getSizeRange();
  580. const startPoint = outline[0] || 0;
  581. const containerInlineSize = this.getContainerInlineSize();
  582. const inlineGap = this.getInlineGap();
  583. const contentGap = this.getContentGap();
  584. const groups = splitItems(items, path);
  585. let passedItems!: number[];
  586. const groupsLength = groups.length;
  587. let contentPos = startPoint;
  588. let displayedSize = 0;
  589. let passedPoint!: number[];
  590. groups.forEach((groupItems, rowIndex) => {
  591. const groupItemslength = groupItems.length;
  592. let rowSize = this._getExpectedRowSize(groupItems, true);
  593. if (isCroppedSize) {
  594. rowSize = Math.max(sizeRange[0], Math.min(rowSize, sizeRange[1]));
  595. }
  596. const allGap = inlineGap * (length - 1);
  597. const itemInfos = groupItems.map((item, index) => {
  598. const itemInlineSize = getExpectedItemInlineSize(item, rowSize);
  599. return {
  600. index,
  601. item,
  602. inlineSize: itemInlineSize,
  603. orgInlineSize: itemInlineSize,
  604. maxInlineSize: itemInlineSize,
  605. minInlineSize: itemInlineSize,
  606. };
  607. });
  608. const expectedInlineSize = this._getExpectedInlineSize(groupItems, rowSize);
  609. const scale = (containerInlineSize - allGap) / (expectedInlineSize - allGap);
  610. const noGapExpectedContainerInlineSize = expectedInlineSize - allGap;
  611. const noGapContainerInlineSize = containerInlineSize - allGap;
  612. if (stretch && expectedInlineSize && noGapContainerInlineSize !== noGapExpectedContainerInlineSize) {
  613. // passed이고 마지막 그룹의 경우 stretchSize가 containerSize보다 작으면 pass!
  614. if (
  615. passUnstretchRow && noGapExpectedContainerInlineSize < noGapContainerInlineSize
  616. && (isEndDirection ? rowIndex === groupsLength - 1 : rowIndex === 0)
  617. ) {
  618. passedPoint = [contentPos];
  619. passedItems = groupItems.map((_, i) => itemsLength - groupItemslength + i);
  620. const inlineSizes = this._getExpectedInlineSizes(groupItems, rowSize);
  621. itemInfos.forEach((info, i) => {
  622. info.minInlineSize = inlineSizes[i].minSize;
  623. info.maxInlineSize = inlineSizes[i].maxSize;
  624. info.inlineSize = between(info.inlineSize, info.minInlineSize, info.maxInlineSize);
  625. });
  626. } else {
  627. const { infos } = this._getStretchItemInfos(groupItems, rowSize);
  628. itemInfos.forEach((info, i) => {
  629. info.inlineSize = infos[i].size;
  630. info.minInlineSize = infos[i].minSize;
  631. info.maxInlineSize = infos[i].maxSize;
  632. });
  633. }
  634. }
  635. itemInfos.forEach((info, i) => {
  636. const {
  637. item,
  638. inlineSize,
  639. } = info;
  640. let nextInlineSize = inlineSize;
  641. const prevItem = groupItems[i - 1];
  642. const inlinePos = prevItem
  643. ? prevItem.cssInlinePos! + prevItem.cssInlineSize! + inlineGap
  644. : 0;
  645. if (isCroppedSize) {
  646. nextInlineSize *= scale;
  647. }
  648. const gridData = item.gridData;
  649. gridData.orgInlineSize = info.orgInlineSize;
  650. gridData.orgContentSize = rowSize;
  651. gridData.minInlineSize = info.minInlineSize;
  652. gridData.maxInlineSize = info.maxInlineSize;
  653. item.setCSSGridRect({
  654. inlinePos,
  655. contentPos,
  656. inlineSize: nextInlineSize,
  657. contentSize: rowSize,
  658. });
  659. });
  660. contentPos += contentGap + rowSize;
  661. if (displayedRow < 0 || rowIndex < displayedRow) {
  662. displayedSize = contentPos;
  663. }
  664. });
  665. if (isEndDirection) {
  666. // previous group's end outline is current group's start outline
  667. return {
  668. start: [startPoint],
  669. end: [displayedSize],
  670. passedItems,
  671. passed: passedPoint,
  672. };
  673. }
  674. // always start is lower than end.
  675. // contentPos is endPoinnt
  676. const height = contentPos - startPoint;
  677. items.forEach((item) => {
  678. item.cssContentPos! -= height;
  679. });
  680. return {
  681. passedItems,
  682. passed: passedPoint ? [passedPoint[0] - height] : null,
  683. start: [startPoint - height],
  684. end: [startPoint], // endPoint - height = startPoint
  685. };
  686. }
  687. public getComputedOutlineLength() {
  688. return 1;
  689. }
  690. public getComputedOutlineSize() {
  691. return this.getContainerInlineSize();
  692. }
  693. private _getRowRange() {
  694. const rowRange = this.rowRange;
  695. return isObject(rowRange) ? rowRange : [rowRange, rowRange];
  696. }
  697. private _getColumnRange() {
  698. const columnRange = this.columnRange;
  699. return isObject(columnRange) ? columnRange : [columnRange, columnRange];
  700. }
  701. private _getSizeRange() {
  702. const sizeRange = this.sizeRange;
  703. return isObject(sizeRange) ? sizeRange : [sizeRange, sizeRange];
  704. }
  705. }
  706. export interface JustifiedGrid extends Properties<typeof JustifiedGrid> {
  707. }
  708. /**
  709. * The minimum and maximum number of items per line.
  710. * @ko 한 줄에 들어가는 아이템의 최소, 최대 개수.
  711. * @name Grid.JustifiedGrid#columnRange
  712. * @type {$ts:Grid.JustifiedGrid.JustifiedGridOptions["columnRange"]}
  713. * @default [1, 8]
  714. * @example
  715. * ```js
  716. * import { JustifiedGrid } from "@egjs/grid";
  717. *
  718. * const grid = new JustifiedGrid(container, {
  719. * columnRange: [1, 8],
  720. * });
  721. *
  722. * grid.columnRange = [3, 6];
  723. * ```
  724. */
  725. /**
  726. * The minimum and maximum number of rows in a group, 0 is not set.
  727. * @ko 한 그룹에 들어가는 행의 최소, 최대 개수, 0은 미설정이다.
  728. * @name Grid.JustifiedGrid#rowRange
  729. * @type {$ts:Grid.JustifiedGrid.JustifiedGridOptions["rowRange"]}
  730. * @default 0
  731. * @example
  732. * ```js
  733. * import { JustifiedGrid } from "@egjs/grid";
  734. *
  735. * const grid = new JustifiedGrid(container, {
  736. * rowRange: 0,
  737. * });
  738. *
  739. * grid.rowRange = [3, 4];
  740. * ```
  741. */
  742. /**
  743. * The minimum and maximum size by which the item is adjusted. If it is not calculated, it may deviate from the minimum and maximum sizes.
  744. * @ko 아이템이 조정되는 최소, 최대 사이즈. 계산이 되지 않는 경우 최소, 최대 사이즈를 벗어날 수 있다.
  745. * @name Grid.JustifiedGrid#sizeRange
  746. * @type {$ts:Grid.JustifiedGrid.JustifiedGridOptions["sizeRange"]}
  747. * @default [0, Infinity]
  748. * @example
  749. * ```js
  750. * import { JustifiedGrid } from "@egjs/grid";
  751. *
  752. * const grid = new JustifiedGrid(container, {
  753. * sizeRange: [0, Infinity],
  754. * });
  755. *
  756. * grid.sizeRange = [200, 800];
  757. * ```
  758. */
  759. /**
  760. * Maximum number of rows to be counted for container size. You can hide it on the screen by setting overflow: hidden. -1 is not set.
  761. * @ko - 컨테이너 크기에 계산될 최대 row 개수. overflow: hidden을 설정하면 화면에 가릴 수 있다. -1은 미설정이다.
  762. * @name Grid.JustifiedGrid#displayedRow
  763. * @type {$ts:Grid.JustifiedGrid.JustifiedGridOptions["displayedRow"]}
  764. * @default -1
  765. * @example
  766. * ```js
  767. * import { JustifiedGrid } from "@egjs/grid";
  768. *
  769. * const grid = new JustifiedGrid(container, {
  770. * displayedRow: -1,
  771. * });
  772. *
  773. * grid.displayedRow = 3;
  774. * ```
  775. */
  776. /**
  777. * Whether to crop when the row size is out of sizeRange. If set to true, this ratio can be broken.
  778. * @ko - row 사이즈가 sizeRange에 벗어나면 크롭할지 여부. true로 설정하면 비율이 깨질 수 있다.
  779. * @name Grid.JustifiedGrid#isCroppedSize
  780. * @type {$ts:Grid.JustifiedGrid.JustifiedGridOptions["isCroppedSize"]}
  781. * @default false
  782. * @example
  783. * ```js
  784. * import { JustifiedGrid } from "@egjs/grid";
  785. *
  786. * const grid = new JustifiedGrid(container, {
  787. * sizeRange: [200, 250],
  788. * isCroppedSize: false,
  789. * });
  790. *
  791. * grid.isCroppedSize = true;
  792. * ```
  793. */
comments powered by Disqus