Source: src/GridItem.ts

  1. /**
  2. * egjs-grid
  3. * Copyright (c) 2021-present NAVER Corp.
  4. * MIT license
  5. */
  6. import { DOMRect, GridRect } from "./types";
  7. import { MOUNT_STATE, RECT_NAMES, UPDATE_STATE } from "./consts";
  8. /**
  9. * @typedef
  10. * @memberof Grid.GridItem
  11. * @property - The item key. <ko>아이템 키.</ko>
  12. * @property - The item index. <ko>아이템 index.</ko>
  13. * @property - The element for the item. <ko>아이템에 있는 엘리먼트.</ko>
  14. * @property - State of whether the element has been added to the container. <ko>element가 container에 추가되었는지 상태.</ko>
  15. * @property - The update state of the element's rect. <ko>element의 rect의 업데이트 상태.</ko>
  16. * @property - Whether the element's rect was updated for the first time. <ko>처음으로 element의 rect를 업데이트 했는지 여부.</ko>
  17. * @property - Attributes set as `data-grid-` of element. <ko>element의 `data-grid-`으로 설정된 속성들.</ko>
  18. * @property - cssText of the first style when an element is added to the container. <ko>element가 container에 추가됐을 때 처음 style의 cssText.</ko>
  19. * @property - The element's rect before first rendering. <ko>처음 렌더링 하기 전 엘리먼트의 rect.</ko>
  20. * @property - The updated element's rect before rendering. <ko>렌더링 하기 전 업데이트 된 엘리먼트의 rect.</ko>
  21. * @property - The CSS rect of the item to be rendered by being applied to the Grid. <ko>Grid에 적용되어 렌더링을 하기 위한 item의 CSS rect</ko>
  22. * @property - Additional data of the item. <ko>item의 추가적인 데이터들.</ko>
  23. * @property - Grid ready data for rendering. <ko>렌더링을 하기 위한 grid의 준비 데이터.</ko>
  24. */
  25. export interface GridItemStatus {
  26. key?: string | number;
  27. index?: number;
  28. element?: HTMLElement | null;
  29. mountState?: MOUNT_STATE;
  30. updateState?: UPDATE_STATE;
  31. isFirstUpdate?: boolean;
  32. attributes?: Record<string, any>;
  33. orgCSSText?: string;
  34. orgRect?: Required<DOMRect>;
  35. rect?: Required<DOMRect>;
  36. cssRect?: DOMRect;
  37. data?: Record<string, any>;
  38. gridData?: Record<string, any>;
  39. }
  40. /**
  41. * @memberof Grid
  42. * @implements Grid.GridItem.GridItemStatus
  43. */
  44. class GridItem {
  45. /**
  46. * Whether or not it will be updated upon request.
  47. * @inner
  48. */
  49. public isUpdating = false;
  50. /**
  51. * Whether the item needs to be updated again
  52. * @inner
  53. */
  54. public shouldReupdate = false;
  55. public hasTransition = false;
  56. public transitionDuration = "";
  57. public isRestoreOrgCSSText = true;
  58. /**
  59. * @constructor
  60. * @param horizontal - Direction of the scroll movement. (true: horizontal, false: vertical) <ko>스크롤 이동 방향. (true: 가로방향, false: 세로방향)</ko>
  61. * @param itemStatus - Default status object of GridItem module. <ko>GridItem 모듈의 기본 status 객체.</ko>
  62. */
  63. constructor(
  64. protected horizontal: boolean,
  65. itemStatus: Partial<GridItemStatus> = {},
  66. ) {
  67. const element = itemStatus.element;
  68. const status: Required<GridItemStatus> = {
  69. key: "",
  70. index: 0,
  71. orgRect: { left: 0, top: 0, width: 0, height: 0 },
  72. rect: { left: 0, top: 0, width: 0, height: 0 },
  73. cssRect: {},
  74. attributes: {},
  75. data: {},
  76. isFirstUpdate: false,
  77. mountState: MOUNT_STATE.UNCHECKED,
  78. updateState: UPDATE_STATE.NEED_UPDATE,
  79. element: element || null,
  80. orgCSSText: element?.style.cssText ?? "",
  81. gridData: {},
  82. ...itemStatus,
  83. };
  84. for (const name in status) {
  85. this[name] = status[name];
  86. }
  87. }
  88. /**
  89. * The size in inline direction before first rendering. "width" if horizontal is false, "height" otherwise.
  90. * @ko 첫 렌더링 되기 전의 inline 방향의 사이즈. horizontal이 false면 "width", 아니면 "height".
  91. * @member Grid.GridItem#orgInlineSize
  92. */
  93. public get orgInlineSize() {
  94. const name = this._names.inlineSize;
  95. return this.orgRect[name] || this.rect[name];
  96. }
  97. /**
  98. * The size in content direction before first rendering. "height" if horizontal is false, "width" otherwise.
  99. * @ko 첫 렌더링 되기 전의 content 방향의 사이즈. horizontal이 false면 "height", 아니면 "width".
  100. * @member Grid.GridItem#orgContentSize
  101. */
  102. public get orgContentSize() {
  103. const name = this._names.contentSize;
  104. return this.orgRect[name] || this.rect[name];
  105. }
  106. /**
  107. * The size in inline direction. "width" if horizontal is false, "height" otherwise.
  108. * @ko inline 방향의 사이즈. horizontal이 false면 "width", 아니면 "height".
  109. * @member Grid.GridItem#inlineSize
  110. */
  111. public get inlineSize() {
  112. return this.rect[this._names.inlineSize];
  113. }
  114. /**
  115. * The size in content direction. "height" if horizontal is false, "width" otherwise.
  116. * @ko content 방향의 사이즈. horizontal이 false면 "height", 아니면 "width".
  117. * @member Grid.GridItem#contentSize
  118. */
  119. public get contentSize() {
  120. return this.rect[this._names.contentSize];
  121. }
  122. /**
  123. * The CSS size in inline direction applied to the Grid. "width" if horizontal is false, "height" otherwise.
  124. * @ko Grid에 적용된 inline 방향의 CSS 사이즈. horizontal이 false면 "width", 아니면 "height".
  125. * @member Grid.GridItem#cssInlineSize
  126. */
  127. public get cssInlineSize() {
  128. return this.cssRect[this._names.inlineSize];
  129. }
  130. /**
  131. * The CSS size in content direction applied to the Grid. "height" if horizontal is false, "width" otherwise.
  132. * @ko Grid에 적용된 content 방향의 CSS 사이즈. horizontal이 false면 "height", 아니면 "width".
  133. * @member Grid.GridItem#cssContentSize
  134. */
  135. public get cssContentSize() {
  136. return this.cssRect[this._names.contentSize];
  137. }
  138. /**
  139. * The CSS pos in inline direction applied to the Grid. "left" if horizontal is false, "top" otherwise.
  140. * @ko Grid에 적용된 inline 방향의 CSS 포지션. horizontal이 false면 "left", 아니면 "top".
  141. * @member Grid.GridItem#cssInlinePos
  142. */
  143. public get cssInlinePos() {
  144. return this.cssRect[this._names.inlinePos];
  145. }
  146. /**
  147. * The CSS pos in content direction applied to the Grid. "top" if horizontal is false, "left" otherwise.
  148. * @ko Grid에 적용된 content 방향의 CSS 포지션. horizontal이 false면 "top", 아니면 "left".
  149. * @member Grid.GridItem#cssContentPos
  150. */
  151. public get cssContentPos() {
  152. return this.cssRect[this._names.contentPos];
  153. }
  154. public set cssInlinePos(inlinePos: number | undefined) {
  155. this.cssRect[this._names.inlinePos] = inlinePos;
  156. }
  157. public set cssContentPos(contentPos: number | undefined) {
  158. this.cssRect[this._names.contentPos] = contentPos;
  159. }
  160. public set cssInlineSize(inlineSize: number | undefined) {
  161. this.cssRect[this._names.inlineSize] = inlineSize;
  162. }
  163. public set cssContentSize(contentSize: number | undefined) {
  164. this.cssRect[this._names.contentSize] = contentSize;
  165. }
  166. /**
  167. * Calculated size in the direction of the inline applied to the grid. "width" if horizontal is false, "height" otherwise.
  168. * @ko Grid에 적용된 inline 방향의 계산된 사이즈. horizontal이 false면 "width", 아니면 "height".
  169. * @member Grid.GridItem#computedInlineSize
  170. */
  171. public get computedInlineSize() {
  172. const name = this._names.inlineSize;
  173. return this.cssRect[name] || this.rect[name] || this.orgRect[name];
  174. }
  175. /**
  176. * Calculated size in the direction of the content applied to the grid. "height" if horizontal is false, "width" otherwise.
  177. * @ko Grid에 적용된 content 방향의 계산된 사이즈. horizontal이 false면 "height", 아니면 "width".
  178. * @member Grid.GridItem#computedContentSize
  179. */
  180. public get computedContentSize() {
  181. const name = this._names.contentSize;
  182. return this.cssRect[name] || this.rect[name] || this.orgRect[name];
  183. }
  184. /**
  185. * Calculated position in the direction of the inline applied to the grid. "left" if horizontal is false, "top" otherwise.
  186. * @ko Grid에 적용된 content 방향의 계산된 포지션. horizontal이 false면 "left", 아니면 "top".
  187. * @member Grid.GridItem#computedInlinePos
  188. */
  189. public get computedInlinePos() {
  190. const name = this._names.inlinePos;
  191. return this.cssRect[name] ?? this.rect[name];
  192. }
  193. /**
  194. * Calculated position in the direction of the content applied to the grid. "top" if horizontal is false, "left" otherwise.
  195. * @ko Grid에 적용된 content 방향의 계산된 포지션. horizontal이 false면 "top", 아니면 "left".
  196. * @member Grid.GridItem#computedContentPos
  197. */
  198. public get computedContentPos() {
  199. const name = this._names.contentPos;
  200. return this.cssRect[name] ?? this.rect[name];
  201. }
  202. /**
  203. * Set CSS Rect through GridRect.
  204. * @ko GridRect을 통해 CSS Rect를 설정한다.
  205. * @param - The style for setting CSS rect. <ko>CSS rect를 설정하기 위한 스타일.</ko>
  206. */
  207. public setCSSGridRect(gridRect: GridRect) {
  208. const names = RECT_NAMES[this.horizontal ? "horizontal" : "vertical"];
  209. const rect: DOMRect = {};
  210. for (const name in gridRect) {
  211. rect[names[name]] = gridRect[name];
  212. }
  213. this.cssRect = rect;
  214. }
  215. /**
  216. * Returns the status of the item.
  217. * @ko 아이템의 상태를 반환한다.
  218. */
  219. public getStatus(): Required<GridItemStatus> {
  220. return {
  221. index: this.index,
  222. mountState: this.mountState,
  223. updateState: this.updateState,
  224. attributes: this.attributes,
  225. orgCSSText: this.orgCSSText,
  226. isFirstUpdate: this.isFirstUpdate,
  227. element: null,
  228. key: this.key,
  229. orgRect: this.orgRect,
  230. rect: this.rect,
  231. cssRect: this.cssRect,
  232. gridData: this.gridData,
  233. data: this.data,
  234. };
  235. }
  236. /**
  237. * Returns minimized status of the item.
  238. * @ko 아이템의 간소화된 상태를 반환한다.
  239. */
  240. public getMinimizedStatus(): Partial<GridItemStatus> {
  241. const status: Partial<GridItemStatus> = {
  242. orgRect: this.orgRect,
  243. rect: this.rect,
  244. cssRect: this.cssRect,
  245. attributes: this.attributes,
  246. gridData: this.gridData,
  247. };
  248. const {
  249. key,
  250. mountState,
  251. updateState,
  252. isFirstUpdate,
  253. orgCSSText,
  254. } = this;
  255. if (typeof key !== "undefined") {
  256. status.key = key;
  257. }
  258. if (mountState !== MOUNT_STATE.UNCHECKED) {
  259. status.mountState = mountState;
  260. }
  261. if (updateState !== UPDATE_STATE.NEED_UPDATE) {
  262. status.updateState = updateState;
  263. }
  264. if (isFirstUpdate) {
  265. status.isFirstUpdate = true;
  266. }
  267. if (orgCSSText) {
  268. status.orgCSSText = orgCSSText;
  269. }
  270. return status;
  271. }
  272. private get _names() {
  273. return this.horizontal ? RECT_NAMES.horizontal : RECT_NAMES.vertical;
  274. }
  275. }
  276. interface GridItem extends Required<GridItemStatus> {
  277. }
  278. export { GridItem };
comments powered by Disqus