opacacms 0.1.1 → 0.1.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 (212) hide show
  1. package/package.json +36 -1
  2. package/bun.lock +0 -34
  3. package/global.d.ts +0 -11
  4. package/src/admin/api-client.ts +0 -63
  5. package/src/admin/auth-client.ts +0 -40
  6. package/src/admin/custom-field.ts +0 -179
  7. package/src/admin/index.ts +0 -15
  8. package/src/admin/react.tsx +0 -72
  9. package/src/admin/router.ts +0 -9
  10. package/src/admin/stores/admin-queries.ts +0 -121
  11. package/src/admin/stores/auth.ts +0 -61
  12. package/src/admin/stores/column-visibility.ts +0 -67
  13. package/src/admin/stores/config.ts +0 -15
  14. package/src/admin/stores/media.ts +0 -95
  15. package/src/admin/stores/query.ts +0 -13
  16. package/src/admin/stores/ui.ts +0 -29
  17. package/src/admin/ui/admin-client.tsx +0 -283
  18. package/src/admin/ui/admin-layout.tsx +0 -276
  19. package/src/admin/ui/components/ColumnVisibilityToggle.tsx +0 -141
  20. package/src/admin/ui/components/DataDetailSheet.tsx +0 -141
  21. package/src/admin/ui/components/DataDetailView.tsx +0 -175
  22. package/src/admin/ui/components/Table.tsx +0 -67
  23. package/src/admin/ui/components/fields/ArrayField.tsx +0 -166
  24. package/src/admin/ui/components/fields/BlocksField.tsx +0 -202
  25. package/src/admin/ui/components/fields/BooleanField.tsx +0 -50
  26. package/src/admin/ui/components/fields/CollapsibleField.tsx +0 -75
  27. package/src/admin/ui/components/fields/DateField.tsx +0 -45
  28. package/src/admin/ui/components/fields/FileField.tsx +0 -322
  29. package/src/admin/ui/components/fields/GroupField.tsx +0 -50
  30. package/src/admin/ui/components/fields/JoinField.tsx +0 -23
  31. package/src/admin/ui/components/fields/NumberField.tsx +0 -46
  32. package/src/admin/ui/components/fields/RadioField.tsx +0 -62
  33. package/src/admin/ui/components/fields/RelationshipField.tsx +0 -278
  34. package/src/admin/ui/components/fields/RowField.tsx +0 -40
  35. package/src/admin/ui/components/fields/SelectField.tsx +0 -59
  36. package/src/admin/ui/components/fields/TabsField.tsx +0 -101
  37. package/src/admin/ui/components/fields/TextAreaField.tsx +0 -54
  38. package/src/admin/ui/components/fields/TextField.tsx +0 -49
  39. package/src/admin/ui/components/fields/VirtualField.tsx +0 -53
  40. package/src/admin/ui/components/fields/index.tsx +0 -371
  41. package/src/admin/ui/components/fields/richtext-editor/index.tsx +0 -211
  42. package/src/admin/ui/components/fields/richtext-editor/nodes/ImageComponent.tsx +0 -142
  43. package/src/admin/ui/components/fields/richtext-editor/nodes/ImageNode.tsx +0 -95
  44. package/src/admin/ui/components/fields/richtext-editor/plugins/ComponentPickerPlugin.tsx +0 -226
  45. package/src/admin/ui/components/fields/richtext-editor/plugins/EditableSyncPlugin.tsx +0 -16
  46. package/src/admin/ui/components/fields/richtext-editor/plugins/NotionToolbarPlugin.tsx +0 -184
  47. package/src/admin/ui/components/fields/richtext-editor/plugins/SimpleToolbarPlugin.tsx +0 -240
  48. package/src/admin/ui/components/fields/richtext-editor/plugins/ValueSyncPlugin.tsx +0 -40
  49. package/src/admin/ui/components/fields/utils.ts +0 -1
  50. package/src/admin/ui/components/link.tsx +0 -41
  51. package/src/admin/ui/components/media/AssetManagerModal.tsx +0 -334
  52. package/src/admin/ui/components/toast.tsx +0 -72
  53. package/src/admin/ui/components/ui/accordion.tsx +0 -51
  54. package/src/admin/ui/components/ui/alert-dialog.tsx +0 -98
  55. package/src/admin/ui/components/ui/blocks.tsx +0 -32
  56. package/src/admin/ui/components/ui/breadcrumbs.tsx +0 -59
  57. package/src/admin/ui/components/ui/button.tsx +0 -26
  58. package/src/admin/ui/components/ui/collapsible.tsx +0 -124
  59. package/src/admin/ui/components/ui/dialog.tsx +0 -79
  60. package/src/admin/ui/components/ui/group.tsx +0 -20
  61. package/src/admin/ui/components/ui/index.ts +0 -17
  62. package/src/admin/ui/components/ui/input.tsx +0 -12
  63. package/src/admin/ui/components/ui/join.tsx +0 -53
  64. package/src/admin/ui/components/ui/label.tsx +0 -11
  65. package/src/admin/ui/components/ui/radio-group.tsx +0 -75
  66. package/src/admin/ui/components/ui/relationship-detail-sheet.tsx +0 -122
  67. package/src/admin/ui/components/ui/relationship.tsx +0 -58
  68. package/src/admin/ui/components/ui/scroll-area.tsx +0 -19
  69. package/src/admin/ui/components/ui/select.tsx +0 -187
  70. package/src/admin/ui/components/ui/separator.tsx +0 -21
  71. package/src/admin/ui/components/ui/sheet.tsx +0 -106
  72. package/src/admin/ui/components/ui/tabs.tsx +0 -116
  73. package/src/admin/ui/components/ui/utils.ts +0 -3
  74. package/src/admin/ui/hooks/use-debounce.ts +0 -15
  75. package/src/admin/ui/styles/_locale-switcher.scss +0 -33
  76. package/src/admin/ui/styles/accordion.scss +0 -60
  77. package/src/admin/ui/styles/animations.scss +0 -41
  78. package/src/admin/ui/styles/asset-manager.scss +0 -547
  79. package/src/admin/ui/styles/badge.scss +0 -13
  80. package/src/admin/ui/styles/base.scss +0 -22
  81. package/src/admin/ui/styles/button.scss +0 -161
  82. package/src/admin/ui/styles/card.scss +0 -13
  83. package/src/admin/ui/styles/collapsible.scss +0 -75
  84. package/src/admin/ui/styles/data-detail.scss +0 -92
  85. package/src/admin/ui/styles/dialog.scss +0 -102
  86. package/src/admin/ui/styles/empty-state.scss +0 -22
  87. package/src/admin/ui/styles/group.scss +0 -19
  88. package/src/admin/ui/styles/index.scss +0 -33
  89. package/src/admin/ui/styles/input.scss +0 -80
  90. package/src/admin/ui/styles/label.scss +0 -12
  91. package/src/admin/ui/styles/layout.scss +0 -56
  92. package/src/admin/ui/styles/lexical.scss +0 -469
  93. package/src/admin/ui/styles/loading.scss +0 -102
  94. package/src/admin/ui/styles/media-registry.scss +0 -597
  95. package/src/admin/ui/styles/pagination.scss +0 -20
  96. package/src/admin/ui/styles/radio-group.scss +0 -66
  97. package/src/admin/ui/styles/row.scss +0 -17
  98. package/src/admin/ui/styles/scrollbar.scss +0 -36
  99. package/src/admin/ui/styles/select.scss +0 -121
  100. package/src/admin/ui/styles/separator.scss +0 -14
  101. package/src/admin/ui/styles/sheet.scss +0 -152
  102. package/src/admin/ui/styles/sidebar.scss +0 -148
  103. package/src/admin/ui/styles/switch.scss +0 -59
  104. package/src/admin/ui/styles/table.scss +0 -207
  105. package/src/admin/ui/styles/tabs.scss +0 -62
  106. package/src/admin/ui/styles/toast.scss +0 -45
  107. package/src/admin/ui/styles/variables.scss +0 -24
  108. package/src/admin/ui/views/collection-list-view.tsx +0 -720
  109. package/src/admin/ui/views/dashboard-view.tsx +0 -263
  110. package/src/admin/ui/views/document-edit-view.tsx +0 -384
  111. package/src/admin/ui/views/global-edit-view.tsx +0 -226
  112. package/src/admin/ui/views/init-view.tsx +0 -182
  113. package/src/admin/ui/views/login-view.tsx +0 -123
  114. package/src/admin/ui/views/media-registry-view.tsx +0 -1104
  115. package/src/admin/ui/views/settings-view.tsx +0 -729
  116. package/src/admin/webcomponent.tsx +0 -15
  117. package/src/auth/index.ts +0 -194
  118. package/src/auth/migrations.ts +0 -87
  119. package/src/auth/premissions.ts +0 -46
  120. package/src/cli/commands/generate-types.ts +0 -116
  121. package/src/cli/commands/init.ts +0 -95
  122. package/src/cli/commands/migrate-commands.ts +0 -160
  123. package/src/cli/commands/seed-command.ts +0 -11
  124. package/src/cli/d1-mock.ts +0 -101
  125. package/src/cli/index.test.ts +0 -84
  126. package/src/cli/index.ts +0 -183
  127. package/src/cli/r2-mock.ts +0 -217
  128. package/src/cli/seeding.ts +0 -409
  129. package/src/client.ts +0 -181
  130. package/src/config-utils.ts +0 -102
  131. package/src/config.ts +0 -49
  132. package/src/db/adapter.ts +0 -53
  133. package/src/db/better-sqlite.ts +0 -630
  134. package/src/db/bun-sqlite.ts +0 -646
  135. package/src/db/d1.ts +0 -711
  136. package/src/db/index.ts +0 -2
  137. package/src/db/kysely/data-mapper.ts +0 -142
  138. package/src/db/kysely/field-mapper.ts +0 -148
  139. package/src/db/kysely/migration-generator.ts +0 -223
  140. package/src/db/kysely/query-builder.ts +0 -92
  141. package/src/db/kysely/schema-builder.ts +0 -439
  142. package/src/db/kysely/sql-utils.ts +0 -13
  143. package/src/db/migration.ts +0 -40
  144. package/src/db/postgres.ts +0 -621
  145. package/src/db/sqlite.ts +0 -658
  146. package/src/db/system-schema.ts +0 -121
  147. package/src/index.ts +0 -11
  148. package/src/runtimes/README.md +0 -59
  149. package/src/runtimes/bun.ts +0 -49
  150. package/src/runtimes/cloudflare-workers.ts +0 -38
  151. package/src/runtimes/next.ts +0 -26
  152. package/src/runtimes/node.ts +0 -52
  153. package/src/schema/collection.ts +0 -184
  154. package/src/schema/fields/base.ts +0 -164
  155. package/src/schema/fields/index.ts +0 -427
  156. package/src/schema/global.ts +0 -145
  157. package/src/schema/index.ts +0 -4
  158. package/src/schema/infer.ts +0 -72
  159. package/src/server/admin-router.ts +0 -20
  160. package/src/server/admin.ts +0 -142
  161. package/src/server/assets.ts +0 -306
  162. package/src/server/collection-router.ts +0 -55
  163. package/src/server/handlers.ts +0 -722
  164. package/src/server/middlewares/admin.ts +0 -27
  165. package/src/server/middlewares/auth.ts +0 -89
  166. package/src/server/middlewares/context.ts +0 -17
  167. package/src/server/middlewares/cors.ts +0 -24
  168. package/src/server/middlewares/database-init.ts +0 -74
  169. package/src/server/middlewares/rate-limit.ts +0 -71
  170. package/src/server/router.ts +0 -47
  171. package/src/server/setup-middlewares.ts +0 -58
  172. package/src/server/system-router.ts +0 -35
  173. package/src/server.ts +0 -9
  174. package/src/storage/adapters/cloudflare-r2.ts +0 -136
  175. package/src/storage/adapters/local.ts +0 -146
  176. package/src/storage/adapters/s3.ts +0 -186
  177. package/src/storage/errors.ts +0 -46
  178. package/src/storage/index.ts +0 -6
  179. package/src/storage/types.ts +0 -39
  180. package/src/types.ts +0 -605
  181. package/src/utils/lexical.ts +0 -37
  182. package/src/utils/logger.ts +0 -73
  183. package/src/validation.ts +0 -429
  184. package/src/validator.ts +0 -179
  185. package/test/admin-custom-field.test.ts +0 -162
  186. package/test/admin-react-field.test.tsx +0 -134
  187. package/test/api-features.test.ts +0 -78
  188. package/test/api.test.ts +0 -178
  189. package/test/auth.test.ts +0 -62
  190. package/test/cli-integration.test.ts +0 -148
  191. package/test/cli.test.ts +0 -25
  192. package/test/db/postgres.test.ts +0 -95
  193. package/test/db/sqlite-filter.test.ts +0 -53
  194. package/test/db/sqlite.test.ts +0 -82
  195. package/test/engine-features.test.ts +0 -79
  196. package/test/globals.test.ts +0 -74
  197. package/test/integration-tmp/db-app/opacacms.config.ts +0 -15
  198. package/test/integration-tmp/my-sqlite-app/opacacms.config.ts +0 -25
  199. package/test/integration-tmp/my-test-app/index.ts +0 -8
  200. package/test/integration-tmp/my-test-app/opacacms.config.ts +0 -16
  201. package/test/integration-tmp/my-test-app/package.json +0 -12
  202. package/test/populate.test.ts +0 -79
  203. package/test/runtimes.test.ts +0 -43
  204. package/test/schema-builder.test.ts +0 -107
  205. package/test/schema-features.test.ts +0 -63
  206. package/test/seeding.test.ts +0 -68
  207. package/test/storage/local.test.ts +0 -72
  208. package/test/storage/s3.test.ts +0 -60
  209. package/test/structural-data.test.ts +0 -100
  210. package/test/test-setup.ts +0 -11
  211. package/test/validation.test.ts +0 -162
  212. package/tsconfig.json +0 -42
