sales-frontend-components 0.0.50 → 0.0.52

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.esm.js CHANGED
@@ -1,12 +1,94 @@
1
- import { jsxs, jsx } from 'react/jsx-runtime';
2
- import { flexRender, useReactTable, getSortedRowModel, getCoreRowModel, getPaginationRowModel } from '@tanstack/react-table';
3
- import { Table, DatePicker, SegmentGroup, FormField, Button, Icon } from 'sales-frontend-design-system';
4
- import styles from './data-table/data-table.module.scss';
5
- import React, { useState, useRef, useEffect, useCallback, useImperativeHandle } from 'react';
1
+ import { jsx, jsxs } from 'react/jsx-runtime';
6
2
  import { useController } from 'react-hook-form';
7
- import styles$1 from './form/form-signautre/form-signature.module.scss';
8
- import styles$2 from './step-indicator/step-indicator.module.scss';
9
- import styles$3 from './camera/camera.module.scss';
3
+ import { DatePicker, SegmentGroup, FormField, Icon } from 'sales-frontend-design-system';
4
+ import React, { useState, useEffect, useRef, useCallback } from 'react';
5
+ import styles from './step-indicator/step-indicator.module.scss';
6
+ import styles$1 from './camera/camera.module.scss';
7
+
8
+ const FormDatePicker = ({
9
+ name,
10
+ control,
11
+ disabled,
12
+ ...props
13
+ }) => {
14
+ const { field, fieldState } = useController({ name, control, disabled });
15
+ const baseProps = {
16
+ ...props,
17
+ id: field.name,
18
+ defaultValue: field.value,
19
+ disabled: field.disabled,
20
+ error: fieldState.invalid,
21
+ onBlur: field.onBlur
22
+ };
23
+ return /* @__PURE__ */ jsx(DatePicker, { ...baseProps });
24
+ };
25
+
26
+ const FormSegmentGroup = ({
27
+ name,
28
+ control,
29
+ disabled,
30
+ ...props
31
+ }) => {
32
+ const { field, fieldState } = useController({ name, control, disabled });
33
+ return /* @__PURE__ */ jsx(
34
+ SegmentGroup,
35
+ {
36
+ ...props,
37
+ tabIndex: 0,
38
+ id: field.name,
39
+ ref: field.ref,
40
+ defaultValue: field.value,
41
+ disabled: field.disabled,
42
+ error: fieldState.invalid,
43
+ onBlur: field.onBlur,
44
+ onValueChange: (selected) => {
45
+ field.onChange(selected);
46
+ props.onValueChange?.(selected);
47
+ }
48
+ }
49
+ );
50
+ };
51
+
52
+ const FormTextField = ({
53
+ name,
54
+ control,
55
+ disabled,
56
+ error,
57
+ onBlur,
58
+ onChange,
59
+ rootProps,
60
+ ...props
61
+ }) => {
62
+ const { field, fieldState } = useController({ name, control, disabled });
63
+ return /* @__PURE__ */ jsx(
64
+ FormField.TextField,
65
+ {
66
+ ...props,
67
+ id: field.name,
68
+ size: "medium",
69
+ autoComplete: "off",
70
+ name: field.name,
71
+ value: field.value ?? "",
72
+ disabled: field.disabled,
73
+ error: fieldState.invalid || error,
74
+ onChange: (e) => {
75
+ field.onChange(e);
76
+ onChange?.(e);
77
+ },
78
+ onBlur: (e) => {
79
+ field.onBlur();
80
+ onBlur?.(e);
81
+ },
82
+ rootProps: {
83
+ ...rootProps,
84
+ onClear: () => {
85
+ field.onChange("");
86
+ rootProps?.onClear?.();
87
+ }
88
+ }
89
+ }
90
+ );
91
+ };
10
92
 
