@ai-table/grid 0.0.11 → 0.0.13

Sign up to get free protection for your applications and to get access to all the features.
Files changed (324) hide show
  1. package/angular-konva/components/container.component.d.ts +7 -0
  2. package/angular-konva/components/container.component.d.ts.map +1 -0
  3. package/angular-konva/components/container.token.d.ts +3 -0
  4. package/angular-konva/components/container.token.d.ts.map +1 -0
  5. package/angular-konva/components/index.d.ts +4 -0
  6. package/angular-konva/components/index.d.ts.map +1 -0
  7. package/angular-konva/components/shape.component.d.ts +61 -0
  8. package/angular-konva/components/shape.component.d.ts.map +1 -0
  9. package/angular-konva/components/stage.component.d.ts +40 -0
  10. package/angular-konva/components/stage.component.d.ts.map +1 -0
  11. package/angular-konva/index.d.ts +5 -0
  12. package/angular-konva/index.d.ts.map +1 -0
  13. package/angular-konva/interfaces/component.d.ts +49 -0
  14. package/angular-konva/interfaces/component.d.ts.map +1 -0
  15. package/angular-konva/interfaces/config.d.ts +26 -0
  16. package/angular-konva/interfaces/config.d.ts.map +1 -0
  17. package/angular-konva/interfaces/event-object.d.ts +7 -0
  18. package/angular-konva/interfaces/event-object.d.ts.map +1 -0
  19. package/angular-konva/interfaces/index.d.ts +5 -0
  20. package/angular-konva/interfaces/index.d.ts.map +1 -0
  21. package/angular-konva/interfaces/shape.d.ts +42 -0
  22. package/angular-konva/interfaces/shape.d.ts.map +1 -0
  23. package/angular-konva/utils/apply-node-props.d.ts +4 -0
  24. package/angular-konva/utils/apply-node-props.d.ts.map +1 -0
  25. package/angular-konva/utils/common.d.ts +8 -0
  26. package/angular-konva/utils/common.d.ts.map +1 -0
  27. package/angular-konva/utils/index.d.ts +5 -0
  28. package/angular-konva/utils/index.d.ts.map +1 -0
  29. package/angular-konva/utils/types.d.ts +4 -0
  30. package/angular-konva/utils/types.d.ts.map +1 -0
  31. package/angular-konva/utils/update-picture.d.ts +4 -0
  32. package/angular-konva/utils/update-picture.d.ts.map +1 -0
  33. package/components/cell-editors/abstract-cell-editor.component.d.ts +7 -5
  34. package/components/cell-editors/abstract-cell-editor.component.d.ts.map +1 -1
  35. package/components/cell-editors/cell-editor.scss +38 -25
  36. package/components/cell-editors/cell-editor.variables.scss +5 -0
  37. package/components/cell-editors/date/date-editor.component.d.ts.map +1 -1
  38. package/components/cell-editors/link/edit-link/edit-link.component.d.ts +33 -0
  39. package/components/cell-editors/link/edit-link/edit-link.component.d.ts.map +1 -0
  40. package/components/cell-editors/link/link-editor.component.d.ts +36 -0
  41. package/components/cell-editors/link/link-editor.component.d.ts.map +1 -0
  42. package/components/cell-editors/link/link-editor.component.scss +21 -0
  43. package/components/cell-editors/progress/progress-editor.component.d.ts +0 -1
  44. package/components/cell-editors/progress/progress-editor.component.d.ts.map +1 -1
  45. package/components/cell-editors/progress/progress-editor.component.scss +12 -4
  46. package/components/cell-editors/rating/rating-editor.component.d.ts +0 -1
  47. package/components/cell-editors/rating/rating-editor.component.d.ts.map +1 -1
  48. package/components/cell-editors/rating/rating-editor.component.scss +10 -0
  49. package/components/cell-editors/select/select-editor.component.d.ts +1 -3
  50. package/components/cell-editors/select/select-editor.component.d.ts.map +1 -1
  51. package/components/cell-editors/text/text-editor.component.d.ts +0 -1
  52. package/components/cell-editors/text/text-editor.component.d.ts.map +1 -1
  53. package/components/cell-views/select/option.component.d.ts.map +1 -1
  54. package/components/cell-views/select/option.scss +10 -0
  55. package/components/field-menu/field-menu.component.d.ts +8 -3
  56. package/components/field-menu/field-menu.component.d.ts.map +1 -1
  57. package/components/field-property-editor/field-property-editor.component.d.ts +4 -2
  58. package/components/field-property-editor/field-property-editor.component.d.ts.map +1 -1
  59. package/components/index.d.ts +3 -1
  60. package/components/index.d.ts.map +1 -1
  61. package/constants/colors.d.ts +18 -0
  62. package/constants/colors.d.ts.map +1 -0
  63. package/constants/grid.d.ts +17 -1
  64. package/constants/grid.d.ts.map +1 -1
  65. package/constants/icon.d.ts +19 -0
  66. package/constants/icon.d.ts.map +1 -0
  67. package/constants/index.d.ts +5 -1
  68. package/constants/index.d.ts.map +1 -1
  69. package/constants/table.d.ts +60 -0
  70. package/constants/table.d.ts.map +1 -0
  71. package/constants/text.d.ts +22 -0
  72. package/constants/text.d.ts.map +1 -0
  73. package/core/context.d.ts +14 -0
  74. package/core/context.d.ts.map +1 -0
  75. package/core/coordinate.d.ts +96 -0
  76. package/core/coordinate.d.ts.map +1 -0
  77. package/core/index.d.ts +3 -2
  78. package/core/index.d.ts.map +1 -1
  79. package/core/types/ai-table.d.ts +56 -0
  80. package/core/types/ai-table.d.ts.map +1 -0
  81. package/core/types/core.d.ts +29 -23
  82. package/core/types/core.d.ts.map +1 -1
  83. package/core/types/index.d.ts +1 -1
  84. package/core/types/index.d.ts.map +1 -1
  85. package/core/utils/common.d.ts +1 -2
  86. package/core/utils/common.d.ts.map +1 -1
  87. package/core/utils/field.d.ts +1 -1
  88. package/core/utils/field.d.ts.map +1 -1
  89. package/core/utils/index.d.ts +0 -1
  90. package/core/utils/index.d.ts.map +1 -1
  91. package/core/utils/queries.d.ts +5 -5
  92. package/core/utils/queries.d.ts.map +1 -1
  93. package/dom-grid.component.d.ts +11 -0
  94. package/dom-grid.component.d.ts.map +1 -0
  95. package/esm2022/angular-konva/components/container.component.mjs +29 -0
  96. package/esm2022/angular-konva/components/container.token.mjs +3 -0
  97. package/esm2022/angular-konva/components/index.mjs +4 -0
  98. package/esm2022/angular-konva/components/shape.component.mjs +142 -0
  99. package/esm2022/angular-konva/components/stage.component.mjs +123 -0
  100. package/esm2022/angular-konva/index.mjs +5 -0
  101. package/esm2022/angular-konva/interfaces/component.mjs +4 -0
  102. package/esm2022/angular-konva/interfaces/config.mjs +2 -0
  103. package/esm2022/angular-konva/interfaces/event-object.mjs +2 -0
  104. package/esm2022/angular-konva/interfaces/index.mjs +5 -0
  105. package/esm2022/angular-konva/interfaces/shape.mjs +42 -0
  106. package/esm2022/angular-konva/utils/apply-node-props.mjs +67 -0
  107. package/esm2022/angular-konva/utils/common.mjs +48 -0
  108. package/esm2022/angular-konva/utils/index.mjs +5 -0
  109. package/esm2022/angular-konva/utils/types.mjs +2 -0
  110. package/esm2022/angular-konva/utils/update-picture.mjs +7 -0
  111. package/esm2022/components/cell-editors/abstract-cell-editor.component.mjs +12 -7
  112. package/esm2022/components/cell-editors/date/date-editor.component.mjs +6 -3
  113. package/esm2022/components/cell-editors/link/edit-link/edit-link.component.mjs +66 -0
  114. package/esm2022/components/cell-editors/link/link-editor.component.mjs +117 -0
  115. package/esm2022/components/cell-editors/number/number-editor.component.mjs +2 -2
  116. package/esm2022/components/cell-editors/progress/progress-editor.component.mjs +3 -6
  117. package/esm2022/components/cell-editors/rating/rating-editor.component.mjs +4 -7
  118. package/esm2022/components/cell-editors/select/select-editor.component.mjs +11 -16
  119. package/esm2022/components/cell-editors/text/text-editor.component.mjs +10 -11
  120. package/esm2022/components/cell-views/select/option.component.mjs +6 -5
  121. package/esm2022/components/field-menu/field-menu.component.mjs +11 -8
  122. package/esm2022/components/field-property-editor/field-property-editor.component.mjs +17 -12
  123. package/esm2022/components/index.mjs +4 -2
  124. package/esm2022/constants/colors.mjs +18 -0
  125. package/esm2022/constants/editor.mjs +2 -3
  126. package/esm2022/constants/grid.mjs +22 -4
  127. package/esm2022/constants/icon.mjs +26 -0
  128. package/esm2022/constants/index.mjs +6 -2
  129. package/esm2022/constants/table.mjs +62 -0
  130. package/esm2022/constants/text.mjs +23 -0
  131. package/esm2022/core/context.mjs +25 -0
  132. package/esm2022/core/coordinate.mjs +222 -0
  133. package/esm2022/core/index.mjs +4 -3
  134. package/esm2022/core/types/ai-table.mjs +44 -0
  135. package/esm2022/core/types/core.mjs +6 -1
  136. package/esm2022/core/types/index.mjs +2 -2
  137. package/esm2022/core/utils/common.mjs +3 -20
  138. package/esm2022/core/utils/field.mjs +9 -5
  139. package/esm2022/core/utils/index.mjs +1 -2
  140. package/esm2022/core/utils/queries.mjs +4 -4
  141. package/esm2022/dom-grid.component.mjs +82 -0
  142. package/esm2022/grid-base.component.mjs +154 -0
  143. package/esm2022/grid.component.mjs +336 -160
  144. package/esm2022/public-api.mjs +9 -5
  145. package/esm2022/renderer/components/add-field-column.component.mjs +68 -0
  146. package/esm2022/renderer/components/cells.component.mjs +35 -0
  147. package/esm2022/renderer/components/field-head.component.mjs +130 -0
  148. package/esm2022/renderer/components/field-icon.component.mjs +66 -0
  149. package/esm2022/renderer/components/frozen-cells.component.mjs +36 -0
  150. package/esm2022/renderer/components/frozen-heads.component.mjs +117 -0
  151. package/esm2022/renderer/components/frozen-placeholder-cells.component.mjs +38 -0
  152. package/esm2022/renderer/components/heads.component.mjs +38 -0
  153. package/esm2022/renderer/components/hover-row-heads.component.mjs +107 -0
  154. package/esm2022/renderer/components/icon.component.mjs +80 -0
  155. package/esm2022/renderer/components/index.mjs +14 -0
  156. package/esm2022/renderer/components/other-rows.component.mjs +68 -0
  157. package/esm2022/renderer/components/placeholder-cells.component.mjs +33 -0
  158. package/esm2022/renderer/components/text.component.mjs +45 -0
  159. package/esm2022/renderer/creations/create-active-cell-border.mjs +68 -0
  160. package/esm2022/renderer/creations/create-cells.mjs +135 -0
  161. package/esm2022/renderer/creations/create-heads.mjs +45 -0
  162. package/esm2022/renderer/drawers/add-row-layout-drawer.mjs +97 -0
  163. package/esm2022/renderer/drawers/cell-drawer.mjs +586 -0
  164. package/esm2022/renderer/drawers/drawer.mjs +936 -0
  165. package/esm2022/renderer/drawers/layout-drawer.mjs +58 -0
  166. package/esm2022/renderer/drawers/record-row-layout-drawer.mjs +101 -0
  167. package/esm2022/renderer/index.mjs +4 -0
  168. package/esm2022/renderer/renderer.component.mjs +174 -0
  169. package/esm2022/services/event.service.mjs +115 -11
  170. package/esm2022/services/field.service.mjs +28 -5
  171. package/esm2022/services/index.mjs +4 -0
  172. package/esm2022/services/selection.service.mjs +8 -1
  173. package/esm2022/types/avatar.mjs +27 -0
  174. package/esm2022/types/canvas.mjs +2 -0
  175. package/esm2022/types/cell.mjs +2 -0
  176. package/esm2022/types/component-config.mjs +7 -0
  177. package/esm2022/types/field.mjs +1 -1
  178. package/esm2022/types/grid.mjs +16 -2
  179. package/esm2022/types/index.mjs +8 -2
  180. package/esm2022/types/layout.mjs +2 -0
  181. package/esm2022/types/row.mjs +6 -0
  182. package/esm2022/utils/build.mjs +39 -0
  183. package/esm2022/utils/cell.mjs +71 -0
  184. package/esm2022/utils/common.mjs +49 -0
  185. package/esm2022/utils/get-placeholder-cells.mjs +66 -0
  186. package/esm2022/utils/get-text-width.mjs +30 -0
  187. package/esm2022/utils/image-cache.mjs +57 -0
  188. package/esm2022/utils/index.mjs +12 -0
  189. package/esm2022/utils/os.mjs +16 -0
  190. package/esm2022/utils/position.mjs +48 -0
  191. package/esm2022/utils/style.mjs +25 -0
  192. package/esm2022/utils/text-measure.mjs +122 -0
  193. package/esm2022/utils/visible-range.mjs +38 -0
  194. package/fesm2022/ai-table-grid.mjs +5526 -693
  195. package/fesm2022/ai-table-grid.mjs.map +1 -1
  196. package/grid-base.component.d.ts +52 -0
  197. package/grid-base.component.d.ts.map +1 -0
  198. package/grid.component.d.ts +43 -40
  199. package/grid.component.d.ts.map +1 -1
  200. package/package.json +5 -2
  201. package/public-api.d.ts +8 -4
  202. package/public-api.d.ts.map +1 -1
  203. package/renderer/components/add-field-column.component.d.ts +16 -0
  204. package/renderer/components/add-field-column.component.d.ts.map +1 -0
  205. package/renderer/components/cells.component.d.ts +14 -0
  206. package/renderer/components/cells.component.d.ts.map +1 -0
  207. package/renderer/components/field-head.component.d.ts +73 -0
  208. package/renderer/components/field-head.component.d.ts.map +1 -0
  209. package/renderer/components/field-icon.component.d.ts +17 -0
  210. package/renderer/components/field-icon.component.d.ts.map +1 -0
  211. package/renderer/components/frozen-cells.component.d.ts +14 -0
  212. package/renderer/components/frozen-cells.component.d.ts.map +1 -0
  213. package/renderer/components/frozen-heads.component.d.ts +47 -0
  214. package/renderer/components/frozen-heads.component.d.ts.map +1 -0
  215. package/renderer/components/frozen-placeholder-cells.component.d.ts +22 -0
  216. package/renderer/components/frozen-placeholder-cells.component.d.ts.map +1 -0
  217. package/renderer/components/heads.component.d.ts +9 -0
  218. package/renderer/components/heads.component.d.ts.map +1 -0
  219. package/renderer/components/hover-row-heads.component.d.ts +11 -0
  220. package/renderer/components/hover-row-heads.component.d.ts.map +1 -0
  221. package/renderer/components/icon.component.d.ts +37 -0
  222. package/renderer/components/icon.component.d.ts.map +1 -0
  223. package/renderer/components/index.d.ts +14 -0
  224. package/renderer/components/index.d.ts.map +1 -0
  225. package/renderer/components/other-rows.component.d.ts +31 -0
  226. package/renderer/components/other-rows.component.d.ts.map +1 -0
  227. package/renderer/components/placeholder-cells.component.d.ts +22 -0
  228. package/renderer/components/placeholder-cells.component.d.ts.map +1 -0
  229. package/renderer/components/text.component.d.ts +99 -0
  230. package/renderer/components/text.component.d.ts.map +1 -0
  231. package/renderer/creations/create-active-cell-border.d.ts +7 -0
  232. package/renderer/creations/create-active-cell-border.d.ts.map +1 -0
  233. package/renderer/creations/create-cells.d.ts +8 -0
  234. package/renderer/creations/create-cells.d.ts.map +1 -0
  235. package/renderer/creations/create-heads.d.ts +3 -0
  236. package/renderer/creations/create-heads.d.ts.map +1 -0
  237. package/renderer/drawers/add-row-layout-drawer.d.ts +12 -0
  238. package/renderer/drawers/add-row-layout-drawer.d.ts.map +1 -0
  239. package/renderer/drawers/cell-drawer.d.ts +22 -0
  240. package/renderer/drawers/cell-drawer.d.ts.map +1 -0
  241. package/renderer/drawers/drawer.d.ts +136 -0
  242. package/renderer/drawers/drawer.d.ts.map +1 -0
  243. package/renderer/drawers/layout-drawer.d.ts +22 -0
  244. package/renderer/drawers/layout-drawer.d.ts.map +1 -0
  245. package/renderer/drawers/record-row-layout-drawer.d.ts +14 -0
  246. package/renderer/drawers/record-row-layout-drawer.d.ts.map +1 -0
  247. package/renderer/index.d.ts +4 -0
  248. package/renderer/index.d.ts.map +1 -0
  249. package/renderer/renderer.component.d.ts +64 -0
  250. package/renderer/renderer.component.d.ts.map +1 -0
  251. package/services/event.service.d.ts +17 -3
  252. package/services/event.service.d.ts.map +1 -1
  253. package/services/field.service.d.ts +5 -4
  254. package/services/field.service.d.ts.map +1 -1
  255. package/services/index.d.ts +4 -0
  256. package/services/index.d.ts.map +1 -0
  257. package/services/selection.service.d.ts.map +1 -1
  258. package/styles/styles.scss +71 -11
  259. package/types/avatar.d.ts +24 -0
  260. package/types/avatar.d.ts.map +1 -0
  261. package/types/canvas.d.ts +83 -0
  262. package/types/canvas.d.ts.map +1 -0
  263. package/types/cell.d.ts +43 -0
  264. package/types/cell.d.ts.map +1 -0
  265. package/types/component-config.d.ts +50 -0
  266. package/types/component-config.d.ts.map +1 -0
  267. package/types/field.d.ts +27 -1
  268. package/types/field.d.ts.map +1 -1
  269. package/types/grid.d.ts +80 -2
  270. package/types/grid.d.ts.map +1 -1
  271. package/types/index.d.ts +7 -1
  272. package/types/index.d.ts.map +1 -1
  273. package/types/layout.d.ts +11 -0
  274. package/types/layout.d.ts.map +1 -0
  275. package/types/row.d.ts +27 -0
  276. package/types/row.d.ts.map +1 -0
  277. package/utils/build.d.ts +5 -0
  278. package/utils/build.d.ts.map +1 -0
  279. package/utils/cell.d.ts +19 -0
  280. package/utils/cell.d.ts.map +1 -0
  281. package/utils/common.d.ts +20 -0
  282. package/utils/common.d.ts.map +1 -0
  283. package/utils/get-placeholder-cells.d.ts +16 -0
  284. package/utils/get-placeholder-cells.d.ts.map +1 -0
  285. package/utils/get-text-width.d.ts +9 -0
  286. package/utils/get-text-width.d.ts.map +1 -0
  287. package/utils/image-cache.d.ts +16 -0
  288. package/utils/image-cache.d.ts.map +1 -0
  289. package/utils/index.d.ts +12 -0
  290. package/utils/index.d.ts.map +1 -0
  291. package/utils/os.d.ts +4 -0
  292. package/utils/os.d.ts.map +1 -0
  293. package/utils/position.d.ts +20 -0
  294. package/utils/position.d.ts.map +1 -0
  295. package/utils/style.d.ts +4 -0
  296. package/utils/style.d.ts.map +1 -0
  297. package/utils/text-measure.d.ts +32 -0
  298. package/utils/text-measure.d.ts.map +1 -0
  299. package/utils/visible-range.d.ts +13 -0
  300. package/utils/visible-range.d.ts.map +1 -0
  301. package/components/cell-editors/link/number-editor.component.d.ts +0 -8
  302. package/components/cell-editors/link/number-editor.component.d.ts.map +0 -1
  303. package/constants/field.d.ts +0 -20
  304. package/constants/field.d.ts.map +0 -1
  305. package/core/action/field.d.ts +0 -12
  306. package/core/action/field.d.ts.map +0 -1
  307. package/core/action/general.d.ts +0 -5
  308. package/core/action/general.d.ts.map +0 -1
  309. package/core/action/index.d.ts +0 -12
  310. package/core/action/index.d.ts.map +0 -1
  311. package/core/action/record.d.ts +0 -12
  312. package/core/action/record.d.ts.map +0 -1
  313. package/core/types/action.d.ts +0 -64
  314. package/core/types/action.d.ts.map +0 -1
  315. package/core/utils/weak-map.d.ts +0 -3
  316. package/core/utils/weak-map.d.ts.map +0 -1
  317. package/esm2022/components/cell-editors/link/number-editor.component.mjs +0 -25
  318. package/esm2022/constants/field.mjs +0 -25
  319. package/esm2022/core/action/field.mjs +0 -57
  320. package/esm2022/core/action/general.mjs +0 -116
  321. package/esm2022/core/action/index.mjs +0 -9
  322. package/esm2022/core/action/record.mjs +0 -44
  323. package/esm2022/core/types/action.mjs +0 -18
  324. package/esm2022/core/utils/weak-map.mjs +0 -2
