Source: src/grids/MasonryGrid.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, UPDATE_STATE } from "../consts";
  8. import { GridOptions, Properties, GridOutlines, GridAlign, MasonryGridVerticalAlign } from "../types";
  9. import { range, GetterSetter } from "../utils";
  10. import { GridItem } from "../GridItem";
  11. function getColumnPoint(
  12. outline: number[],
  13. columnIndex: number,
  14. columnCount: number,
  15. pointCaculationName: "max" | "min",
  16. ) {
  17. return Math[pointCaculationName](...outline.slice(columnIndex, columnIndex + columnCount));
  18. }
  19. function getColumnIndex(
  20. outline: number[],
  21. columnCount: number,
  22. nearestCalculationName: "max" | "min",
  23. startPos: number,
  24. ) {
  25. const length = outline.length - columnCount + 1;
  26. const pointCaculationName = nearestCalculationName === "max" ? "min" : "max";
  27. const indexCaculationName = nearestCalculationName === "max" ? "lastIndexOf" : "indexOf";
  28. const points = range(length).map((index) => {
  29. const point = getColumnPoint(outline, index, columnCount, pointCaculationName);
  30. return Math[pointCaculationName](startPos, point);
  31. });
  32. return points[indexCaculationName](Math[nearestCalculationName](...points));
  33. }
  34. /**
  35. * @typedef
  36. * @memberof Grid.MasonryGrid
  37. * @extends Grid.GridOptions
  38. */
  39. export interface MasonryGridOptions extends GridOptions {
  40. /**
  41. * The number of columns. If the number of columns is 0, it is automatically calculated according to the size of the container. Can be used instead of outlineLength.
  42. * <ko>열의 개수. 열의 개수가 0이라면, 컨테이너의 사이즈에 의해 계산이 된다. outlineLength 대신 사용할 수 있다.</ko>
  43. * @default 0
  44. */
  45. column?: number;
  46. /**
  47. * The size of the columns. If it is 0, it is calculated as the size of the first item in items. Can be used instead of outlineSize.
  48. * <ko>열의 사이즈. 만약 열의 사이즈가 0이면, 아이템들의 첫번째 아이템의 사이즈로 계산이 된다. outlineSize 대신 사용할 수 있다.</ko>
  49. * @default 0
  50. */
  51. columnSize?: number;
  52. /**
  53. * The size ratio(inlineSize / contentSize) of the columns. 0 is not set.
  54. * <ko>열의 사이즈 비율(inlineSize / contentSize). 0은 미설정이다.</ko>
  55. * @default 0
  56. */
  57. columnSizeRatio?: number;
  58. /**
  59. * Align of the position of the items. If you want to use `stretch`, be sure to set `column`, `columnSize` or `maxStretchColumnSize` option. ("start", "center", "end", "justify", "stretch")
  60. * <ko>아이템들의 위치의 정렬. `stretch`를 사용하고 싶다면 `column`, `columnSize` 또는 `maxStretchColumnSize` 옵션을 설정해라. ("start", "center", "end", "justify", "stretch")</ko>
  61. * @default "justify"
  62. */
  63. align?: GridAlign;
  64. /**
  65. * Content direction alignment of items. “Masonry” is sorted in the form of masonry. Others are applied as content direction alignment, similar to vertical-align of inline-block.
  66. * If you set multiple columns (`data-grid-column`), the screen may look strange.
  67. * <ko>아이템들의 Content 방향의 정렬. "masonry"는 masonry 형태로 정렬이 된다. 그 외는 inline-block의 vertical-align과 유사하게 content 방향 정렬로 적용이 된다.칼럼(`data-grid-column` )을 여러개 설정하면 화면이 이상하게 보일 수 있다. </ko>
  68. * @default "masonry"
  69. */
  70. contentAlign?: MasonryGridVerticalAlign;
  71. /**
  72. * Difference Threshold for Counting Columns. Since offsetSize is calculated by rounding, the number of columns may not be accurate.
  73. * <ko>칼럼 개수를 계산하기 위한 차이 임계값. offset 사이즈는 반올림으로 게산하기 때문에 정확하지 않을 수 있다.</ko>
  74. * @default 1
  75. */
  76. columnCalculationThreshold?: number;
  77. /**
  78. * If stretch is used, the column can be automatically calculated by setting the maximum size of the column that can be stretched.
  79. * <ko>stretch를 사용한 경우 최대로 늘릴 수 있는 column의 사이즈를 설정하여 column을 자동 계산할 수 있다.</ko>
  80. * @default Infinity
  81. */
  82. maxStretchColumnSize?: number;
  83. }
  84. /**
  85. * MasonryGrid is a grid that stacks items with the same width as a stack of bricks. Adjust the width of all images to the same size, find the lowest height column, and insert a new item.
  86. * @ko MasonryGrid는 벽돌을 쌓아 올린 모양처럼 동일한 너비를 가진 아이템를 쌓는 레이아웃이다. 모든 이미지의 너비를 동일한 크기로 조정하고, 가장 높이가 낮은 열을 찾아 새로운 이미지를 삽입한다. 따라서 배치된 아이템 사이에 빈 공간이 생기지는 않지만 배치된 레이아웃의 아래쪽은 울퉁불퉁해진다.
  87. * @memberof Grid
  88. * @param {HTMLElement | string} container - A base element for a module <ko>모듈을 적용할 기준 엘리먼트</ko>
  89. * @param {Grid.MasonryGrid.MasonryGridOptions} options - The option object of the MasonryGrid module <ko>MasonryGrid 모듈의 옵션 객체</ko>
  90. */
  91. @GetterSetter
  92. export class MasonryGrid extends Grid<MasonryGridOptions> {
  93. public static propertyTypes = {
  94. ...Grid.propertyTypes,
  95. column: PROPERTY_TYPE.RENDER_PROPERTY,
  96. columnSize: PROPERTY_TYPE.RENDER_PROPERTY,
  97. columnSizeRatio: PROPERTY_TYPE.RENDER_PROPERTY,
  98. align: PROPERTY_TYPE.RENDER_PROPERTY,
  99. columnCalculationThreshold: PROPERTY_TYPE.RENDER_PROPERTY,
  100. maxStretchColumnSize: PROPERTY_TYPE.RENDER_PROPERTY,
  101. contentAlign: PROPERTY_TYPE.RENDER_PROPERTY,
  102. };
  103. public static defaultOptions: Required<MasonryGridOptions> = {
  104. ...Grid.defaultOptions,
  105. align: "justify",
  106. column: 0,
  107. columnSize: 0,
  108. columnSizeRatio: 0,
  109. columnCalculationThreshold: 0.5,
  110. maxStretchColumnSize: Infinity,
  111. contentAlign: "masonry",
  112. };
  113. public applyGrid(items: GridItem[], direction: "start" | "end", outline: number[]): GridOutlines {
  114. items.forEach((item) => {
  115. item.isRestoreOrgCSSText = false;
  116. });
  117. const columnSize = this.getComputedOutlineSize(items);
  118. const column = this.getComputedOutlineLength(items);
  119. const {
  120. align,
  121. observeChildren,
  122. columnSizeRatio,
  123. contentAlign,
  124. } = this.options;
  125. const inlineGap = this.getContentGap();
  126. const contentGap = this.getContentGap();
  127. const outlineLength = outline.length;
  128. const itemsLength = items.length;
  129. const alignPoses = this._getAlignPoses(column, columnSize);
  130. const isEndDirection = direction === "end";
  131. const nearestCalculationName = isEndDirection ? "min" : "max";
  132. const pointCalculationName = isEndDirection ? "max" : "min";
  133. let startOutline = [0];
  134. if (outlineLength === column) {
  135. startOutline = outline.slice();
  136. } else {
  137. const point = outlineLength ? Math[pointCalculationName](...outline) : 0;
  138. startOutline = range(column).map(() => point);
  139. }
  140. let endOutline = startOutline.slice();
  141. const columnDist = column > 1 ? alignPoses[1] - alignPoses[0] : 0;
  142. const isStretch = align === "stretch";
  143. const isStartContentAlign = isEndDirection && contentAlign === "start";
  144. let startPos = isEndDirection ? -Infinity : Infinity;
  145. if (isStartContentAlign) {
  146. // support only end direction
  147. startPos = Math.min(...endOutline);
  148. }
  149. for (let i = 0; i < itemsLength; ++i) {
  150. const item = items[isEndDirection ? i : itemsLength - 1 - i];
  151. const columnAttribute = parseInt(item.attributes.column || "1", 10);
  152. const maxColumnAttribute = parseInt(item.attributes.maxColumn || "1", 10);
  153. let contentSize = item.contentSize;
  154. let columnCount = Math.min(
  155. column,
  156. columnAttribute || Math.max(1, Math.ceil((item.inlineSize + inlineGap) / columnDist)),
  157. );
  158. const maxColumnCount = Math.min(column, Math.max(columnCount, maxColumnAttribute));
  159. let columnIndex = getColumnIndex(endOutline, columnCount, nearestCalculationName, startPos);
  160. let contentPos = getColumnPoint(endOutline, columnIndex, columnCount, pointCalculationName);
  161. if (isStartContentAlign && startPos !== contentPos) {
  162. startPos = Math.max(...endOutline);
  163. endOutline = endOutline.map(() => startPos);
  164. contentPos = startPos;
  165. columnIndex = 0;
  166. }
  167. while (columnCount < maxColumnCount) {
  168. const nextEndColumnIndex = columnIndex + columnCount;
  169. const nextColumnIndex = columnIndex - 1;
  170. if (isEndDirection && (nextEndColumnIndex >= column || endOutline[nextEndColumnIndex] > contentPos)) {
  171. break;
  172. }
  173. if (!isEndDirection && (nextColumnIndex < 0 || endOutline[nextColumnIndex] < contentPos)) {
  174. break;
  175. }
  176. if (!isEndDirection) {
  177. --columnIndex;
  178. }
  179. ++columnCount;
  180. }
  181. columnIndex = Math.max(0, columnIndex);
  182. columnCount = Math.min(column - columnIndex, columnCount);
  183. // stretch mode or data-grid-column > "1"
  184. if ((columnAttribute > 0 && columnCount > 1) || isStretch) {
  185. const nextInlineSize = (columnCount - 1) * columnDist + columnSize;
  186. if ((!this._isObserverEnabled() || !observeChildren) && item.cssInlineSize !== nextInlineSize) {
  187. item.shouldReupdate = true;
  188. }
  189. item.cssInlineSize = nextInlineSize;
  190. }
  191. if (columnSizeRatio > 0) {
  192. contentSize = item.computedInlineSize / columnSizeRatio;
  193. item.cssContentSize = contentSize;
  194. }
  195. const inlinePos = alignPoses[columnIndex];
  196. contentPos = isEndDirection ? contentPos : contentPos - contentGap - contentSize;
  197. item.cssInlinePos = inlinePos;
  198. item.cssContentPos = contentPos;
  199. const nextOutlinePoint = isEndDirection ? contentPos + contentSize + contentGap : contentPos;
  200. range(columnCount).forEach((indexOffset) => {
  201. endOutline[columnIndex + indexOffset] = nextOutlinePoint;
  202. });
  203. }
  204. // Finally, check whether startPos and min of the outline match.
  205. // If different, endOutline is updated.
  206. if (isStartContentAlign && startPos !== Math.min(...endOutline)) {
  207. startPos = Math.max(...endOutline);
  208. endOutline = endOutline.map(() => startPos);
  209. }
  210. // if end items, startOutline is low, endOutline is high
  211. // if start items, startOutline is high, endOutline is low
  212. return {
  213. start: isEndDirection ? startOutline : endOutline,
  214. end: isEndDirection ? endOutline : startOutline,
  215. };
  216. }
  217. public getComputedOutlineSize(items = this.items) {
  218. const { align } = this.options;
  219. const inlineGap = this.getInlineGap();
  220. const containerInlineSize = this.getContainerInlineSize();
  221. const columnSizeOption = this.columnSize || this.outlineSize;
  222. const columnOption = this.column || this.outlineLength;
  223. let column = columnOption || 1;
  224. let columnSize = 0;
  225. if (align === "stretch") {
  226. if (!columnOption) {
  227. const maxStretchColumnSize = this.maxStretchColumnSize || Infinity;
  228. column = Math.max(1, Math.ceil((containerInlineSize + inlineGap) / (maxStretchColumnSize + inlineGap)));
  229. }
  230. columnSize = (containerInlineSize + inlineGap) / (column || 1) - inlineGap;
  231. } else if (columnSizeOption) {
  232. columnSize = columnSizeOption;
  233. } else if (items.length) {
  234. let checkedItem = items[0];
  235. for (const item of items) {
  236. const attributes = item.attributes;
  237. const columnAttribute = parseInt(attributes.column || "1", 10);
  238. const maxColumnAttribute = parseInt(attributes.maxColumn || "1", 10);
  239. if (
  240. item.updateState !== UPDATE_STATE.UPDATED
  241. || !item.inlineSize
  242. || columnAttribute !== 1
  243. || maxColumnAttribute !== 1
  244. ) {
  245. continue;
  246. }
  247. checkedItem = item;
  248. break;
  249. }
  250. const inlineSize = checkedItem.inlineSize || 0;
  251. columnSize = inlineSize;
  252. } else {
  253. columnSize = containerInlineSize;
  254. }
  255. return columnSize || 0;
  256. }
  257. public getComputedOutlineLength(items = this.items) {
  258. const inlineGap = this.getInlineGap();
  259. const columnOption = this.column || this.outlineLength;
  260. const columnCalculationThreshold = this.columnCalculationThreshold;
  261. let column = 1;
  262. if (columnOption) {
  263. column = columnOption;
  264. } else {
  265. const columnSize = this.getComputedOutlineSize(items);
  266. column = Math.min(
  267. items.length,
  268. Math.max(
  269. 1,
  270. Math.floor(
  271. (this.getContainerInlineSize() + inlineGap) /
  272. (columnSize - columnCalculationThreshold + inlineGap)
  273. )
  274. )
  275. );
  276. }
  277. return column;
  278. }
  279. private _getAlignPoses(column: number, columnSize: number) {
  280. const { align } = this.options;
  281. const inlineGap = this.getInlineGap();
  282. const containerSize = this.getContainerInlineSize();
  283. const indexes = range(column);
  284. let offset = 0;
  285. let dist = 0;
  286. if (align === "justify" || align === "stretch") {
  287. const countDist = column - 1;
  288. dist = countDist ? Math.max((containerSize - columnSize) / countDist, columnSize + inlineGap) : 0;
  289. offset = Math.min(0, containerSize / 2 - (countDist * dist + columnSize) / 2);
  290. } else {
  291. dist = columnSize + inlineGap;
  292. const totalColumnSize = (column - 1) * dist + columnSize;
  293. if (align === "center") {
  294. offset = (containerSize - totalColumnSize) / 2;
  295. } else if (align === "end") {
  296. offset = containerSize - totalColumnSize;
  297. }
  298. }
  299. return indexes.map((i) => {
  300. return offset + i * dist;
  301. });
  302. }
  303. }
  304. export interface MasonryGrid extends Properties<typeof MasonryGrid> {
  305. }
  306. /**
  307. * Align of the position of the items. If you want to use `stretch`, be sure to set `column` or `columnSize` option. ("start", "center", "end", "justify", "stretch")
  308. * @ko 아이템들의 위치의 정렬. `stretch`를 사용하고 싶다면 `column` 또는 `columnSize` 옵션을 설정해라. ("start", "center", "end", "justify", "stretch")
  309. * @name Grid.MasonryGrid#align
  310. * @type {$ts:Grid.MasonryGrid.MasonryGridOptions["align"]}
  311. * @default "justify"
  312. * @example
  313. * ```js
  314. * import { MasonryGrid } from "@egjs/grid";
  315. *
  316. * const grid = new MasonryGrid(container, {
  317. * align: "start",
  318. * });
  319. *
  320. * grid.align = "justify";
  321. * ```
  322. */
  323. /**
  324. * The number of columns. If the number of columns is 0, it is automatically calculated according to the size of the container. Can be used instead of outlineLength.
  325. * @ko 열의 개수. 열의 개수가 0이라면, 컨테이너의 사이즈에 의해 계산이 된다. outlineLength 대신 사용할 수 있다.
  326. * @name Grid.MasonryGrid#column
  327. * @type {$ts:Grid.MasonryGrid.MasonryGridOptions["column"]}
  328. * @default 0
  329. * @example
  330. * ```js
  331. * import { MasonryGrid } from "@egjs/grid";
  332. *
  333. * const grid = new MasonryGrid(container, {
  334. * column: 0,
  335. * });
  336. *
  337. * grid.column = 4;
  338. * ```
  339. */
  340. /**
  341. * The size of the columns. If it is 0, it is calculated as the size of the first item in items. Can be used instead of outlineSize.
  342. * @ko 열의 사이즈. 만약 열의 사이즈가 0이면, 아이템들의 첫번째 아이템의 사이즈로 계산이 된다. outlineSize 대신 사용할 수 있다.
  343. * @name Grid.MasonryGrid#columnSize
  344. * @type {$ts:Grid.MasonryGrid.MasonryGridOptions["columnSize"]}
  345. * @default 0
  346. * @example
  347. * ```js
  348. * import { MasonryGrid } from "@egjs/grid";
  349. *
  350. * const grid = new MasonryGrid(container, {
  351. * columnSize: 0,
  352. * });
  353. *
  354. * grid.columnSize = 200;
  355. * ```
  356. */
  357. /**
  358. * The size ratio(inlineSize / contentSize) of the columns. 0 is not set.
  359. * @ko 열의 사이즈 비율(inlineSize / contentSize). 0은 미설정이다.
  360. * @name Grid.MasonryGrid#columnSizeRatio
  361. * @type {$ts:Grid.MasonryGrid.MasonryGridOptions["columnSizeRatio"]}
  362. * @default 0
  363. * @example
  364. * ```js
  365. * import { MasonryGrid } from "@egjs/grid";
  366. *
  367. * const grid = new MasonryGrid(container, {
  368. * columnSizeRatio: 0,
  369. * });
  370. *
  371. * grid.columnSizeRatio = 0.5;
  372. * ```
  373. */
  374. /**
  375. * If stretch is used, the column can be automatically calculated by setting the maximum size of the column that can be stretched.
  376. * @ko stretch를 사용한 경우 최대로 늘릴 수 있는 column의 사이즈를 설정하여 column을 자동 계산할 수 있다.
  377. * @name Grid.MasonryGrid#maxStretchColumnSize
  378. * @type {$ts:Grid.MasonryGrid.MasonryGridOptions["maxStretchColumnSize"]}
  379. * @default Infinity
  380. * @example
  381. * ```js
  382. * import { MasonryGrid } from "@egjs/grid";
  383. *
  384. * const grid = new MasonryGrid(container, {
  385. * align: "stretch",
  386. * maxStretchColumnSize: 0,
  387. * });
  388. *
  389. * grid.maxStretchColumnSize = 400;
  390. * ```
  391. */
comments powered by Disqus