@nixxie-cms/core 1.0.0 → 1.0.2

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 (187) 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 +44 -15
  100. package/scripts/cli/dist/nixxie-cms-core-scripts-cli.esm.js +44 -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/next-config.ts +29 -0
  115. package/src/admin-ui/templates/no-access.ts +7 -7
  116. package/src/fields/types/bigInt/index.ts +181 -181
  117. package/src/fields/types/bytes/index.ts +275 -275
  118. package/src/fields/types/calendarDay/index.ts +194 -194
  119. package/src/fields/types/checkbox/index.ts +76 -76
  120. package/src/fields/types/decimal/index.ts +182 -182
  121. package/src/fields/types/file/index.ts +168 -168
  122. package/src/fields/types/float/index.ts +133 -133
  123. package/src/fields/types/image/index.ts +244 -244
  124. package/src/fields/types/integer/index.ts +156 -156
  125. package/src/fields/types/json/index.ts +77 -77
  126. package/src/fields/types/multiselect/index.ts +212 -212
  127. package/src/fields/types/password/index.ts +241 -241
  128. package/src/fields/types/relationship/index.ts +381 -381
  129. package/src/fields/types/relationship/views/RelationshipTable.tsx +190 -190
  130. package/src/fields/types/select/index.ts +226 -226
  131. package/src/fields/types/text/index.ts +207 -207
  132. package/src/fields/types/timestamp/index.ts +116 -116
  133. package/src/fields/types/virtual/index.ts +108 -108
  134. package/src/helpers.ts +342 -316
  135. package/src/index.ts +4 -0
  136. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/id-field-view.tsx +167 -167
  137. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/App/index.tsx +22 -22
  138. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/CreateItemPage/index.tsx +71 -71
  139. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/HomePage/index.tsx +333 -333
  140. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ItemPage/common.tsx +358 -358
  141. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ItemPage/index.tsx +483 -483
  142. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/FilterAdd.tsx +221 -221
  143. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/PaginationControls.tsx +170 -170
  144. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/Tag.tsx +72 -72
  145. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/index.tsx +1006 -1006
  146. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/NoAccessPage/index.tsx +24 -24
  147. package/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/artifacts.ts +5 -5
  148. package/src/lib/context/createContext.ts +165 -161
  149. package/src/lib/core/initialise-lists.ts +1097 -1097
  150. package/src/lib/id-field.ts +214 -214
  151. package/src/lib/telemetry.ts +342 -342
  152. package/src/schema.ts +237 -233
  153. package/src/scripts/telemetry.ts +1 -1
  154. package/src/types/config/index.ts +400 -333
  155. package/src/types/config/lists.ts +4 -4
  156. package/src/types/context.ts +700 -530
  157. package/src/types/next-fields.ts +499 -499
  158. package/src/types/telemetry.ts +51 -51
  159. package/tests/telemetry.test.ts +361 -361
  160. package/CHANGELOG.md +0 -3158
  161. package/___internal-do-not-use-will-break-in-patch/admin-ui/id-field-view/package.json +0 -4
  162. package/___internal-do-not-use-will-break-in-patch/admin-ui/next-config/package.json +0 -4
  163. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/App/package.json +0 -4
  164. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/CreateItemPage/package.json +0 -4
  165. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/HomePage/package.json +0 -4
  166. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ItemPage/package.json +0 -4
  167. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ListPage/package.json +0 -4
  168. package/___internal-do-not-use-will-break-in-patch/admin-ui/pages/NoAccessPage/package.json +0 -4
  169. package/___internal-do-not-use-will-break-in-patch/artifacts/package.json +0 -4
  170. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/id-field-view.d.ts.map +0 -1
  171. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/App/index.d.ts.map +0 -1
  172. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/CreateItemPage/index.d.ts.map +0 -1
  173. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/HomePage/index.d.ts.map +0 -1
  174. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ItemPage/index.d.ts.map +0 -1
  175. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/ListPage/index.d.ts.map +0 -1
  176. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/admin-ui/pages/NoAccessPage/index.d.ts.map +0 -1
  177. package/dist/declarations/src/___internal-do-not-use-will-break-in-patch/artifacts.d.ts.map +0 -1
  178. /package/dist/{common-1a350e11.cjs.js → common-5933f758.cjs.js} +0 -0
  179. /package/dist/{common-29fc82e6.esm.js → common-ea5c441a.esm.js} +0 -0
  180. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/id-field-view.d.ts +0 -0
  181. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/App/index.d.ts +0 -0
  182. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/CreateItemPage/index.d.ts +0 -0
  183. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/HomePage/index.d.ts +0 -0
  184. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ItemPage/index.d.ts +0 -0
  185. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/ListPage/index.d.ts +0 -0
  186. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/admin-ui/pages/NoAccessPage/index.d.ts +0 -0
  187. /package/dist/declarations/src/{___internal-do-not-use-will-break-in-patch → internal-unstable}/artifacts.d.ts +0 -0