@@ -1,191 +1,367 @@
1
- import { CommonModule, NgClass, NgComponentOutlet, NgForOf, NgTemplateOutlet } from '@angular/common';
2
- import { ChangeDetectionStrategy, Component, computed, DestroyRef, ElementRef, inject, input, model, NgZone, output } from '@angular/core';
1
+ import { afterNextRender, ChangeDetectionStrategy, Component, computed, effect, signal, viewChild } from '@angular/core';
3
2
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
4
- import { FormsModule } from '@angular/forms';
5
- import { ThyAction } from 'ngx-tethys/action';
6
- import { ThyAvatarModule } from 'ngx-tethys/avatar';
7
- import { ThyCheckboxModule } from 'ngx-tethys/checkbox';
8
- import { ThyDatePickerFormatPipe } from 'ngx-tethys/date-picker';
9
- import { ThyDropdownDirective, ThyDropdownMenuComponent } from 'ngx-tethys/dropdown';
10
- import { ThyFlexibleText } from 'ngx-tethys/flexible-text';
11
- import { ThyIcon } from 'ngx-tethys/icon';
12
- import { ThyPopoverModule } from 'ngx-tethys/popover';
13
- import { ThyProgress } from 'ngx-tethys/progress';
14
- import { ThyRate } from 'ngx-tethys/rate';
15
- import { ThyStopPropagationDirective } from 'ngx-tethys/shared';
16
- import { ThyTag } from 'ngx-tethys/tag';
17
- import { mergeWith } from 'rxjs/operators';
18
- import { ProgressEditorComponent, SelectOptionComponent } from './components';
19
- import { FieldMenu } from './components/field-menu/field-menu.component';
20
- import { AITableFieldPropertyEditor } from './components/field-property-editor/field-property-editor.component';
21
- import { DBL_CLICK_EDIT_TYPE, DefaultFieldMenus, MOUSEOVER_EDIT_TYPE } from './constants';
22
- import { Actions, AITableFieldType, AITableSelectOptionStyle, createAITable, createDefaultField, getDefaultRecord } from './core';
23
- import { IsSelectRecordPipe, MemberSettingPipe, SelectOptionPipe, SelectOptionsPipe, SelectSettingPipe, UserPipe } from './pipes/grid.pipe';
3
+ import { filter, fromEvent } from 'rxjs';
4
+ import { AI_TABLE_CELL, AI_TABLE_FIELD_ADD_BUTTON, AI_TABLE_FIELD_ADD_BUTTON_WIDTH, AI_TABLE_FIELD_HEAD, AI_TABLE_FIELD_HEAD_HEIGHT, AI_TABLE_FIELD_HEAD_MORE, AI_TABLE_FIELD_HEAD_SELECT_CHECKBOX, AI_TABLE_POPOVER_LEFT_OFFSET, AI_TABLE_ROW_ADD_BUTTON, AI_TABLE_ROW_HEAD_WIDTH, AI_TABLE_ROW_SELECT_CHECKBOX, DBL_CLICK_EDIT_TYPE, DEFAULT_POINT_POSITION, DEFAULT_SCROLL_STATE, MOUSEOVER_EDIT_TYPE } from './constants';
5
+ import { AITable, Coordinate, RendererContext } from './core';
6
+ import { AITableGridBase } from './grid-base.component';
7
+ import { AITableRenderer } from './renderer/renderer.component';
24
8
  import { AITableGridEventService } from './services/event.service';
25
- import { AI_TABLE_GRID_FIELD_SERVICE_MAP, AITableGridFieldService } from './services/field.service';
9
+ import { AITableGridFieldService } from './services/field.service';
26
10
  import { AITableGridSelectionService } from './services/selection.service';
11
+ import { AITableMouseDownType } from './types';
12
+ import { buildGridLinearRows, getColumnIndicesMap, getDetailByTargetName, handleMouseStyle, isWindows } from './utils';
13
+ import { getMousePosition } from './utils/position';
27
14
  import * as i0 from "@angular/core";