11
93
  function getDefaultExportFromCjs (x) {
12
94
  return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
@@ -99,629 +181,7 @@ function requireBind () {
99
181
  var bindExports = requireBind();
100
182
  var classNames = /*@__PURE__*/getDefaultExportFromCjs(bindExports);
101
183
 
102
- const cx$3 = classNames.bind(styles);
103
- const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
104
- return /* @__PURE__ */ jsxs(
105
- Table,
106
- {
107
- ...props,
108
- selectable: true,
109
- style: {
110
- // minWidth: table.getCenterTotalSize() // 테이블 전체 너비를 컬럼 너비의 합으로 설정
111
- },
112
- children: [
113
- /* @__PURE__ */ jsx("thead", { children: table.getHeaderGroups().map((headerGroup) => /* @__PURE__ */ jsx("tr", { children: headerGroup.headers.map((header) => /* @__PURE__ */ jsxs(
114
- "th",
115
- {
116
- colSpan: header.colSpan,
117
- className: cx$3({
118
- "is-resizing": header.column.getIsResizing()
119
- // 3. 리사이징 중일 때 클래스 추가
120
- }),
121
- style: header.column.columnDef.size ? { width: header.getSize(), minWidth: header.getSize(), maxWidth: header.getSize() } : void 0,
122
- children: [
123
- header.isPlaceholder ? null : flexRender(header.column.columnDef.header, header.getContext()),
124
- header.column.getCanResize() && /* @__PURE__ */ jsx(
125
- "div",
126
- {
127
- onMouseDown: header.getResizeHandler(),
128
- onTouchStart: header.getResizeHandler(),
129
- className: cx$3("resize-handle")
130
- }
131
- )
132
- ]
133
- },
134
- header.id
135
- )) }, headerGroup.id)) }),
136
- /* @__PURE__ */ jsx("tbody", { children: !isLoading && !isError && table.getRowModel().rows.length > 0 ? table.getRowModel().rows.map((row) => (
137
- // TODO aria-disabled 스타일을 어떻게 적용할 것인지?
138
- /* @__PURE__ */ jsx("tr", { "aria-selected": row.getIsSelected(), children: row.getVisibleCells().map((cell) => /* @__PURE__ */ jsx(
139
- "td",
140
- {
141
- style: cell.column.columnDef.size ? {
142
- width: cell.column.getSize(),
143
- minWidth: cell.column.getSize(),
144
- maxWidth: cell.column.getSize()
145
- } : void 0,
146
- children: flexRender(cell.column.columnDef.cell, cell.getContext())
147
- },
148
- cell.id
149
- )) }, row.id)
150
- )) : /* @__PURE__ */ jsx("tr", { children: /* @__PURE__ */ jsx("td", { className: cx$3("feedback-cell"), colSpan: table.getVisibleLeafColumns().length, children: isLoading ? /* @__PURE__ */ jsx("div", { children: "Loading..." }) : /* @__PURE__ */ jsx("div", { children: msgText || (isError ? "Error" : "No data") }) }) }) })
151
- ]
152
- }
153
- );
154
- };
155
-
156
- const useTable = ({
157
- data,
158
- columns,
159
- defaultColumn = { size: 150, enableResizing: true },
160
- initialState,
161
- onSortingChange,
162
- enableRowSelection,
163
- enableColumnResizing,
164
- mode = "SSR",
165
- meta,
166
- debug
167
- }) => {
168
- const [sorting, setSorting] = React.useState(initialState?.sorting || []);
169
- const [rowSelection, setRowSelection] = useState(initialState?.rowSelection || {});
170
- const [pagination, setPagination] = React.useState(initialState?.pagination || {});
171
- const [columnVisibility, setColumnVisibility] = React.useState(initialState?.columnVisibility || {});
172
- const getOption = (mode2) => {
173
- if (mode2 === "SSR") {
174
- return {
175
- manualSorting: true,
176
- manualPagination: true,
177
- onSortingChange
178
- };
179
- } else {
180
- return {
181
- getPaginationRowModel: getPaginationRowModel(),
182
- onPaginationChange: setPagination,
183
- onSortingChange: setSorting
184
- };
185
- }
186
- };
187
- const table = useReactTable({
188
- meta,
189
- data,
190
- columns,
191
- columnResizeMode: "onChange",
192
- defaultColumn,
193
- enableColumnResizing,
194
- enableRowSelection,
195
- initialState,
196
- state: {
197
- sorting,
198
- rowSelection,
199
- pagination,
200
- columnVisibility
201
- },
202
- onRowSelectionChange: setRowSelection,
203
- onColumnVisibilityChange: setColumnVisibility,
204
- getCoreRowModel: getCoreRowModel(),
205
- getSortedRowModel: getSortedRowModel(),
206
- ...getOption(mode),
207
- debugTable: debug,
208
- debugHeaders: debug,
209
- debugColumns: debug
210
- });
211
- return table;
212
- };
213
-
214
- const FormDatePicker = ({
215
- name,
216
- control,
217
- disabled,
218
- ...props
219
- }) => {
220
- const { field, fieldState } = useController({ name, control, disabled });
221
- const baseProps = {
222
- ...props,
223
- id: field.name,
224
- defaultValue: field.value,
225
- disabled: field.disabled,
226
- error: fieldState.invalid,
227
- onBlur: field.onBlur
228
- };
229
- return /* @__PURE__ */ jsx(DatePicker, { ...baseProps });
230
- };
231
-
232
- const FormSegmentGroup = ({
233
- name,
234
- control,
235
- disabled,
236
- ...props
237
- }) => {
238
- const { field, fieldState } = useController({ name, control, disabled });
239
- return /* @__PURE__ */ jsx(
240
- SegmentGroup,
241
- {
242
- ...props,
243
- tabIndex: 0,
244
- id: field.name,
245
- ref: field.ref,
246
- defaultValue: field.value,
247
- disabled: field.disabled,
248
- error: fieldState.invalid,
249
- onBlur: field.onBlur,
250
- onValueChange: (selected) => {
251
- field.onChange(selected);
252
- props.onValueChange?.(selected);
253
- }
254
- }
255
- );
256
- };
257
-
258
- const FormTextField = ({
259
- name,
260
- control,
261
- disabled,
262
- error,
263
- onBlur,
264
- onChange,
265
- rootProps,
266
- ...props
267
- }) => {
268
- const { field, fieldState } = useController({ name, control, disabled });
269
- return /* @__PURE__ */ jsx(
270
- FormField.TextField,
271
- {
272
- ...props,
273
- id: field.name,
274
- size: "medium",
275
- autoComplete: "off",
276
- name: field.name,
277
- value: field.value ?? "",
278
- disabled: field.disabled,
279
- error: fieldState.invalid || error,
280
- onChange: (e) => {
281
- field.onChange(e);
282
- onChange?.(e);
283
- },
284
- onBlur: (e) => {
285
- field.onBlur();
286
- onBlur?.(e);
287
- },
288
- rootProps: {
289
- ...rootProps,
290
- onClear: () => {
291
- field.onChange("");
292
- rootProps?.onClear?.();
293
- }
294
- }
295
- }
296
- );
297
- };
298
-
299
- const cx$2 = classNames.bind(styles$1);
300
- const FormSignature = ({
301
- label,
302
- error,
303
- errorMsg,
304
- onSubmit,
305
- onRecentSignatureLoad,
306
- onChangeSignature,
307
- onReset,
308
- isShowRecentSignatureButton,
309
- isShowResetButton,
310
- ref,
311
- canvasWidth = 632,
312
- canvasHeight = 230
313
- }) => {
314
- const canvasRef = useRef(null);
315
- const containerRef = useRef(null);
316
- const [isDrawing, setIsDrawing] = useState(false);
317
- const [isEmpty, setIsEmpty] = useState(true);
318
- const [signatureData, setSignatureData] = useState("");
319
- const [isLoadingSignature, setIsLoadingSignature] = useState(false);
320
- const [isMousePressed, setIsMousePressed] = useState(false);
321
- const [totalLineLength, setTotalLineLength] = useState(0);
322
- const [lastPoint, setLastPoint] = useState(null);
323
- const [strokeLengths, setStrokeLengths] = useState([]);
324
- const [currentStrokeLength, setCurrentStrokeLength] = useState(0);
325
- useEffect(() => {
326
- const canvas = canvasRef.current;
327
- if (!canvas) {
328
- return;
329
- }
330
- const ctx = canvas.getContext("2d");
331
- if (!ctx) {
332
- return;
333
- }
334
- ctx.strokeStyle = "#000000";
335
- ctx.lineWidth = Math.max(1, canvasWidth / 400);
336
- ctx.lineCap = "round";
337
- ctx.lineJoin = "round";
338
- }, [canvasWidth]);
339
- useEffect(() => {
340
- const handleGlobalMouseUp = () => {
341
- setIsMousePressed(false);
342
- if (isDrawing) {
343
- stopDrawing();
344
- }
345
- };
346
- document.addEventListener("mouseup", handleGlobalMouseUp);
347
- return () => {
348
- document.removeEventListener("mouseup", handleGlobalMouseUp);
349
- };
350
- }, [isDrawing]);
351
- const getCanvasCoordinates = useCallback((e) => {
352
- const canvas = canvasRef.current;
353
- if (!canvas) {
354
- return { x: 0, y: 0 };
355
- }
356
- const rect = canvas.getBoundingClientRect();
357
- const scaleX = canvas.width / rect.width;
358
- const scaleY = canvas.height / rect.height;
359
- return {
360
- x: (e.clientX - rect.left) * scaleX,
361
- y: (e.clientY - rect.top) * scaleY
362
- };
363
- }, []);
364
- const calculateDistance = useCallback((point1, point2) => {
365
- return Math.sqrt(Math.pow(point2.x - point1.x, 2) + Math.pow(point2.y - point1.y, 2));
366
- }, []);
367
- const startDrawing = useCallback(
368
- (e) => {
369
- const canvas = canvasRef.current;
370
- if (!canvas) {
371
- return;
372
- }
373
- const ctx = canvas.getContext("2d");
374
- if (!ctx) {
375
- return;
376
- }
377
- const { x, y } = getCanvasCoordinates(e);
378
- setIsDrawing(true);
379
- setIsMousePressed(true);
380
- setIsEmpty(false);
381
- setLastPoint({ x, y });
382
- setCurrentStrokeLength(0);
383
- ctx.beginPath();
384
- ctx.moveTo(x, y);
385
- },
386
- [getCanvasCoordinates]
387
- );
388
- const draw = useCallback(
389
- (e) => {
390
- if (!isDrawing) {
391
- return;
392
- }
393
- const canvas = canvasRef.current;
394
- if (!canvas) {
395
- return;
396
- }
397
- const ctx = canvas.getContext("2d");
398
- if (!ctx) {
399
- return;
400
- }
401
- const { x, y } = getCanvasCoordinates(e);
402
- const currentPoint = { x, y };
403
- if (lastPoint) {
404
- const distance = calculateDistance(lastPoint, currentPoint);
405
- if (distance > 0.5) {
406
- setCurrentStrokeLength((prev) => prev + distance);
407
- setTotalLineLength((prev) => prev + distance);
408
- }
409
- }
410
- ctx.lineTo(x, y);
411
- ctx.stroke();
412
- setLastPoint(currentPoint);
413
- const data = canvas.toDataURL();
414
- setSignatureData(data);
415
- onChangeSignature?.();
416
- },
417
- [isDrawing, getCanvasCoordinates, calculateDistance, lastPoint, onChangeSignature]
418
- );
419
- const stopDrawing = useCallback(() => {
420
- if (isDrawing && currentStrokeLength > 0) {
421
- setStrokeLengths((prev) => [...prev, currentStrokeLength]);
422
- }
423
- setIsDrawing(false);
424
- setLastPoint(null);
425
- }, [isDrawing, currentStrokeLength]);
426
- const handleMouseEnter = useCallback(
427
- (e) => {
428
- if (isMousePressed && !isDrawing) {
429
- const canvas = canvasRef.current;
430
- if (!canvas) {
431
- return;
432
- }
433
- const ctx = canvas.getContext("2d");
434
- if (!ctx) {
435
- return;
436
- }
437
- const { x, y } = getCanvasCoordinates(e);
438
- setIsDrawing(true);
439
- setLastPoint({ x, y });
440
- ctx.beginPath();
441
- ctx.moveTo(x, y);
442
- }
443
- },
444
- [isMousePressed, isDrawing, getCanvasCoordinates]
445
- );
446
- const handleMouseLeave = useCallback(() => {
447
- if (isDrawing) {
448
- setIsDrawing(false);
449
- setLastPoint(null);
450
- }
451
- }, [isDrawing]);
452
- const clearSignature = useCallback(() => {
453
- const canvas = canvasRef.current;
454
- if (!canvas) {
455
- return;
456
- }
457
- const ctx = canvas.getContext("2d");
458
- if (!ctx) {
459
- return;
460
- }
461
- ctx.clearRect(0, 0, canvasWidth, canvasHeight);
462
- setIsEmpty(true);
463
- setSignatureData("");
464
- setTotalLineLength(0);
465
- setLastPoint(null);
466
- setStrokeLengths([]);
467
- setCurrentStrokeLength(0);
468
- }, [canvasWidth, canvasHeight]);
469
- const loadSignatureToCanvas = useCallback(
470
- (imageDataURL) => {
471
- return new Promise((resolve, reject) => {
472
- const canvas = canvasRef.current;
473
- if (!canvas) {
474
- reject(new Error("Canvas not found"));
475
- return;
476
- }
477
- const ctx = canvas.getContext("2d");
478
- if (!ctx) {
479
- reject(new Error("Canvas context not found"));
480
- return;
481
- }
482
- const img = new Image();
483
- img.onload = () => {
484
- ctx.clearRect(0, 0, canvasWidth, canvasHeight);
485
- ctx.drawImage(img, 0, 0, canvasWidth, canvasHeight);
486
- setIsEmpty(false);
487
- setSignatureData(imageDataURL);
488
- setTotalLineLength(0);
489
- setStrokeLengths([]);
490
- setCurrentStrokeLength(0);
491
- setLastPoint(null);
492
- resolve();
493
- };
494
- img.onerror = () => {
495
- reject(new Error("Failed to load signature image"));
496
- };
497
- img.src = imageDataURL;
498
- });
499
- },
500
- [canvasWidth, canvasHeight]
501
- );
502
- const validateForm = useCallback(() => {
503
- return !(!signatureData || isEmpty);
504
- }, [signatureData, isEmpty]);
505
- const getSignatureDataObject = useCallback(() => {
506
- if (!signatureData || isEmpty) {
507
- return null;
508
- }
509
- return {
510
- signature: signatureData,
511
- totalLineLength: Math.round(totalLineLength),
512
- strokeCount: strokeLengths.length,
513
- strokeLengths: strokeLengths.map((length) => Math.round(length)),
514
- averageStrokeLength: strokeLengths.length > 0 ? Math.round(totalLineLength / strokeLengths.length) : 0
515
- };
516
- }, [signatureData, isEmpty, totalLineLength, strokeLengths]);
517
- const handleSubmit = useCallback(() => {
518
- if (!validateForm()) {
519
- return;
520
- }
521
- const data = getSignatureDataObject();
522
- if (data) {
523
- console.log("\uC81C\uCD9C\uB41C \uB370\uC774\uD130:", data);
524
- onSubmit(data);
525
- }
526
- }, [validateForm, getSignatureDataObject, onSubmit]);
527
- const handleReset = useCallback(() => {
528
- clearSignature();
529
- onReset?.();
530
- }, [clearSignature, onReset]);
531
- const handleRecentSignatureLoad = useCallback(async () => {
532
- if (!onRecentSignatureLoad) {
533
- return;
534
- }
535
- try {
536
- setIsLoadingSignature(true);
537
- const loadedSignature = await onRecentSignatureLoad();
538
- if (loadedSignature) {
539
- await loadSignatureToCanvas(loadedSignature);
540
- console.log("\uCD5C\uADFC \uC11C\uBA85\uC744 \uC131\uACF5\uC801\uC73C\uB85C \uBD88\uB7EC\uC654\uC2B5\uB2C8\uB2E4.");
541
- } else {
542
- alert("\uBD88\uB7EC\uC62C \uCD5C\uADFC \uC11C\uBA85\uC774 \uC5C6\uC2B5\uB2C8\uB2E4.");
543
- }
544
- } catch (error2) {
545
- console.error("\uCD5C\uADFC \uC11C\uBA85 \uBD88\uB7EC\uC624\uAE30 \uC2E4\uD328:", error2);
546
- alert("\uCD5C\uADFC \uC11C\uBA85\uC744 \uBD88\uB7EC\uC624\uB294\uB370 \uC2E4\uD328\uD588\uC2B5\uB2C8\uB2E4.");
547
- } finally {
548
- setIsLoadingSignature(false);
549
- }
550
- }, [onRecentSignatureLoad, loadSignatureToCanvas]);
551
- useImperativeHandle(
552
- ref,
553
- () => ({
554
- submit: handleSubmit,
555
- clear: clearSignature,
556
- getSignatureData: getSignatureDataObject,
557
- isValid: validateForm
558
- }),
559
- [handleSubmit, clearSignature, getSignatureDataObject, validateForm]
560
- );
561
- return /* @__PURE__ */ jsxs("div", { children: [
562
- /* @__PURE__ */ jsxs("div", { className: cx$2("form-label"), children: [
563
- /* @__PURE__ */ jsx("label", { className: "typo-subtitle3 text-body_1", children: label }),
564
- /* @__PURE__ */ jsxs("div", { className: cx$2("form-label-button-box"), children: [
565
- isShowResetButton && /* @__PURE__ */ jsx(
566
- Button,
567
- {
568
- appearance: "filled",
569
- variant: "neutral",
570
- size: "xsmall",
571
- onClick: handleReset,
572
- className: "text-nowrap-i",
573
- disabled: isEmpty,
574
- children: "\uB2E4\uC2DC \uC791\uC131"
575
- }
576
- ),
577
- isShowRecentSignatureButton && /* @__PURE__ */ jsx(
578
- Button,
579
- {
580
- appearance: "filled",
581
- variant: "neutral",
582
- size: "xsmall",
583
- className: "text-nowrap-i",
584
- onClick: handleRecentSignatureLoad,
585
- disabled: isLoadingSignature,
586
- children: isLoadingSignature ? "\uBD88\uB7EC\uC624\uB294 \uC911..." : "\uCD5C\uADFC \uC131\uBA85 \uBD88\uB7EC\uC624\uAE30"
587
- }
588
- )
589
- ] })
590
- ] }),
591
- /* @__PURE__ */ jsx("div", { className: cx$2("form-signature", error && "is-error"), ref: containerRef, children: /* @__PURE__ */ jsx(
592
- "canvas",
593
- {
594
- ref: canvasRef,
595
- width: canvasWidth,
596
- height: canvasHeight,
597
- className: "border border-gray-200 rounded cursor-crosshair bg-white",
598
- style: { width: "100%", height: `${canvasHeight}px` },
599
- onMouseDown: startDrawing,
600
- onMouseMove: draw,
601
- onMouseUp: stopDrawing,
602
- onMouseEnter: handleMouseEnter,
603
- onMouseLeave: handleMouseLeave
604
- }
605
- ) }),
606
- error && /* @__PURE__ */ jsx("div", { className: cx$2("error-msg-box"), children: /* @__PURE__ */ jsx("span", { className: cx$2("error-msg"), children: errorMsg || "\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694." }) })
607
- ] });
608
- };
609
-
610
- const useFormSignature = ({ onSubmit, onError, onClear, minLength = 100 } = {}) => {
611
- const signatureRef = useRef(null);
612
- const [signatureData, setSignatureData] = useState(null);
613
- const [isError, setIsError] = useState(false);
614
- const [errorMessage, setErrorMessage] = useState("");
615
- const [isLoadedFromRecent, setIsLoadedFromRecent] = useState(false);
616
- const handleSignatureSubmit = useCallback(
617
- (data) => {
618
- setSignatureData(data);
619
- setIsError(false);
620
- setErrorMessage("");
621
- onSubmit?.(data);
622
- },
623
- [onSubmit]
624
- );
625
- const submit = useCallback(() => {
626
- try {
627
- if (!signatureRef.current?.isValid()) {
628
- setIsError(true);
629
- setErrorMessage("\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694.");
630
- return false;
631
- }
632
- const data = signatureRef.current.getSignatureData();
633
- if (!data) {
634
- setIsError(true);
635
- setErrorMessage("\uC11C\uBA85 \uB370\uC774\uD130\uB97C \uAC00\uC838\uC62C \uC218 \uC5C6\uC2B5\uB2C8\uB2E4.");
636
- return false;
637
- }
638
- if (!isLoadedFromRecent && data.totalLineLength < minLength) {
639
- setIsError(true);
640
- setErrorMessage(`\uC11C\uBA85\uC744 \uB2E4\uC2DC \uD655\uC778\uD574 \uC8FC\uC138\uC694.`);
641
- return false;
642
- }
643
- setIsError(false);
644
- setErrorMessage("");
645
- signatureRef.current.submit();
646
- return true;
647
- } catch (error) {
648
- const errorObj = error instanceof Error ? error : new Error("\uC54C \uC218 \uC5C6\uB294 \uC624\uB958\uAC00 \uBC1C\uC0DD\uD588\uC2B5\uB2C8\uB2E4.");
649
- setIsError(true);
650
- setErrorMessage(errorObj.message);
651
- onError?.(errorObj);
652
- return false;
653
- }
654
- }, [onError, minLength, isLoadedFromRecent]);
655
- const clear = useCallback(() => {
656
- signatureRef.current?.clear();
657
- setSignatureData(null);
658
- setIsError(false);
659
- setErrorMessage("");
660
- setIsLoadedFromRecent(false);
661
- onClear?.();
662
- }, [onClear]);
663
- const loadRecentSignature = useCallback(async () => {
664
- setIsLoadedFromRecent(true);
665
- setIsError(false);
666
- setErrorMessage("");
667
- return "";
668
- }, []);
669
- const isValid = useCallback(() => {
670
- const basicValid = signatureRef.current?.isValid() ?? false;
671
- if (!basicValid) {
672
- return false;
673
- }
674
- const data = signatureRef.current?.getSignatureData();
675
- if (!data) {
676
- return false;
677
- }
678
- if (isLoadedFromRecent) {
679
- return true;
680
- }
681
- return data.totalLineLength >= minLength;
682
- }, [minLength, isLoadedFromRecent]);
683
- const getSignatureData = useCallback(() => {
684
- return signatureRef.current?.getSignatureData() ?? null;
685
- }, []);
686
- const handleSignatureChange = useCallback(() => {
687
- setIsError(false);
688
- setErrorMessage("");
689
- }, []);
690
- const handleReset = useCallback(() => {
691
- setSignatureData(null);
692
- setIsError(false);
693
- setErrorMessage("");
694
- setIsLoadedFromRecent(false);
695
- onClear?.();
696
- }, [onClear]);
697
- const getErrorMessage = useCallback(() => {
698
- return errorMessage;
699
- }, [errorMessage]);
700
- return {
701
- // ref
702
- signatureRef,
703
- // state
704
- signatureData,
705
- isError,
706
- isLoadedFromRecent,
707
- // 최근 서명 불러옴 여부 노출
708
- // actions
709
- submit,
710
- clear,
711
- loadRecentSignature,
712
- handleSignatureChange,
713
- // 일반적인 서명 변경
714
- handleReset,
715
- // 다시 작성
716
- handleSignatureSubmit,
717
- // utilities
718
- isValid,
719
- getSignatureData,
720
- getErrorMessage
721
- };
722
- };
723
-
724
- const cx$1 = classNames.bind(styles$2);
184
+ const cx$1 = classNames.bind(styles);
725
185
  const StepIndicator = ({
726
186
  items,
727
187
  onClickItem,
@@ -753,7 +213,7 @@ const StepIndicator = ({
753
213
  ] });
754
214
  };
755
215
 
756
- const cx = classNames.bind(styles$3);
216
+ const cx = classNames.bind(styles$1);
757
217
  function Attachment({
758
218
  photos,
759
219
  onAddPhoto,
@@ -993,19 +453,18 @@ const DEFAULT_DOWNLOAD_PROPS = {
993
453
  width: 500,
994
454
  height: 500
995
455
  };
996
- function usePaint(paintProps = {}) {
456
+ function useCanvasPaint(paintProps = {}) {
997
457
  const { pen = { strokeWeight: 5, strokeColor: "black" }, onChange, onStart, onEnd } = paintProps;
998
- const canvasRef = useRef(null);
458
+ const [canvasRefState, setCanvasRefState] = useState(null);
999
459
  const contextRef = useRef(null);
1000
460
  const [isPainting, setIsPainting] = useState(false);
1001
461
  const [history, setHistory] = useState([]);
1002
462
  const [historyIndex, setHistoryIndex] = useState(-1);
1003
463
  const saveState = useCallback(() => {
1004
- const canvas = canvasRef.current;
1005
- if (!canvas) {
464
+ if (!canvasRefState) {
1006
465
  return;
1007
466
  }
1008
- const dataUrl = canvas.toDataURL();
467
+ const dataUrl = canvasRefState.toDataURL();
1009
468
  const newHistory = history.slice(0, historyIndex + 1);
1010
469
  if (newHistory.length >= HISTORY_SIZE) {
1011
470
  newHistory.shift();
@@ -1014,39 +473,37 @@ function usePaint(paintProps = {}) {
1014
473
  setHistory(newHistory);
1015
474
  setHistoryIndex(newHistory.length - 1);
1016
475
  onChange && onChange();
1017
- }, [history, historyIndex, onChange]);
476
+ }, [history, historyIndex, onChange, canvasRefState]);
1018
477
  const restoreState = useCallback(
1019
478
  (index) => {
1020
- const canvas = canvasRef.current;
1021
479
  const context = contextRef.current;
1022
- if (!canvas || !context || !history[index]) {
480
+ if (!canvasRefState || !context || !history[index]) {
1023
481
  return;
1024
482
  }
1025
483
  const dataUrl = history[index];
1026
484
  const img = new Image();
1027
485
  img.onload = () => {
1028
- context.clearRect(0, 0, canvas.width, canvas.height);
486
+ context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
1029
487
  context.drawImage(img, 0, 0);
1030
488
  };
1031
489
  img.src = dataUrl;
1032
490
  onChange && onChange();
1033
491
  },
1034
- [history, onChange]
492
+ [history, onChange, canvasRefState]
1035
493
  );
1036
494
  useEffect(() => {
1037
- const canvas = canvasRef.current;
1038
- if (!canvas) {
495
+ if (!canvasRefState) {
1039
496
  return;
1040
497
  }
1041
- const context = canvas.getContext("2d", { willReadFrequently: true });
498
+ const context = canvasRefState.getContext("2d", { willReadFrequently: true });
1042
499
  if (!context) {
1043
500
  return;
1044
501
  }
1045
502
  contextRef.current = context;
1046
- const initialDataUrl = canvas.toDataURL();
503
+ const initialDataUrl = canvasRefState.toDataURL();
1047
504
  setHistory([initialDataUrl]);
1048
505
  setHistoryIndex(0);
1049
- }, []);
506
+ }, [canvasRefState]);
1050
507
  useEffect(() => {
1051
508
  const context = contextRef.current;
1052
509
  if (context) {
@@ -1054,19 +511,53 @@ function usePaint(paintProps = {}) {
1054
511
  context.strokeStyle = pen.strokeColor;
1055
512
  }
1056
513
  }, [pen]);
514
+ const getEventPosition = useCallback(
515
+ (event) => {
516
+ if (!canvasRefState) {
517
+ return null;
518
+ }
519
+ const rect = canvasRefState.getBoundingClientRect();
520
+ const scaleX = canvasRefState.width / rect.width;
521
+ const scaleY = canvasRefState.height / rect.height;
522
+ let clientX;
523
+ let clientY;
524
+ if (event instanceof MouseEvent) {
525
+ clientX = event.clientX;
526
+ clientY = event.clientY;
527
+ } else if (event instanceof TouchEvent) {
528
+ const touch = event.touches[0];
529
+ if (!touch) {
530
+ return null;
531
+ }
532
+ clientX = touch.clientX;
533
+ clientY = touch.clientY;
534
+ } else {
535
+ return null;
536
+ }
537
+ const offsetX = (clientX - rect.left) * scaleX;
538
+ const offsetY = (clientY - rect.top) * scaleY;
539
+ return { offsetX, offsetY };
540
+ },
541
+ [canvasRefState]
542
+ );
1057
543
  const startPainting = useCallback(
1058
544
  (event) => {
545
+ event.preventDefault();
1059
546
  const context = contextRef.current;
1060
547
  if (!context) {
1061
548
  return;
1062
549
  }
550
+ const pos = getEventPosition(event);
551
+ if (!pos) {
552
+ return;
553
+ }
1063
554
  onStart && onStart();
1064
- const { offsetX, offsetY } = event;
555
+ const { offsetX, offsetY } = pos;
1065
556
  context.beginPath();
1066
557
  context.moveTo(offsetX, offsetY);
1067
558
  setIsPainting(true);
1068
559
  },
1069
- [onStart]
560
+ [onStart, getEventPosition]
1070
561
  );
1071
562
  const stopPainting = useCallback(() => {
1072
563
  const context = contextRef.current;
@@ -1080,47 +571,48 @@ function usePaint(paintProps = {}) {
1080
571
  }, [isPainting, onEnd, saveState]);
1081
572
  const paint = useCallback(
1082
573
  (event) => {
574
+ event.preventDefault();
1083
575
  if (!isPainting || !contextRef.current) {
1084
576
  return;
1085
577
  }
1086
- const { offsetX, offsetY } = event;
578
+ const pos = getEventPosition(event);
579
+ if (!pos) {
580
+ return;
581
+ }
582
+ const { offsetX, offsetY } = pos;
1087
583
  contextRef.current.lineTo(offsetX, offsetY);
1088
584
  contextRef.current.stroke();
1089
585
  },
1090
- [isPainting]
1091
- );
1092
- const handleMouseEnter = useCallback(
1093
- (event) => {
1094
- if (event.buttons === 1 && !isPainting) {
1095
- startPainting(event);
1096
- }
1097
- },
1098
- [isPainting, startPainting]
586
+ [isPainting, getEventPosition]
1099
587
  );
1100
588
  useEffect(() => {
1101
- const canvas = canvasRef.current;
1102
- if (!canvas) {
589
+ if (!canvasRefState) {
1103
590
  return;
1104
591
  }
1105
- canvas.addEventListener("mousedown", startPainting);
1106
- canvas.addEventListener("mouseup", stopPainting);
1107
- canvas.addEventListener("mousemove", paint);
1108
- canvas.addEventListener("mouseleave", stopPainting);
1109
- canvas.addEventListener("mouseenter", handleMouseEnter);
592
+ canvasRefState.addEventListener("mousedown", startPainting);
593
+ canvasRefState.addEventListener("mouseup", stopPainting);
594
+ canvasRefState.addEventListener("mousemove", paint);
595
+ canvasRefState.addEventListener("mouseleave", stopPainting);
596
+ canvasRefState.addEventListener("touchstart", startPainting);
597
+ canvasRefState.addEventListener("touchend", stopPainting);
598
+ canvasRefState.addEventListener("touchmove", paint);
599
+ canvasRefState.addEventListener("touchcancel", stopPainting);
1110
600
  return () => {
1111
- canvas.removeEventListener("mousedown", startPainting);
1112
- canvas.removeEventListener("mouseup", stopPainting);
1113
- canvas.removeEventListener("mousemove", paint);
1114
- canvas.removeEventListener("mouseleave", stopPainting);
1115
- canvas.removeEventListener("mouseenter", handleMouseEnter);
601
+ canvasRefState.removeEventListener("mousedown", startPainting);
602
+ canvasRefState.removeEventListener("mouseup", stopPainting);
603
+ canvasRefState.removeEventListener("mousemove", paint);
604
+ canvasRefState.removeEventListener("mouseleave", stopPainting);
605
+ canvasRefState.removeEventListener("touchstart", startPainting);
606
+ canvasRefState.removeEventListener("touchend", stopPainting);
607
+ canvasRefState.removeEventListener("touchmove", paint);
608
+ canvasRefState.removeEventListener("touchcancel", stopPainting);
1116
609
  };
1117
- }, [startPainting, stopPainting, paint, handleMouseEnter]);
610
+ }, [startPainting, stopPainting, paint, canvasRefState]);
1118
611
  const clear = () => {
1119
- const canvas = canvasRef.current;
1120
612
  const context = contextRef.current;
1121
- if (canvas && context) {
1122
- context.clearRect(0, 0, canvas.width, canvas.height);
1123
- const initialDataUrl = canvas.toDataURL();
613
+ if (canvasRefState && context) {
614
+ context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
615
+ const initialDataUrl = canvasRefState.toDataURL();
1124
616
  setHistory([initialDataUrl]);
1125
617
  setHistoryIndex(0);
1126
618
  onChange && onChange();
@@ -1141,15 +633,14 @@ function usePaint(paintProps = {}) {
1141
633
  }
1142
634
  };
1143
635
  const loadImage = (base64String) => {
1144
- const canvas = canvasRef.current;
1145
636
  const context = contextRef.current;
1146
- if (!canvas || !context) {
637
+ if (!canvasRefState || !context) {
1147
638
  return;
1148
639
  }
1149
640
  const img = new Image();
1150
641
  img.onload = () => {
1151
- context.clearRect(0, 0, canvas.width, canvas.height);
1152
- context.drawImage(img, 0, 0, canvas.width, canvas.height);
642
+ context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
643
+ context.drawImage(img, 0, 0, canvasRefState.width, canvasRefState.height);
1153
644
  saveState();
1154
645
  };
1155
646
  img.src = base64String;
@@ -1157,7 +648,7 @@ function usePaint(paintProps = {}) {
1157
648
  const getBase64String = (imageProps) => {
1158
649
  const props = { ...DEFAULT_DOWNLOAD_PROPS, ...imageProps };
1159
650
  const { fileExtendsion, transparent, backgroundColor, width, height } = props;
1160
- const originalCanvas = canvasRef.current;
651
+ const originalCanvas = canvasRefState;
1161
652
  if (!originalCanvas) {
1162
653
  return;
1163
654
  }
@@ -1177,7 +668,7 @@ function usePaint(paintProps = {}) {
1177
668
  const download = (downloadProps) => {
1178
669
  const props = { ...DEFAULT_DOWNLOAD_PROPS, ...downloadProps };
1179
670
  const { fileName, fileExtendsion, transparent, backgroundColor, width, height } = props;
1180
- const originalCanvas = canvasRef.current;
671
+ const originalCanvas = canvasRefState;
1181
672
  if (!originalCanvas) {
1182
673
  return;
1183
674
  }
@@ -1197,16 +688,39 @@ function usePaint(paintProps = {}) {
1197
688
  link.click();
1198
689
  }
1199
690
  };
691
+ const isCanvasBlank = () => {
692
+ if (!canvasRefState) {
693
+ return true;
694
+ }
695
+ const { width, height } = canvasRefState;
696
+ const ctx = canvasRefState.getContext("2d");
697
+ if (!ctx) {
698
+ console.error("2D context not available");
699
+ return true;
700
+ }
701
+ const imageData = ctx.getImageData(0, 0, width, height);
702
+ const { data } = imageData;
703
+ for (let i = 0; i < data.length; i += 4) {
704
+ if (data[i + 3] !== 0) {
705
+ return false;
706
+ }
707
+ }
708
+ return true;
709
+ };
1200
710
  return {
1201
711
  clear,
1202
712
  undo,
1203
713
  redo,
1204
714
  loadImage,
1205
715
  download,
1206
- canvasRef,
1207
- getBase64String
716
+ getBase64String,
717
+ isCanvasBlank,
718
+ setCanvasRefState,
719
+ canvasRefState
1208
720
  };
1209
721
  }
1210
722
 
1211
- export { Attachment, DataTable, FormDatePicker, FormSegmentGroup, FormSignature, FormTextField, StepIndicator, resize, useCamera, useFormSignature, usePaint, useTable };
723
+ const testSignatureBase64Data = "";
724
+
725
+ export { Attachment, FormDatePicker, FormSegmentGroup, FormTextField, StepIndicator, resize, testSignatureBase64Data, useCamera, useCanvasPaint };
1212
726
  //# sourceMappingURL=index.esm.js.map