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,175 +0,0 @@
1
- import { RichTextEditor } from "./fields/richtext-editor";
2
- import { Input } from "./ui/input";
3
- import { cn } from "./ui/utils";
4
-
5
- interface DataDetailViewProps {
6
- data: any;
7
- label?: string;
8
- depth?: number;
9
- isEditing?: boolean;
10
- onChange?: (data: any) => void;
11
- }
12
-
13
- export const DataDetailView: React.FC<DataDetailViewProps> = ({
14
- data,
15
- label,
16
- depth = 0,
17
- isEditing,
18
- onChange,
19
- }) => {
20
- if (data === null || data === undefined) {
21
- return <span className="opaca-text-muted">None</span>;
22
- }
23
-
24
- // Handle arrays (Blocks or HasMany Relationships)
25
- if (Array.isArray(data)) {
26
- if (data.length === 0) return <span className="opaca-text-muted">Empty</span>;
27
-
28
- return (
29
- <div className="opaca-detail-list">
30
- {data.map((item, index) => (
31
- <div key={item || index} className="opaca-detail-item-card">
32
- {item?.blockType && (
33
- <div className="opaca-detail-item-header">
34
- <span className="opaca-badge">{item.blockType}</span>
35
- </div>
36
- )}
37
- <div className="opaca-detail-item-content">
38
- {typeof item === "object" ? (
39
- <DataDetailView
40
- data={item}
41
- depth={depth + 1}
42
- isEditing={isEditing}
43
- onChange={(val) => {
44
- const newData = [...data];
45
- newData[index] = val;
46
- onChange?.(newData);
47
- }}
48
- />
49
- ) : isEditing ? (
50
- <Input
51
- value={String(item)}
52
- onChange={(e) => {
53
- const newData = [...data];
54
- newData[index] = e.target.value;
55
- onChange?.(newData);
56
- }}
57
- />
58
- ) : (
59
- <span className="opaca-detail-value">{String(item)}</span>
60
- )}
61
- </div>
62
- </div>
63
- ))}
64
- </div>
65
- );
66
- }
67
-
68
- // Handle objects or strings that might be Lexical JSON
69
- const isLexicalJSON = (val: unknown): boolean => {
70
- if (!val) return false;
71
-
72
- if (typeof val === "object") {
73
- return val !== null && "root" in val;
74
- }
75
-
76
- if (typeof val === "string") {
77
- const trimmed = val.trim();
78
-
79
- if (!trimmed.startsWith("{")) return false;
80
-
81
- try {
82
- const parsed = JSON.parse(trimmed);
83
- return parsed && typeof parsed === "object" && "root" in parsed;
84
- } catch {
85
- return false;
86
- }
87
- }
88
-
89
- return false;
90
- };
91
-
92
- if (isLexicalJSON(data) && depth === 0) {
93
- let valueToEdit = data;
94
- if (typeof data === "object") {
95
- try {
96
- valueToEdit = JSON.stringify(data);
97
- } catch (e) {}
98
- }
99
-
100
- return (
101
- <div className="opaca-detail-richtext-preview">
102
- <RichTextEditor
103
- value={valueToEdit}
104
- onChange={(val) => {
105
- try {
106
- onChange?.(JSON.parse(val));
107
- } catch (e) {
108
- onChange?.(val);
109
- }
110
- }}
111
- readOnly={!isEditing}
112
- />
113
- </div>
114
- );
115
- }
116
-
117
- // Handle objects (Groups or Block contents)
118
- if (typeof data === "object" && data !== null) {
119
- const keys = Object.keys(data).filter(
120
- (k) => k !== "blockType" && k !== "id" && k !== "_order" && !k.startsWith("_"),
121
- );
122
- if (keys.length === 0) return <span className="opaca-text-muted">Empty Object</span>;
123
-
124
- return (
125
- <div className={cn("opaca-detail-grid", depth > 0 && "nested")}>
126
- {keys.map((key) => (
127
- <div key={key} className="opaca-detail-row">
128
- <span className="opaca-detail-label">
129
- {key.replace(/([A-Z])/g, " $1").replace(/^./, (str) => str.toUpperCase())}
130
- </span>
131
- <div className="opaca-detail-value-container">
132
- {typeof data[key] === "object" && data[key] !== null ? (
133
- <DataDetailView
134
- data={data[key]}
135
- depth={depth + 1}
136
- isEditing={isEditing}
137
- onChange={(val) => {
138
- onChange?.({ ...data, [key]: val });
139
- }}
140
- />
141
- ) : isEditing ? (
142
- <Input
143
- value={
144
- data[key] === true
145
- ? "true"
146
- : data[key] === false
147
- ? "false"
148
- : String(data[key] ?? "")
149
- }
150
- onChange={(e) => {
151
- let val: any = e.target.value;
152
- if (val === "true") val = true;
153
- if (val === "false") val = false;
154
- onChange?.({ ...data, [key]: val });
155
- }}
156
- />
157
- ) : (
158
- <span className="opaca-detail-value">
159
- {data[key] === true
160
- ? "Yes"
161
- : data[key] === false
162
- ? "No"
163
- : String(data[key] ?? "-")}
164
- </span>
165
- )}
166
- </div>
167
- </div>
168
- ))}
169
- </div>
170
- );
171
- }
172
-
173
- // Fallback for primitives
174
- return <span className="opaca-detail-value">{String(data)}</span>;
175
- };
@@ -1,67 +0,0 @@
1
- import React from "react";
2
-
3
- const Table = React.forwardRef<HTMLTableElement, React.HTMLAttributes<HTMLTableElement>>(
4
- ({ className, ...props }, ref) => (
5
- <div className="opaca-table-wrapper">
6
- <table ref={ref} className={`opaca-new-table ${className || ""}`} {...props} />
7
- </div>
8
- ),
9
- );
10
- Table.displayName = "Table";
11
-
12
- const TableHeader = React.forwardRef<
13
- HTMLTableSectionElement,
14
- React.HTMLAttributes<HTMLTableSectionElement>
15
- >(({ className, ...props }, ref) => (
16
- <thead ref={ref} className={`opaca-table-header ${className || ""}`} {...props} />
17
- ));
18
- TableHeader.displayName = "TableHeader";
19
-
20
- const TableBody = React.forwardRef<
21
- HTMLTableSectionElement,
22
- React.HTMLAttributes<HTMLTableSectionElement>
23
- >(({ className, ...props }, ref) => (
24
- <tbody ref={ref} className={`opaca-table-body ${className || ""}`} {...props} />
25
- ));
26
- TableBody.displayName = "TableBody";
27
-
28
- const TableFooter = React.forwardRef<
29
- HTMLTableSectionElement,
30
- React.HTMLAttributes<HTMLTableSectionElement>
31
- >(({ className, ...props }, ref) => (
32
- <tfoot ref={ref} className={`opaca-table-footer ${className || ""}`} {...props} />
33
- ));
34
- TableFooter.displayName = "TableFooter";
35
-
36
- const TableRow = React.forwardRef<HTMLTableRowElement, React.HTMLAttributes<HTMLTableRowElement>>(
37
- ({ className, ...props }, ref) => (
38
- <tr ref={ref} className={`opaca-table-row ${className || ""}`} {...props} />
39
- ),
40
- );
41
- TableRow.displayName = "TableRow";
42
-
43
- const TableHead = React.forwardRef<
44
- HTMLTableCellElement,
45
- React.ThHTMLAttributes<HTMLTableCellElement>
46
- >(({ className, ...props }, ref) => (
47
- <th ref={ref} className={`opaca-table-head ${className || ""}`} {...props} />
48
- ));
49
- TableHead.displayName = "TableHead";
50
-
51
- const TableCell = React.forwardRef<
52
- HTMLTableCellElement,
53
- React.TdHTMLAttributes<HTMLTableCellElement>
54
- >(({ className, ...props }, ref) => (
55
- <td ref={ref} className={`opaca-table-cell ${className || ""}`} {...props} />
56
- ));
57
- TableCell.displayName = "TableCell";
58
-
59
- const TableCaption = React.forwardRef<
60
- HTMLTableCaptionElement,
61
- React.HTMLAttributes<HTMLTableCaptionElement>
62
- >(({ className, ...props }, ref) => (
63
- <caption ref={ref} className={`opaca-table-caption ${className || ""}`} {...props} />
64
- ));
65
- TableCaption.displayName = "TableCaption";
66
-
67
- export { Table, TableBody, TableCaption, TableCell, TableFooter, TableHead, TableHeader, TableRow };
@@ -1,166 +0,0 @@
1
- import { Plus, Trash2 } from "lucide-react";
2
- import type React from "react";
3
- import { Button } from "../ui/button";
4
- import { capitalize } from "./utils";
5
-
6
- interface ArrayFieldProps {
7
- name: string;
8
- label?: string;
9
- fields: any[];
10
- value: any[];
11
- onChange: (val: any[]) => void;
12
- disabled?: boolean;
13
- readOnly?: boolean;
14
- renderField: (field: any, value: any, onChange: (val: any) => void) => React.ReactNode;
15
- }
16
-
17
- export const ArrayField: React.FC<ArrayFieldProps> = ({
18
- name,
19
- label,
20
- fields = [],
21
- value = [],
22
- onChange,
23
- disabled,
24
- readOnly,
25
- renderField,
26
- }) => {
27
- const addItem = () => {
28
- const currentValue = Array.isArray(value) ? value : [];
29
- const initialData: Record<string, any> = {};
30
-
31
- fields.forEach((f) => {
32
- if (f.name) {
33
- if (f.defaultValue !== undefined) {
34
- initialData[f.name] = f.defaultValue;
35
- } else if (f.type === "boolean") {
36
- initialData[f.name] = false;
37
- } else if (f.type === "number") {
38
- initialData[f.name] = 0;
39
- } else if (f.type === "relationship" && f.hasMany) {
40
- initialData[f.name] = [];
41
- } else if (f.type === "blocks") {
42
- initialData[f.name] = [];
43
- } else if (f.type === "array") {
44
- initialData[f.name] = [];
45
- } else {
46
- initialData[f.name] = "";
47
- }
48
- }
49
- });
50
-
51
- onChange([...currentValue, initialData]);
52
- };
53
-
54
- const removeItem = (index: number) => {
55
- const currentValue = Array.isArray(value) ? value : [];
56
- const newValue = [...currentValue];
57
- newValue.splice(index, 1);
58
- onChange(newValue);
59
- };
60
-
61
- const updateItemData = (index: number, fieldName: string, fieldValue: any) => {
62
- const currentValue = Array.isArray(value) ? value : [];
63
- const newValue = [...currentValue];
64
- newValue[index] = {
65
- ...newValue[index],
66
- [fieldName]: fieldValue,
67
- };
68
- onChange(newValue);
69
- };
70
-
71
- const updateRawItemData = (index: number, fieldValue: any) => {
72
- const currentValue = Array.isArray(value) ? value : [];
73
- const newValue = [...currentValue];
74
- newValue[index] = fieldValue;
75
- onChange(newValue);
76
- };
77
-
78
- return (
79
- <div className="opaca-array-field">
80
- <div
81
- style={{
82
- display: "flex",
83
- justifyContent: "space-between",
84
- alignItems: "center",
85
- marginBottom: "0.75rem",
86
- }}
87
- >
88
- <label htmlFor={name} className="opaca-label" style={{ marginBottom: 0 }}>
89
- {label || capitalize(name)}
90
- </label>
91
- </div>
92
-
93
- <div style={{ display: "flex", flexDirection: "column", gap: "1rem" }}>
94
- {(Array.isArray(value) ? value : []).map((item, index) => (
95
- <div
96
- key={item || index}
97
- style={{
98
- border: "1px solid var(--opaca-border)",
99
- borderRadius: "var(--opaca-radius-lg)",
100
- backgroundColor: "rgba(255, 255, 255, 0.02)",
101
- padding: "1rem",
102
- position: "relative",
103
- }}
104
- >
105
- <div
106
- style={{
107
- display: "flex",
108
- justifyContent: "flex-end",
109
- marginBottom: "0.5rem",
110
- }}
111
- >
112
- {!disabled && !readOnly && (
113
- <Button
114
- type="button"
115
- variant="ghost"
116
- size="icon"
117
- onClick={() => removeItem(index)}
118
- style={{
119
- color: "var(--opaca-text-dim)",
120
- height: "24px",
121
- width: "24px",
122
- }}
123
- >
124
- <Trash2 size={14} />
125
- </Button>
126
- )}
127
- </div>
128
-
129
- <div style={{ display: "flex", flexDirection: "column", gap: "1rem" }}>
130
- {fields.map((field, fieldIndex) => (
131
- <div key={field.name || fieldIndex} className="array-field-item">
132
- {renderField(field, field.name ? item[field.name] : item, (val) => {
133
- if (field.name) {
134
- updateItemData(index, field.name, val);
135
- } else {
136
- updateRawItemData(index, val);
137
- }
138
- })}
139
- </div>
140
- ))}
141
- </div>
142
- </div>
143
- ))}
144
-
145
- {!disabled && !readOnly && (
146
- <Button
147
- type="button"
148
- variant="outline"
149
- size="sm"
150
- onClick={addItem}
151
- style={{
152
- display: "flex",
153
- alignItems: "center",
154
- gap: "6px",
155
- fontSize: "0.75rem",
156
- width: "fit-content",
157
- }}
158
- >
159
- <Plus size={12} />
160
- Add Item
161
- </Button>
162
- )}
163
- </div>
164
- </div>
165
- );
166
- };
@@ -1,202 +0,0 @@
1
- import { Plus, Trash2 } from "lucide-react";
2
- import type React from "react";
3
- import { Blocks } from "../ui/blocks";
4
- import { Button } from "../ui/button";
5
- import { capitalize } from "./utils";
6
-
7
- interface BlocksFieldProps {
8
- name: string;
9
- label?: string;
10
- blocks: Array<{
11
- slug: string;
12
- fields: any[];
13
- label?: string;
14
- }>;
15
- value: any[];
16
- onChange: (val: any[]) => void;
17
- disabled?: boolean;
18
- readOnly?: boolean;
19
- renderField: (field: any, value: any, onChange: (val: any) => void) => React.ReactNode;
20
- }
21
-
22
- export const BlocksField: React.FC<BlocksFieldProps> = ({
23
- name,
24
- label,
25
- blocks,
26
- value = [],
27
- onChange,
28
- disabled,
29
- readOnly,
30
- renderField,
31
- }) => {
32
- const addBlock = (blockSlug: string) => {
33
- const currentValue = Array.isArray(value) ? value : [];
34
- const blockDef = blocks.find((b) => b.slug === blockSlug);
35
-
36
- const initialData: Record<string, any> = { blockType: blockSlug };
37
-
38
- if (blockDef) {
39
- blockDef.fields.forEach((f) => {
40
- if (f.name) {
41
- if (f.defaultValue !== undefined) {
42
- initialData[f.name] = f.defaultValue;
43
- } else if (f.type === "boolean") {
44
- initialData[f.name] = false;
45
- } else if (f.type === "number") {
46
- initialData[f.name] = 0;
47
- } else if (f.type === "relationship" && f.hasMany) {
48
- initialData[f.name] = [];
49
- } else if (f.type === "blocks") {
50
- initialData[f.name] = [];
51
- } else {
52
- initialData[f.name] = "";
53
- }
54
- }
55
- });
56
- }
57
-
58
- onChange([...currentValue, initialData]);
59
- };
60
-
61
- const removeBlock = (index: number) => {
62
- const currentValue = Array.isArray(value) ? value : [];
63
- const newValue = [...currentValue];
64
- newValue.splice(index, 1);
65
- onChange(newValue);
66
- };
67
-
68
- const updateBlockData = (index: number, fieldName: string, fieldValue: any) => {
69
- const currentValue = Array.isArray(value) ? value : [];
70
- const newValue = [...currentValue];
71
- newValue[index] = {
72
- ...newValue[index],
73
- [fieldName]: fieldValue,
74
- };
75
- onChange(newValue);
76
- };
77
-
78
- return (
79
- <Blocks label={label || name} className="opaca-blocks-field">
80
- <div style={{ display: "flex", flexDirection: "column", gap: "24px" }}>
81
- {(Array.isArray(value) ? value : []).map((item, index) => {
82
- const blockDef = blocks.find((b) => b.slug === item.blockType);
83
- if (!blockDef) return null;
84
-
85
- return (
86
- <div
87
- key={`${item.blockType}-${index}`}
88
- style={{
89
- border: "1px solid var(--opaca-border)",
90
- borderRadius: "var(--opaca-radius-lg)",
91
- backgroundColor: "rgba(255, 255, 255, 0.02)",
92
- padding: "1.25rem",
93
- position: "relative",
94
- transition: "border-color var(--opaca-transition)",
95
- }}
96
- className="opaca-block-item"
97
- >
98
- <div
99
- style={{
100
- display: "flex",
101
- justifyContent: "space-between",
102
- alignItems: "center",
103
- marginBottom: "1.25rem",
104
- borderBottom: "1px solid var(--opaca-border)",
105
- paddingBottom: "0.75rem",
106
- }}
107
- >
108
- <div style={{ display: "flex", alignItems: "center", gap: "8px" }}>
109
- <div
110
- style={{
111
- width: "8px",
112
- height: "8px",
113
- borderRadius: "50%",
114
- backgroundColor: "var(--opaca-primary)",
115
- }}
116
- ></div>
117
- <h4
118
- style={{
119
- margin: 0,
120
- fontSize: "0.8125rem",
121
- fontWeight: 600,
122
- color: "var(--opaca-text)",
123
- }}
124
- >
125
- {blockDef.label || capitalize(blockDef.slug)}
126
- </h4>
127
- </div>
128
- {!disabled && !readOnly && (
129
- <Button
130
- type="button"
131
- variant="ghost"
132
- size="icon"
133
- onClick={() => removeBlock(index)}
134
- style={{ color: "var(--opaca-text-dim)", height: "24px", width: "24px" }}
135
- >
136
- <Trash2 size={14} />
137
- </Button>
138
- )}
139
- </div>
140
- <div style={{ display: "flex", flexDirection: "column", gap: "1.25rem" }}>
141
- {blockDef.fields.map((field) => (
142
- <div key={field.name} className="block-field-item">
143
- {renderField(field, item[field.name], (val) =>
144
- updateBlockData(index, field.name, val),
145
- )}
146
- </div>
147
- ))}
148
- </div>
149
- </div>
150
- );
151
- })}
152
-
153
- {!disabled && !readOnly && (
154
- <div
155
- style={{
156
- marginTop: "0.5rem",
157
- padding: "1.25rem",
158
- border: "1px dashed var(--opaca-border)",
159
- borderRadius: "var(--opaca-radius-lg)",
160
- textAlign: "center",
161
- }}
162
- >
163
- <p
164
- style={{
165
- fontSize: "0.75rem",
166
- fontWeight: 500,
167
- color: "var(--opaca-text-muted)",
168
- textTransform: "uppercase",
169
- letterSpacing: "0.03em",
170
- marginBottom: "1rem",
171
- }}
172
- >
173
- Add a new block:
174
- </p>
175
- <div
176
- style={{
177
- display: "flex",
178
- flexWrap: "wrap",
179
- gap: "0.75rem",
180
- justifyContent: "center",
181
- }}
182
- >
183
- {blocks.map((block) => (
184
- <Button
185
- type="button"
186
- key={block.slug}
187
- variant="outline"
188
- size="sm"
189
- onClick={() => addBlock(block.slug)}
190
- style={{ display: "flex", alignItems: "center", gap: "6px", fontSize: "0.75rem" }}
191
- >
192
- <Plus size={12} />
193
- {block.label || capitalize(block.slug)}
194
- </Button>
195
- ))}
196
- </div>
197
- </div>
198
- )}
199
- </div>
200
- </Blocks>
201
- );
202
- };
@@ -1,50 +0,0 @@
1
- import type React from "react";
2
-
3
- interface BooleanFieldProps {
4
- name: string;
5
- label?: string;
6
- value: boolean;
7
- onChange: (val: boolean) => void;
8
- disabled?: boolean;
9
- readOnly?: boolean;
10
- error?: string;
11
- renderType?: "switch" | "toggle" | "checkbox";
12
- }
13
-
14
- export const BooleanField: React.FC<BooleanFieldProps> = ({
15
- name,
16
- label,
17
- value,
18
- onChange,
19
- disabled,
20
- readOnly,
21
- error,
22
- renderType = "switch",
23
- }) => {
24
- const isDisabled = disabled || readOnly;
25
-
26
- return (
27
- <div className="opaca-form-group">
28
- <label className="opaca-label" htmlFor={`field-${name}`}>
29
- {label || name}
30
- </label>
31
-
32
- <label className="opaca-switch">
33
- <input
34
- id={`field-${name}`}
35
- type="checkbox"
36
- checked={!!value}
37
- disabled={isDisabled}
38
- onChange={(e) => onChange(e.target.checked)}
39
- />
40
- <div className="opaca-switch-track">
41
- <div className="opaca-switch-thumb" />
42
- </div>
43
- <span className="opaca-switch-label" style={{ minWidth: "60px" }}>
44
- {value ? "Enabled" : "Disabled"}
45
- </span>
46
- </label>
47
- {error && <span className="opaca-field-error">{error}</span>}
48
- </div>
49
- );
50
- };