@@ -1,385 +1,389 @@
1
- import { type ReactNode, type PropsWithChildren, useState, type ChangeEvent } from 'react'
2
- import { useRouter } from 'next/router'
3
-
4
- import { css } from '@keystar/ui/style'
5
-
6
- import type { ListMeta } from '../../types'
7
- import { useNixxie } from '../context'
8
-
9
- // ================================================================
10
- // Helpers
11
- // ================================================================
12
-
13
- export function getHrefFromList(list: Pick<ListMeta, 'path' | 'isSingleton'>) {
14
- return `/${list.path}${list.isSingleton ? '/1' : ''}`
15
- }
16
-
17
- // ================================================================
18
- // Nav Item — minimal, text-only, precise
19
- // ================================================================
20
-
21
- type NavItemProps = {
22
- href: string
23
- children: ReactNode
24
- onClick?: () => void
25
- indent?: boolean
26
- }
27
-
28
- export function NavItem({ href, children, onClick, indent }: NavItemProps) {
29
- const router = useRouter()
30
- const segment = href.split('/')[1]
31
- const isActive =
32
- router.pathname === href ||
33
- (segment && router.pathname.split('/')[1] === segment && href !== '/')
34
-
35
- return (
36
- <a
37
- href={href}
38
- onClick={e => {
39
- e.preventDefault()
40
- onClick?.()
41
- router.push(href)
42
- }}
43
- aria-current={isActive ? 'page' : undefined}
44
- className={css({
45
- position: 'relative',
46
- display: 'flex',
47
- alignItems: 'center',
48
- paddingInline: indent ? '32px 16px' : '16px',
49
- paddingBlock: '7px',
50
- fontSize: 13,
51
- fontWeight: isActive ? 500 : 400,
52
- color: isActive ? '#0a0a0a' : '#636363',
53
- backgroundColor: isActive ? '#f5f5f5' : 'transparent',
54
- textDecoration: 'none',
55
- borderRadius: '0 6px 6px 0',
56
- marginRight: '12px',
57
- lineHeight: '1.4',
58
- cursor: 'pointer',
59
- userSelect: 'none',
60
- transition: 'color 120ms, background-color 120ms',
61
-
62
- '&:hover': {
63
- color: '#0a0a0a',
64
- backgroundColor: '#f5f5f5',
65
- },
66
- })}
67
- >
68
- {/* Active indicator — thin solid bar on the left edge */}
69
- {isActive && (
70
- <span
71
- className={css({
72
- position: 'absolute',
73
- left: 0,
74
- top: '20%',
75
- bottom: '20%',
76
- width: 2,
77
- borderRadius: '0 2px 2px 0',
78
- backgroundColor: '#000000',
79
- })}
80
- />
81
- )}
82
- <span
83
- className={css({
84
- overflow: 'hidden',
85
- textOverflow: 'ellipsis',
86
- whiteSpace: 'nowrap',
87
- })}
88
- >
89
- {children}
90
- </span>
91
- </a>
92
- )
93
- }
94
-
95
- // ================================================================
96
- // Section label
97
- // ================================================================
98
-
99
- function SectionLabel({ children, action }: { children: ReactNode; action?: ReactNode }) {
100
- return (
101
- <div
102
- className={css({
103
- display: 'flex',
104
- alignItems: 'center',
105
- justifyContent: 'space-between',
106
- paddingInline: '16px 12px',
107
- paddingBlock: '12px 4px',
108
- })}
109
- >
110
- <span
111
- className={css({
112
- fontSize: 10,
113
- fontWeight: 600,
114
- letterSpacing: '0.10em',
115
- textTransform: 'uppercase',
116
- color: '#b8b8b8',
117
- })}
118
- >
119
- {children}
120
- </span>
121
- {action}
122
- </div>
123
- )
124
- }
125
-
126
- // ================================================================
127
- // Divider
128
- // ================================================================
129
-
130
- function NavDivider() {
131
- return (
132
- <div
133
- className={css({
134
- height: 1,
135
- marginInline: '16px',
136
- marginBlock: '6px',
137
- backgroundColor: '#f2f2f2',
138
- })}
139
- />
140
- )
141
- }
142
-
143
- // ================================================================
144
- // Search shortcut — Cmd+K trigger
145
- // ================================================================
146
-
147
- function SearchTrigger({ onClick }: { onClick?: () => void }) {
148
- return (
149
- <button
150
- onClick={onClick}
151
- className={css({
152
- display: 'flex',
153
- alignItems: 'center',
154
- gap: 8,
155
- width: 'calc(100% - 24px)',
156
- marginInline: '12px',
157
- marginBottom: '8px',
158
- paddingInline: '10px',
159
- paddingBlock: '7px',
160
- borderRadius: 6,
161
- border: '1px solid #ebebeb',
162
- backgroundColor: '#f5f5f5',
163
- color: '#a3a3a3',
164
- fontSize: 12.5,
165
- cursor: 'pointer',
166
- textAlign: 'left',
167
- transition: 'border-color 140ms, background 140ms, color 140ms',
168
-
169
- '&:hover': {
170
- borderColor: '#dedede',
171
- backgroundColor: '#efefef',
172
- color: '#737373',
173
- },
174
- '&:focus-visible': {
175
- outline: '2px solid #000',
176
- outlineOffset: 2,
177
- },
178
- })}
179
- >
180
- <svg width="12" height="12" viewBox="0 0 12 12" fill="none" style={{ flexShrink: 0 }}>
181
- <circle cx="5" cy="5" r="3.5" stroke="currentColor" strokeWidth="1.25" />
182
- <path d="M8 8L10.5 10.5" stroke="currentColor" strokeWidth="1.25" strokeLinecap="round" />
183
- </svg>
184
- <span className={css({ flex: 1 })}>Search…</span>
185
- <kbd
186
- className={css({
187
- fontSize: 10.5,
188
- color: '#c8c8c8',
189
- fontFamily: 'inherit',
190
- letterSpacing: '0.02em',
191
- })}
192
- >
193
- ⌘K
194
- </kbd>
195
- </button>
196
- )
197
- }
198
-
199
- // ================================================================
200
- // List filter — inline nav search
201
- // ================================================================
202
-
203
- function NavSearch({ value, onChange }: { value: string; onChange: (v: string) => void }) {
204
- return (
205
- <div className={css({ paddingInline: '16px 12px', paddingBottom: '4px' })}>
206
- <input
207
- type="search"
208
- value={value}
209
- onChange={(e: ChangeEvent<HTMLInputElement>) => onChange(e.target.value)}
210
- placeholder="Filter…"
211
- className={css({
212
- width: '100%',
213
- paddingInline: '10px',
214
- paddingBlock: '5px',
215
- fontSize: 12.5,
216
- border: '1px solid #ebebeb',
217
- borderRadius: 5,
218
- backgroundColor: '#f5f5f5',
219
- color: '#0a0a0a',
220
- outline: 'none',
221
- boxSizing: 'border-box',
222
- transition: 'border-color 140ms',
223
- '&:focus': { borderColor: '#b0b0b0' },
224
- '&::placeholder': { color: '#c8c8c8' },
225
- '&::-webkit-search-cancel-button': { display: 'none' },
226
- })}
227
- />
228
- </div>
229
- )
230
- }
231
-
232
- // ================================================================
233
- // Collections section
234
- // ================================================================
235
-
236
- function CollectionsSection({
237
- lists,
238
- onNavItemClick,
239
- }: {
240
- lists: ListMeta[]
241
- onNavItemClick?: () => void
242
- }) {
243
- const [search, setSearch] = useState('')
244
-
245
- const filtered = search.trim()
246
- ? lists.filter(l => l.label.toLowerCase().includes(search.toLowerCase()))
247
- : lists
248
-
249
- return (
250
- <>
251
- <SectionLabel>Collections</SectionLabel>
252
-
253
- {lists.length > 6 && <NavSearch value={search} onChange={setSearch} />}
254
-
255
- {filtered.length === 0 ? (
256
- <p className={css({ margin: '4px 16px', fontSize: 12.5, color: '#c8c8c8' })}>
257
- No results
258
- </p>
259
- ) : (
260
- filtered.map(list => (
261
- <NavItem key={list.key} href={getHrefFromList(list)} onClick={onNavItemClick}>
262
- {list.label}
263
- {list.isSingleton && (
264
- <span
265
- className={css({
266
- marginLeft: 6,
267
- fontSize: 10,
268
- fontWeight: 500,
269
- color: '#c8c8c8',
270
- letterSpacing: '0.04em',
271
- textTransform: 'uppercase',
272
- })}
273
- >
274
- Single
275
- </span>
276
- )}
277
- </NavItem>
278
- ))
279
- )}
280
- </>
281
- )
282
- }
283
-
284
- // ================================================================
285
- // Root Navigation
286
- // ================================================================
287
-
288
- type InternalNavigationProps = {
289
- onNavItemClick?: () => void
290
- onCmdK?: () => void
291
- }
292
-
293
- export function Navigation({ onNavItemClick, onCmdK }: InternalNavigationProps) {
294
- const { adminConfig, lists: allLists, apiPath } = useNixxie()
295
- const visibleLists = Object.values(allLists).filter(x => !x.hideNavigation)
296
-
297
- if (adminConfig?.components?.Navigation) {
298
- return <adminConfig.components.Navigation lists={visibleLists} />
299
- }
300
-
301
- return (
302
- <nav
303
- aria-label="Main navigation"
304
- className={css({ display: 'flex', flexDirection: 'column', height: '100%' })}
305
- >
306
- {/* Search trigger */}
307
- <div className={css({ paddingTop: '10px' })}>
308
- <SearchTrigger onClick={onCmdK} />
309
- </div>
310
-
311
- {/* Dashboard */}
312
- <NavItem href="/" onClick={onNavItemClick}>
313
- Dashboard
314
- </NavItem>
315
-
316
- <NavDivider />
317
-
318
- {/* Collections + developer */}
319
- <div
320
- className={css({ flex: 1, overflowY: 'auto', overflowX: 'hidden', paddingBottom: '8px' })}
321
- >
322
- {visibleLists.length > 0 ? (
323
- <CollectionsSection lists={visibleLists} onNavItemClick={onNavItemClick} />
324
- ) : (
325
- <p className={css({ margin: '8px 16px', fontSize: 12.5, color: '#c8c8c8' })}>
326
- No collections
327
- </p>
328
- )}
329
-
330
- {process.env.NODE_ENV !== 'production' && apiPath && (
331
- <>
332
- <NavDivider />
333
- <SectionLabel>Developer</SectionLabel>
334
- <NavItem href={apiPath} onClick={onNavItemClick}>
335
- API Explorer
336
- </NavItem>
337
- </>
338
- )}
339
- </div>
340
- </nav>
341
- )
342
- }
343
-
344
- // ================================================================
345
- // Legacy public exports (used by @nixxie-cms/auth and custom navigations)
346
- // ================================================================
347
-
348
- export function NavList({ children }: PropsWithChildren) {
349
- return <>{children}</>
350
- }
351
-
352
- export function NavContainer({ children }: PropsWithChildren) {
353
- return (
354
- <div
355
- className={css({
356
- display: 'flex',
357
- flexDirection: 'column',
358
- height: '100%',
359
- paddingBlock: '8px',
360
- })}
361
- >
362
- {children}
363
- </div>
364
- )
365
- }
366
-
367
- export function NavFooter({ children }: PropsWithChildren) {
368
- return (
369
- <div
370
- className={css({
371
- paddingInline: '12px',
372
- paddingBlock: '10px',
373
- borderTop: '1px solid #f2f2f2',
374
- marginTop: 'auto',
375
- })}
376
- >
377
- {children}
378
- </div>
379
- )
380
- }
381
-
382
- /** @deprecated */
383
- export function DeveloperResourcesMenu() {
384
- return null
385
- }
1
+ import { type ReactNode, type PropsWithChildren, useState, type ChangeEvent } from 'react'
2
+ import { useRouter } from 'next/router'
3
+
4
+ import { css } from '@keystar/ui/style'
5
+
6
+ import type { ListMeta } from '../../types'
7
+ import { useNixxie } from '../context'
8
+
9
+ // ================================================================
10
+ // Helpers
11
+ // ================================================================
12
+
13
+ export function getHrefFromList(list: Pick<ListMeta, 'path' | 'isSingleton'>) {
14
+ return `/${list.path}${list.isSingleton ? '/1' : ''}`
15
+ }
16
+
17
+ // ================================================================
18
+ // Nav Item — minimal, text-only, precise
19
+ // ================================================================
20
+
21
+ type NavItemProps = {
22
+ href: string
23
+ children: ReactNode
24
+ onClick?: () => void
25
+ indent?: boolean
26
+ }
27
+
28
+ export function NavItem({ href, children, onClick, indent }: NavItemProps) {
29
+ const router = useRouter()
30
+ const segment = href.split('/')[1]
31
+ const isActive =
32
+ router.pathname === href ||
33
+ (segment && router.pathname.split('/')[1] === segment && href !== '/')
34
+
35
+ return (
36
+ <a
37
+ href={href}
38
+ onClick={e => {
39
+ e.preventDefault()
40
+ onClick?.()
41
+ router.push(href)
42
+ }}
43
+ aria-current={isActive ? 'page' : undefined}
44
+ className={css({
45
+ position: 'relative',
46
+ display: 'flex',
47
+ alignItems: 'center',
48
+ paddingInline: indent ? '32px 16px' : '16px',
49
+ paddingBlock: '7px',
50
+ fontSize: 13,
51
+ fontWeight: isActive ? 500 : 400,
52
+ color: isActive ? '#0a0a0a' : '#636363',
53
+ backgroundColor: isActive ? '#f5f5f5' : 'transparent',
54
+ textDecoration: 'none',
55
+ borderRadius: '0 6px 6px 0',
56
+ marginRight: '12px',
57
+ lineHeight: '1.4',
58
+ cursor: 'pointer',
59
+ userSelect: 'none',
60
+ transition: 'color 120ms, background-color 120ms',
61
+
62
+ '&:hover': {
63
+ color: '#0a0a0a',
64
+ backgroundColor: '#f5f5f5',
65
+ },
66
+ })}
67
+ >
68
+ {/* Active indicator — thin solid bar on the left edge */}
69
+ {isActive && (
70
+ <span
71
+ className={css({
72
+ position: 'absolute',
73
+ left: 0,
74
+ top: '20%',
75
+ bottom: '20%',
76
+ width: 2,
77
+ borderRadius: '0 2px 2px 0',
78
+ backgroundColor: '#000000',
79
+ })}
80
+ />
81
+ )}
82
+ <span
83
+ className={css({
84
+ overflow: 'hidden',
85
+ textOverflow: 'ellipsis',
86
+ whiteSpace: 'nowrap',
87
+ })}
88
+ >
89
+ {children}
90
+ </span>
91
+ </a>
92
+ )
93
+ }
94
+
95
+ // ================================================================
96
+ // Section label
97
+ // ================================================================
98
+
99
+ function SectionLabel({ children, action }: { children: ReactNode; action?: ReactNode }) {
100
+ return (
101
+ <div
102
+ className={css({
103
+ display: 'flex',
104
+ alignItems: 'center',
105
+ justifyContent: 'space-between',
106
+ paddingInline: '16px 12px',
107
+ paddingBlock: '12px 4px',
108
+ })}
109
+ >
110
+ <span
111
+ className={css({
112
+ fontSize: 10,
113
+ fontWeight: 600,
114
+ letterSpacing: '0.10em',
115
+ textTransform: 'uppercase',
116
+ color: '#b8b8b8',
117
+ })}
118
+ >
119
+ {children}
120
+ </span>
121
+ {action}
122
+ </div>
123
+ )
124
+ }
125
+
126
+ // ================================================================
127
+ // Divider
128
+ // ================================================================
129
+
130
+ function NavDivider() {
131
+ return (
132
+ <div
133
+ className={css({
134
+ height: 1,
135
+ marginInline: '16px',
136
+ marginBlock: '6px',
137
+ backgroundColor: '#f2f2f2',
138
+ })}
139
+ />
140
+ )
141
+ }
142
+
143
+ // ================================================================
144
+ // Search shortcut — Cmd+K trigger
145
+ // ================================================================
146
+
147
+ function SearchTrigger({ onClick }: { onClick?: () => void }) {
148
+ return (
149
+ <button
150
+ onClick={onClick}
151
+ className={css({
152
+ display: 'flex',
153
+ alignItems: 'center',
154
+ gap: 8,
155
+ width: 'calc(100% - 24px)',
156
+ marginInline: '12px',
157
+ marginBottom: '8px',
158
+ paddingInline: '10px',
159
+ paddingBlock: '7px',
160
+ borderRadius: 6,
161
+ border: '1px solid #ebebeb',
162
+ backgroundColor: '#f5f5f5',
163
+ color: '#a3a3a3',
164
+ fontSize: 12.5,
165
+ cursor: 'pointer',
166
+ textAlign: 'left',
167
+ transition: 'border-color 140ms, background 140ms, color 140ms',
168
+
169
+ '&:hover': {
170
+ borderColor: '#dedede',
171
+ backgroundColor: '#efefef',
172
+ color: '#737373',
173
+ },
174
+ '&:focus-visible': {
175
+ outline: '2px solid #000',
176
+ outlineOffset: 2,
177
+ },
178
+ })}
179
+ >
180
+ <svg width="12" height="12" viewBox="0 0 12 12" fill="none" style={{ flexShrink: 0 }}>
181
+ <circle cx="5" cy="5" r="3.5" stroke="currentColor" strokeWidth="1.25" />
182
+ <path d="M8 8L10.5 10.5" stroke="currentColor" strokeWidth="1.25" strokeLinecap="round" />
183
+ </svg>
184
+ <span className={css({ flex: 1 })}>Search…</span>
185
+ <kbd
186
+ className={css({
187
+ fontSize: 10.5,
188
+ color: '#c8c8c8',
189
+ fontFamily: 'inherit',
190
+ letterSpacing: '0.02em',
191
+ })}
192
+ >
193
+ ⌘K
194
+ </kbd>
195
+ </button>
196
+ )
197
+ }
198
+
199
+ // ================================================================
200
+ // List filter — inline nav search
201
+ // ================================================================
202
+
203
+ function NavSearch({ value, onChange }: { value: string; onChange: (v: string) => void }) {
204
+ return (
205
+ <div className={css({ paddingInline: '16px 12px', paddingBottom: '4px' })}>
206
+ <input
207
+ type="search"
208
+ value={value}
209
+ onChange={(e: ChangeEvent<HTMLInputElement>) => onChange(e.target.value)}
210
+ placeholder="Filter…"
211
+ className={css({
212
+ width: '100%',
213
+ paddingInline: '10px',
214
+ paddingBlock: '5px',
215
+ fontSize: 12.5,
216
+ border: '1px solid #ebebeb',
217
+ borderRadius: 5,
218
+ backgroundColor: '#f5f5f5',
219
+ color: '#0a0a0a',
220
+ outline: 'none',
221
+ boxSizing: 'border-box',
222
+ transition: 'border-color 140ms',
223
+ '&:focus': { borderColor: '#b0b0b0' },
224
+ '&::placeholder': { color: '#c8c8c8' },
225
+ '&::-webkit-search-cancel-button': { display: 'none' },
226
+ })}
227
+ />
228
+ </div>
229
+ )
230
+ }
231
+
232
+ // ================================================================
233
+ // Collections section
234
+ // ================================================================
235
+
236
+ function CollectionsSection({
237
+ lists,
238
+ onNavItemClick,
239
+ }: {
240
+ lists: ListMeta[]
241
+ onNavItemClick?: () => void
242
+ }) {
243
+ const [search, setSearch] = useState('')
244
+
245
+ // The filter input is only shown for longer lists; gate filtering on the same condition so a
246
+ // stale search value can't strand the user on "No results" after the list count drops to <= 6.
247
+ const showSearch = lists.length > 6
248
+ const filtered =
249
+ showSearch && search.trim()
250
+ ? lists.filter(l => l.label.toLowerCase().includes(search.toLowerCase()))
251
+ : lists
252
+
253
+ return (
254
+ <>
255
+ <SectionLabel>Collections</SectionLabel>
256
+
257
+ {showSearch && <NavSearch value={search} onChange={setSearch} />}
258
+
259
+ {filtered.length === 0 ? (
260
+ <p className={css({ margin: '4px 16px', fontSize: 12.5, color: '#c8c8c8' })}>
261
+ No results
262
+ </p>
263
+ ) : (
264
+ filtered.map(list => (
265
+ <NavItem key={list.key} href={getHrefFromList(list)} onClick={onNavItemClick}>
266
+ {list.label}
267
+ {list.isSingleton && (
268
+ <span
269
+ className={css({
270
+ marginLeft: 6,
271
+ fontSize: 10,
272
+ fontWeight: 500,
273
+ color: '#c8c8c8',
274
+ letterSpacing: '0.04em',
275
+ textTransform: 'uppercase',
276
+ })}
277
+ >
278
+ Single
279
+ </span>
280
+ )}
281
+ </NavItem>
282
+ ))
283
+ )}
284
+ </>
285
+ )
286
+ }
287
+
288
+ // ================================================================
289
+ // Root Navigation
290
+ // ================================================================
291
+
292
+ type InternalNavigationProps = {
293
+ onNavItemClick?: () => void
294
+ onCmdK?: () => void
295
+ }
296
+
297
+ export function Navigation({ onNavItemClick, onCmdK }: InternalNavigationProps) {
298
+ const { adminConfig, lists: allLists, apiPath } = useNixxie()
299
+ const visibleLists = Object.values(allLists).filter(x => !x.hideNavigation)
300
+
301
+ if (adminConfig?.components?.Navigation) {
302
+ return <adminConfig.components.Navigation lists={visibleLists} />
303
+ }
304
+
305
+ return (
306
+ <nav
307
+ aria-label="Main navigation"
308
+ className={css({ display: 'flex', flexDirection: 'column', height: '100%' })}
309
+ >
310
+ {/* Search trigger */}
311
+ <div className={css({ paddingTop: '10px' })}>
312
+ <SearchTrigger onClick={onCmdK} />
313
+ </div>
314
+
315
+ {/* Dashboard */}
316
+ <NavItem href="/" onClick={onNavItemClick}>
317
+ Dashboard
318
+ </NavItem>
319
+
320
+ <NavDivider />
321
+
322
+ {/* Collections + developer */}
323
+ <div
324
+ className={css({ flex: 1, overflowY: 'auto', overflowX: 'hidden', paddingBottom: '8px' })}
325
+ >
326
+ {visibleLists.length > 0 ? (
327
+ <CollectionsSection lists={visibleLists} onNavItemClick={onNavItemClick} />
328
+ ) : (
329
+ <p className={css({ margin: '8px 16px', fontSize: 12.5, color: '#c8c8c8' })}>
330
+ No collections
331
+ </p>
332
+ )}
333
+
334
+ {process.env.NODE_ENV !== 'production' && apiPath && (
335
+ <>
336
+ <NavDivider />
337
+ <SectionLabel>Developer</SectionLabel>
338
+ <NavItem href={apiPath} onClick={onNavItemClick}>
339
+ API Explorer
340
+ </NavItem>
341
+ </>
342
+ )}
343
+ </div>
344
+ </nav>
345
+ )
346
+ }
347
+
348
+ // ================================================================
349
+ // Legacy public exports (used by @nixxie-cms/auth and custom navigations)
350
+ // ================================================================
351
+
352
+ export function NavList({ children }: PropsWithChildren) {
353
+ return <>{children}</>
354
+ }
355
+
356
+ export function NavContainer({ children }: PropsWithChildren) {
357
+ return (
358
+ <div
359
+ className={css({
360
+ display: 'flex',
361
+ flexDirection: 'column',
362
+ height: '100%',
363
+ paddingBlock: '8px',
364
+ })}
365
+ >
366
+ {children}
367
+ </div>
368
+ )
369
+ }
370
+
371
+ export function NavFooter({ children }: PropsWithChildren) {
372
+ return (
373
+ <div
374
+ className={css({
375
+ paddingInline: '12px',
376
+ paddingBlock: '10px',
377
+ borderTop: '1px solid #f2f2f2',
378
+ marginTop: 'auto',
379
+ })}
380
+ >
381
+ {children}
382
+ </div>
383
+ )
384
+ }
385
+
386
+ /** @deprecated */
387
+ export function DeveloperResourcesMenu() {
388
+ return null
389
+ }