Source: src/Grid.ts

  1. /**
  2. * egjs-grid
  3. * Copyright (c) 2021-present NAVER Corp.
  4. * MIT license
  5. */
  6. import Component from "@egjs/component";
  7. import { DEFAULT_GRID_OPTIONS, GRID_PROPERTY_TYPES, MOUNT_STATE, UPDATE_STATE } from "./consts";
  8. import { ContainerManager } from "./ContainerManager";
  9. import {
  10. DestroyOptions, GridEvents, GridOptions,
  11. GridOutlines, GridStatus, Properties, RenderOptions,
  12. OnRenderComplete,
  13. } from "./types";
  14. import ImReady from "@egjs/imready";
  15. import { ItemRenderer } from "./ItemRenderer";
  16. import { GetterSetter, getMountedElements, isNumber, isString, getUpdatedItems } from "./utils";
  17. import { diff } from "@egjs/children-differ";
  18. import { GridItem } from "./GridItem";
  19. import { ResizeWatcherResizeEvent } from "./ResizeWatcher";
  20. /**
  21. * @extends eg.Component
  22. */
  23. @GetterSetter
  24. abstract class Grid<Options extends GridOptions = GridOptions> extends Component<GridEvents> {
  25. public static defaultOptions: Required<GridOptions> = DEFAULT_GRID_OPTIONS;
  26. public static propertyTypes = GRID_PROPERTY_TYPES;
  27. public options: Required<Options>;
  28. protected containerElement: HTMLElement;
  29. protected containerManager: ContainerManager;
  30. protected itemRenderer!: ItemRenderer;
  31. protected items: GridItem[] = [];
  32. protected outlines: GridOutlines = {
  33. start: [],
  34. end: [],
  35. };
  36. private _renderTimer = 0;
  37. private _im: ImReady;
  38. /**
  39. * Apply the CSS rect of items to fit the Grid and calculate the outline.
  40. * @ko Grid에 맞게 아이템들의 CSS rect를 적용하고 outline을 계산한다.
  41. * @abstract
  42. * @method Grid#applyGrid
  43. * @param {"start" | "end"} direcion - The direction to apply the Grid. ("end": start to end, "start": end to start) <ko>Grid를 적용할 방향. ("end": 시작에서 끝 방향, "start": 끝에서 시작 방향)</ko>
  44. * @param {number[]} outline - The start outline to apply the Grid. <ko>Grid를 적용할 시작 outline.</ko>
  45. */
  46. public abstract applyGrid(items: GridItem[], direction: "start" | "end", outline: number[]): GridOutlines;
  47. /**
  48. * @param - A base element for a module <ko>모듈을 적용할 기준 엘리먼트</ko>
  49. * @param - The option object of the Grid module <ko>Grid 모듈의 옵션 객체</ko>
  50. */
  51. constructor(containerElement: HTMLElement | string, options: Partial<Options> = {}) {
  52. super();
  53. this.options = {
  54. ...((this.constructor as typeof Grid)
  55. .defaultOptions as Required<Options>),
  56. ...options,
  57. };
  58. this.containerElement = isString(containerElement)
  59. ? document.querySelector<HTMLElement>(containerElement)!
  60. : containerElement;
  61. const {
  62. isEqualSize,
  63. isConstantSize,
  64. useTransform,
  65. horizontal,
  66. percentage,
  67. externalContainerManager,
  68. externalItemRenderer,
  69. resizeDebounce,
  70. maxResizeDebounce,
  71. autoResize,
  72. useRoundedSize,
  73. useResizeObserver,
  74. } = this.options;
  75. // TODO: 테스트용 설정
  76. this.containerManager = externalContainerManager!
  77. || new ContainerManager(this.containerElement, {
  78. horizontal,
  79. resizeDebounce,
  80. maxResizeDebounce,
  81. autoResize,
  82. useResizeObserver,
  83. }).on("resize", this._onResize);
  84. this.itemRenderer = externalItemRenderer!
  85. || new ItemRenderer({
  86. useTransform,
  87. isEqualSize,
  88. isConstantSize,
  89. percentage,
  90. useRoundedSize,
  91. });
  92. this._init();
  93. }
  94. /**
  95. * Return Container Element.
  96. * @ko 컨테이너 엘리먼트를 반환한다.
  97. */
  98. public getContainerElement(): HTMLElement {
  99. return this.containerElement;
  100. }
  101. /**
  102. * Return items.
  103. * @ko 아이템들을 반환한다.
  104. */
  105. public getItems(): GridItem[] {
  106. return this.items;
  107. }
  108. /**
  109. * Returns the children of the container element.
  110. * @ko 컨테이너 엘리먼트의 children을 반환한다.
  111. */
  112. public getChildren(): HTMLElement[] {
  113. return [].slice.call(this.containerElement.children);
  114. }
  115. /**
  116. * Set items.
  117. * @ko 아이템들을 설정한다.
  118. * @param items - The items to set. <ko>설정할 아이템들</ko>
  119. */
  120. public setItems(items: GridItem[]): this {
  121. items.forEach((item, i) => {
  122. item.index = i;
  123. });
  124. const options = this.options;
  125. if (options.useResizeObserver && options.observeChildren) {
  126. const containerManager = this.containerManager;
  127. containerManager.unobserveChildren(getMountedElements(this.items));
  128. containerManager.observeChildren(getMountedElements(items));
  129. }
  130. this.items = items;
  131. return this;
  132. }
  133. /**
  134. * Gets the container's inline size. ("width" if horizontal is false, otherwise "height")
  135. * @ko container의 inline 사이즈를 가져온다. (horizontal이 false면 "width", 아니면 "height")
  136. */
  137. public getContainerInlineSize(): number {
  138. return this.containerManager.getInlineSize()!;
  139. }
  140. /**
  141. * Returns the outlines of the start and end of the Grid.
  142. * @ko Grid의 처음과 끝의 outline을 반환한다.
  143. */
  144. public getOutlines(): GridOutlines {
  145. return this.outlines;
  146. }
  147. /**
  148. * Set outlines.
  149. * @ko 아웃라인을 설정한다.
  150. * @param outlines - The outlines to set. <ko>설정할 아웃라인.</ko>
  151. */
  152. public setOutlines(outlines: GridOutlines) {
  153. this.outlines = outlines;
  154. return this;
  155. }
  156. /**
  157. * When elements change, it synchronizes and renders items.
  158. * @ko elements가 바뀐 경우 동기화를 하고 렌더링을 한다.
  159. * @param - Options for rendering. <ko>렌더링을 하기 위한 옵션.</ko>
  160. */
  161. public syncElements(options: RenderOptions = {}) {
  162. const items = this.items;
  163. const { horizontal } = this.options;
  164. const elements: HTMLElement[] = this.getChildren();
  165. const { added, maintained, changed, removed } = diff(this.items.map((item) => item.element!), elements);
  166. const nextItems: GridItem[] = [];
  167. maintained.forEach(([beforeIndex, afterIndex]) => {
  168. nextItems[afterIndex] = items[beforeIndex];
  169. });
  170. added.forEach((index) => {
  171. nextItems[index] = new GridItem(horizontal!, {
  172. element: elements[index],
  173. });
  174. });
  175. this.setItems(nextItems);
  176. if (added.length || removed.length || changed.length) {
  177. this.renderItems(options);
  178. }
  179. return this;
  180. }
  181. /**
  182. * Update the size of the items and render them.
  183. * @ko 아이템들의 사이즈를 업데이트하고 렌더링을 한다.
  184. * @param - Items to be updated. <ko>업데이트할 아이템들.</ko>
  185. * @param - Options for rendering. <ko>렌더링을 하기 위한 옵션.</ko>
  186. */
  187. public updateItems(items: GridItem[] = this.items, options: RenderOptions = {}) {
  188. const useOrgResize = options.useOrgResize;
  189. items.forEach((item) => {
  190. if (useOrgResize) {
  191. const orgRect = item.orgRect;
  192. orgRect.width = 0;
  193. orgRect.height = 0;
  194. }
  195. item.updateState = UPDATE_STATE.NEED_UPDATE;
  196. });
  197. this.checkReady(options);
  198. return this;
  199. }
  200. /**
  201. * Rearrange items to fit the grid and render them. When rearrange is complete, the `renderComplete` event is fired.
  202. * @ko grid에 맞게 아이템을 재배치하고 렌더링을 한다. 배치가 완료되면 `renderComplete` 이벤트가 발생한다.
  203. * @param - Options for rendering. <ko>렌더링을 하기 위한 옵션.</ko>
  204. * @example
  205. * ```js
  206. * import { MasonryGrid } from "@egjs/grid";
  207. * const grid = new MasonryGrid();
  208. *
  209. * grid.on("renderComplete", e => {
  210. * console.log(e);
  211. * });
  212. * grid.renderItems();
  213. * ```
  214. */
  215. public renderItems(options: RenderOptions = {}) {
  216. this._renderItems(options);
  217. return this;
  218. }
  219. /**
  220. * Returns current status such as item's position, size. The returned status can be restored with the setStatus() method.
  221. * @ko 아이템의 위치, 사이즈 등 현재 상태를 반환한다. 반환한 상태는 setStatus() 메서드로 복원할 수 있다.
  222. * @param - Whether to minimize the status of the item. (default: false) <ko>item의 status를 최소화할지 여부. (default: false)</ko>
  223. */
  224. public getStatus(minimize?: boolean): GridStatus {
  225. return {
  226. outlines: this.outlines,
  227. items: this.items.map((item) => minimize ? item.getMinimizedStatus() : item.getStatus()),
  228. containerManager: this.containerManager.getStatus(),
  229. itemRenderer: this.itemRenderer.getStatus(),
  230. };
  231. }
  232. /**
  233. * Set status of the Grid module with the status returned through a call to the getStatus() method.
  234. * @ko getStatus() 메서드에 대한 호출을 통해 반환된 상태로 Grid 모듈의 상태를 설정한다.
  235. */
  236. public setStatus(status: GridStatus) {
  237. const horizontal = this.options.horizontal;
  238. const containerManager = this.containerManager;
  239. const prevInlineSize = containerManager.getInlineSize();
  240. const children = this.getChildren();
  241. this.itemRenderer.setStatus(status.itemRenderer);
  242. containerManager.setStatus(status.containerManager);
  243. this.outlines = status.outlines;
  244. this.items = status.items.map((item, i) => new GridItem(horizontal!, {
  245. ...item,
  246. element: children[i],
  247. }));
  248. this.itemRenderer.renderItems(this.items);
  249. if (prevInlineSize !== containerManager.getInlineSize()) {
  250. this.renderItems({
  251. useResize: true,
  252. });
  253. } else {
  254. window.setTimeout(() => {
  255. this._renderComplete({
  256. direction: this.defaultDirection,
  257. mounted: this.items,
  258. updated: [],
  259. isResize: false,
  260. });
  261. });
  262. }
  263. return this;
  264. }
  265. /**
  266. * Get the inline size corresponding to outline.
  267. * @ko outline에 해당하는 inline 사이즈를 구한다.
  268. * @param items - Items to get outline size. <ko>outline 사이즈를 구하기 위한 아이템들.</ko>
  269. */
  270. // eslint-disable-next-line @typescript-eslint/no-unused-vars
  271. public getComputedOutlineSize(items: GridItem[] = this.items) {
  272. return this.options.outlineSize! || this.getContainerInlineSize();
  273. }
  274. /**
  275. * Get the length corresponding to outline.
  276. * @ko outline에 해당하는 length를 가져온다.
  277. * @param items - Items to get outline length. <ko>outline length를 구하기 위한 아이템들.</ko>
  278. */
  279. // eslint-disable-next-line @typescript-eslint/no-unused-vars
  280. public getComputedOutlineLength(items: GridItem[] = this.items): number {
  281. return this.options.outlineLength! || 1;
  282. }
  283. /**
  284. * Releases the instnace and events and returns the CSS of the container and elements.
  285. * @ko 인스턴스와 이벤트를 해제하고 컨테이너와 엘리먼트들의 CSS를 되돌린다.
  286. * @param Options for destroy. <ko>destory()를 위한 옵션</ko>
  287. */
  288. public destroy(options: DestroyOptions = {}) {
  289. const {
  290. preserveUI = this.options.preserveUIOnDestroy,
  291. } = options;
  292. this.containerManager.destroy({
  293. preserveUI,
  294. });
  295. if (!preserveUI) {
  296. this.items.forEach(({ element, orgCSSText }) => {
  297. if (element) {
  298. element.style.cssText = orgCSSText;
  299. }
  300. });
  301. }
  302. this._im?.destroy();
  303. }
  304. protected getInlineGap(): number {
  305. return this._getDirectionalGap('inline');
  306. }
  307. protected getContentGap(): number {
  308. return this._getDirectionalGap('content');
  309. }
  310. protected checkReady(options: RenderOptions = {}) {
  311. // Grid: renderItems => checkReady => readyItems => applyGrid
  312. const items = this.items;
  313. const updated = items.filter((item) => item.element?.parentNode && item.updateState !== UPDATE_STATE.UPDATED);
  314. const mounted = items.filter((item) => item.element?.parentNode && item.mountState !== MOUNT_STATE.MOUNTED);
  315. const moreUpdated: GridItem[] = [];
  316. mounted.filter((item) => {
  317. if (item.hasTransition) {
  318. return true;
  319. } else {
  320. const element = item.element!;
  321. const transitionDuration = parseFloat(getComputedStyle(element).transitionDuration);
  322. if (transitionDuration > 0) {
  323. item.hasTransition = true;
  324. item.transitionDuration = element.style.transitionDuration;
  325. return true;
  326. }
  327. }
  328. return false;
  329. }).forEach((item) => {
  330. item.element!.style.transitionDuration = "0s";
  331. });
  332. this._im?.destroy();
  333. this._im = new ImReady({
  334. prefix: this.options.attributePrefix,
  335. }).on("preReadyElement", (e) => {
  336. updated[e.index].updateState = UPDATE_STATE.WAIT_LOADING;
  337. }).on("preReady", () => {
  338. // reset org size
  339. updated.forEach((item) => {
  340. const hasOrgSize = item.orgRect.width && item.orgRect.height;
  341. const hasCSSSize = item.cssRect.width || item.cssRect.height;
  342. if (!hasOrgSize && hasCSSSize) {
  343. item.element!.style.cssText = item.orgCSSText;
  344. }
  345. });
  346. this._updateItems(updated);
  347. this.readyItems(mounted, updated, options);
  348. }).on("readyElement", (e) => {
  349. const item = updated[e.index];
  350. item.updateState = UPDATE_STATE.NEED_UPDATE;
  351. // after preReady
  352. if (e.isPreReadyOver) {
  353. if (item.isRestoreOrgCSSText) {
  354. item.element!.style.cssText = item.orgCSSText;
  355. }
  356. this._updateItems([item]);
  357. this.readyItems([], [item], options);
  358. }
  359. }).on("error", (e) => {
  360. const item = updated[e.index];
  361. /**
  362. * This event is fired when an error occurs in the content.
  363. * @ko 콘텐츠 로드에 에러가 날 때 발생하는 이벤트.
  364. * @event Grid#contentError
  365. * @param {Grid.OnContentError} e - The object of data to be sent to an event <ko>이벤트에 전달되는 데이터 객체</ko>
  366. * @example
  367. * ```js
  368. * grid.on("contentError", e => {
  369. * e.update();
  370. * });
  371. * ```
  372. */
  373. this.trigger("contentError", {
  374. element: e.element,
  375. target: e.target,
  376. item,
  377. update: () => {
  378. moreUpdated.push(item);
  379. },
  380. });
  381. }).on("ready", () => {
  382. if (moreUpdated.length) {
  383. this.updateItems(moreUpdated);
  384. }
  385. }).check(updated.map((item) => item.element!));
  386. }
  387. protected scheduleRender() {
  388. this._clearRenderTimer();
  389. this._renderTimer = window.setTimeout(() => {
  390. this.renderItems();
  391. });
  392. }
  393. protected fitOutlines(useFit = this.useFit) {
  394. const outlines = this.outlines;
  395. const startOutline = outlines.start;
  396. const endOutline = outlines.end;
  397. const outlineOffset = startOutline.length ? Math.min(...startOutline) : 0;
  398. // If the outline is less than 0, a fit occurs forcibly.
  399. if (!useFit && outlineOffset > 0) {
  400. return;
  401. }
  402. outlines.start = startOutline.map((point) => point - outlineOffset);
  403. outlines.end = endOutline.map((point) => point - outlineOffset);
  404. this.items.forEach((item) => {
  405. const contentPos = item.cssContentPos;
  406. if (!isNumber(contentPos)) {
  407. return;
  408. }
  409. item.cssContentPos = contentPos - outlineOffset;
  410. });
  411. }
  412. protected readyItems(mounted: GridItem[], updated: GridItem[], options: RenderOptions) {
  413. const prevOutlines = this.outlines;
  414. const direction = options.direction || this.options.defaultDirection!;
  415. const prevOutline = options.outline || prevOutlines[direction === "end" ? "start" : "end"];
  416. const items = this.items;
  417. let nextOutlines = {
  418. start: [...prevOutline],
  419. end: [...prevOutline],
  420. };
  421. mounted.forEach((item) => {
  422. item.mountState = MOUNT_STATE.MOUNTED;
  423. });
  424. updated.forEach((item) => {
  425. item.isUpdating = true;
  426. });
  427. if (items.length) {
  428. nextOutlines = this.applyGrid(this.items, direction, prevOutline);
  429. }
  430. updated.forEach((item) => {
  431. item.isUpdating = false;
  432. });
  433. this.setOutlines(nextOutlines);
  434. this.fitOutlines();
  435. this.itemRenderer.renderItems(this.items);
  436. this._refreshContainerContentSize();
  437. const transitionMounted = mounted.filter((item) => item.hasTransition);
  438. if (transitionMounted.length) {
  439. this.containerManager.resize();
  440. transitionMounted.forEach((item) => {
  441. const element = item.element!;
  442. element.style.transitionDuration = item.transitionDuration;
  443. });
  444. }
  445. this._renderComplete({
  446. direction,
  447. mounted,
  448. updated,
  449. isResize: !!options.useResize,
  450. });
  451. const shouldReupdateItems = updated.filter((item) => item.shouldReupdate);
  452. if (shouldReupdateItems.length) {
  453. this.updateItems(shouldReupdateItems);
  454. }
  455. }
  456. protected _isObserverEnabled() {
  457. return this.containerManager.isObserverEnabled();
  458. }
  459. protected _updateItems(items: GridItem[]) {
  460. this.itemRenderer.updateEqualSizeItems(items, this.getItems());
  461. }
  462. private _getDirectionalGap(direction: 'inline' | 'content'): number {
  463. const horizontal = this.options.horizontal!;
  464. const gap = this.options.gap!;
  465. if (typeof gap === 'number') return gap;
  466. const isVerticalGap = horizontal && direction === 'inline' || !horizontal && direction === 'content';
  467. return (isVerticalGap ? (gap as any).vertical : (gap as any).horizontal) ?? (DEFAULT_GRID_OPTIONS["gap"] as number);
  468. }
  469. private _renderComplete(e: OnRenderComplete) {
  470. /**
  471. * This event is fired when the Grid has completed rendering.
  472. * @ko Grid가 렌더링이 완료됐을 때 발생하는 이벤트이다.
  473. * @event Grid#renderComplete
  474. * @param {Grid.OnRenderComplete} e - The object of data to be sent to an event <ko>이벤트에 전달되는 데이터 객체</ko>
  475. * @example
  476. * ```js
  477. * grid.on("renderComplete", e => {
  478. * console.log(e.mounted, e.updated, e.useResize);
  479. * });
  480. * ```
  481. */
  482. this.trigger("renderComplete", e);
  483. }
  484. private _clearRenderTimer() {
  485. clearTimeout(this._renderTimer);
  486. this._renderTimer = 0;
  487. }
  488. private _refreshContainerContentSize() {
  489. const {
  490. start: startOutline,
  491. end: endOutline,
  492. } = this.outlines;
  493. const contentGap = this.getContentGap();
  494. const endPoint = endOutline.length ? Math.max(...endOutline) : 0;
  495. const startPoint = startOutline.length ? Math.max(...startOutline) : 0;
  496. const contentSize = Math.max(startPoint, endPoint - contentGap);
  497. this.containerManager.setContentSize(contentSize);
  498. }
  499. private _resizeContainer() {
  500. this.containerManager.resize();
  501. this.itemRenderer.setContainerRect(this.containerManager.getRect());
  502. }
  503. private _onResize = (e: ResizeWatcherResizeEvent) => {
  504. if (e.isResizeContainer) {
  505. this._renderItems({
  506. useResize: true,
  507. }, true);
  508. } else {
  509. const updatedItems = getUpdatedItems(this.items, e.childEntries);
  510. if (updatedItems.length > 0) {
  511. this.updateItems(updatedItems);
  512. }
  513. }
  514. }
  515. private _init() {
  516. this._resizeContainer();
  517. }
  518. private _renderItems(options: RenderOptions = {}, isTrusted?: boolean) {
  519. this._clearRenderTimer();
  520. const isResize = options.useResize || options.useOrgResize;
  521. if (isResize && !isTrusted) {
  522. // Resize container
  523. // isTrusted has already been resized internally.
  524. this._resizeContainer();
  525. this.itemRenderer.resize();
  526. }
  527. if (!this.getItems().length && this.getChildren().length) {
  528. this.syncElements(options);
  529. } else if (isResize) {
  530. // Update all items
  531. this.updateItems(this.items, options);
  532. } else {
  533. // Update only items that need to be updated.
  534. this.checkReady(options);
  535. }
  536. }
  537. }
  538. interface Grid extends Properties<typeof Grid> { }
  539. export default Grid;
  540. /**
  541. * Gap used to create space around items.
  542. * @ko 아이템들 사이의 공간.
  543. * @name Grid#gap
  544. * @type {$ts:Grid.GridOptions["gap"]}
  545. * @default 0
  546. * @example
  547. * ```js
  548. * import { MasonryGrid } from "@egjs/grid";
  549. *
  550. * const grid = new MasonryGrid(container, {
  551. * gap: 0,
  552. * });
  553. *
  554. * grid.gap = 5;
  555. * ```
  556. */
  557. /**
  558. * The default direction value when direction is not set in the render option.
  559. * @ko render옵션에서 direction을 미설정시의 기본 방향값.
  560. * @name Grid#defaultDirection
  561. * @type {$ts:Grid.GridOptions["defaultDirection"]}
  562. * @default "end"
  563. * @example
  564. * ```js
  565. * import { MasonryGrid } from "@egjs/grid";
  566. *
  567. * const grid = new MasonryGrid(container, {
  568. * defaultDirection: "end",
  569. * });
  570. *
  571. * grid.defaultDirection = "start";
  572. * ```
  573. */
  574. /**
  575. * Whether to move the outline to 0 when the top is empty when rendering. However, if it overflows above the top, the outline is forced to 0. (default: true)
  576. * @ko 렌더링시 상단이 비어있을 때 아웃라인을 0으로 이동시킬지 여부. 하지만 상단보다 넘치는 경우 아웃라인을 0으로 강제 이동한다. (default: true)
  577. * @name Grid#useFit
  578. * @type {$ts:Grid.GridOptions["useFit"]}
  579. * @default true
  580. * @example
  581. * ```js
  582. * import { MasonryGrid } from "@egjs/grid";
  583. *
  584. * const grid = new MasonryGrid(container, {
  585. * useFit: true,
  586. * });
  587. *
  588. * grid.useFit = false;
  589. * ```
  590. */
  591. /**
  592. * Whether to preserve the UI of the existing container or item when destroying.
  593. * @ko destroy 시 기존 컨테이너, 아이템의 UI를 보존할지 여부.
  594. * @name Grid#preserveUIOnDestroy
  595. * @type {$ts:Grid.GridOptions["preserveUIOnDestroy"]}
  596. * @default false
  597. * @example
  598. * ```js
  599. * import { MasonryGrid } from "@egjs/grid";
  600. *
  601. * const grid = new MasonryGrid(container, {
  602. * preserveUIOnDestroy: false,
  603. * });
  604. *
  605. * grid.preserveUIOnDestroy = true;
  606. * ```
  607. */
  608. /**
  609. * The number of outlines. If the number of outlines is 0, it is calculated according to the type of grid.
  610. * @ko outline의 개수. 아웃라인의 개수가 0이라면 grid의 종류에 따라 계산이 된다.
  611. * @name Grid#outlineLength
  612. * @type {$ts:Grid.GridOptions["outlineLength"]}
  613. * @default 0
  614. * @example
  615. * ```js
  616. * import { MasonryGrid } from "@egjs/grid";
  617. *
  618. * const grid = new MasonryGrid(container, {
  619. * outlineLength: 0,
  620. * outlineSize: 0,
  621. * });
  622. *
  623. * grid.outlineLength = 3;
  624. * ```
  625. */
  626. /**
  627. * The size of the outline. If the outline size is 0, it is calculated according to the grid type.
  628. * @ko outline의 사이즈. 만약 outline의 사이즈가 0이면, grid의 종류에 따라 계산이 된다.
  629. * @name Grid#outlineSize
  630. * @type {$ts:Grid.GridOptions["outlineSize"]}
  631. * @default 0
  632. * @example
  633. * ```js
  634. * import { MasonryGrid } from "@egjs/grid";
  635. *
  636. * const grid = new MasonryGrid(container, {
  637. * outlineLength: 0,
  638. * outlineSize: 0,
  639. * });
  640. *
  641. * grid.outlineSize = 300;
  642. * ```
  643. */
comments powered by Disqus