@@ -1,240 +0,0 @@
1
- import { $createCodeNode } from "@lexical/code";
2
- import { TOGGLE_LINK_COMMAND } from "@lexical/link";
3
- import {
4
- INSERT_ORDERED_LIST_COMMAND,
5
- INSERT_UNORDERED_LIST_COMMAND,
6
- REMOVE_LIST_COMMAND,
7
- } from "@lexical/list";
8
- import { useLexicalComposerContext } from "@lexical/react/LexicalComposerContext";
9
- import { $createHeadingNode, $createQuoteNode, $isHeadingNode } from "@lexical/rich-text";
10
- import { $setBlocksType } from "@lexical/selection";
11
- import {
12
- $getSelection,
13
- $insertNodes,
14
- $isRangeSelection,
15
- COMMAND_PRIORITY_CRITICAL,
16
- FORMAT_ELEMENT_COMMAND,
17
- FORMAT_TEXT_COMMAND,
18
- SELECTION_CHANGE_COMMAND,
19
- } from "lexical";
20
- import {
21
- Bold,
22
- Code,
23
- Heading1,
24
- Heading2,
25
- Image as ImageIcon,
26
- Italic,
27
- Link as LinkIcon,
28
- List,
29
- ListOrdered,
30
- Quote,
31
- Strikethrough,
32
- } from "lucide-react";
33
- import { useCallback, useEffect, useState } from "react";
34
- import { getCurrentBaseURL } from "../../../../../api-client";
35
- import { AssetManagerModal } from "../../../media/AssetManagerModal";
36
- import { $createImageNode } from "../nodes/ImageNode";
37
-
38
- export function SimpleToolbarPlugin() {
39
- const [isModalOpen, setIsModalOpen] = useState(false);
40
- const [editor] = useLexicalComposerContext();
41
- const [isBold, setIsBold] = useState(false);
42
- const [isItalic, setIsItalic] = useState(false);
43
- const [isStrikethrough, setIsStrikethrough] = useState(false);
44
- const [blockType, setBlockType] = useState("paragraph");
45
-
46
- const updateToolbar = useCallback(() => {
47
- const selection = $getSelection();
48
- if ($isRangeSelection(selection)) {
49
- setIsBold(selection.hasFormat("bold"));
50
- setIsItalic(selection.hasFormat("italic"));
51
- setIsStrikethrough(selection.hasFormat("strikethrough"));
52
-
53
- const anchorNode = selection.anchor.getNode();
54
- const element =
55
- anchorNode.getKey() === "root" ? anchorNode : anchorNode.getTopLevelElementOrThrow();
56
- const elementKey = element.getKey();
57
- const elementDOM = editor.getElementByKey(elementKey);
58
-
59
- if (elementDOM !== null) {
60
- if ($isHeadingNode(element)) {
61
- const tag = element.getTag();
62
- setBlockType(tag);
63
- } else {
64
- const type = element.getType();
65
- setBlockType(type);
66
- }
67
- }
68
- }
69
- }, [editor]);
70
-
71
- useEffect(() => {
72
- return editor.registerCommand(
73
- SELECTION_CHANGE_COMMAND,
74
- (_payload) => {
75
- updateToolbar();
76
- return false;
77
- },
78
- COMMAND_PRIORITY_CRITICAL,
79
- );
80
- }, [editor, updateToolbar]);
81
-
82
- const formatHeading = (level: "h1" | "h2") => {
83
- if (blockType !== level) {
84
- editor.update(() => {
85
- const selection = $getSelection();
86
- if ($isRangeSelection(selection)) {
87
- $setBlocksType(selection, () => $createHeadingNode(level));
88
- }
89
- });
90
- } else {
91
- editor.update(() => {
92
- const selection = $getSelection();
93
- if ($isRangeSelection(selection)) {
94
- $setBlocksType(selection, () => $createHeadingNode("h1")); // Default to paragraph logic generally, simplify for now
95
- }
96
- });
97
- }
98
- };
99
-
100
- const formatQuote = () => {
101
- if (blockType !== "quote") {
102
- editor.update(() => {
103
- const selection = $getSelection();
104
- if ($isRangeSelection(selection)) {
105
- $setBlocksType(selection, () => $createQuoteNode());
106
- }
107
- });
108
- }
109
- };
110
-
111
- const formatCode = () => {
112
- if (blockType !== "code") {
113
- editor.update(() => {
114
- const selection = $getSelection();
115
- if ($isRangeSelection(selection)) {
116
- $setBlocksType(selection, () => $createCodeNode());
117
- }
118
- });
119
- }
120
- };
121
-
122
- const formatList = (type: "bullet" | "number") => {
123
- if (type === "bullet") {
124
- editor.dispatchCommand(INSERT_UNORDERED_LIST_COMMAND, undefined);
125
- } else {
126
- editor.dispatchCommand(INSERT_ORDERED_LIST_COMMAND, undefined);
127
- }
128
- };
129
-
130
- const insertLink = useCallback(() => {
131
- editor.dispatchCommand(TOGGLE_LINK_COMMAND, "https://");
132
- }, [editor]);
133
-
134
- const handleInsertImage = (asset: any) => {
135
- const url = asset.url || `${getCurrentBaseURL()}/api/assets/${asset.id || asset.assetId}/view`;
136
- editor.update(() => {
137
- const imageNode = $createImageNode(url, asset.filename || "Image");
138
- $insertNodes([imageNode]);
139
- });
140
- setIsModalOpen(false);
141
- };
142
-
143
- return (
144
- <>
145
- <div className="opaca-lexical-toolbar">
146
- <button
147
- type="button"
148
- onClick={() => {
149
- editor.dispatchCommand(FORMAT_TEXT_COMMAND, "bold");
150
- }}
151
- className={`opaca-lexical-btn ${isBold ? "is-active" : ""}`}
152
- >
153
- <Bold size={16} />
154
- </button>
155
- <button
156
- type="button"
157
- onClick={() => {
158
- editor.dispatchCommand(FORMAT_TEXT_COMMAND, "italic");
159
- }}
160
- className={`opaca-lexical-btn ${isItalic ? "is-active" : ""}`}
161
- >
162
- <Italic size={16} />
163
- </button>
164
- <button
165
- type="button"
166
- onClick={() => {
167
- editor.dispatchCommand(FORMAT_TEXT_COMMAND, "strikethrough");
168
- }}
169
- className={`opaca-lexical-btn ${isStrikethrough ? "is-active" : ""}`}
170
- >
171
- <Strikethrough size={16} />
172
- </button>
173
-
174
- <div className="opaca-lexical-divider" />
175
-
176
- <button
177
- type="button"
178
- onClick={() => formatHeading("h1")}
179
- className={`opaca-lexical-btn ${blockType === "h1" ? "is-active" : ""}`}
180
- >
181
- <Heading1 size={16} />
182
- </button>
183
- <button
184
- type="button"
185
- onClick={() => formatHeading("h2")}
186
- className={`opaca-lexical-btn ${blockType === "h2" ? "is-active" : ""}`}
187
- >
188
- <Heading2 size={16} />
189
- </button>
190
-
191
- <button
192
- type="button"
193
- onClick={() => formatList("bullet")}
194
- className={`opaca-lexical-btn ${blockType === "ul" ? "is-active" : ""}`}
195
- >
196
- <List size={16} />
197
- </button>
198
- <button
199
- type="button"
200
- onClick={() => formatList("number")}
201
- className={`opaca-lexical-btn ${blockType === "ol" ? "is-active" : ""}`}
202
- >
203
- <ListOrdered size={16} />
204
- </button>
205
-
206
- <button
207
- type="button"
208
- onClick={formatQuote}
209
- className={`opaca-lexical-btn ${blockType === "quote" ? "is-active" : ""}`}
210
- >
211
- <Quote size={16} />
212
- </button>
213
- <button
214
- type="button"
215
- onClick={formatCode}
216
- className={`opaca-lexical-btn ${blockType === "code" ? "is-active" : ""}`}
217
- >
218
- <Code size={16} />
219
- </button>
220
-
221
- <div className="opaca-lexical-divider" />
222
-
223
- <button type="button" onClick={insertLink} className={`opaca-lexical-btn`}>
224
- <LinkIcon size={16} />
225
- </button>
226
- <button type="button" onClick={() => setIsModalOpen(true)} className={`opaca-lexical-btn`}>
227
- <ImageIcon size={16} />
228
- </button>
229
- </div>
230
-
231
- {isModalOpen && (
232
- <AssetManagerModal
233
- bucket={"default"}
234
- onClose={() => setIsModalOpen(false)}
235
- onSelect={handleInsertImage}
236
- />
237
- )}
238
- </>
239
- );
240
- }
@@ -1,40 +0,0 @@
1
- import { useLexicalComposerContext } from "@lexical/react/LexicalComposerContext";
2
- import { useEffect, useRef } from "react";
3
- import { normalizeEditorState } from "..";
4
-
5
- interface ValueSyncPluginProps {
6
- value: string;
7
- }
8
-
9
- export function ValueSyncPlugin({ value }: ValueSyncPluginProps) {
10
- const [editor] = useLexicalComposerContext();
11
- const isFirstRender = useRef(true);
12
-
13
- useEffect(() => {
14
- // We only want to sync if the value actually changed from the outside
15
- // and specifically when we're moving from empty/initial to having real data.
16
- // Lexical's initialConfig.editorState only works once.
17
- if (isFirstRender.current) {
18
- isFirstRender.current = false;
19
- return;
20
- }
21
-
22
- if (normalizeEditorState(value)) {
23
- editor.update(() => {
24
- const currentContent = JSON.stringify(editor.getEditorState().toJSON());
25
-
26
- // If the external value is different from internal state, and internal is essentially empty or this is the first real load
27
- if (currentContent !== value) {
28
- try {
29
- const newState = editor.parseEditorState(value);
30
- editor.setEditorState(newState);
31
- } catch (e) {
32
- console.error("Failed to parse editor state in ValueSyncPlugin:", e);
33
- }
34
- }
35
- });
36
- }
37
- }, [editor, value]);
38
-
39
- return null;
40
- }
@@ -1 +0,0 @@
1
- export const capitalize = (s: string) => s.charAt(0).toUpperCase() + s.slice(1);
@@ -1,41 +0,0 @@
1
- import { getPagePath, openPage } from "@nanostores/router";
2
- import type React from "react";
3
- import { $router } from "../../router";
4
-
5
- interface LinkProps extends React.AnchorHTMLAttributes<HTMLAnchorElement> {
6
- href: string;
7
- params?: any;
8
- search?: any;
9
- }
10
-
11
- export function Link({ href, params, search, children, onClick, ...props }: LinkProps) {
12
- const handleClick = (e: React.MouseEvent<HTMLAnchorElement>) => {
13
- if (onClick) onClick(e);
14
-
15
- // Check if it's a normal click (no meta keys, left button)
16
- if (
17
- !e.defaultPrevented &&
18
- e.button === 0 &&
19
- !e.metaKey &&
20
- !e.ctrlKey &&
21
- !e.shiftKey &&
22
- !e.altKey &&
23
- !props.target
24
- ) {
25
- e.preventDefault();
26
- // If href starts with /admin, we use the router
27
- if (href.startsWith("/admin")) {
28
- // We can either pass the path directly or try to find a route
29
- $router.open(href);
30
- } else {
31
- window.location.href = href;
32
- }
33
- }
34
- };
35
-
36
- return (
37
- <a href={href} onClick={handleClick} {...props}>
38
- {children}
39
- </a>
40
- );
41
- }
@@ -1,334 +0,0 @@
1
- import { useStore } from '@nanostores/react';
2
- import {
3
- ChevronRight,
4
- File,
5
- FileText,
6
- FolderPlus,
7
- Image as ImageIcon,
8
- Loader2,
9
- Upload,
10
- X,
11
- } from 'lucide-react';
12
- import React, { useMemo, useRef, useState } from 'react';
13
- import { api, getCurrentBaseURL } from '../../../api-client';
14
- import { $config } from '../../../stores/config';
15
- import {
16
- $assets,
17
- $mediaCurrentFolder,
18
- $mediaSelectedBucket,
19
- $mediaViewMode,
20
- type AssetDoc,
21
- setMediaBucket,
22
- setMediaFolder,
23
- } from '../../../stores/media';
24
- import '../../styles/asset-manager.scss';
25
- import {
26
- Select,
27
- SelectContent,
28
- SelectItem,
29
- SelectLabel,
30
- SelectSeparator,
31
- SelectTrigger,
32
- SelectValue,
33
- } from '../ui';
34
-
35
- interface AssetManagerProps {
36
- onSelect: (asset: {
37
- assetId: string;
38
- url: string;
39
- filename: string;
40
- mimeType: string;
41
- filesize: number;
42
- }) => void;
43
- onClose: () => void;
44
- allowedmime_types?: string[];
45
- maxFileSize?: number;
46
- bucket?: string;
47
- }
48
-
49
- export const AssetManagerModal: React.FC<AssetManagerProps> = ({
50
- onSelect,
51
- onClose,
52
- allowedmime_types,
53
- maxFileSize,
54
- bucket = 'default',
55
- }) => {
56
- const { data, loading: isLoading } = useStore($assets);
57
- const selectedBucket = useStore($mediaSelectedBucket);
58
- const currentFolder = useStore($mediaCurrentFolder);
59
- const viewMode = useStore($mediaViewMode);
60
- const config = useStore($config);
61
-
62
- const [isUploading, setIsUploading] = useState<boolean>(false);
63
- const [isDragActive, setIsDragActive] = useState<boolean>(false);
64
-
65
- const fileInputRef = useRef<HTMLInputElement>(null);
66
-
67
- const buckets = useMemo(() => {
68
- return Object.keys(config?.storages || {});
69
- }, [config?.storages]);
70
-
71
- const assets = data?.docs || [];
72
- const folders = data?.folders || [];
73
-
74
- const handleUpload = async (file: File) => {
75
- // ... same logic but use selectedBucket and currentFolder
76
- if (allowedmime_types && !allowedmime_types.includes(file.type)) {
77
- alert(`Invalid file type. Allowed: ${allowedmime_types.join(', ')}`);
78
- return;
79
- }
80
- if (maxFileSize && file.size > maxFileSize) {
81
- alert(`File too large. Max size: ${maxFileSize / 1024 / 1024}MB`);
82
- return;
83
- }
84
-
85
- setIsUploading(true);
86
-
87
- const formData = new FormData();
88
- formData.append('file', file);
89
-
90
- try {
91
- const bucketToUse = selectedBucket === 'all' ? 'default' : selectedBucket;
92
- const resp = await api
93
- .post(
94
- `api/__system/assets/upload?bucket=${bucketToUse}&folder=${currentFolder}`,
95
- {
96
- body: formData,
97
- onDownloadProgress: (_progress: unknown) => {
98
- // Ky uses onDownloadProgress, but xhr.upload.onprogress is for upload.
99
- },
100
- },
101
- )
102
- .json<AssetDoc>();
103
-
104
- setIsUploading(false);
105
- $assets.revalidate(); // Refresh the list
106
- // Auto select the new file
107
- onSelect({
108
- assetId: resp.id,
109
- url: resp.url || `${getCurrentBaseURL()}/api/assets/${resp.id}/view`,
110
- filename: resp.filename,
111
- mimeType: resp.mimeType || resp.mime_type || '',
112
- filesize: resp.filesize,
113
- });
114
- } catch (err: unknown) {
115
- alert(
116
- `Upload failed: ${err instanceof Error ? err.message : 'Unknown error'}`,
117
- );
118
- setIsUploading(false);
119
- }
120
- };
121
-
122
- const handleDrop = (e: React.DragEvent) => {
123
- e.preventDefault();
124
- setIsDragActive(false);
125
- if (
126
- e.dataTransfer.files &&
127
- e.dataTransfer.files.length > 0 &&
128
- e.dataTransfer.files[0]
129
- ) {
130
- handleUpload(e.dataTransfer.files[0]);
131
- }
132
- };
133
-
134
- const getFileIcon = (mime: string) => {
135
- if (!mime) return <File size={40} className="text-gray-400" />;
136
- if (mime.startsWith('image/'))
137
- return <ImageIcon size={40} className="text-blue-400" />;
138
- if (mime.startsWith('video/'))
139
- return <FileText size={40} className="text-purple-400" />;
140
- if (mime.includes('pdf'))
141
- return <FileText size={40} className="text-red-400" />;
142
- return <File size={40} className="text-gray-400" />;
143
- };
144
-
145
- return (
146
- <div className="asset-manager-overlay">
147
- <div className="asset-manager-container">
148
- {/* Header */}
149
- <div className="asset-manager-header">
150
- <div>
151
- <h2>Media Library</h2>
152
- <div className="asset-manager-breadcrumbs">
153
- <button type="button" onClick={() => setMediaFolder('')}>
154
- Home
155
- </button>
156
- {currentFolder
157
- .split('/')
158
- .filter(Boolean)
159
- .map((part: string, i: number, arr: string[]) => (
160
- <React.Fragment key={part || i}>
161
- <ChevronRight size={14} className="breadcrumb-separator" />
162
- <button
163
- type="button"
164
- onClick={() =>
165
- setMediaFolder(arr.slice(0, i + 1).join('/'))
166
- }
167
- >
168
- {part}
169
- </button>
170
- </React.Fragment>
171
- ))}
172
- </div>
173
- </div>
174
- <div className="header-actions">
175
- <Select
176
- value={selectedBucket}
177
- onValueChange={(val: string) => setMediaBucket(val)}
178
- >
179
- <SelectTrigger className="bucket-selector">
180
- <SelectValue placeholder="Bucket" />
181
- </SelectTrigger>
182
- <SelectContent>
183
- <SelectItem value="all">All Buckets</SelectItem>
184
- <SelectSeparator />
185
- <SelectLabel>Storage</SelectLabel>
186
- {buckets.map((b) => (
187
- <SelectItem key={b} value={b}>
188
- {b.toUpperCase()}
189
- </SelectItem>
190
- ))}
191
- </SelectContent>
192
- </Select>
193
- <button type="button" onClick={onClose} className="close-button">
194
- <X size={20} />
195
- </button>
196
- </div>
197
- </div>
198
-
199
- {/* Upload Zone */}
200
- <button
201
- type="button"
202
- onDragOver={(e) => {
203
- e.preventDefault();
204
- setIsDragActive(true);
205
- }}
206
- onDragLeave={() => setIsDragActive(false)}
207
- onDrop={handleDrop}
208
- onClick={() => !isUploading && fileInputRef.current?.click()}
209
- aria-label="Upload file"
210
- className={`asset-manager-upload-zone ${isDragActive ? 'is-drag-active' : ''} ${isUploading ? 'uploading' : ''}`}
211
- >
212
- <input
213
- type="file"
214
- ref={fileInputRef}
215
- style={{ display: 'none' }}
216
- accept={allowedmime_types?.join(',')}
217
- onChange={(e) => {
218
- if (
219
- e.target.files &&
220
- e.target.files.length > 0 &&
221
- e.target.files[0]
222
- ) {
223
- handleUpload(e.target.files[0]);
224
- }
225
- }}
226
- />
227
- {isUploading ? (
228
- <div className="uploading-status">
229
- <Loader2 className="opaca-spin" size={24} />
230
- <span className="status-text">Uploading...</span>
231
- </div>
232
- ) : (
233
- <>
234
- <Upload size={24} className="upload-icon" />
235
- <p className="upload-prompt">
236
- Drag & drop a file here, or <span>click to browse</span>.
237
- </p>
238
- </>
239
- )}
240
- </button>
241
-
242
- {/* Grid */}
243
- <div className="asset-manager-grid-container">
244
- {isLoading ? (
245
- <div className="loading-assets">
246
- <Loader2 className="opaca-spin" size={32} />
247
- <p>Loading assets...</p>
248
- </div>
249
- ) : assets.length === 0 && folders.length === 0 ? (
250
- <div className="no-assets">
251
- <ImageIcon size={48} className="empty-icon" />
252
- <p>No assets found here.</p>
253
- </div>
254
- ) : (
255
- <div className={`asset-manager-grid mode-${viewMode}`}>
256
- {/* Folders */}
257
- {folders.map((folder: any) => (
258
- <button
259
- type="button"
260
- key={`folder-${folder.name}`}
261
- className="asset-manager-card folder-card"
262
- onClick={() =>
263
- setMediaFolder(
264
- currentFolder
265
- ? `${currentFolder}/${folder.name}`
266
- : folder.name,
267
- )
268
- }
269
- >
270
- <div className="asset-thumb">
271
- <FolderPlus size={40} className="folder-icon" />
272
- </div>
273
- <div className="asset-info">
274
- <span className="filename">{folder.name}</span>
275
- <span className="file-meta">Folder</span>
276
- </div>
277
- </button>
278
- ))}
279
-
280
- {/* Assets */}
281
- {assets.map((asset: AssetDoc) => (
282
- <button
283
- type="button"
284
- key={asset.id}
285
- className="asset-manager-card asset-card"
286
- onClick={() => {
287
- const baseUrl = getCurrentBaseURL();
288
- onSelect({
289
- assetId: asset.id,
290
- url: `${baseUrl}/api/assets/${asset.id}/view`,
291
- filename: asset.filename,
292
- mimeType: asset.mimeType || asset.mime_type || '',
293
- filesize: asset.filesize,
294
- });
295
- }}
296
- >
297
- <div className="asset-thumb">
298
- {(() => {
299
- const mime = asset.mimeType || asset.mime_type;
300
- return mime?.startsWith('image/') ? (
301
- <img
302
- src={`${getCurrentBaseURL()}/api/assets/${asset.id}/view`}
303
- alt={asset.filename}
304
- />
305
- ) : (
306
- getFileIcon(mime || '')
307
- );
308
- })()}
309
- </div>
310
-
311
- <div className="asset-info">
312
- <span className="filename" title={asset.filename}>
313
- {asset.filename}
314
- </span>
315
- <span className="file-meta">
316
- {(asset.mimeType || asset.mime_type || '')
317
- .split('/')[1]
318
- ?.toUpperCase() || 'FILE'}{' '}
319
- • {((asset.filesize || 0) / 1024).toFixed(1)} KB
320
- </span>
321
- </div>
322
-
323
- <div className="selection-overlay">
324
- <div className="select-badge">Select</div>
325
- </div>
326
- </button>
327
- ))}
328
- </div>
329
- )}
330
- </div>
331
- </div>
332
- </div>
333
- );
334
- };