create-brainerce-store 1.28.21 → 1.29.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/index.js +5 -4
- package/messages/en.json +7 -0
- package/messages/he.json +7 -0
- package/package.json +3 -3
- package/templates/nextjs/base/src/app/products/[slug]/product-client-section.tsx +566 -529
- package/templates/nextjs/base/src/components/products/customization-fields.tsx +478 -0
- package/templates/nextjs/base/tsconfig.tsbuildinfo +0 -1
|
@@ -0,0 +1,478 @@
|
|
|
1
|
+
'use client';
|
|
2
|
+
|
|
3
|
+
import { useState } from 'react';
|
|
4
|
+
import type { ProductCustomizationField } from 'brainerce';
|
|
5
|
+
import { useTranslations } from '@/lib/translations';
|
|
6
|
+
import { cn } from '@/lib/utils';
|
|
7
|
+
import { LoadingSpinner } from '@/components/shared/loading-spinner';
|
|
8
|
+
|
|
9
|
+
export type CustomizationValues = Record<string, unknown>;
|
|
10
|
+
|
|
11
|
+
interface CustomizationFieldsProps {
|
|
12
|
+
fields: ProductCustomizationField[];
|
|
13
|
+
values: CustomizationValues;
|
|
14
|
+
onChange: (values: CustomizationValues) => void;
|
|
15
|
+
errors?: Record<string, string>;
|
|
16
|
+
}
|
|
17
|
+
|
|
18
|
+
export function CustomizationFields({
|
|
19
|
+
fields,
|
|
20
|
+
values,
|
|
21
|
+
onChange,
|
|
22
|
+
errors = {},
|
|
23
|
+
}: CustomizationFieldsProps) {
|
|
24
|
+
const t = useTranslations('customization');
|
|
25
|
+
|
|
26
|
+
if (fields.length === 0) return null;
|
|
27
|
+
|
|
28
|
+
const sorted = [...fields].sort((a, b) => a.position - b.position);
|
|
29
|
+
|
|
30
|
+
const setValue = (key: string, value: unknown) => {
|
|
31
|
+
onChange({ ...values, [key]: value });
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
return (
|
|
35
|
+
<div className="border-border space-y-4 border-t pt-4">
|
|
36
|
+
<h2 className="text-foreground text-lg font-semibold">{t('title')}</h2>
|
|
37
|
+
<div className="space-y-4">
|
|
38
|
+
{sorted.map((field) => (
|
|
39
|
+
<CustomizationFieldRow
|
|
40
|
+
key={field.definitionId}
|
|
41
|
+
field={field}
|
|
42
|
+
value={values[field.key]}
|
|
43
|
+
error={errors[field.key]}
|
|
44
|
+
onChange={(v) => setValue(field.key, v)}
|
|
45
|
+
/>
|
|
46
|
+
))}
|
|
47
|
+
</div>
|
|
48
|
+
</div>
|
|
49
|
+
);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
interface RowProps {
|
|
53
|
+
field: ProductCustomizationField;
|
|
54
|
+
value: unknown;
|
|
55
|
+
error?: string;
|
|
56
|
+
onChange: (value: unknown) => void;
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
function CustomizationFieldRow({ field, value, error, onChange }: RowProps) {
|
|
60
|
+
const t = useTranslations('customization');
|
|
61
|
+
const labelText = (
|
|
62
|
+
<span className="text-foreground mb-1.5 block text-sm font-medium">
|
|
63
|
+
{field.name}
|
|
64
|
+
{field.required && <span className="text-destructive ms-1">*</span>}
|
|
65
|
+
</span>
|
|
66
|
+
);
|
|
67
|
+
|
|
68
|
+
const help = field.description ? (
|
|
69
|
+
<p className="text-muted-foreground mt-1 text-xs">{field.description}</p>
|
|
70
|
+
) : null;
|
|
71
|
+
|
|
72
|
+
const errorHint = error ? <p className="text-destructive mt-1 text-xs">{error}</p> : null;
|
|
73
|
+
|
|
74
|
+
const inputBase =
|
|
75
|
+
'border-border bg-background focus:border-primary w-full rounded border px-3 py-2 text-sm outline-none focus:ring-1';
|
|
76
|
+
|
|
77
|
+
switch (field.type) {
|
|
78
|
+
case 'TEXTAREA':
|
|
79
|
+
return (
|
|
80
|
+
<div>
|
|
81
|
+
{labelText}
|
|
82
|
+
<textarea
|
|
83
|
+
value={typeof value === 'string' ? value : ''}
|
|
84
|
+
onChange={(e) => onChange(e.target.value)}
|
|
85
|
+
minLength={field.minLength ?? undefined}
|
|
86
|
+
maxLength={field.maxLength ?? undefined}
|
|
87
|
+
required={field.required}
|
|
88
|
+
rows={3}
|
|
89
|
+
className={cn(inputBase, 'resize-y')}
|
|
90
|
+
/>
|
|
91
|
+
{help}
|
|
92
|
+
{errorHint}
|
|
93
|
+
</div>
|
|
94
|
+
);
|
|
95
|
+
|
|
96
|
+
case 'NUMBER':
|
|
97
|
+
return (
|
|
98
|
+
<div>
|
|
99
|
+
{labelText}
|
|
100
|
+
<input
|
|
101
|
+
type="number"
|
|
102
|
+
value={typeof value === 'number' ? value : ''}
|
|
103
|
+
onChange={(e) => onChange(e.target.value === '' ? undefined : Number(e.target.value))}
|
|
104
|
+
min={field.minValue ?? undefined}
|
|
105
|
+
max={field.maxValue ?? undefined}
|
|
106
|
+
required={field.required}
|
|
107
|
+
className={inputBase}
|
|
108
|
+
/>
|
|
109
|
+
{help}
|
|
110
|
+
{errorHint}
|
|
111
|
+
</div>
|
|
112
|
+
);
|
|
113
|
+
|
|
114
|
+
case 'BOOLEAN':
|
|
115
|
+
return (
|
|
116
|
+
<div>
|
|
117
|
+
<label className="inline-flex items-center gap-2">
|
|
118
|
+
<input
|
|
119
|
+
type="checkbox"
|
|
120
|
+
checked={value === true}
|
|
121
|
+
onChange={(e) => onChange(e.target.checked)}
|
|
122
|
+
className="h-4 w-4"
|
|
123
|
+
/>
|
|
124
|
+
<span className="text-foreground text-sm font-medium">
|
|
125
|
+
{field.name}
|
|
126
|
+
{field.required && <span className="text-destructive ms-1">*</span>}
|
|
127
|
+
</span>
|
|
128
|
+
</label>
|
|
129
|
+
{help}
|
|
130
|
+
{errorHint}
|
|
131
|
+
</div>
|
|
132
|
+
);
|
|
133
|
+
|
|
134
|
+
case 'DATE':
|
|
135
|
+
return (
|
|
136
|
+
<div>
|
|
137
|
+
{labelText}
|
|
138
|
+
<input
|
|
139
|
+
type="date"
|
|
140
|
+
value={typeof value === 'string' ? value : ''}
|
|
141
|
+
onChange={(e) => onChange(e.target.value)}
|
|
142
|
+
required={field.required}
|
|
143
|
+
className={inputBase}
|
|
144
|
+
/>
|
|
145
|
+
{help}
|
|
146
|
+
{errorHint}
|
|
147
|
+
</div>
|
|
148
|
+
);
|
|
149
|
+
|
|
150
|
+
case 'DATETIME':
|
|
151
|
+
return (
|
|
152
|
+
<div>
|
|
153
|
+
{labelText}
|
|
154
|
+
<input
|
|
155
|
+
type="datetime-local"
|
|
156
|
+
value={typeof value === 'string' ? value : ''}
|
|
157
|
+
onChange={(e) => onChange(e.target.value)}
|
|
158
|
+
required={field.required}
|
|
159
|
+
className={inputBase}
|
|
160
|
+
/>
|
|
161
|
+
{help}
|
|
162
|
+
{errorHint}
|
|
163
|
+
</div>
|
|
164
|
+
);
|
|
165
|
+
|
|
166
|
+
case 'COLOR':
|
|
167
|
+
return (
|
|
168
|
+
<div>
|
|
169
|
+
{labelText}
|
|
170
|
+
<input
|
|
171
|
+
type="color"
|
|
172
|
+
value={typeof value === 'string' && value ? value : '#000000'}
|
|
173
|
+
onChange={(e) => onChange(e.target.value)}
|
|
174
|
+
required={field.required}
|
|
175
|
+
className="border-border h-10 w-20 rounded border p-1"
|
|
176
|
+
/>
|
|
177
|
+
{help}
|
|
178
|
+
{errorHint}
|
|
179
|
+
</div>
|
|
180
|
+
);
|
|
181
|
+
|
|
182
|
+
case 'SELECT': {
|
|
183
|
+
const options = field.enumValues ?? [];
|
|
184
|
+
return (
|
|
185
|
+
<div>
|
|
186
|
+
{labelText}
|
|
187
|
+
<select
|
|
188
|
+
value={typeof value === 'string' ? value : ''}
|
|
189
|
+
onChange={(e) => onChange(e.target.value || undefined)}
|
|
190
|
+
required={field.required}
|
|
191
|
+
className={inputBase}
|
|
192
|
+
>
|
|
193
|
+
<option value="">{t('selectPlaceholder')}</option>
|
|
194
|
+
{options.map((opt) => (
|
|
195
|
+
<option key={opt} value={opt}>
|
|
196
|
+
{opt}
|
|
197
|
+
</option>
|
|
198
|
+
))}
|
|
199
|
+
</select>
|
|
200
|
+
{help}
|
|
201
|
+
{errorHint}
|
|
202
|
+
</div>
|
|
203
|
+
);
|
|
204
|
+
}
|
|
205
|
+
|
|
206
|
+
case 'MULTI_SELECT': {
|
|
207
|
+
const options = field.enumValues ?? [];
|
|
208
|
+
const current = Array.isArray(value) ? (value as string[]) : [];
|
|
209
|
+
const toggle = (opt: string) => {
|
|
210
|
+
onChange(current.includes(opt) ? current.filter((v) => v !== opt) : [...current, opt]);
|
|
211
|
+
};
|
|
212
|
+
return (
|
|
213
|
+
<div>
|
|
214
|
+
{labelText}
|
|
215
|
+
<div className="space-y-1.5">
|
|
216
|
+
{options.map((opt) => (
|
|
217
|
+
<label key={opt} className="me-4 inline-flex items-center gap-2">
|
|
218
|
+
<input
|
|
219
|
+
type="checkbox"
|
|
220
|
+
checked={current.includes(opt)}
|
|
221
|
+
onChange={() => toggle(opt)}
|
|
222
|
+
className="h-4 w-4"
|
|
223
|
+
/>
|
|
224
|
+
<span className="text-foreground text-sm">{opt}</span>
|
|
225
|
+
</label>
|
|
226
|
+
))}
|
|
227
|
+
</div>
|
|
228
|
+
{help}
|
|
229
|
+
{errorHint}
|
|
230
|
+
</div>
|
|
231
|
+
);
|
|
232
|
+
}
|
|
233
|
+
|
|
234
|
+
case 'IMAGE':
|
|
235
|
+
return (
|
|
236
|
+
<SingleImageField
|
|
237
|
+
field={field}
|
|
238
|
+
value={typeof value === 'string' ? value : undefined}
|
|
239
|
+
onChange={onChange}
|
|
240
|
+
labelText={labelText}
|
|
241
|
+
help={help}
|
|
242
|
+
errorHint={errorHint}
|
|
243
|
+
/>
|
|
244
|
+
);
|
|
245
|
+
|
|
246
|
+
case 'GALLERY':
|
|
247
|
+
return (
|
|
248
|
+
<GalleryField
|
|
249
|
+
field={field}
|
|
250
|
+
value={Array.isArray(value) ? (value as string[]) : []}
|
|
251
|
+
onChange={onChange}
|
|
252
|
+
labelText={labelText}
|
|
253
|
+
help={help}
|
|
254
|
+
errorHint={errorHint}
|
|
255
|
+
/>
|
|
256
|
+
);
|
|
257
|
+
|
|
258
|
+
case 'URL':
|
|
259
|
+
return (
|
|
260
|
+
<div>
|
|
261
|
+
{labelText}
|
|
262
|
+
<input
|
|
263
|
+
type="url"
|
|
264
|
+
value={typeof value === 'string' ? value : ''}
|
|
265
|
+
onChange={(e) => onChange(e.target.value)}
|
|
266
|
+
required={field.required}
|
|
267
|
+
className={inputBase}
|
|
268
|
+
/>
|
|
269
|
+
{help}
|
|
270
|
+
{errorHint}
|
|
271
|
+
</div>
|
|
272
|
+
);
|
|
273
|
+
|
|
274
|
+
default:
|
|
275
|
+
return (
|
|
276
|
+
<div>
|
|
277
|
+
{labelText}
|
|
278
|
+
<input
|
|
279
|
+
type="text"
|
|
280
|
+
value={typeof value === 'string' ? value : ''}
|
|
281
|
+
onChange={(e) => onChange(e.target.value)}
|
|
282
|
+
minLength={field.minLength ?? undefined}
|
|
283
|
+
maxLength={field.maxLength ?? undefined}
|
|
284
|
+
required={field.required}
|
|
285
|
+
className={inputBase}
|
|
286
|
+
/>
|
|
287
|
+
{help}
|
|
288
|
+
{errorHint}
|
|
289
|
+
</div>
|
|
290
|
+
);
|
|
291
|
+
}
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
interface FileFieldProps {
|
|
295
|
+
field: ProductCustomizationField;
|
|
296
|
+
labelText: React.ReactNode;
|
|
297
|
+
help: React.ReactNode;
|
|
298
|
+
errorHint: React.ReactNode;
|
|
299
|
+
onChange: (value: unknown) => void;
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
function SingleImageField({
|
|
303
|
+
field,
|
|
304
|
+
value,
|
|
305
|
+
onChange,
|
|
306
|
+
labelText,
|
|
307
|
+
help,
|
|
308
|
+
errorHint,
|
|
309
|
+
}: FileFieldProps & { value?: string }) {
|
|
310
|
+
const t = useTranslations('customization');
|
|
311
|
+
const [uploading, setUploading] = useState(false);
|
|
312
|
+
const [uploadError, setUploadError] = useState<string | null>(null);
|
|
313
|
+
|
|
314
|
+
async function handleFile(file: File | null) {
|
|
315
|
+
if (!file) return;
|
|
316
|
+
setUploading(true);
|
|
317
|
+
setUploadError(null);
|
|
318
|
+
try {
|
|
319
|
+
const { getClient } = await import('@/lib/brainerce');
|
|
320
|
+
const client = getClient();
|
|
321
|
+
const result = await client.uploadCustomizationFile(file);
|
|
322
|
+
onChange(result.url);
|
|
323
|
+
} catch (err) {
|
|
324
|
+
console.error('Upload failed', err);
|
|
325
|
+
setUploadError(t('uploadFailed'));
|
|
326
|
+
} finally {
|
|
327
|
+
setUploading(false);
|
|
328
|
+
}
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
return (
|
|
332
|
+
<div>
|
|
333
|
+
{labelText}
|
|
334
|
+
<input
|
|
335
|
+
type="file"
|
|
336
|
+
accept="image/*"
|
|
337
|
+
onChange={(e) => handleFile(e.target.files?.[0] ?? null)}
|
|
338
|
+
disabled={uploading}
|
|
339
|
+
required={field.required && !value}
|
|
340
|
+
className="text-foreground text-sm"
|
|
341
|
+
/>
|
|
342
|
+
{uploading && (
|
|
343
|
+
<p className="text-muted-foreground mt-2 inline-flex items-center gap-2 text-xs">
|
|
344
|
+
<LoadingSpinner size="sm" />
|
|
345
|
+
{t('uploading')}
|
|
346
|
+
</p>
|
|
347
|
+
)}
|
|
348
|
+
{value && !uploading && (
|
|
349
|
+
<div className="mt-2">
|
|
350
|
+
<img src={value} alt="" className="border-border h-20 w-20 rounded border object-cover" />
|
|
351
|
+
</div>
|
|
352
|
+
)}
|
|
353
|
+
{uploadError && <p className="text-destructive mt-1 text-xs">{uploadError}</p>}
|
|
354
|
+
{help}
|
|
355
|
+
{errorHint}
|
|
356
|
+
</div>
|
|
357
|
+
);
|
|
358
|
+
}
|
|
359
|
+
|
|
360
|
+
function GalleryField({
|
|
361
|
+
field,
|
|
362
|
+
value,
|
|
363
|
+
onChange,
|
|
364
|
+
labelText,
|
|
365
|
+
help,
|
|
366
|
+
errorHint,
|
|
367
|
+
}: FileFieldProps & { value: string[] }) {
|
|
368
|
+
const t = useTranslations('customization');
|
|
369
|
+
const [uploading, setUploading] = useState(false);
|
|
370
|
+
const [uploadError, setUploadError] = useState<string | null>(null);
|
|
371
|
+
|
|
372
|
+
async function handleFiles(fileList: FileList | null) {
|
|
373
|
+
if (!fileList || fileList.length === 0) return;
|
|
374
|
+
setUploading(true);
|
|
375
|
+
setUploadError(null);
|
|
376
|
+
try {
|
|
377
|
+
const { getClient } = await import('@/lib/brainerce');
|
|
378
|
+
const client = getClient();
|
|
379
|
+
const newUrls: string[] = [];
|
|
380
|
+
for (const file of Array.from(fileList)) {
|
|
381
|
+
const result = await client.uploadCustomizationFile(file);
|
|
382
|
+
newUrls.push(result.url);
|
|
383
|
+
}
|
|
384
|
+
onChange([...value, ...newUrls]);
|
|
385
|
+
} catch (err) {
|
|
386
|
+
console.error('Upload failed', err);
|
|
387
|
+
setUploadError(t('uploadFailed'));
|
|
388
|
+
} finally {
|
|
389
|
+
setUploading(false);
|
|
390
|
+
}
|
|
391
|
+
}
|
|
392
|
+
|
|
393
|
+
function removeAt(idx: number) {
|
|
394
|
+
onChange(value.filter((_, i) => i !== idx));
|
|
395
|
+
}
|
|
396
|
+
|
|
397
|
+
return (
|
|
398
|
+
<div>
|
|
399
|
+
{labelText}
|
|
400
|
+
<input
|
|
401
|
+
type="file"
|
|
402
|
+
accept="image/*"
|
|
403
|
+
multiple
|
|
404
|
+
onChange={(e) => handleFiles(e.target.files)}
|
|
405
|
+
disabled={uploading}
|
|
406
|
+
className="text-foreground text-sm"
|
|
407
|
+
/>
|
|
408
|
+
{uploading && (
|
|
409
|
+
<p className="text-muted-foreground mt-2 inline-flex items-center gap-2 text-xs">
|
|
410
|
+
<LoadingSpinner size="sm" />
|
|
411
|
+
{t('uploading')}
|
|
412
|
+
</p>
|
|
413
|
+
)}
|
|
414
|
+
{value.length > 0 && (
|
|
415
|
+
<div className="mt-2 flex flex-wrap gap-2">
|
|
416
|
+
{value.map((url, idx) => (
|
|
417
|
+
<div key={`${url}-${idx}`} className="relative">
|
|
418
|
+
<img
|
|
419
|
+
src={url}
|
|
420
|
+
alt=""
|
|
421
|
+
className="border-border h-16 w-16 rounded border object-cover"
|
|
422
|
+
/>
|
|
423
|
+
<button
|
|
424
|
+
type="button"
|
|
425
|
+
onClick={() => removeAt(idx)}
|
|
426
|
+
aria-label={t('removeImage')}
|
|
427
|
+
className="bg-destructive text-destructive-foreground absolute -end-1.5 -top-1.5 flex h-5 w-5 items-center justify-center rounded-full text-xs"
|
|
428
|
+
>
|
|
429
|
+
×
|
|
430
|
+
</button>
|
|
431
|
+
</div>
|
|
432
|
+
))}
|
|
433
|
+
</div>
|
|
434
|
+
)}
|
|
435
|
+
{uploadError && <p className="text-destructive mt-1 text-xs">{uploadError}</p>}
|
|
436
|
+
{help}
|
|
437
|
+
{errorHint}
|
|
438
|
+
</div>
|
|
439
|
+
);
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
export function validateCustomization(
|
|
443
|
+
fields: ProductCustomizationField[],
|
|
444
|
+
values: CustomizationValues
|
|
445
|
+
): Record<string, string> {
|
|
446
|
+
const errors: Record<string, string> = {};
|
|
447
|
+
for (const field of fields) {
|
|
448
|
+
const value = values[field.key];
|
|
449
|
+
const empty =
|
|
450
|
+
value === undefined ||
|
|
451
|
+
value === null ||
|
|
452
|
+
value === '' ||
|
|
453
|
+
(Array.isArray(value) && value.length === 0);
|
|
454
|
+
|
|
455
|
+
if (field.required && empty) {
|
|
456
|
+
errors[field.key] = 'required';
|
|
457
|
+
continue;
|
|
458
|
+
}
|
|
459
|
+
if (empty) continue;
|
|
460
|
+
|
|
461
|
+
if (field.type === 'SELECT') {
|
|
462
|
+
const allowed = field.enumValues ?? [];
|
|
463
|
+
if (typeof value !== 'string' || !allowed.includes(value)) {
|
|
464
|
+
errors[field.key] = 'invalidOption';
|
|
465
|
+
}
|
|
466
|
+
}
|
|
467
|
+
if (field.type === 'MULTI_SELECT') {
|
|
468
|
+
const allowed = field.enumValues ?? [];
|
|
469
|
+
if (
|
|
470
|
+
!Array.isArray(value) ||
|
|
471
|
+
!(value as unknown[]).every((v) => typeof v === 'string' && allowed.includes(v as string))
|
|
472
|
+
) {
|
|
473
|
+
errors[field.key] = 'invalidOption';
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
}
|
|
477
|
+
return errors;
|
|
478
|
+
}
|