@nixxie-cms/core 1.0.0 → 1.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (186) hide show
  1. package/README.md +2 -2
  2. package/admin-ui/components/dist/nixxie-cms-core-admin-ui-components.cjs.js +4 -4
  3. package/admin-ui/components/dist/nixxie-cms-core-admin-ui-components.esm.js +4 -4
  4. package/admin-ui/context/dist/nixxie-cms-core-admin-ui-context.cjs.js +2 -2
  5. package/admin-ui/context/dist/nixxie-cms-core-admin-ui-context.esm.js +2 -2
  6. package/context/dist/nixxie-cms-core-context.cjs.js +2 -2
  7. package/context/dist/nixxie-cms-core-context.esm.js +2 -2
  8. package/dist/{CreateItemDialog-33335548.esm.js → CreateItemDialog-7008b050.esm.js} +1 -1
  9. package/dist/{CreateItemDialog-56cf59b7.cjs.js → CreateItemDialog-a0cab315.cjs.js} +1 -1
  10. package/dist/{PageContainer-7db73317.esm.js → PageContainer-5ae731cc.esm.js} +25 -18
  11. package/dist/{PageContainer-27c27f10.cjs.js → PageContainer-abd7159f.cjs.js} +25 -18
  12. package/dist/{admin-meta-graphql-6f7f5331.esm.js → admin-meta-graphql-0e6e606e.esm.js} +1 -1
  13. package/dist/{admin-meta-graphql-c8f926e9.cjs.js → admin-meta-graphql-306c224a.cjs.js} +1 -1
  14. package/dist/{context-3132c3ed.esm.js → context-af9957ed.esm.js} +2 -2
  15. package/dist/{context-e7a45152.cjs.js → context-b5204629.cjs.js} +2 -2
  16. package/dist/declarations/src/admin-ui/components/Navigation.d.ts.map +1 -1
  17. package/dist/declarations/src/admin-ui/components/PageContainer.d.ts.map +1 -1
  18. package/dist/declarations/src/helpers.d.ts.map +1 -1
  19. package/dist/declarations/src/index.d.ts +1 -0
  20. package/dist/declarations/src/index.d.ts.map +1 -1
  21. package/dist/declarations/src/internal-unstable/admin-ui/id-field-view.d.ts.map +1 -0
  22. package/dist/declarations/src/internal-unstable/admin-ui/pages/App/index.d.ts.map +1 -0
  23. package/dist/declarations/src/internal-unstable/admin-ui/pages/CreateItemPage/index.d.ts.map +1 -0
  24. package/dist/declarations/src/internal-unstable/admin-ui/pages/HomePage/index.d.ts.map +1 -0
  25. package/dist/declarations/src/internal-unstable/admin-ui/pages/ItemPage/index.d.ts.map +1 -0
  26. package/dist/declarations/src/internal-unstable/admin-ui/pages/ListPage/index.d.ts.map +1 -0
  27. package/dist/declarations/src/internal-unstable/admin-ui/pages/NoAccessPage/index.d.ts.map +1 -0
  28. package/dist/declarations/src/internal-unstable/artifacts.d.ts.map +1 -0
  29. package/dist/declarations/src/lib/core/initialise-lists.d.ts +1 -1
  30. package/dist/declarations/src/schema.d.ts.map +1 -1
  31. package/dist/declarations/src/types/config/index.d.ts +60 -1
  32. package/dist/declarations/src/types/config/index.d.ts.map +1 -1
  33. package/dist/declarations/src/types/config/lists.d.ts +4 -4
  34. package/dist/declarations/src/types/context.d.ts +150 -0
  35. package/dist/declarations/src/types/context.d.ts.map +1 -1
  36. package/dist/declarations/src/types/next-fields.d.ts +1 -1
  37. package/dist/{express-e9ed9a7d.cjs.js → express-455ae20c.cjs.js} +1 -1
  38. package/dist/{express-6743b918.esm.js → express-7559ca2d.esm.js} +1 -1
  39. package/dist/{index-ac01583b.cjs.js → index-89635494.cjs.js} +4 -4
  40. package/dist/{index-24b78415.esm.js → index-baa799e0.esm.js} +4 -4
  41. package/dist/nixxie-cms-core.cjs.js +104 -77
  42. package/dist/nixxie-cms-core.esm.js +104 -77
  43. package/dist/{non-null-graphql-5315718c.esm.js → non-null-graphql-a84ed64d.esm.js} +1 -1
  44. package/dist/{non-null-graphql-17b83ddc.cjs.js → non-null-graphql-add6bb3d.cjs.js} +1 -1
  45. package/dist/{resolve-hooks-66fe8a8e.cjs.js → resolve-hooks-165a9ce2.cjs.js} +1 -1
  46. package/dist/{resolve-hooks-17aafd37.esm.js → resolve-hooks-6813a045.esm.js} +2 -2
  47. package/dist/{system-dfec2f0a.esm.js → system-03e49e4f.esm.js} +8 -4
  48. package/dist/{system-48c5f6df.cjs.js → system-a321642d.cjs.js} +8 -4
  49. package/dist/{useFilter-0b5a1ee6.esm.js → useFilter-9b6db1f9.esm.js} +1 -1
  50. package/dist/{useFilter-1a4e6900.cjs.js → useFilter-acc9d413.cjs.js} +1 -1
  51. package/fields/dist/nixxie-cms-core-fields.cjs.js +16 -16
  52. package/fields/dist/nixxie-cms-core-fields.esm.js +17 -17
  53. package/fields/types/bytes/dist/nixxie-cms-core-fields-types-bytes.cjs.js +3 -3
  54. package/fields/types/bytes/dist/nixxie-cms-core-fields-types-bytes.esm.js +3 -3
  55. package/fields/types/bytes/views/dist/nixxie-cms-core-fields-types-bytes-views.cjs.js +1 -1
  56. package/fields/types/bytes/views/dist/nixxie-cms-core-fields-types-bytes-views.esm.js +1 -1
  57. package/fields/types/password/dist/nixxie-cms-core-fields-types-password.cjs.js +3 -3
  58. package/fields/types/password/dist/nixxie-cms-core-fields-types-password.esm.js +3 -3
  59. package/fields/types/relationship/views/dist/nixxie-cms-core-fields-types-relationship-views.cjs.js +4 -4
  60. package/fields/types/relationship/views/dist/nixxie-cms-core-fields-types-relationship-views.esm.js +4 -4
  61. package/fields/types/select/views/dist/nixxie-cms-core-fields-types-select-views.cjs.js +1 -1
  62. package/fields/types/select/views/dist/nixxie-cms-core-fields-types-select-views.esm.js +1 -1
  63. package/fields/types/text/views/dist/nixxie-cms-core-fields-types-text-views.cjs.js +1 -1
  64. package/fields/types/text/views/dist/nixxie-cms-core-fields-types-text-views.esm.js +1 -1
  65. package/internal-unstable/admin-ui/id-field-view/dist/nixxie-cms-core-internal-unstable-admin-ui-id-field-view.cjs.d.ts +2 -0
  66. package/internal-unstable/admin-ui/id-field-view/dist/nixxie-cms-core-internal-unstable-admin-ui-id-field-view.cjs.js +244 -0
  67. package/internal-unstable/admin-ui/id-field-view/dist/nixxie-cms-core-internal-unstable-admin-ui-id-field-view.esm.js +235 -0
  68. package/internal-unstable/admin-ui/id-field-view/package.json +4 -0
  69. package/internal-unstable/admin-ui/next-config/package.json +4 -0
  70. package/internal-unstable/admin-ui/pages/App/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-App.cjs.d.ts +2 -0
  71. package/internal-unstable/admin-ui/pages/App/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-App.cjs.js +59 -0
  72. package/internal-unstable/admin-ui/pages/App/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-App.esm.js +55 -0
  73. package/internal-unstable/admin-ui/pages/App/package.json +4 -0
  74. package/internal-unstable/admin-ui/pages/CreateItemPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-CreateItemPage.cjs.d.ts +2 -0
  75. package/internal-unstable/admin-ui/pages/CreateItemPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-CreateItemPage.cjs.js +116 -0
  76. package/internal-unstable/admin-ui/pages/CreateItemPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-CreateItemPage.esm.js +112 -0
  77. package/internal-unstable/admin-ui/pages/CreateItemPage/package.json +4 -0
  78. package/internal-unstable/admin-ui/pages/HomePage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-HomePage.cjs.d.ts +2 -0
  79. package/internal-unstable/admin-ui/pages/HomePage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-HomePage.cjs.js +336 -0
  80. package/internal-unstable/admin-ui/pages/HomePage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-HomePage.esm.js +332 -0
  81. package/internal-unstable/admin-ui/pages/HomePage/package.json +4 -0
  82. package/internal-unstable/admin-ui/pages/ItemPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-ItemPage.cjs.d.ts +2 -0
  83. package/internal-unstable/admin-ui/pages/ItemPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-ItemPage.cjs.js +463 -0
  84. package/internal-unstable/admin-ui/pages/ItemPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-ItemPage.esm.js +455 -0
  85. package/internal-unstable/admin-ui/pages/ItemPage/package.json +4 -0
  86. package/internal-unstable/admin-ui/pages/ListPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-ListPage.cjs.d.ts +2 -0
  87. package/internal-unstable/admin-ui/pages/ListPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-ListPage.cjs.js +1195 -0
  88. package/internal-unstable/admin-ui/pages/ListPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-ListPage.esm.js +1187 -0
  89. package/internal-unstable/admin-ui/pages/ListPage/package.json +4 -0
  90. package/internal-unstable/admin-ui/pages/NoAccessPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-NoAccessPage.cjs.d.ts +2 -0
  91. package/internal-unstable/admin-ui/pages/NoAccessPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-NoAccessPage.cjs.js +40 -0
  92. package/internal-unstable/admin-ui/pages/NoAccessPage/dist/nixxie-cms-core-internal-unstable-admin-ui-pages-NoAccessPage.esm.js +35 -0
  93. package/internal-unstable/admin-ui/pages/NoAccessPage/package.json +4 -0
  94. package/internal-unstable/artifacts/dist/nixxie-cms-core-internal-unstable-artifacts.cjs.d.ts +2 -0
  95. package/internal-unstable/artifacts/dist/nixxie-cms-core-internal-unstable-artifacts.cjs.js +51 -0
  96. package/internal-unstable/artifacts/dist/nixxie-cms-core-internal-unstable-artifacts.esm.js +38 -0
  97. package/internal-unstable/artifacts/package.json +4 -0
  98. package/package.json +44 -44
  99. package/scripts/cli/dist/nixxie-cms-core-scripts-cli.cjs.js +15 -15
  100. package/scripts/cli/dist/nixxie-cms-core-scripts-cli.esm.js +15 -15
  101. package/scripts/dist/nixxie-cms-core-scripts.cjs.js +3 -3
  102. package/scripts/dist/nixxie-cms-core-scripts.esm.js +3 -3
  103. package/src/admin-ui/admin-meta-graphql.ts +168 -168
  104. package/src/admin-ui/components/CommandPalette.tsx +433 -431
  105. package/src/admin-ui/components/Navigation.tsx +389 -385
  106. package/src/admin-ui/components/PageContainer.tsx +311 -310
  107. package/src/admin-ui/components/WelcomeDialog.tsx +1 -1
  108. package/src/admin-ui/context.tsx +338 -338
  109. package/src/admin-ui/templates/app.ts +60 -60
  110. package/src/admin-ui/templates/create-item.ts +5 -5
  111. package/src/admin-ui/templates/home.ts +2 -2
  112. package/src/admin-ui/templates/item.tsx +5 -5
  113. package/src/admin-ui/templates/list.tsx +5 -5
  114. package/src/admin-ui/templates/no-access.ts +7 -7
  115. package/src/fields/types/bigInt/index.ts +181 -181
  116. package/src/fields/types/bytes/index.ts +275 -275
  117. package/src/fields/types/calendarDay/index.ts +194 -194
  118. package/src/fields/types/checkbox/index.ts +76 -76
  119. package/src/fields/types/decimal/index.ts +182 -182
  120. package/src/fields/types/file/index.ts +168 -168
  121. package/src/fields/types/float/index.ts +133 -133
  122. package/src/fields/types/image/index.ts +244 -244
  123. package/src/fields/types/integer/index.ts +156 -156
  124. package/src/fields/types/json/index.ts +77 -77
  125. package/src/fields/types/multiselect/index.ts +212 -212
  126. package/src/fields/types/password/index.ts +241 -241
  127. package/src/fields/types/relationship/index.ts +381 -381
  128. package/src/fields/types/relationship/views/RelationshipTable.tsx +190 -190
  129. package/src/fields/types/select/index.ts +226 -226
  130. package/src/fields/types/text/index.ts +207 -207
  131. package/src/fields/types/timestamp/index.ts +116 -116
  132. package/src/fields/types/virtual/index.ts +108 -108
  133. package/src/helpers.ts +342 -316
  134. package/src/index.ts +4 -0
  135. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/id-field-view.tsx +167 -167
  136. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/App/index.tsx +22 -22
  137. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/CreateItemPage/index.tsx +71 -71
  138. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/HomePage/index.tsx +333 -333
  139. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ItemPage/common.tsx +358 -358
  140. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ItemPage/index.tsx +483 -483
  141. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/FilterAdd.tsx +221 -221
  142. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/PaginationControls.tsx +170 -170
  143. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/Tag.tsx +72 -72
  144. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/index.tsx +1006 -1006
  145. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/NoAccessPage/index.tsx +24 -24
  146. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/artifacts.ts +5 -5
  147. package/src/lib/context/createContext.ts +165 -161
  148. package/src/lib/core/initialise-lists.ts +1097 -1097
  149. package/src/lib/id-field.ts +214 -214
  150. package/src/lib/telemetry.ts +342 -342
  151. package/src/schema.ts +237 -233
  152. package/src/scripts/telemetry.ts +1 -1
  153. package/src/types/config/index.ts +400 -333
  154. package/src/types/config/lists.ts +4 -4
  155. package/src/types/context.ts +700 -530
  156. package/src/types/next-fields.ts +499 -499
  157. package/src/types/telemetry.ts +51 -51
  158. package/tests/telemetry.test.ts +361 -361
  159. package/CHANGELOG.md +0 -3158
  160. package/___internal-do-not-use-will-break-in-patch/admin-ui/id-field-view/package.json +0 -4
  161. package/___internal-do-not-use-will-break-in-patch/admin-ui/next-config/package.json +0 -4
  162. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/App/package.json +0 -4
  163. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/CreateItemPage/package.json +0 -4
  164. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/HomePage/package.json +0 -4
  165. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ItemPage/package.json +0 -4
  166. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ListPage/package.json +0 -4
  167. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/NoAccessPage/package.json +0 -4
  168. package/___internal-do-not-use-will-break-in-patch/artifacts/package.json +0 -4
  169. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/id-field-view.d.ts.map +0 -1
  170. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/App/index.d.ts.map +0 -1
  171. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/CreateItemPage/index.d.ts.map +0 -1
  172. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/HomePage/index.d.ts.map +0 -1
  173. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ItemPage/index.d.ts.map +0 -1
  174. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ListPage/index.d.ts.map +0 -1
  175. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/NoAccessPage/index.d.ts.map +0 -1
  176. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/artifacts.d.ts.map +0 -1
  177. /package/dist/{common-1a350e11.cjs.js → common-5933f758.cjs.js} +0 -0
  178. /package/dist/{common-29fc82e6.esm.js → common-ea5c441a.esm.js} +0 -0
  179. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/id-field-view.d.ts +0 -0
  180. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/App/index.d.ts +0 -0
  181. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/CreateItemPage/index.d.ts +0 -0
  182. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/HomePage/index.d.ts +0 -0
  183. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ItemPage/index.d.ts +0 -0
  184. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/index.d.ts +0 -0
  185. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/NoAccessPage/index.d.ts +0 -0
  186. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/artifacts.d.ts +0 -0