28
- import * as i1 from "@angular/common";
29
- import * as i2 from "@angular/forms";
30
- import * as i3 from "ngx-tethys/checkbox";
31
- import * as i4 from "ngx-tethys/avatar";
32
- export class AITableGrid {
15
+ export class AITableGrid extends AITableGridBase {
33
16
  constructor() {
34
- this.aiRecords = model.required();
35
- this.aiFields = model.required();
36
- this.aiFieldConfig = input();
37
- this.aiReadonly = input();
38
- this.aiPlugins = input();
39
- this.aiReferences = input();
40
- this.aiBuildRenderDataFn = input();
41
- this.AITableFieldType = AITableFieldType;
42
- this.AITableSelectOptionStyle = AITableSelectOptionStyle;
43
- this.gridData = computed(() => {
44
- if (this.aiBuildRenderDataFn && this.aiBuildRenderDataFn() && this.aiTable) {
45
- return this.aiBuildRenderDataFn()(this.aiTable);
46
- }
17
+ super();
18
+ this.fieldHeadHeight = AI_TABLE_FIELD_HEAD_HEIGHT;
19
+ this.ADD_BUTTON_WIDTH = AI_TABLE_FIELD_ADD_BUTTON_WIDTH;
20
+ this.containerRect = signal({ width: 0, height: 0 });
21
+ this.hasContainerRect = computed(() => {
22
+ return this.containerRect().width > 0 && this.containerRect().height > 0;
23
+ });
24
+ this.container = viewChild('container');
25
+ this.verticalBarRef = viewChild('verticalBar');
26
+ this.horizontalBarRef = viewChild('horizontalBar');
27
+ this.linearRows = computed(() => {
28
+ return buildGridLinearRows(this.gridData().records);
29
+ });
30
+ this.visibleColumnsMap = computed(() => {
31
+ const columns = AITable.getVisibleFields(this.aiTable);
32
+ return new Map(columns?.map((item, index) => [item._id, index]));
33
+ });
34
+ this.visibleRowsIndexMap = computed(() => {
35
+ return new Map(this.linearRows().map((row, index) => [row._id, index]));
36
+ });
37
+ this.containerElement = computed(() => {
38
+ return this.container().nativeElement;
39
+ });
40
+ this.rendererConfig = computed(() => {
41
+ const fields = AITable.getVisibleFields(this.aiTable);
42
+ const coordinate = new Coordinate({
43
+ container: this.containerElement(),
44
+ rowHeight: AI_TABLE_FIELD_HEAD_HEIGHT,
45
+ rowCount: this.linearRows().length,
46
+ columnCount: fields.length,
47
+ rowInitSize: AI_TABLE_FIELD_HEAD_HEIGHT,
48
+ columnInitSize: AI_TABLE_ROW_HEAD_WIDTH,
49
+ rowIndicesMap: {},
50
+ columnIndicesMap: getColumnIndicesMap(fields),
51
+ frozenColumnCount: 1
52
+ });
47
53
  return {
48
- records: this.aiRecords(),
49
- fields: this.aiFields()
54
+ aiTable: this.aiTable,
55
+ container: this.containerElement(),
56
+ coordinate: coordinate,
57
+ containerWidth: this.containerRect().width,
58
+ containerHeight: this.containerRect().height,
59
+ references: this.aiReferences()
50
60
  };
51
61
  });
52
- this.isSelectedAll = computed(() => {
53
- return this.aiTable.selection().selectedRecords.size === this.aiRecords().length;
54
- });
55
- this.onChange = output();
56
- this.aiTableInitialized = output();
57
- this.ngZone = inject(NgZone);
58
- this.elementRef = inject(ElementRef);
59
- this.destroyRef = inject(DestroyRef);
60
- this.aiTableGridFieldService = inject(AITableGridFieldService);
61
- this.aiTableGridEventService = inject(AITableGridEventService);
62
- this.aiTableGridSelectionService = inject(AITableGridSelectionService);
63
- }
64
- ngOnInit() {
65
- this.initAITable();
66
- this.initService();
67
- this.buildFieldMenus();
68
- this.subscribeEvents();
69
- }
70
- initAITable() {
71
- this.aiTable = createAITable(this.aiRecords, this.aiFields);
72
- this.aiPlugins()?.forEach((plugin) => {
73
- this.aiTable = plugin(this.aiTable);
74
- });
75
- this.aiTableInitialized.emit(this.aiTable);
76
- this.aiTable.onChange = () => {
77
- this.onChange.emit({
78
- records: this.aiRecords(),
79
- fields: this.aiFields(),
80
- actions: this.aiTable.actions
62
+ this.coordinate = computed(() => {
63
+ return this.rendererConfig().coordinate;
64
+ });
65
+ this.scrollTotalHeight = computed(() => {
66
+ return Math.max(this.coordinate().totalHeight, this.containerRect().height - this.fieldHeadHeight);
67
+ });
68
+ this.scrollAction = (options) => {
69
+ if (this.timer) {
70
+ cancelAnimationFrame(this.timer);
71
+ }
72
+ this.timer = requestAnimationFrame(() => {
73
+ const { deltaX, deltaY, shiftKey } = options;
74
+ const fixedDeltaY = shiftKey && isWindows ? 0 : deltaY;
75
+ const fixedDeltaX = shiftKey && isWindows ? deltaY : deltaX;
76
+ const horizontalBar = this.horizontalBarRef()?.nativeElement;
77
+ const verticalBar = this.verticalBarRef()?.nativeElement;
78
+ if (horizontalBar) {
79
+ horizontalBar.scrollLeft = horizontalBar.scrollLeft + fixedDeltaX;
80
+ }
81
+ if (verticalBar) {
82
+ verticalBar.scrollTop = verticalBar.scrollTop + fixedDeltaY;
83
+ }
84
+ options.callback && options.callback();
85
+ this.timer = null;
86
+ });
87
+ };
88
+ this.resetScrolling = () => {
89
+ this.aiTable.context.setScrollState({
90
+ isScrolling: false
81
91
  });
82
92
  };
93
+ this.horizontalScroll = (e) => {
94
+ const { scrollLeft } = e.target;
95
+ this.aiTable.context.setScrollState({
96
+ scrollLeft,
97
+ isScrolling: true
98
+ });
99
+ this.resetScrolling();
100
+ };
101
+ afterNextRender(() => {
102
+ this.setContainerRect();
103
+ this.bindGlobalMousedown();
104
+ this.containerResizeListener();
105
+ this.bindWheel();
106
+ });
107
+ effect(() => {
108
+ if (this.hasContainerRect() && this.horizontalBarRef() && this.verticalBarRef()) {
109
+ this.bindScrollBarScroll();
110
+ }
111
+ });
112
+ effect(() => {
113
+ if (this.aiTable.context?.pointPosition()) {
114
+ this.toggleHoverCellEditor();
115
+ }
116
+ });
83
117
  }
84
- initService() {
85
- this.aiTableGridEventService.initialize(this.aiTable, this.aiFieldConfig()?.fieldPropertyEditor);
86
- this.aiTableGridSelectionService.initialize(this.aiTable);
87
- this.aiTableGridEventService.registerEvents(this.elementRef.nativeElement);
88
- this.aiTableGridFieldService.initAIFieldConfig(this.aiFieldConfig());
89
- AI_TABLE_GRID_FIELD_SERVICE_MAP.set(this.aiTable, this.aiTableGridFieldService);
118
+ ngOnInit() {
119
+ super.ngOnInit();
120
+ this.initContext();
90
121
  }
91
- buildFieldMenus() {
92
- this.fieldMenus = this.aiFieldConfig()?.fieldMenus ?? DefaultFieldMenus;
122
+ ngOnDestroy() {
123
+ this.resizeObserver?.disconnect();
93
124
  }
94
- addRecord() {
95
- Actions.addRecord(this.aiTable, getDefaultRecord(this.aiFields()), [this.aiRecords().length]);
125
+ initContext() {
126
+ this.aiTable.context = new RendererContext({
127
+ linearRows: this.linearRows,
128
+ visibleColumnsMap: this.visibleColumnsMap,
129
+ visibleRowsIndexMap: this.visibleRowsIndexMap,
130
+ pointPosition: signal(DEFAULT_POINT_POSITION),
131
+ scrollState: signal(DEFAULT_SCROLL_STATE),
132
+ scrollAction: this.scrollAction
133
+ });
96
134
  }
97
- selectRecord(recordId) {
98
- this.aiTableGridSelectionService.selectRecord(recordId);
135
+ stageMousemove(e) {
136
+ if (this.timer) {
137
+ cancelAnimationFrame(this.timer);
138
+ }
139
+ this.timer = requestAnimationFrame(() => {
140
+ const targetName = e.event.target.name();
141
+ const gridStage = e.event.currentTarget.getStage();
142
+ const pos = gridStage?.getPointerPosition();
143
+ if (pos == null)
144
+ return;
145
+ const { context } = this.aiTable;
146
+ const { x, y } = pos;
147
+ const curMousePosition = getMousePosition(x, y, this.coordinate(), AITable.getVisibleFields(this.aiTable), context, targetName);
148
+ handleMouseStyle(curMousePosition.realTargetName, curMousePosition.areaType, this.containerElement());
149
+ context.setPointPosition(curMousePosition);
150
+ this.timer = null;
151
+ });
99
152
  }
100
- toggleSelectAll(checked) {
101
- this.aiTableGridSelectionService.toggleSelectAll(checked);
153
+ stageMousedown(e) {
154
+ const mouseEvent = e.event.evt;
155
+ const _targetName = e.event.target.name();
156
+ const { targetName, fieldId, recordId } = getDetailByTargetName(_targetName);
157
+ switch (targetName) {
158
+ case AI_TABLE_FIELD_HEAD:
159
+ mouseEvent.preventDefault();
160
+ if (!fieldId)
161
+ return;
162
+ this.aiTableGridSelectionService.selectField(fieldId);
163
+ return;
164
+ case AI_TABLE_CELL:
165
+ if (!recordId || !fieldId)
166
+ return;
167
+ this.aiTableGridEventService.closeCellEditor();
168
+ this.aiTableGridSelectionService.selectCell(recordId, fieldId);
169
+ return;
170
+ case AI_TABLE_ROW_ADD_BUTTON:
171
+ case AI_TABLE_FIELD_ADD_BUTTON:
172
+ case AI_TABLE_ROW_SELECT_CHECKBOX:
173
+ case AI_TABLE_FIELD_HEAD_SELECT_CHECKBOX:
174
+ return;
175
+ default:
176
+ this.aiTableGridSelectionService.clearSelection();
177
+ }
102
178
  }
103
- addField(gridColumnBlank) {
104
- const field = createDefaultField(this.aiTable, AITableFieldType.text);
105
- this.aiTableGridFieldService.editFieldProperty(gridColumnBlank, this.aiTable, field, false);
179
+ stageClick(e) {
180
+ const mouseEvent = e.event.evt;
181
+ mouseEvent.preventDefault();
182
+ const { context } = this.aiTable;
183
+ const { targetName, rowIndex: pointRowIndex } = context.pointPosition();
184
+ if (mouseEvent.button !== AITableMouseDownType.Left)
185
+ return;
186
+ switch (targetName) {
187
+ case AI_TABLE_ROW_ADD_BUTTON: {
188
+ this.aiTableGridSelectionService.clearSelection();
189
+ this.addRecord();
190
+ return;
191
+ }
192
+ case AI_TABLE_ROW_SELECT_CHECKBOX: {
193
+ const pointRecordId = context.linearRows()[pointRowIndex]?._id;
194
+ this.selectRecord(pointRecordId);
195
+ return;
196
+ }
197
+ case AI_TABLE_FIELD_HEAD_SELECT_CHECKBOX: {
198
+ const isChecked = this.aiTable.selection().selectedRecords.size === this.aiTable.records().length;
199
+ this.toggleSelectAll(!isChecked);
200
+ return;
201
+ }
202
+ case AI_TABLE_FIELD_ADD_BUTTON: {
203
+ this.aiTableGridSelectionService.clearSelection();
204
+ const fieldGroupRect = e.event.target.getParent()?.getClientRect();
205
+ const containerRect = this.containerElement().getBoundingClientRect();
206
+ this.addField(this.containerElement(), {
207
+ x: fieldGroupRect.x + containerRect.x,
208
+ y: containerRect.y + fieldGroupRect.y + fieldGroupRect.height
209
+ });
210
+ return;
211
+ }
212
+ case AI_TABLE_FIELD_HEAD_MORE:
213
+ mouseEvent.preventDefault();
214
+ const _targetName = e.event.target.name();
215
+ const { fieldId } = getDetailByTargetName(_targetName);
216
+ if (fieldId) {
217
+ const moreRect = e.event.target.getClientRect();
218
+ const fieldGroupRect = e.event.target.getParent()?.getParent()?.getClientRect();
219
+ const containerRect = this.containerElement().getBoundingClientRect();
220
+ const position = {
221
+ x: containerRect.x + moreRect.x,
222
+ y: containerRect.y + moreRect.y + moreRect.height
223
+ };
224
+ const editOriginPosition = {
225
+ x: AI_TABLE_POPOVER_LEFT_OFFSET + fieldGroupRect.x,
226
+ y: containerRect.y + fieldGroupRect.y + fieldGroupRect.height
227
+ };
228
+ const editOrigin = this.containerElement().querySelector('.konvajs-content');
229
+ this.aiTableGridFieldService.openFieldMenu(this.aiTable, {
230
+ fieldId: fieldId,
231
+ fieldMenus: this.fieldMenus,
232
+ origin: this.containerElement(),
233
+ position,
234
+ editOrigin: editOrigin,
235
+ editOriginPosition
236
+ });
237
+ }
238
+ return;
239
+ }
106
240
  }
107
- subscribeEvents() {
108
- this.ngZone.runOutsideAngular(() => {
109
- this.aiTableGridEventService.dblClickEvent$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {
110
- this.dblClick(event);
111
- });
112
- this.aiTableGridEventService.mousedownEvent$
113
- .pipe(mergeWith(this.aiTableGridEventService.globalMousedownEvent$), takeUntilDestroyed(this.destroyRef))
114
- .subscribe((event) => {
115
- this.aiTableGridSelectionService.updateSelect(event);
116
- });
117
- this.aiTableGridEventService.mouseoverEvent$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {
118
- this.mouseoverHandle(event);
119
- });
120
- this.aiTableGridEventService.globalMouseoverEvent$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {
121
- this.closeHoverCellEditor(event);
241
+ stageDblclick(e) {
242
+ const _targetName = e.event.target.name();
243
+ const { fieldId, recordId } = getDetailByTargetName(_targetName);
244
+ if (!recordId || !fieldId) {
245
+ return;
246
+ }
247
+ const field = this.aiTable.fieldsMap()[fieldId];
248
+ const fieldType = field.type;
249
+ if (!DBL_CLICK_EDIT_TYPE.includes(fieldType)) {
250
+ return;
251
+ }
252
+ const popoverRef = this.aiTableGridEventService.openCellEditor(this.aiTable, {
253
+ container: this.containerElement(),
254
+ coordinate: this.coordinate(),
255
+ fieldId: fieldId,
256
+ recordId: recordId
257
+ });
258
+ if (popoverRef && !(this.aiFieldConfig()?.fieldPropertyEditor && this.aiFieldConfig()?.fieldPropertyEditor[fieldType])) {
259
+ popoverRef.componentInstance.updateFieldValue.subscribe((value) => {
260
+ this.aiUpdateFieldValue.emit(value);
122
261
  });
262
+ }
263
+ }
264
+ bindWheel() {
265
+ fromEvent(this.containerElement(), 'wheel', { passive: false })
266
+ .pipe(takeUntilDestroyed(this.destroyRef))
267
+ .subscribe((e) => {
268
+ e.preventDefault();
269
+ this.aiTableGridEventService.closeCellEditor();
270
+ this.scrollAction({ deltaX: e.deltaX, deltaY: e.deltaY, shiftKey: e.shiftKey });
123
271
  });
124
272
  }
125
- dblClick(event) {
126
- const cellDom = event.target.closest('.grid-cell');
127
- const type = cellDom && cellDom.getAttribute('type');
128
- if (type && DBL_CLICK_EDIT_TYPE.includes(type)) {
129
- this.aiTableGridEventService.openEdit(cellDom);
130
- }
273
+ bindScrollBarScroll() {
274
+ fromEvent(this.horizontalBarRef().nativeElement, 'scroll', { passive: true })
275
+ .pipe(takeUntilDestroyed(this.destroyRef))
276
+ .subscribe((e) => {
277
+ this.horizontalScroll(e);
278
+ });
279
+ fromEvent(this.verticalBarRef().nativeElement, 'scroll', { passive: true })
280
+ .pipe(takeUntilDestroyed(this.destroyRef))
281
+ .subscribe((e) => {
282
+ this.verticalScroll(e);
283
+ });
131
284
  }
132
- mouseoverHandle(event) {
133
- if (this.mouseoverRef) {
134
- this.mouseoverRef?.close();
135
- }
136
- const cellDom = event.target.closest('.grid-cell');
137
- const type = cellDom && cellDom.getAttribute('type');
138
- if (type && MOUSEOVER_EDIT_TYPE.includes(type)) {
139
- this.mouseoverRef = this.aiTableGridEventService.openEdit(cellDom);
140
- }
285
+ bindGlobalMousedown() {
286
+ fromEvent(document, 'mousedown', { passive: true })
287
+ .pipe(filter((e) => e.target instanceof Element && !this.containerElement().contains(e.target)), takeUntilDestroyed(this.destroyRef))
288
+ .subscribe(() => {
289
+ this.aiTableGridSelectionService.clearSelection();
290
+ });
291
+ }
292
+ verticalScroll(e) {
293
+ const { scrollTop } = e.target;
294
+ this.aiTable.context.setScrollState({
295
+ scrollTop,
296
+ isScrolling: true
297
+ });
298
+ this.resetScrolling();
299
+ }
300
+ setContainerRect() {
301
+ this.containerRect.set({
302
+ width: this.containerElement().offsetWidth,
303
+ height: this.containerElement().offsetHeight
304
+ });
305
+ }
306
+ containerResizeListener() {
307
+ this.resizeObserver = new ResizeObserver(() => {
308
+ const containerWidth = this.containerElement().offsetWidth;
309
+ const totalWidth = this.coordinate().totalWidth + this.ADD_BUTTON_WIDTH;
310
+ this.setContainerRect();
311
+ if (containerWidth >= totalWidth) {
312
+ this.aiTable.context.setScrollState({ scrollLeft: 0 });
313
+ return;
314
+ }
315
+ });
316
+ this.resizeObserver.observe(this.containerElement());
141
317
  }
142
- closeHoverCellEditor(e) {
143
- if (this.mouseoverRef) {
144
- const hasGrid = e.target && e.target.closest('.ai-table-grid');
145
- const hasCellEditor = e.target && e.target.closest('.grid-cell-editor');
146
- if (!hasGrid && !hasCellEditor) {
147
- this.mouseoverRef.close();
318
+ toggleHoverCellEditor() {
319
+ const { realTargetName } = this.aiTable.context?.pointPosition();
320
+ const { targetName, fieldId, recordId } = getDetailByTargetName(realTargetName);
321
+ const editingCell = this.aiTableGridEventService.getCurrentEditCell();
322
+ if (targetName === AI_TABLE_CELL && recordId && fieldId) {
323
+ const field = this.aiTable.fieldsMap()[fieldId];
324
+ if (!field) {
325
+ return;
326
+ }
327
+ const fieldType = field.type;
328
+ const editingFieldType = editingCell ? this.aiTable.fieldsMap()[editingCell.fieldId].type : null;
329
+ const isEditingFieldTypeHovered = editingFieldType ? MOUSEOVER_EDIT_TYPE.includes(editingFieldType) : false;
330
+ const isFieldTypeHovered = MOUSEOVER_EDIT_TYPE.includes(fieldType);
331
+ if (editingCell && isEditingFieldTypeHovered) {
332
+ this.aiTableGridEventService.closeCellEditor();
333
+ }
334
+ if (!editingCell && !isFieldTypeHovered) {
335
+ this.aiTableGridEventService.closeCellEditor();
336
+ return;
337
+ }
338
+ if (editingCell && ((!isEditingFieldTypeHovered && isFieldTypeHovered) || !isFieldTypeHovered)) {
339
+ return;
340
+ }
341
+ setTimeout(() => {
342
+ this.aiTableGridEventService.openCellEditor(this.aiTable, {
343
+ container: this.containerElement(),
344
+ coordinate: this.coordinate(),
345
+ fieldId: fieldId,
346
+ recordId: recordId,
347
+ isHoverEdit: true
348
+ });
349
+ });
350
+ }
351
+ else {
352
+ // 鼠标位于非单元格区域时,如果当前有 mouseover 编辑元素,则结束编辑
353
+ if (editingCell && MOUSEOVER_EDIT_TYPE.includes(this.aiTable.fieldsMap()[editingCell.fieldId].type)) {
354
+ this.aiTableGridEventService.closeCellEditor();
148
355
  }
149
356
  }
150
357
  }
151
358
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.0", ngImport: i0, type: AITableGrid, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
152
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.0", type: AITableGrid, isStandalone: true, selector: "ai-table-grid", inputs: { aiRecords: { classPropertyName: "aiRecords", publicName: "aiRecords", isSignal: true, isRequired: true, transformFunction: null }, aiFields: { classPropertyName: "aiFields", publicName: "aiFields", isSignal: true, isRequired: true, transformFunction: null }, aiFieldConfig: { classPropertyName: "aiFieldConfig", publicName: "aiFieldConfig", isSignal: true, isRequired: false, transformFunction: null }, aiReadonly: { classPropertyName: "aiReadonly", publicName: "aiReadonly", isSignal: true, isRequired: false, transformFunction: null }, aiPlugins: { classPropertyName: "aiPlugins", publicName: "aiPlugins", isSignal: true, isRequired: false, transformFunction: null }, aiReferences: { classPropertyName: "aiReferences", publicName: "aiReferences", isSignal: true, isRequired: false, transformFunction: null }, aiBuildRenderDataFn: { classPropertyName: "aiBuildRenderDataFn", publicName: "aiBuildRenderDataFn", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { aiRecords: "aiRecordsChange", aiFields: "aiFieldsChange", onChange: "onChange", aiTableInitialized: "aiTableInitialized" }, host: { classAttribute: "ai-table-grid" }, providers: [AITableGridEventService, AITableGridFieldService, AITableGridSelectionService], ngImport: i0, template: "<div class=\"grid-header d-flex\">\n <div class=\"grid-column-checkbox grid-cell grid-checkbox\">\n <label thyCheckbox thyLabelText=\"\" [ngModel]=\"isSelectedAll()\" (ngModelChange)=\"toggleSelectAll($event)\"></label>\n </div>\n @for (field of gridData().fields; track field._id) {\n <div\n class=\"grid-cell grid-field\"\n #fieldAction\n [attr.fieldId]=\"field._id\"\n [ngClass]=\"{ highlight: aiTable.selection().selectedFields.has(field._id) }\"\n [ngStyle]=\"{ width: field.width + 'px' }\"\n >\n <span class=\"text-truncate\">\n <thy-icon [thyIconName]=\"field.icon!\" class=\"mr-2 text-muted\"></thy-icon>\n <span>{{ field.name }}</span>\n </span>\n <a\n class=\"grid-field-action\"\n thyAction\n thyActiveClass=\"active\"\n thyIcon=\"more-vertical\"\n [thyDropdown]=\"fieldMenu\"\n href=\"javascript:;\"\n >\n <thy-dropdown-menu #fieldMenu>\n <field-menu [origin]=\"fieldAction\" [fieldId]=\"field._id\" [aiTable]=\"aiTable\" [fieldMenus]=\"fieldMenus\"></field-menu>\n </thy-dropdown-menu>\n </a>\n </div>\n }\n <div class=\"grid-column-blank cursor-pointer\" #gridColumnBlank (click)=\"addField(gridColumnBlank)\">\n <thy-icon thyIconName=\"plus\"></thy-icon>\n </div>\n</div>\n<div class=\"grid-body d-flex\">\n @for (record of gridData().records; track record._id; let index = $index) {\n <div class=\"grid-row d-flex\" [ngClass]=\"{ highlight: (record._id | isSelectRecord: aiTable.selection()) }\">\n <div class=\"grid-row-index grid-checkbox\">\n <label\n [ngClass]=\"(record._id | isSelectRecord: aiTable.selection()) ? 'checked-box' : 'unchecked-box'\"\n thyCheckbox\n thyLabelText=\"\"\n [ngModel]=\"record._id | isSelectRecord: aiTable.selection()\"\n (ngModelChange)=\"selectRecord(record._id)\"\n ></label>\n <span [ngClass]=\"(record._id | isSelectRecord: aiTable.selection()) ? 'grid-row-no-number' : 'grid-row-number'\">\n {{ index + 1 }}\n </span>\n </div>\n @for (field of gridData().fields; track field._id) {\n <div\n #cell\n class=\"grid-cell\"\n [ngClass]=\"{\n highlight: aiTable.selection().selectedCells.has(record._id) || aiTable.selection().selectedFields.has(field._id),\n selected: aiTable.selection().selectedCells.get(record._id)?.hasOwnProperty(field._id)\n }\"\n [attr.type]=\"[field.type]\"\n [attr.fieldId]=\"[field._id]\"\n [attr.recordId]=\"[record._id]\"\n [ngStyle]=\"{ width: field.width + 'px' }\"\n >\n @switch (field.type) {\n @case (AITableFieldType.select) {\n @let fieldValue = record.values[field._id];\n @let settings = field.settings! | selectSetting;\n @let options = settings['options'];\n @let optionStyle = settings['option_style'] || AITableSelectOptionStyle.tag;\n @let isTagStyle = optionStyle === AITableSelectOptionStyle.tag;\n\n @if (!settings['is_multiple'] && fieldValue | selectOption: options; as selectedOption) {\n @if (isTagStyle) {\n <select-option class=\"mb-1 mr-1\" [field]=\"field\" [displayOption]=\"selectedOption\"></select-option>\n } @else {\n <div thyTag class=\"mb-1 mr-1\">\n <select-option [field]=\"field\" [displayOption]=\"selectedOption\"></select-option>\n </div>\n }\n } @else {\n @let maxShowCount = 2;\n\n <div class=\"d-flex\">\n @if (fieldValue | selectOptions: options; as selectedOptions) {\n @for (option of selectedOptions; track option!._id; let i = $index) {\n @if (i + 1 <= maxShowCount) {\n @if (isTagStyle) {\n <select-option\n class=\"mb-1 mr-1\"\n [field]=\"field\"\n [displayOption]=\"option!\"\n ></select-option>\n } @else {\n <div thyTag class=\"mb-1 mr-1\">\n <select-option [field]=\"field\" [displayOption]=\"option!\"></select-option>\n </div>\n }\n }\n }\n\n @let selectedLength = selectedOptions.length || 0;\n @if (selectedOptions && maxShowCount < selectedLength) {\n @let shape = isTagStyle ? 'pill' : 'rectangle';\n @let isHidden = maxShowCount >= selectedLength;\n\n <thy-tag\n class=\"cursor-pointer\"\n [class.multi-property-value-hidden]=\"isHidden\"\n [thyShape]=\"shape\"\n >\n <span class=\"text-truncate\"> +{{ selectedLength - maxShowCount }} </span>\n </thy-tag>\n }\n }\n </div>\n }\n }\n @case (AITableFieldType.date) {\n {{ record.values[field._id].timestamp | thyDatePickerFormat }}\n }\n @case (AITableFieldType.updatedAt) {\n <div class=\"d-block user-select-none\">\n <span class=\"text-truncate\">\n {{ record.values[field._id] | thyDatePickerFormat: 'yyyy-MM-dd HH:mm' }}\n </span>\n </div>\n }\n @case (AITableFieldType.createdAt) {\n <div class=\"d-block user-select-none\">\n <span class=\"text-truncate\">\n {{ record.values[field._id] | thyDatePickerFormat: 'yyyy-MM-dd HH:mm' }}\n </span>\n </div>\n }\n @case (AITableFieldType.rate) {\n <thy-rate [ngModel]=\"record.values[field._id]\"></thy-rate>\n }\n @case (AITableFieldType.link) {\n <a\n class=\"d-block\"\n target=\"_blank\"\n [href]=\"record.values[field._id]?.url\"\n thyStopPropagation\n thyFlexibleText\n [thyTooltipContent]=\"record.values[field._id]?.text\"\n >\n {{ record.values[field._id]?.text }}\n </a>\n }\n @case (AITableFieldType.progress) {\n <thy-progress\n class=\"w-100\"\n [thyValue]=\"record.values[field._id] || 0\"\n [thySize]=\"record.values[field._id]?.config?.size || 'md'\"\n [thyMax]=\"record.values[field._id]?.config?.max || 100\"\n [thyType]=\"record.values[field._id]?.config?.progressType || 'success'\"\n >\n <span> {{ record.values[field._id] || 0 }}{{ record.values[field._id]?.config?.suffix || '%' }} </span>\n </thy-progress>\n }\n @case (AITableFieldType.member) {\n @let settings = field.settings! | memberSetting;\n\n @if (!settings!['is_multiple']) {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n @if (recordValues && recordValues.length) {\n <thy-avatar\n [thyName]=\"recordValues[0].display_name!\"\n [thySrc]=\"recordValues[0].avatar!\"\n thySize=\"xs\"\n thyShowName=\"true\"\n ></thy-avatar>\n }\n } @else {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n <thy-avatar-list thyAvatarSize=\"xs\">\n @for (item of recordValues; track $index) {\n <thy-avatar [thyName]=\"item.display_name!\" [thySrc]=\"item.avatar!\"></thy-avatar>\n }\n </thy-avatar-list>\n }\n }\n @case (AITableFieldType.createdBy) {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n @if (recordValues && recordValues.length) {\n <thy-avatar\n [thyName]=\"recordValues[0].display_name!\"\n [thySrc]=\"recordValues[0].avatar!\"\n thySize=\"xs\"\n thyShowName=\"true\"\n ></thy-avatar>\n }\n }\n @case (AITableFieldType.updatedBy) {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n @if (recordValues && recordValues.length) {\n <thy-avatar\n [thyName]=\"recordValues[0].display_name!\"\n [thySrc]=\"recordValues[0].avatar!\"\n thySize=\"xs\"\n thyShowName=\"true\"\n ></thy-avatar>\n }\n }\n @default {\n <span class=\"text-truncate\"> {{ record.values[field._id] }}</span>\n }\n }\n <div class=\"autofill-container\"></div>\n </div>\n }\n <div class=\"grid-column-blank\"></div>\n </div>\n }\n <div class=\"grid-row-insert grid-row cursor-pointer\" (click)=\"addRecord()\">\n <thy-icon thyIconName=\"plus\"></thy-icon>\n </div>\n</div>\n\n<div #activeBorder class=\"active-border\"></div>\n", dependencies: [{ kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i2.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i2.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "pipe", type: SelectOptionPipe, name: "selectOption" }, { kind: "pipe", type: SelectOptionsPipe, name: "selectOptions" }, { kind: "component", type: ThyTag, selector: "thy-tag,[thyTag]", inputs: ["thyTag", "thyShape", "thyColor", "thyTheme", "thySize", "thyHoverable"] }, { kind: "ngmodule", type: ThyPopoverModule }, { kind: "component", type: ThyIcon, selector: "thy-icon, [thy-icon]", inputs: ["thyIconType", "thyTwotoneColor", "thyIconName", "thyIconRotate", "thyIconSet", "thyIconLegging", "thyIconLinearGradient"] }, { kind: "component", type: ThyRate, selector: "thy-rate", inputs: ["thyCount", "thyDisabled", "thyAllowHalf", "thyAllowClear", "thyTooltips", "thyIconTemplate"], outputs: ["thyItemHoverChange"] }, { kind: "component", type: ThyProgress, selector: "thy-progress", inputs: ["thyType", "thySize", "thyValue", "thyMax", "thyTips", "thyShape", "thyGapDegree", "thyGapPosition", "thyStrokeWidth"] }, { kind: "pipe", type: ThyDatePickerFormatPipe, name: "thyDatePickerFormat" }, { kind: "component", type: ThyFlexibleText, selector: "thy-flexible-text,[thyFlexibleText]", inputs: ["thyTooltipTrigger", "thyContainerClass", "thyTooltipContent", "thyTooltipPlacement", "thyTooltipOffset"], exportAs: ["thyFlexibleText"] }, { kind: "directive", type: ThyStopPropagationDirective, selector: "[thyStopPropagation]", inputs: ["thyStopPropagation"] }, { kind: "component", type: FieldMenu, selector: "field-menu", inputs: ["fieldId", "aiTable", "fieldMenus", "origin"] }, { kind: "component", type: ThyAction, selector: "thy-action, [thyAction]", inputs: ["thyType", "thyIcon", "thyActionIcon", "thyActive", "thyActionActive", "thyTheme", "thyHoverIcon", "thyDisabled"] }, { kind: "directive", type: ThyDropdownDirective, selector: "[thyDropdown]", inputs: ["thyDropdownMenu", "thyDropdown", "thyTrigger", "thyShowDelay", "thyHideDelay", "thyActiveClass", "thyPopoverOptions", "thyPlacement", "thyMenuInsideClosable", "thyPanelClass"], outputs: ["thyActiveChange"] }, { kind: "component", type: ThyDropdownMenuComponent, selector: "thy-dropdown-menu", inputs: ["thyWidth", "thyImmediateRender"] }, { kind: "ngmodule", type: ThyCheckboxModule }, { kind: "component", type: i3.ThyCheckbox, selector: "thy-checkbox,[thy-checkbox],[thyCheckbox]", inputs: ["thyIndeterminate"] }, { kind: "ngmodule", type: ThyAvatarModule }, { kind: "component", type: i4.ThyAvatar, selector: "thy-avatar", inputs: ["thyShowName", "thySrc", "thyName", "thySize", "thyShowRemove", "thyRemovable", "thyImgClass", "thyDisabled", "thyLoading", "thyFetchPriority"], outputs: ["thyOnRemove", "thyRemove", "thyError"] }, { kind: "component", type: i4.ThyAvatarList, selector: "thy-avatar-list", inputs: ["thyMode", "thyAvatarSize"] }, { kind: "pipe", type: IsSelectRecordPipe, name: "isSelectRecord" }, { kind: "component", type: SelectOptionComponent, selector: "select-option", inputs: ["field", "displayOption"] }, { kind: "pipe", type: UserPipe, name: "user" }, { kind: "pipe", type: SelectSettingPipe, name: "selectSetting" }, { kind: "pipe", type: MemberSettingPipe, name: "memberSetting" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
359
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.0", type: AITableGrid, isStandalone: true, selector: "ai-table-grid", host: { classAttribute: "ai-table-grid" }, providers: [AITableGridEventService, AITableGridFieldService, AITableGridSelectionService], viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true, isSignal: true }, { propertyName: "verticalBarRef", first: true, predicate: ["verticalBar"], descendants: true, isSignal: true }, { propertyName: "horizontalBarRef", first: true, predicate: ["horizontalBar"], descendants: true, isSignal: true }], usesInheritance: true, ngImport: i0, template: "<div #container class=\"ai-table-grid-view\">\n @if (hasContainerRect()) {\n <ai-table-renderer\n [config]=\"rendererConfig()\"\n (koMousemove)=\"stageMousemove($event)\"\n (koMousedown)=\"stageMousedown($event)\"\n (koClick)=\"stageClick($event)\"\n (koDblclick)=\"stageDblclick($event)\"\n >\n <div #horizontalBar class=\"ai-table-horizontal-scroll-bar-wrapper\" [style.width.px]=\"containerRect().width\">\n <div class=\"ai-table-scroll-bar-inner\" [style.width.px]=\"coordinate().totalWidth + ADD_BUTTON_WIDTH\"></div>\n </div>\n <div\n #verticalBar\n class=\"ai-table-vertical-scroll-bar-wrapper\"\n [style.height.px]=\"containerRect().height - fieldHeadHeight\"\n [style.top.px]=\"fieldHeadHeight\"\n >\n <div class=\"ai-table-scroll-bar-inner\" [style.height.px]=\"scrollTotalHeight()\"></div>\n </div>\n </ai-table-renderer>\n }\n</div>\n", dependencies: [{ kind: "component", type: AITableRenderer, selector: "ai-table-renderer", inputs: ["config"], outputs: ["koMousemove", "koMousedown", "koWheel", "koClick", "koDblclick"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
153
360
  }
154
361
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.0", ngImport: i0, type: AITableGrid, decorators: [{
155
362
  type: Component,
156
363
  args: [{ selector: 'ai-table-grid', standalone: true, changeDetection: ChangeDetectionStrategy.OnPush, host: {
157
364
  class: 'ai-table-grid'
158
- }, imports: [
159
- NgForOf,
160
- NgClass,
161
- NgComponentOutlet,
162
- CommonModule,
163
- FormsModule,
164
- SelectOptionPipe,
165
- SelectOptionsPipe,
166
- ThyTag,
167
- ThyPopoverModule,
168
- ThyIcon,
169
- ThyRate,
170
- ThyProgress,
171
- AITableFieldPropertyEditor,
172
- ThyDatePickerFormatPipe,
173
- ThyFlexibleText,
174
- ThyStopPropagationDirective,
175
- FieldMenu,
176
- ThyAction,
177
- ThyDropdownDirective,
178
- ThyDropdownMenuComponent,
179
- ThyCheckboxModule,
180
- ProgressEditorComponent,
181
- ThyAvatarModule,
182
- NgTemplateOutlet,
183
- IsSelectRecordPipe,
184
- ProgressEditorComponent,
185
- SelectOptionComponent,
186
- UserPipe,
187
- SelectSettingPipe,
188
- MemberSettingPipe
189
- ], providers: [AITableGridEventService, AITableGridFieldService, AITableGridSelectionService], template: "<div class=\"grid-header d-flex\">\n <div class=\"grid-column-checkbox grid-cell grid-checkbox\">\n <label thyCheckbox thyLabelText=\"\" [ngModel]=\"isSelectedAll()\" (ngModelChange)=\"toggleSelectAll($event)\"></label>\n </div>\n @for (field of gridData().fields; track field._id) {\n <div\n class=\"grid-cell grid-field\"\n #fieldAction\n [attr.fieldId]=\"field._id\"\n [ngClass]=\"{ highlight: aiTable.selection().selectedFields.has(field._id) }\"\n [ngStyle]=\"{ width: field.width + 'px' }\"\n >\n <span class=\"text-truncate\">\n <thy-icon [thyIconName]=\"field.icon!\" class=\"mr-2 text-muted\"></thy-icon>\n <span>{{ field.name }}</span>\n </span>\n <a\n class=\"grid-field-action\"\n thyAction\n thyActiveClass=\"active\"\n thyIcon=\"more-vertical\"\n [thyDropdown]=\"fieldMenu\"\n href=\"javascript:;\"\n >\n <thy-dropdown-menu #fieldMenu>\n <field-menu [origin]=\"fieldAction\" [fieldId]=\"field._id\" [aiTable]=\"aiTable\" [fieldMenus]=\"fieldMenus\"></field-menu>\n </thy-dropdown-menu>\n </a>\n </div>\n }\n <div class=\"grid-column-blank cursor-pointer\" #gridColumnBlank (click)=\"addField(gridColumnBlank)\">\n <thy-icon thyIconName=\"plus\"></thy-icon>\n </div>\n</div>\n<div class=\"grid-body d-flex\">\n @for (record of gridData().records; track record._id; let index = $index) {\n <div class=\"grid-row d-flex\" [ngClass]=\"{ highlight: (record._id | isSelectRecord: aiTable.selection()) }\">\n <div class=\"grid-row-index grid-checkbox\">\n <label\n [ngClass]=\"(record._id | isSelectRecord: aiTable.selection()) ? 'checked-box' : 'unchecked-box'\"\n thyCheckbox\n thyLabelText=\"\"\n [ngModel]=\"record._id | isSelectRecord: aiTable.selection()\"\n (ngModelChange)=\"selectRecord(record._id)\"\n ></label>\n <span [ngClass]=\"(record._id | isSelectRecord: aiTable.selection()) ? 'grid-row-no-number' : 'grid-row-number'\">\n {{ index + 1 }}\n </span>\n </div>\n @for (field of gridData().fields; track field._id) {\n <div\n #cell\n class=\"grid-cell\"\n [ngClass]=\"{\n highlight: aiTable.selection().selectedCells.has(record._id) || aiTable.selection().selectedFields.has(field._id),\n selected: aiTable.selection().selectedCells.get(record._id)?.hasOwnProperty(field._id)\n }\"\n [attr.type]=\"[field.type]\"\n [attr.fieldId]=\"[field._id]\"\n [attr.recordId]=\"[record._id]\"\n [ngStyle]=\"{ width: field.width + 'px' }\"\n >\n @switch (field.type) {\n @case (AITableFieldType.select) {\n @let fieldValue = record.values[field._id];\n @let settings = field.settings! | selectSetting;\n @let options = settings['options'];\n @let optionStyle = settings['option_style'] || AITableSelectOptionStyle.tag;\n @let isTagStyle = optionStyle === AITableSelectOptionStyle.tag;\n\n @if (!settings['is_multiple'] && fieldValue | selectOption: options; as selectedOption) {\n @if (isTagStyle) {\n <select-option class=\"mb-1 mr-1\" [field]=\"field\" [displayOption]=\"selectedOption\"></select-option>\n } @else {\n <div thyTag class=\"mb-1 mr-1\">\n <select-option [field]=\"field\" [displayOption]=\"selectedOption\"></select-option>\n </div>\n }\n } @else {\n @let maxShowCount = 2;\n\n <div class=\"d-flex\">\n @if (fieldValue | selectOptions: options; as selectedOptions) {\n @for (option of selectedOptions; track option!._id; let i = $index) {\n @if (i + 1 <= maxShowCount) {\n @if (isTagStyle) {\n <select-option\n class=\"mb-1 mr-1\"\n [field]=\"field\"\n [displayOption]=\"option!\"\n ></select-option>\n } @else {\n <div thyTag class=\"mb-1 mr-1\">\n <select-option [field]=\"field\" [displayOption]=\"option!\"></select-option>\n </div>\n }\n }\n }\n\n @let selectedLength = selectedOptions.length || 0;\n @if (selectedOptions && maxShowCount < selectedLength) {\n @let shape = isTagStyle ? 'pill' : 'rectangle';\n @let isHidden = maxShowCount >= selectedLength;\n\n <thy-tag\n class=\"cursor-pointer\"\n [class.multi-property-value-hidden]=\"isHidden\"\n [thyShape]=\"shape\"\n >\n <span class=\"text-truncate\"> +{{ selectedLength - maxShowCount }} </span>\n </thy-tag>\n }\n }\n </div>\n }\n }\n @case (AITableFieldType.date) {\n {{ record.values[field._id].timestamp | thyDatePickerFormat }}\n }\n @case (AITableFieldType.updatedAt) {\n <div class=\"d-block user-select-none\">\n <span class=\"text-truncate\">\n {{ record.values[field._id] | thyDatePickerFormat: 'yyyy-MM-dd HH:mm' }}\n </span>\n </div>\n }\n @case (AITableFieldType.createdAt) {\n <div class=\"d-block user-select-none\">\n <span class=\"text-truncate\">\n {{ record.values[field._id] | thyDatePickerFormat: 'yyyy-MM-dd HH:mm' }}\n </span>\n </div>\n }\n @case (AITableFieldType.rate) {\n <thy-rate [ngModel]=\"record.values[field._id]\"></thy-rate>\n }\n @case (AITableFieldType.link) {\n <a\n class=\"d-block\"\n target=\"_blank\"\n [href]=\"record.values[field._id]?.url\"\n thyStopPropagation\n thyFlexibleText\n [thyTooltipContent]=\"record.values[field._id]?.text\"\n >\n {{ record.values[field._id]?.text }}\n </a>\n }\n @case (AITableFieldType.progress) {\n <thy-progress\n class=\"w-100\"\n [thyValue]=\"record.values[field._id] || 0\"\n [thySize]=\"record.values[field._id]?.config?.size || 'md'\"\n [thyMax]=\"record.values[field._id]?.config?.max || 100\"\n [thyType]=\"record.values[field._id]?.config?.progressType || 'success'\"\n >\n <span> {{ record.values[field._id] || 0 }}{{ record.values[field._id]?.config?.suffix || '%' }} </span>\n </thy-progress>\n }\n @case (AITableFieldType.member) {\n @let settings = field.settings! | memberSetting;\n\n @if (!settings!['is_multiple']) {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n @if (recordValues && recordValues.length) {\n <thy-avatar\n [thyName]=\"recordValues[0].display_name!\"\n [thySrc]=\"recordValues[0].avatar!\"\n thySize=\"xs\"\n thyShowName=\"true\"\n ></thy-avatar>\n }\n } @else {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n <thy-avatar-list thyAvatarSize=\"xs\">\n @for (item of recordValues; track $index) {\n <thy-avatar [thyName]=\"item.display_name!\" [thySrc]=\"item.avatar!\"></thy-avatar>\n }\n </thy-avatar-list>\n }\n }\n @case (AITableFieldType.createdBy) {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n @if (recordValues && recordValues.length) {\n <thy-avatar\n [thyName]=\"recordValues[0].display_name!\"\n [thySrc]=\"recordValues[0].avatar!\"\n thySize=\"xs\"\n thyShowName=\"true\"\n ></thy-avatar>\n }\n }\n @case (AITableFieldType.updatedBy) {\n @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n @if (recordValues && recordValues.length) {\n <thy-avatar\n [thyName]=\"recordValues[0].display_name!\"\n [thySrc]=\"recordValues[0].avatar!\"\n thySize=\"xs\"\n thyShowName=\"true\"\n ></thy-avatar>\n }\n }\n @default {\n <span class=\"text-truncate\"> {{ record.values[field._id] }}</span>\n }\n }\n <div class=\"autofill-container\"></div>\n </div>\n }\n <div class=\"grid-column-blank\"></div>\n </div>\n }\n <div class=\"grid-row-insert grid-row cursor-pointer\" (click)=\"addRecord()\">\n <thy-icon thyIconName=\"plus\"></thy-icon>\n </div>\n</div>\n\n<div #activeBorder class=\"active-border\"></div>\n" }]
190
- }] });
191
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"grid.component.js","sourceRoot":"","sources":["../../../packages/grid/src/grid.component.ts","../../../packages/grid/src/grid.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule, NgClass, NgComponentOutlet, NgForOf, NgTemplateOutlet } from '@angular/common';\nimport {\n    ChangeDetectionStrategy,\n    Component,\n    computed,\n    DestroyRef,\n    ElementRef,\n    inject,\n    input,\n    model,\n    NgZone,\n    OnInit,\n    output,\n    signal\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { FormsModule } from '@angular/forms';\nimport { ThyAction } from 'ngx-tethys/action';\nimport { ThyAvatarModule } from 'ngx-tethys/avatar';\nimport { ThyCheckboxModule } from 'ngx-tethys/checkbox';\nimport { ThyDatePickerFormatPipe } from 'ngx-tethys/date-picker';\nimport { ThyDropdownDirective, ThyDropdownMenuComponent } from 'ngx-tethys/dropdown';\nimport { ThyFlexibleText } from 'ngx-tethys/flexible-text';\nimport { ThyIcon } from 'ngx-tethys/icon';\nimport { ThyPopoverModule, ThyPopoverRef } from 'ngx-tethys/popover';\nimport { ThyProgress } from 'ngx-tethys/progress';\nimport { ThyRate } from 'ngx-tethys/rate';\nimport { ThyStopPropagationDirective } from 'ngx-tethys/shared';\nimport { ThyTag } from 'ngx-tethys/tag';\nimport { mergeWith } from 'rxjs/operators';\nimport { ProgressEditorComponent, SelectOptionComponent } from './components';\nimport { FieldMenu } from './components/field-menu/field-menu.component';\nimport { AITableFieldPropertyEditor } from './components/field-property-editor/field-property-editor.component';\nimport { DBL_CLICK_EDIT_TYPE, DefaultFieldMenus, MOUSEOVER_EDIT_TYPE } from './constants';\nimport {\n    Actions,\n    AIPlugin,\n    AITable,\n    AITableChangeOptions,\n    AITableFields,\n    AITableFieldType,\n    AITableRecords,\n    AITableSelectOptionStyle,\n    AITableValue,\n    createAITable,\n    createDefaultField,\n    getDefaultRecord\n} from './core';\nimport { IsSelectRecordPipe, MemberSettingPipe, SelectOptionPipe, SelectOptionsPipe, SelectSettingPipe, UserPipe } from './pipes/grid.pipe';\nimport { AITableGridEventService } from './services/event.service';\nimport { AI_TABLE_GRID_FIELD_SERVICE_MAP, AITableGridFieldService } from './services/field.service';\nimport { AITableGridSelectionService } from './services/selection.service';\nimport { AIFieldConfig, AITableFieldMenuItem, AITableReferences } from './types';\n\n@Component({\n    selector: 'ai-table-grid',\n    templateUrl: './grid.component.html',\n    standalone: true,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    host: {\n        class: 'ai-table-grid'\n    },\n    imports: [\n        NgForOf,\n        NgClass,\n        NgComponentOutlet,\n        CommonModule,\n        FormsModule,\n        SelectOptionPipe,\n        SelectOptionsPipe,\n        ThyTag,\n        ThyPopoverModule,\n        ThyIcon,\n        ThyRate,\n        ThyProgress,\n        AITableFieldPropertyEditor,\n        ThyDatePickerFormatPipe,\n        ThyFlexibleText,\n        ThyStopPropagationDirective,\n        FieldMenu,\n        ThyAction,\n        ThyDropdownDirective,\n        ThyDropdownMenuComponent,\n        ThyCheckboxModule,\n        ProgressEditorComponent,\n        ThyAvatarModule,\n        NgTemplateOutlet,\n        IsSelectRecordPipe,\n        ProgressEditorComponent,\n        SelectOptionComponent,\n        UserPipe,\n        SelectSettingPipe,\n        MemberSettingPipe\n    ],\n    providers: [AITableGridEventService, AITableGridFieldService, AITableGridSelectionService]\n})\nexport class AITableGrid implements OnInit {\n    aiRecords = model.required<AITableRecords>();\n\n    aiFields = model.required<AITableFields>();\n\n    aiFieldConfig = input<AIFieldConfig>();\n\n    aiReadonly = input<boolean>();\n\n    aiPlugins = input<AIPlugin[]>();\n\n    aiReferences = input<AITableReferences>();\n\n    aiBuildRenderDataFn = input<(aiTable: AITable) => AITableValue>();\n\n    AITableFieldType = AITableFieldType;\n\n    AITableSelectOptionStyle = AITableSelectOptionStyle;\n\n    aiTable!: AITable;\n\n    gridData = computed(() => {\n        if (this.aiBuildRenderDataFn && this.aiBuildRenderDataFn() && this.aiTable) {\n            return this.aiBuildRenderDataFn()!(this.aiTable);\n        }\n        return {\n            records: this.aiRecords(),\n            fields: this.aiFields()\n        };\n    });\n\n    isSelectedAll = computed(() => {\n        return this.aiTable.selection().selectedRecords.size === this.aiRecords().length;\n    });\n\n    onChange = output<AITableChangeOptions>();\n\n    aiTableInitialized = output<AITable>();\n\n    fieldMenus!: AITableFieldMenuItem[];\n\n    mouseoverRef!: ThyPopoverRef<any>;\n\n    private ngZone = inject(NgZone);\n\n    private elementRef = inject(ElementRef);\n\n    private destroyRef = inject(DestroyRef);\n\n    private aiTableGridFieldService = inject(AITableGridFieldService);\n\n    private aiTableGridEventService = inject(AITableGridEventService);\n\n    public aiTableGridSelectionService = inject(AITableGridSelectionService);\n\n    ngOnInit(): void {\n        this.initAITable();\n        this.initService();\n        this.buildFieldMenus();\n        this.subscribeEvents();\n    }\n\n    initAITable() {\n        this.aiTable = createAITable(this.aiRecords, this.aiFields);\n        this.aiPlugins()?.forEach((plugin) => {\n            this.aiTable = plugin(this.aiTable);\n        });\n        this.aiTableInitialized.emit(this.aiTable);\n        this.aiTable.onChange = () => {\n            this.onChange.emit({\n                records: this.aiRecords(),\n                fields: this.aiFields(),\n                actions: this.aiTable.actions\n            });\n        };\n    }\n\n    initService() {\n        this.aiTableGridEventService.initialize(this.aiTable, this.aiFieldConfig()?.fieldPropertyEditor);\n        this.aiTableGridSelectionService.initialize(this.aiTable);\n        this.aiTableGridEventService.registerEvents(this.elementRef.nativeElement);\n        this.aiTableGridFieldService.initAIFieldConfig(this.aiFieldConfig());\n        AI_TABLE_GRID_FIELD_SERVICE_MAP.set(this.aiTable, this.aiTableGridFieldService);\n    }\n\n    buildFieldMenus() {\n        this.fieldMenus = this.aiFieldConfig()?.fieldMenus ?? DefaultFieldMenus;\n    }\n\n    addRecord() {\n        Actions.addRecord(this.aiTable, getDefaultRecord(this.aiFields()), [this.aiRecords().length]);\n    }\n\n    selectRecord(recordId: string) {\n        this.aiTableGridSelectionService.selectRecord(recordId);\n    }\n\n    toggleSelectAll(checked: boolean) {\n        this.aiTableGridSelectionService.toggleSelectAll(checked);\n    }\n\n    addField(gridColumnBlank: HTMLElement) {\n        const field = createDefaultField(this.aiTable, AITableFieldType.text);\n        this.aiTableGridFieldService.editFieldProperty(gridColumnBlank, this.aiTable, field, false);\n    }\n\n    private subscribeEvents() {\n        this.ngZone.runOutsideAngular(() => {\n            this.aiTableGridEventService.dblClickEvent$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {\n                this.dblClick(event);\n            });\n            this.aiTableGridEventService.mousedownEvent$\n                .pipe(mergeWith(this.aiTableGridEventService.globalMousedownEvent$), takeUntilDestroyed(this.destroyRef))\n                .subscribe((event) => {\n                    this.aiTableGridSelectionService.updateSelect(event);\n                });\n\n            this.aiTableGridEventService.mouseoverEvent$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {\n                this.mouseoverHandle(event);\n            });\n            this.aiTableGridEventService.globalMouseoverEvent$.pipe(takeUntilDestroyed(this.destroyRef)).subscribe((event) => {\n                this.closeHoverCellEditor(event);\n            });\n        });\n    }\n\n    private dblClick(event: MouseEvent) {\n        const cellDom = (event.target as HTMLElement).closest('.grid-cell') as HTMLElement;\n        const type = cellDom && (cellDom.getAttribute('type')! as AITableFieldType);\n        if (type && DBL_CLICK_EDIT_TYPE.includes(type)) {\n            this.aiTableGridEventService.openEdit(cellDom);\n        }\n    }\n\n    private mouseoverHandle(event: MouseEvent) {\n        if (this.mouseoverRef) {\n            this.mouseoverRef?.close();\n        }\n        const cellDom = (event.target as HTMLElement).closest('.grid-cell') as HTMLElement;\n        const type = cellDom && (cellDom.getAttribute('type')! as AITableFieldType);\n        if (type && MOUSEOVER_EDIT_TYPE.includes(type)) {\n            this.mouseoverRef = this.aiTableGridEventService.openEdit(cellDom);\n        }\n    }\n\n    private closeHoverCellEditor(e: MouseEvent) {\n        if (this.mouseoverRef) {\n            const hasGrid = e.target && (e.target as HTMLElement).closest('.ai-table-grid');\n            const hasCellEditor = e.target && (e.target as HTMLElement).closest('.grid-cell-editor');\n            if (!hasGrid && !hasCellEditor) {\n                this.mouseoverRef.close();\n            }\n        }\n    }\n}\n","<div class=\"grid-header d-flex\">\n    <div class=\"grid-column-checkbox grid-cell grid-checkbox\">\n        <label thyCheckbox thyLabelText=\"\" [ngModel]=\"isSelectedAll()\" (ngModelChange)=\"toggleSelectAll($event)\"></label>\n    </div>\n    @for (field of gridData().fields; track field._id) {\n        <div\n            class=\"grid-cell grid-field\"\n            #fieldAction\n            [attr.fieldId]=\"field._id\"\n            [ngClass]=\"{ highlight: aiTable.selection().selectedFields.has(field._id) }\"\n            [ngStyle]=\"{ width: field.width + 'px' }\"\n        >\n            <span class=\"text-truncate\">\n                <thy-icon [thyIconName]=\"field.icon!\" class=\"mr-2 text-muted\"></thy-icon>\n                <span>{{ field.name }}</span>\n            </span>\n            <a\n                class=\"grid-field-action\"\n                thyAction\n                thyActiveClass=\"active\"\n                thyIcon=\"more-vertical\"\n                [thyDropdown]=\"fieldMenu\"\n                href=\"javascript:;\"\n            >\n                <thy-dropdown-menu #fieldMenu>\n                    <field-menu [origin]=\"fieldAction\" [fieldId]=\"field._id\" [aiTable]=\"aiTable\" [fieldMenus]=\"fieldMenus\"></field-menu>\n                </thy-dropdown-menu>\n            </a>\n        </div>\n    }\n    <div class=\"grid-column-blank cursor-pointer\" #gridColumnBlank (click)=\"addField(gridColumnBlank)\">\n        <thy-icon thyIconName=\"plus\"></thy-icon>\n    </div>\n</div>\n<div class=\"grid-body d-flex\">\n    @for (record of gridData().records; track record._id; let index = $index) {\n        <div class=\"grid-row d-flex\" [ngClass]=\"{ highlight: (record._id | isSelectRecord: aiTable.selection()) }\">\n            <div class=\"grid-row-index grid-checkbox\">\n                <label\n                    [ngClass]=\"(record._id | isSelectRecord: aiTable.selection()) ? 'checked-box' : 'unchecked-box'\"\n                    thyCheckbox\n                    thyLabelText=\"\"\n                    [ngModel]=\"record._id | isSelectRecord: aiTable.selection()\"\n                    (ngModelChange)=\"selectRecord(record._id)\"\n                ></label>\n                <span [ngClass]=\"(record._id | isSelectRecord: aiTable.selection()) ? 'grid-row-no-number' : 'grid-row-number'\">\n                    {{ index + 1 }}\n                </span>\n            </div>\n            @for (field of gridData().fields; track field._id) {\n                <div\n                    #cell\n                    class=\"grid-cell\"\n                    [ngClass]=\"{\n                        highlight: aiTable.selection().selectedCells.has(record._id) || aiTable.selection().selectedFields.has(field._id),\n                        selected: aiTable.selection().selectedCells.get(record._id)?.hasOwnProperty(field._id)\n                    }\"\n                    [attr.type]=\"[field.type]\"\n                    [attr.fieldId]=\"[field._id]\"\n                    [attr.recordId]=\"[record._id]\"\n                    [ngStyle]=\"{ width: field.width + 'px' }\"\n                >\n                    @switch (field.type) {\n                        @case (AITableFieldType.select) {\n                            @let fieldValue = record.values[field._id];\n                            @let settings = field.settings! | selectSetting;\n                            @let options = settings['options'];\n                            @let optionStyle = settings['option_style'] || AITableSelectOptionStyle.tag;\n                            @let isTagStyle = optionStyle === AITableSelectOptionStyle.tag;\n\n                            @if (!settings['is_multiple'] && fieldValue | selectOption: options; as selectedOption) {\n                                @if (isTagStyle) {\n                                    <select-option class=\"mb-1 mr-1\" [field]=\"field\" [displayOption]=\"selectedOption\"></select-option>\n                                } @else {\n                                    <div thyTag class=\"mb-1 mr-1\">\n                                        <select-option [field]=\"field\" [displayOption]=\"selectedOption\"></select-option>\n                                    </div>\n                                }\n                            } @else {\n                                @let maxShowCount = 2;\n\n                                <div class=\"d-flex\">\n                                    @if (fieldValue | selectOptions: options; as selectedOptions) {\n                                        @for (option of selectedOptions; track option!._id; let i = $index) {\n                                            @if (i + 1 <= maxShowCount) {\n                                                @if (isTagStyle) {\n                                                    <select-option\n                                                        class=\"mb-1 mr-1\"\n                                                        [field]=\"field\"\n                                                        [displayOption]=\"option!\"\n                                                    ></select-option>\n                                                } @else {\n                                                    <div thyTag class=\"mb-1 mr-1\">\n                                                        <select-option [field]=\"field\" [displayOption]=\"option!\"></select-option>\n                                                    </div>\n                                                }\n                                            }\n                                        }\n\n                                        @let selectedLength = selectedOptions.length || 0;\n                                        @if (selectedOptions && maxShowCount < selectedLength) {\n                                            @let shape = isTagStyle ? 'pill' : 'rectangle';\n                                            @let isHidden = maxShowCount >= selectedLength;\n\n                                            <thy-tag\n                                                class=\"cursor-pointer\"\n                                                [class.multi-property-value-hidden]=\"isHidden\"\n                                                [thyShape]=\"shape\"\n                                            >\n                                                <span class=\"text-truncate\"> +{{ selectedLength - maxShowCount }} </span>\n                                            </thy-tag>\n                                        }\n                                    }\n                                </div>\n                            }\n                        }\n                        @case (AITableFieldType.date) {\n                            {{ record.values[field._id].timestamp | thyDatePickerFormat }}\n                        }\n                        @case (AITableFieldType.updatedAt) {\n                            <div class=\"d-block user-select-none\">\n                                <span class=\"text-truncate\">\n                                    {{ record.values[field._id] | thyDatePickerFormat: 'yyyy-MM-dd HH:mm' }}\n                                </span>\n                            </div>\n                        }\n                        @case (AITableFieldType.createdAt) {\n                            <div class=\"d-block user-select-none\">\n                                <span class=\"text-truncate\">\n                                    {{ record.values[field._id] | thyDatePickerFormat: 'yyyy-MM-dd HH:mm' }}\n                                </span>\n                            </div>\n                        }\n                        @case (AITableFieldType.rate) {\n                            <thy-rate [ngModel]=\"record.values[field._id]\"></thy-rate>\n                        }\n                        @case (AITableFieldType.link) {\n                            <a\n                                class=\"d-block\"\n                                target=\"_blank\"\n                                [href]=\"record.values[field._id]?.url\"\n                                thyStopPropagation\n                                thyFlexibleText\n                                [thyTooltipContent]=\"record.values[field._id]?.text\"\n                            >\n                                {{ record.values[field._id]?.text }}\n                            </a>\n                        }\n                        @case (AITableFieldType.progress) {\n                            <thy-progress\n                                class=\"w-100\"\n                                [thyValue]=\"record.values[field._id] || 0\"\n                                [thySize]=\"record.values[field._id]?.config?.size || 'md'\"\n                                [thyMax]=\"record.values[field._id]?.config?.max || 100\"\n                                [thyType]=\"record.values[field._id]?.config?.progressType || 'success'\"\n                            >\n                                <span> {{ record.values[field._id] || 0 }}{{ record.values[field._id]?.config?.suffix || '%' }} </span>\n                            </thy-progress>\n                        }\n                        @case (AITableFieldType.member) {\n                            @let settings = field.settings! | memberSetting;\n\n                            @if (!settings!['is_multiple']) {\n                                @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n                                @if (recordValues && recordValues.length) {\n                                    <thy-avatar\n                                        [thyName]=\"recordValues[0].display_name!\"\n                                        [thySrc]=\"recordValues[0].avatar!\"\n                                        thySize=\"xs\"\n                                        thyShowName=\"true\"\n                                    ></thy-avatar>\n                                }\n                            } @else {\n                                @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n                                <thy-avatar-list thyAvatarSize=\"xs\">\n                                    @for (item of recordValues; track $index) {\n                                        <thy-avatar [thyName]=\"item.display_name!\" [thySrc]=\"item.avatar!\"></thy-avatar>\n                                    }\n                                </thy-avatar-list>\n                            }\n                        }\n                        @case (AITableFieldType.createdBy) {\n                            @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n                            @if (recordValues && recordValues.length) {\n                                <thy-avatar\n                                    [thyName]=\"recordValues[0].display_name!\"\n                                    [thySrc]=\"recordValues[0].avatar!\"\n                                    thySize=\"xs\"\n                                    thyShowName=\"true\"\n                                ></thy-avatar>\n                            }\n                        }\n                        @case (AITableFieldType.updatedBy) {\n                            @let recordValues = record.values[field._id] | user: aiReferences()!;\n\n                            @if (recordValues && recordValues.length) {\n                                <thy-avatar\n                                    [thyName]=\"recordValues[0].display_name!\"\n                                    [thySrc]=\"recordValues[0].avatar!\"\n                                    thySize=\"xs\"\n                                    thyShowName=\"true\"\n                                ></thy-avatar>\n                            }\n                        }\n                        @default {\n                            <span class=\"text-truncate\"> {{ record.values[field._id] }}</span>\n                        }\n                    }\n                    <div class=\"autofill-container\"></div>\n                </div>\n            }\n            <div class=\"grid-column-blank\"></div>\n        </div>\n    }\n    <div class=\"grid-row-insert grid-row cursor-pointer\" (click)=\"addRecord()\">\n        <thy-icon thyIconName=\"plus\"></thy-icon>\n    </div>\n</div>\n\n<div #activeBorder class=\"active-border\"></div>\n"]}
365
+ }, imports: [AITableRenderer], providers: [AITableGridEventService, AITableGridFieldService, AITableGridSelectionService], template: "<div #container class=\"ai-table-grid-view\">\n @if (hasContainerRect()) {\n <ai-table-renderer\n [config]=\"rendererConfig()\"\n (koMousemove)=\"stageMousemove($event)\"\n (koMousedown)=\"stageMousedown($event)\"\n (koClick)=\"stageClick($event)\"\n (koDblclick)=\"stageDblclick($event)\"\n >\n <div #horizontalBar class=\"ai-table-horizontal-scroll-bar-wrapper\" [style.width.px]=\"containerRect().width\">\n <div class=\"ai-table-scroll-bar-inner\" [style.width.px]=\"coordinate().totalWidth + ADD_BUTTON_WIDTH\"></div>\n </div>\n <div\n #verticalBar\n class=\"ai-table-vertical-scroll-bar-wrapper\"\n [style.height.px]=\"containerRect().height - fieldHeadHeight\"\n [style.top.px]=\"fieldHeadHeight\"\n >\n <div class=\"ai-table-scroll-bar-inner\" [style.height.px]=\"scrollTotalHeight()\"></div>\n </div>\n </ai-table-renderer>\n }\n</div>\n" }]
366
+ }], ctorParameters: () => [] });
367
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"grid.component.js","sourceRoot":"","sources":["../../../packages/grid/src/grid.component.ts","../../../packages/grid/src/grid.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    afterNextRender,\n    ChangeDetectionStrategy,\n    Component,\n    computed,\n    effect,\n    ElementRef,\n    OnDestroy,\n    OnInit,\n    Signal,\n    signal,\n    viewChild\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { filter, fromEvent } from 'rxjs';\nimport { KoEventObject } from './angular-konva';\nimport { AbstractEditCellEditor } from './components/cell-editors/abstract-cell-editor.component';\nimport {\n    AI_TABLE_CELL,\n    AI_TABLE_FIELD_ADD_BUTTON,\n    AI_TABLE_FIELD_ADD_BUTTON_WIDTH,\n    AI_TABLE_FIELD_HEAD,\n    AI_TABLE_FIELD_HEAD_HEIGHT,\n    AI_TABLE_FIELD_HEAD_MORE,\n    AI_TABLE_FIELD_HEAD_SELECT_CHECKBOX,\n    AI_TABLE_POPOVER_LEFT_OFFSET,\n    AI_TABLE_ROW_ADD_BUTTON,\n    AI_TABLE_ROW_HEAD_WIDTH,\n    AI_TABLE_ROW_SELECT_CHECKBOX,\n    DBL_CLICK_EDIT_TYPE,\n    DEFAULT_POINT_POSITION,\n    DEFAULT_SCROLL_STATE,\n    MOUSEOVER_EDIT_TYPE\n} from './constants';\nimport { AITable, Coordinate, RendererContext } from './core';\nimport { AITableGridBase } from './grid-base.component';\nimport { AITableRenderer } from './renderer/renderer.component';\nimport { AITableGridEventService } from './services/event.service';\nimport { AITableGridFieldService } from './services/field.service';\nimport { AITableGridSelectionService } from './services/selection.service';\nimport { AITableMouseDownType, AITableRendererConfig, ScrollActionOptions } from './types';\nimport { buildGridLinearRows, getColumnIndicesMap, getDetailByTargetName, handleMouseStyle, isWindows } from './utils';\nimport { getMousePosition } from './utils/position';\n\n@Component({\n    selector: 'ai-table-grid',\n    templateUrl: './grid.component.html',\n    standalone: true,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    host: {\n        class: 'ai-table-grid'\n    },\n    imports: [AITableRenderer],\n    providers: [AITableGridEventService, AITableGridFieldService, AITableGridSelectionService]\n})\nexport class AITableGrid extends AITableGridBase implements OnInit, OnDestroy {\n    timer!: number | null;\n\n    resizeObserver!: ResizeObserver;\n\n    fieldHeadHeight = AI_TABLE_FIELD_HEAD_HEIGHT;\n\n    ADD_BUTTON_WIDTH = AI_TABLE_FIELD_ADD_BUTTON_WIDTH;\n\n    containerRect = signal({ width: 0, height: 0 });\n\n    hasContainerRect = computed(() => {\n        return this.containerRect().width > 0 && this.containerRect().height > 0;\n    });\n\n    container = viewChild<ElementRef>('container');\n\n    verticalBarRef = viewChild<ElementRef>('verticalBar');\n\n    horizontalBarRef = viewChild<ElementRef>('horizontalBar');\n\n    linearRows = computed(() => {\n        return buildGridLinearRows(this.gridData().records);\n    });\n\n    visibleColumnsMap = computed(() => {\n        const columns = AITable.getVisibleFields(this.aiTable);\n        return new Map(columns?.map((item, index) => [item._id, index]));\n    });\n\n    visibleRowsIndexMap = computed(() => {\n        return new Map(this.linearRows().map((row, index) => [row._id, index]));\n    });\n\n    containerElement = computed(() => {\n        return this.container()!.nativeElement;\n    });\n\n    rendererConfig: Signal<AITableRendererConfig> = computed(() => {\n        const fields = AITable.getVisibleFields(this.aiTable);\n        const coordinate = new Coordinate({\n            container: this.containerElement(),\n            rowHeight: AI_TABLE_FIELD_HEAD_HEIGHT,\n            rowCount: this.linearRows().length,\n            columnCount: fields.length,\n            rowInitSize: AI_TABLE_FIELD_HEAD_HEIGHT,\n            columnInitSize: AI_TABLE_ROW_HEAD_WIDTH,\n            rowIndicesMap: {},\n            columnIndicesMap: getColumnIndicesMap(fields),\n            frozenColumnCount: 1\n        });\n        return {\n            aiTable: this.aiTable,\n            container: this.containerElement(),\n            coordinate: coordinate,\n            containerWidth: this.containerRect().width,\n            containerHeight: this.containerRect().height,\n            references: this.aiReferences()\n        };\n    });\n\n    coordinate = computed(() => {\n        return this.rendererConfig().coordinate;\n    });\n\n    scrollTotalHeight = computed(() => {\n        return Math.max(this.coordinate().totalHeight, this.containerRect().height - this.fieldHeadHeight);\n    });\n\n    constructor() {\n        super();\n        afterNextRender(() => {\n            this.setContainerRect();\n            this.bindGlobalMousedown();\n            this.containerResizeListener();\n            this.bindWheel();\n        });\n        effect(() => {\n            if (this.hasContainerRect() && this.horizontalBarRef() && this.verticalBarRef()) {\n                this.bindScrollBarScroll();\n            }\n        });\n        effect(() => {\n            if (this.aiTable.context?.pointPosition()) {\n                this.toggleHoverCellEditor();\n            }\n        });\n    }\n\n    override ngOnInit(): void {\n        super.ngOnInit();\n        this.initContext();\n    }\n\n    ngOnDestroy(): void {\n        this.resizeObserver?.disconnect();\n    }\n\n    private initContext() {\n        this.aiTable.context = new RendererContext({\n            linearRows: this.linearRows,\n            visibleColumnsMap: this.visibleColumnsMap,\n            visibleRowsIndexMap: this.visibleRowsIndexMap,\n            pointPosition: signal(DEFAULT_POINT_POSITION),\n            scrollState: signal(DEFAULT_SCROLL_STATE),\n            scrollAction: this.scrollAction\n        });\n    }\n\n    stageMousemove(e: KoEventObject<MouseEvent>) {\n        if (this.timer) {\n            cancelAnimationFrame(this.timer);\n        }\n        this.timer = requestAnimationFrame(() => {\n            const targetName = e.event.target.name();\n            const gridStage = e.event.currentTarget.getStage();\n            const pos = gridStage?.getPointerPosition();\n            if (pos == null) return;\n            const { context } = this.aiTable;\n            const { x, y } = pos;\n            const curMousePosition = getMousePosition(\n                x,\n                y,\n                this.coordinate(),\n                AITable.getVisibleFields(this.aiTable),\n                context!,\n                targetName\n            );\n            handleMouseStyle(curMousePosition.realTargetName, curMousePosition.areaType, this.containerElement());\n            context!.setPointPosition(curMousePosition);\n            this.timer = null;\n        });\n    }\n\n    stageMousedown(e: KoEventObject<MouseEvent>) {\n        const mouseEvent = e.event.evt;\n        const _targetName = e.event.target.name();\n\n        const { targetName, fieldId, recordId } = getDetailByTargetName(_targetName);\n        switch (targetName) {\n            case AI_TABLE_FIELD_HEAD:\n                mouseEvent.preventDefault();\n                if (!fieldId) return;\n                this.aiTableGridSelectionService.selectField(fieldId);\n                return;\n            case AI_TABLE_CELL:\n                if (!recordId || !fieldId) return;\n                this.aiTableGridEventService.closeCellEditor();\n                this.aiTableGridSelectionService.selectCell(recordId, fieldId);\n                return;\n            case AI_TABLE_ROW_ADD_BUTTON:\n            case AI_TABLE_FIELD_ADD_BUTTON:\n            case AI_TABLE_ROW_SELECT_CHECKBOX:\n            case AI_TABLE_FIELD_HEAD_SELECT_CHECKBOX:\n                return;\n            default:\n                this.aiTableGridSelectionService.clearSelection();\n        }\n    }\n\n    stageClick(e: KoEventObject<MouseEvent>) {\n        const mouseEvent = e.event.evt;\n        mouseEvent.preventDefault();\n        const { context } = this.aiTable;\n        const { targetName, rowIndex: pointRowIndex } = context!.pointPosition();\n        if (mouseEvent.button !== AITableMouseDownType.Left) return;\n        switch (targetName) {\n            case AI_TABLE_ROW_ADD_BUTTON: {\n                this.aiTableGridSelectionService.clearSelection();\n                this.addRecord();\n                return;\n            }\n            case AI_TABLE_ROW_SELECT_CHECKBOX: {\n                const pointRecordId = context!.linearRows()[pointRowIndex]?._id;\n                this.selectRecord(pointRecordId);\n                return;\n            }\n            case AI_TABLE_FIELD_HEAD_SELECT_CHECKBOX: {\n                const isChecked = this.aiTable.selection().selectedRecords.size === this.aiTable.records().length;\n                this.toggleSelectAll(!isChecked);\n                return;\n            }\n            case AI_TABLE_FIELD_ADD_BUTTON: {\n                this.aiTableGridSelectionService.clearSelection();\n                const fieldGroupRect = e.event.target.getParent()?.getClientRect()!;\n                const containerRect = this.containerElement().getBoundingClientRect();\n                this.addField(this.containerElement(), {\n                    x: fieldGroupRect.x + containerRect.x,\n                    y: containerRect.y + fieldGroupRect.y + fieldGroupRect.height\n                });\n                return;\n            }\n            case AI_TABLE_FIELD_HEAD_MORE:\n                mouseEvent.preventDefault();\n                const _targetName = e.event.target.name();\n                const { fieldId } = getDetailByTargetName(_targetName);\n                if (fieldId) {\n                    const moreRect = e.event.target.getClientRect();\n                    const fieldGroupRect = e.event.target.getParent()?.getParent()?.getClientRect()!;\n                    const containerRect = this.containerElement().getBoundingClientRect();\n\n                    const position = {\n                        x: containerRect.x + moreRect.x,\n                        y: containerRect.y + moreRect.y + moreRect.height\n                    };\n                    const editOriginPosition = {\n                        x: AI_TABLE_POPOVER_LEFT_OFFSET + fieldGroupRect.x,\n                        y: containerRect.y + fieldGroupRect.y + fieldGroupRect.height\n                    };\n\n                    const editOrigin = this.containerElement().querySelector('.konvajs-content') as HTMLElement;\n                    this.aiTableGridFieldService.openFieldMenu(this.aiTable, {\n                        fieldId: fieldId,\n                        fieldMenus: this.fieldMenus,\n                        origin: this.containerElement(),\n                        position,\n                        editOrigin: editOrigin,\n                        editOriginPosition\n                    });\n                }\n                return;\n        }\n    }\n\n    stageDblclick(e: KoEventObject<MouseEvent>) {\n        const _targetName = e.event.target.name();\n        const { fieldId, recordId } = getDetailByTargetName(_targetName);\n        if (!recordId || !fieldId) {\n            return;\n        }\n        const field = this.aiTable.fieldsMap()[fieldId];\n        const fieldType = field.type;\n        if (!DBL_CLICK_EDIT_TYPE.includes(fieldType)) {\n            return;\n        }\n        const popoverRef = this.aiTableGridEventService.openCellEditor(this.aiTable, {\n            container: this.containerElement(),\n            coordinate: this.coordinate(),\n            fieldId: fieldId!,\n            recordId: recordId!\n        });\n        if (popoverRef && !(this.aiFieldConfig()?.fieldPropertyEditor && this.aiFieldConfig()?.fieldPropertyEditor[fieldType])) {\n            (popoverRef.componentInstance as AbstractEditCellEditor<any>).updateFieldValue.subscribe((value) => {\n                this.aiUpdateFieldValue.emit(value);\n            });\n        }\n    }\n\n    private bindWheel() {\n        fromEvent<WheelEvent>(this.containerElement(), 'wheel', { passive: false })\n            .pipe(takeUntilDestroyed(this.destroyRef))\n            .subscribe((e: WheelEvent) => {\n                e.preventDefault();\n                this.aiTableGridEventService.closeCellEditor();\n                this.scrollAction({ deltaX: e.deltaX, deltaY: e.deltaY, shiftKey: e.shiftKey });\n            });\n    }\n\n    scrollAction = (options: ScrollActionOptions) => {\n        if (this.timer) {\n            cancelAnimationFrame(this.timer);\n        }\n        this.timer = requestAnimationFrame(() => {\n            const { deltaX, deltaY, shiftKey } = options;\n            const fixedDeltaY = shiftKey && isWindows ? 0 : deltaY;\n            const fixedDeltaX = shiftKey && isWindows ? deltaY : deltaX;\n            const horizontalBar = this.horizontalBarRef()?.nativeElement;\n            const verticalBar = this.verticalBarRef()?.nativeElement;\n            if (horizontalBar) {\n                horizontalBar.scrollLeft = horizontalBar.scrollLeft + fixedDeltaX;\n            }\n            if (verticalBar) {\n                verticalBar.scrollTop = verticalBar.scrollTop + fixedDeltaY;\n            }\n            options.callback && options.callback();\n            this.timer = null;\n        });\n    };\n\n    private bindScrollBarScroll() {\n        fromEvent<WheelEvent>(this.horizontalBarRef()!.nativeElement, 'scroll', { passive: true })\n            .pipe(takeUntilDestroyed(this.destroyRef))\n            .subscribe((e) => {\n                this.horizontalScroll(e);\n            });\n\n        fromEvent<WheelEvent>(this.verticalBarRef()!.nativeElement, 'scroll', { passive: true })\n            .pipe(takeUntilDestroyed(this.destroyRef))\n            .subscribe((e) => {\n                this.verticalScroll(e);\n            });\n    }\n\n    private bindGlobalMousedown() {\n        fromEvent<MouseEvent>(document, 'mousedown', { passive: true })\n            .pipe(\n                filter((e) => e.target instanceof Element && !this.containerElement().contains(e.target)),\n                takeUntilDestroyed(this.destroyRef)\n            )\n            .subscribe(() => {\n                this.aiTableGridSelectionService.clearSelection();\n            });\n    }\n\n    private resetScrolling = () => {\n        this.aiTable.context!.setScrollState({\n            isScrolling: false\n        });\n    };\n\n    private horizontalScroll = (e: any) => {\n        const { scrollLeft } = e.target;\n        this.aiTable.context!.setScrollState({\n            scrollLeft,\n            isScrolling: true\n        });\n        this.resetScrolling();\n    };\n\n    private verticalScroll(e: any) {\n        const { scrollTop } = e.target;\n        this.aiTable.context!.setScrollState({\n            scrollTop,\n            isScrolling: true\n        });\n        this.resetScrolling();\n    }\n\n    private setContainerRect() {\n        this.containerRect.set({\n            width: this.containerElement().offsetWidth,\n            height: this.containerElement().offsetHeight\n        });\n    }\n\n    private containerResizeListener() {\n        this.resizeObserver = new ResizeObserver(() => {\n            const containerWidth = this.containerElement().offsetWidth;\n            const totalWidth = this.coordinate().totalWidth + this.ADD_BUTTON_WIDTH;\n            this.setContainerRect();\n            if (containerWidth >= totalWidth) {\n                this.aiTable.context!.setScrollState({ scrollLeft: 0 });\n                return;\n            }\n        });\n        this.resizeObserver.observe(this.containerElement());\n    }\n\n    private toggleHoverCellEditor() {\n        const { realTargetName } = this.aiTable.context?.pointPosition()!;\n        const { targetName, fieldId, recordId } = getDetailByTargetName(realTargetName!);\n        const editingCell = this.aiTableGridEventService.getCurrentEditCell();\n\n        if (targetName === AI_TABLE_CELL && recordId && fieldId) {\n            const field = this.aiTable.fieldsMap()[fieldId];\n\n            if (!field) {\n                return;\n            }\n\n            const fieldType = field.type;\n            const editingFieldType = editingCell ? this.aiTable.fieldsMap()[editingCell.fieldId].type : null;\n            const isEditingFieldTypeHovered = editingFieldType ? MOUSEOVER_EDIT_TYPE.includes(editingFieldType) : false;\n            const isFieldTypeHovered = MOUSEOVER_EDIT_TYPE.includes(fieldType);\n\n            if (editingCell && isEditingFieldTypeHovered) {\n                this.aiTableGridEventService.closeCellEditor();\n            }\n\n            if (!editingCell && !isFieldTypeHovered) {\n                this.aiTableGridEventService.closeCellEditor();\n                return;\n            }\n\n            if (editingCell && ((!isEditingFieldTypeHovered && isFieldTypeHovered) || !isFieldTypeHovered)) {\n                return;\n            }\n\n            setTimeout(() => {\n                this.aiTableGridEventService.openCellEditor(this.aiTable, {\n                    container: this.containerElement(),\n                    coordinate: this.coordinate(),\n                    fieldId: fieldId!,\n                    recordId: recordId!,\n                    isHoverEdit: true\n                });\n            });\n        } else {\n            // 鼠标位于非单元格区域时，如果当前有 mouseover 编辑元素，则结束编辑\n            if (editingCell && MOUSEOVER_EDIT_TYPE.includes(this.aiTable.fieldsMap()[editingCell.fieldId].type)) {\n                this.aiTableGridEventService.closeCellEditor();\n            }\n        }\n    }\n}\n","<div #container class=\"ai-table-grid-view\">\n    @if (hasContainerRect()) {\n        <ai-table-renderer\n            [config]=\"rendererConfig()\"\n            (koMousemove)=\"stageMousemove($event)\"\n            (koMousedown)=\"stageMousedown($event)\"\n            (koClick)=\"stageClick($event)\"\n            (koDblclick)=\"stageDblclick($event)\"\n        >\n            <div #horizontalBar class=\"ai-table-horizontal-scroll-bar-wrapper\" [style.width.px]=\"containerRect().width\">\n                <div class=\"ai-table-scroll-bar-inner\" [style.width.px]=\"coordinate().totalWidth + ADD_BUTTON_WIDTH\"></div>\n            </div>\n            <div\n                #verticalBar\n                class=\"ai-table-vertical-scroll-bar-wrapper\"\n                [style.height.px]=\"containerRect().height - fieldHeadHeight\"\n                [style.top.px]=\"fieldHeadHeight\"\n            >\n                <div class=\"ai-table-scroll-bar-inner\" [style.height.px]=\"scrollTotalHeight()\"></div>\n            </div>\n        </ai-table-renderer>\n    }\n</div>\n"]}