@@ -1,190 +1,190 @@
1
- import { useMemo, useState } from 'react'
2
- import type { SortDescriptor } from '@keystar/ui/table'
3
- import { Cell, Column, Row, TableBody, TableHeader, TableView } from '@keystar/ui/table'
4
- import { Field as KeystarField } from '@keystar/ui/field'
5
- import { useList } from '../../../../admin-ui/context'
6
- import type { controller } from '.'
7
- import { textSelectIcon } from '@keystar/ui/icon/icons/textSelectIcon'
8
- import { EmptyState } from '../../../../admin-ui/components/EmptyState'
9
- import { gql, type TypedDocumentNode, useQuery } from '../../../../admin-ui/apollo'
10
- import { Text } from '@keystar/ui/typography'
11
- import { GraphQLErrorNotice } from '../../../../admin-ui/components'
12
- import { PaginationControls } from '../../../../___internal-do-not-use-will-break-in-patch/admin-ui/pages/ListPage/PaginationControls'
13
- import type { CountRelationshipValue } from './types'
14
- import { useRelatedItemHref, useRelatedItemLabel } from './ContextualActions'
15
- import { ActionButton } from '@keystar/ui/button'
16
- import { Icon } from '@keystar/ui/icon'
17
- import { arrowUpRightIcon } from '@keystar/ui/icon/icons/arrowUpRightIcon'
18
- import { TooltipTrigger, Tooltip } from '@keystar/ui/tooltip'
19
- import { ProgressCircle } from '@keystar/ui/progress'
20
-
21
- export function RelationshipTable({
22
- field,
23
- value,
24
- }: {
25
- field: ReturnType<typeof controller>
26
- value: CountRelationshipValue
27
- }) {
28
- if (!field.refFieldKey) {
29
- throw new Error('refFieldKey is required for displayMode: table')
30
- }
31
- const [currentPage, setCurrentPage] = useState(1)
32
- const [pageSize, setPageSize] = useState(50)
33
- const list = useList(field.refListKey)
34
- const hasManyOnRefList: boolean = (list.fields[field.refFieldKey].controller as any).many
35
-
36
- const selectedFields = field.columns ?? list.initialColumns
37
- const columns = useMemo(() => {
38
- return selectedFields.map(path => {
39
- const field = list.fields[path]
40
- return {
41
- id: path,
42
- label: field.label,
43
- allowsSorting: field.isOrderable,
44
- }
45
- })
46
- }, [selectedFields, list])
47
-
48
- const [sort, setSort] = useState<SortDescriptor>(() => {
49
- if (field.initialSort) {
50
- return {
51
- column: field.initialSort.field,
52
- direction: field.initialSort.direction === 'ASC' ? 'ascending' : 'descending',
53
- }
54
- }
55
- return { column: 'id', direction: 'ascending' }
56
- })
57
-
58
- const { data, error, loading } = useQuery(
59
- useMemo(() => {
60
- const selectedGqlFields = [...selectedFields]
61
- .map(fieldPath => list.fields[fieldPath].controller.graphqlSelection)
62
- .join('\n')
63
-
64
- // TODO: FIXME: this is bad
65
- return gql`
66
- query (
67
- $where: ${list.graphql.names.whereInputName},
68
- $take: Int!,
69
- $skip: Int!,
70
- $orderBy: [${list.graphql.names.listOrderName}!]
71
- ) {
72
- items: ${list.graphql.names.listQueryName}(
73
- where: $where,
74
- take: $take,
75
- skip: $skip,
76
- orderBy: $orderBy
77
- ) {
78
- ${selectedFields.includes('id') ? '' : 'id'}
79
- ${selectedGqlFields}
80
- }
81
- }
82
- ` as TypedDocumentNode<
83
- { items: Record<string, unknown>[] },
84
- {
85
- where: Record<string, any>
86
- take: number
87
- skip: number
88
- orderBy: { [key: string]: 'asc' | 'desc' }[]
89
- }
90
- >
91
- }, [list, selectedFields]),
92
- {
93
- fetchPolicy: 'cache-and-network',
94
- errorPolicy: 'all',
95
- variables: {
96
- where: {
97
- [field.refFieldKey]: hasManyOnRefList
98
- ? { some: { id: { equals: value.id } } }
99
- : { id: { equals: value.id } },
100
- },
101
- take: pageSize,
102
- skip: (currentPage - 1) * pageSize,
103
- orderBy: [
104
- { [sort.column]: sort.direction === 'ascending' ? ('asc' as const) : ('desc' as const) },
105
- ],
106
- },
107
- }
108
- )
109
- const relatedItemLabel = useRelatedItemLabel(field)
110
- const relatedItemHref = useRelatedItemHref({ field, value })
111
- const items: Record<string, unknown>[] = data?.items ?? []
112
- return (
113
- <KeystarField label={field.label} description={field.description}>
114
- {inputProps => (
115
- <>
116
- <TableView
117
- {...inputProps}
118
- selectionMode="none"
119
- onSortChange={sort => setSort(sort)}
120
- sortDescriptor={sort}
121
- density="compact"
122
- UNSAFE_style={{ minHeight: 29 * 10, maxHeight: 29 * 10 }}
123
- overflowMode="truncate"
124
- renderEmptyState={() =>
125
- loading ? (
126
- <ProgressCircle isIndeterminate />
127
- ) : error ? (
128
- <GraphQLErrorNotice errors={[error]} />
129
- ) : (
130
- <EmptyState
131
- icon={textSelectIcon}
132
- title="Empty related items"
133
- message="There are no related items."
134
- />
135
- )
136
- }
137
- flex
138
- >
139
- <TableHeader columns={columns}>
140
- {({ label, id, ...options }) => (
141
- <Column key={id} isRowHeader {...options}>
142
- {label}
143
- </Column>
144
- )}
145
- </TableHeader>
146
- <TableBody items={items}>
147
- {row => {
148
- return (
149
- <Row href={`/${list.path}/${row?.id}`}>
150
- {key => {
151
- const field = list.fields[key]
152
- const value = row[key]
153
- const CellContent = field.views.Cell
154
- return (
155
- <Cell>
156
- {CellContent ? (
157
- <CellContent value={value} field={field.controller} item={row} />
158
- ) : (
159
- <Text>{value?.toString()}</Text>
160
- )}
161
- </Cell>
162
- )
163
- }}
164
- </Row>
165
- )
166
- }}
167
- </TableBody>
168
- </TableView>
169
- <PaginationControls
170
- currentPage={currentPage}
171
- pageSize={pageSize}
172
- plural={list.plural}
173
- singular={list.singular}
174
- total={value.count}
175
- onChangePage={page => setCurrentPage(page)}
176
- onChangePageSize={size => setPageSize(size)}
177
- extraActions={
178
- <TooltipTrigger>
179
- <ActionButton href={relatedItemHref!}>
180
- <Icon src={arrowUpRightIcon} />
181
- </ActionButton>
182
- <Tooltip>{relatedItemLabel}</Tooltip>
183
- </TooltipTrigger>
184
- }
185
- />
186
- </>
187
- )}
188
- </KeystarField>
189
- )
190
- }
1
+ import { useMemo, useState } from 'react'
2
+ import type { SortDescriptor } from '@keystar/ui/table'
3
+ import { Cell, Column, Row, TableBody, TableHeader, TableView } from '@keystar/ui/table'
4
+ import { Field as KeystarField } from '@keystar/ui/field'
5
+ import { useList } from '../../../../admin-ui/context'
6
+ import type { controller } from '.'
7
+ import { textSelectIcon } from '@keystar/ui/icon/icons/textSelectIcon'
8
+ import { EmptyState } from '../../../../admin-ui/components/EmptyState'
9
+ import { gql, type TypedDocumentNode, useQuery } from '../../../../admin-ui/apollo'
10
+ import { Text } from '@keystar/ui/typography'
11
+ import { GraphQLErrorNotice } from '../../../../admin-ui/components'
12
+ import { PaginationControls } from '../../../../internal-unstable/admin-ui/pages/ListPage/PaginationControls'
13
+ import type { CountRelationshipValue } from './types'
14
+ import { useRelatedItemHref, useRelatedItemLabel } from './ContextualActions'
15
+ import { ActionButton } from '@keystar/ui/button'
16
+ import { Icon } from '@keystar/ui/icon'
17
+ import { arrowUpRightIcon } from '@keystar/ui/icon/icons/arrowUpRightIcon'
18
+ import { TooltipTrigger, Tooltip } from '@keystar/ui/tooltip'
19
+ import { ProgressCircle } from '@keystar/ui/progress'
20
+
21
+ export function RelationshipTable({
22
+ field,
23
+ value,
24
+ }: {
25
+ field: ReturnType<typeof controller>
26
+ value: CountRelationshipValue
27
+ }) {
28
+ if (!field.refFieldKey) {
29
+ throw new Error('refFieldKey is required for displayMode: table')
30
+ }
31
+ const [currentPage, setCurrentPage] = useState(1)
32
+ const [pageSize, setPageSize] = useState(50)
33
+ const list = useList(field.refListKey)
34
+ const hasManyOnRefList: boolean = (list.fields[field.refFieldKey].controller as any).many
35
+
36
+ const selectedFields = field.columns ?? list.initialColumns
37
+ const columns = useMemo(() => {
38
+ return selectedFields.map(path => {
39
+ const field = list.fields[path]
40
+ return {
41
+ id: path,
42
+ label: field.label,
43
+ allowsSorting: field.isOrderable,
44
+ }
45
+ })
46
+ }, [selectedFields, list])
47
+
48
+ const [sort, setSort] = useState<SortDescriptor>(() => {
49
+ if (field.initialSort) {
50
+ return {
51
+ column: field.initialSort.field,
52
+ direction: field.initialSort.direction === 'ASC' ? 'ascending' : 'descending',
53
+ }
54
+ }
55
+ return { column: 'id', direction: 'ascending' }
56
+ })
57
+
58
+ const { data, error, loading } = useQuery(
59
+ useMemo(() => {
60
+ const selectedGqlFields = [...selectedFields]
61
+ .map(fieldPath => list.fields[fieldPath].controller.graphqlSelection)
62
+ .join('\n')
63
+
64
+ // TODO: FIXME: this is bad
65
+ return gql`
66
+ query (
67
+ $where: ${list.graphql.names.whereInputName},
68
+ $take: Int!,
69
+ $skip: Int!,
70
+ $orderBy: [${list.graphql.names.listOrderName}!]
71
+ ) {
72
+ items: ${list.graphql.names.listQueryName}(
73
+ where: $where,
74
+ take: $take,
75
+ skip: $skip,
76
+ orderBy: $orderBy
77
+ ) {
78
+ ${selectedFields.includes('id') ? '' : 'id'}
79
+ ${selectedGqlFields}
80
+ }
81
+ }
82
+ ` as TypedDocumentNode<
83
+ { items: Record<string, unknown>[] },
84
+ {
85
+ where: Record<string, any>
86
+ take: number
87
+ skip: number
88
+ orderBy: { [key: string]: 'asc' | 'desc' }[]
89
+ }
90
+ >
91
+ }, [list, selectedFields]),
92
+ {
93
+ fetchPolicy: 'cache-and-network',
94
+ errorPolicy: 'all',
95
+ variables: {
96
+ where: {
97
+ [field.refFieldKey]: hasManyOnRefList
98
+ ? { some: { id: { equals: value.id } } }
99
+ : { id: { equals: value.id } },
100
+ },
101
+ take: pageSize,
102
+ skip: (currentPage - 1) * pageSize,
103
+ orderBy: [
104
+ { [sort.column]: sort.direction === 'ascending' ? ('asc' as const) : ('desc' as const) },
105
+ ],
106
+ },
107
+ }
108
+ )
109
+ const relatedItemLabel = useRelatedItemLabel(field)
110
+ const relatedItemHref = useRelatedItemHref({ field, value })
111
+ const items: Record<string, unknown>[] = data?.items ?? []
112
+ return (
113
+ <KeystarField label={field.label} description={field.description}>
114
+ {inputProps => (
115
+ <>
116
+ <TableView
117
+ {...inputProps}
118
+ selectionMode="none"
119
+ onSortChange={sort => setSort(sort)}
120
+ sortDescriptor={sort}
121
+ density="compact"
122
+ UNSAFE_style={{ minHeight: 29 * 10, maxHeight: 29 * 10 }}
123
+ overflowMode="truncate"
124
+ renderEmptyState={() =>
125
+ loading ? (
126
+ <ProgressCircle isIndeterminate />
127
+ ) : error ? (
128
+ <GraphQLErrorNotice errors={[error]} />
129
+ ) : (
130
+ <EmptyState
131
+ icon={textSelectIcon}
132
+ title="Empty related items"
133
+ message="There are no related items."
134
+ />
135
+ )
136
+ }
137
+ flex
138
+ >
139
+ <TableHeader columns={columns}>
140
+ {({ label, id, ...options }) => (
141
+ <Column key={id} isRowHeader {...options}>
142
+ {label}
143
+ </Column>
144
+ )}
145
+ </TableHeader>
146
+ <TableBody items={items}>
147
+ {row => {
148
+ return (
149
+ <Row href={`/${list.path}/${row?.id}`}>
150
+ {key => {
151
+ const field = list.fields[key]
152
+ const value = row[key]
153
+ const CellContent = field.views.Cell
154
+ return (
155
+ <Cell>
156
+ {CellContent ? (
157
+ <CellContent value={value} field={field.controller} item={row} />
158
+ ) : (
159
+ <Text>{value?.toString()}</Text>
160
+ )}
161
+ </Cell>
162
+ )
163
+ }}
164
+ </Row>
165
+ )
166
+ }}
167
+ </TableBody>
168
+ </TableView>
169
+ <PaginationControls
170
+ currentPage={currentPage}
171
+ pageSize={pageSize}
172
+ plural={list.plural}
173
+ singular={list.singular}
174
+ total={value.count}
175
+ onChangePage={page => setCurrentPage(page)}
176
+ onChangePageSize={size => setPageSize(size)}
177
+ extraActions={
178
+ <TooltipTrigger>
179
+ <ActionButton href={relatedItemHref!}>
180
+ <Icon src={arrowUpRightIcon} />
181
+ </ActionButton>
182
+ <Tooltip>{relatedItemLabel}</Tooltip>
183
+ </TooltipTrigger>
184
+ }
185
+ />
186
+ </>
187
+ )}
188
+ </KeystarField>
189
+ )
190
+ }