sales-frontend-components 0.0.34 → 0.0.36
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/form/form-signautre/form-signature.module.scss +39 -0
- package/dist/form/form-signautre/form.signature.module.scss +39 -0
- package/dist/index.cjs.js +400 -6
- package/dist/index.cjs.js.map +1 -1
- package/dist/index.d.ts +56 -2
- package/dist/index.esm.js +401 -9
- package/dist/index.esm.js.map +1 -1
- package/package.json +10 -10
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
@use 'sales-frontend-design-system/design-system-context' as *;
|
|
2
|
+
|
|
3
|
+
.form-label {
|
|
4
|
+
display: flex;
|
|
5
|
+
gap: padding(10);
|
|
6
|
+
align-items: center;
|
|
7
|
+
justify-content: space-between;
|
|
8
|
+
padding-bottom: padding(xxxsmall);
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
.form-label-button-box {
|
|
12
|
+
display: flex;
|
|
13
|
+
gap: padding(xxsmall);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
.form-signature {
|
|
17
|
+
background: colors(surface-neutral_1);
|
|
18
|
+
border: 1px solid colors(border-neutral_3);
|
|
19
|
+
border-radius: radius-basic(medium);
|
|
20
|
+
|
|
21
|
+
&.is-error {
|
|
22
|
+
border-color: colors(additional-semantic_2);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
.error-msg {
|
|
27
|
+
padding: padding(xxxsmall) 0;
|
|
28
|
+
color: colors(additional-semantic_2);
|
|
29
|
+
|
|
30
|
+
@include apply-typography(body2);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
.error-msg-box {
|
|
34
|
+
display: flex;
|
|
35
|
+
flex-direction: column;
|
|
36
|
+
gap: padding(xxxsmall);
|
|
37
|
+
justify-content: space-between;
|
|
38
|
+
padding-top: padding(xsmall);
|
|
39
|
+
}
|
|
@@ -0,0 +1,39 @@
|
|
|
1
|
+
@use 'sales-frontend-design-system/design-system-context' as *;
|
|
2
|
+
|
|
3
|
+
.form-label {
|
|
4
|
+
display: flex;
|
|
5
|
+
gap: padding(10);
|
|
6
|
+
align-items: center;
|
|
7
|
+
justify-content: space-between;
|
|
8
|
+
padding-bottom: padding(xxxsmall);
|
|
9
|
+
}
|
|
10
|
+
|
|
11
|
+
.form-label-button-box {
|
|
12
|
+
display: flex;
|
|
13
|
+
gap: padding(xxsmall);
|
|
14
|
+
}
|
|
15
|
+
|
|
16
|
+
.form-signature {
|
|
17
|
+
background: colors(surface-neutral_1);
|
|
18
|
+
border: 1px solid colors(border-neutral_3);
|
|
19
|
+
border-radius: radius-basic(medium);
|
|
20
|
+
|
|
21
|
+
&.is-error {
|
|
22
|
+
border-color: colors(additional-semantic_2);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
.error-msg {
|
|
27
|
+
padding: padding(xxxsmall) 0;
|
|
28
|
+
color: colors(additional-semantic_2);
|
|
29
|
+
|
|
30
|
+
@include apply-typography(body2);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
.error-msg-box {
|
|
34
|
+
display: flex;
|
|
35
|
+
flex-direction: column;
|
|
36
|
+
gap: padding(xxxsmall);
|
|
37
|
+
justify-content: space-between;
|
|
38
|
+
padding-top: padding(xsmall);
|
|
39
|
+
}
|
package/dist/index.cjs.js
CHANGED
|
@@ -6,7 +6,8 @@ var styles = require('./data-table/data-table.module.scss');
|
|
|
6
6
|
var salesFrontendDesignSystem = require('sales-frontend-design-system');
|
|
7
7
|
var React = require('react');
|
|
8
8
|
var reactHookForm = require('react-hook-form');
|
|
9
|
-
var styles$1 = require('./
|
|
9
|
+
var styles$1 = require('./form/form-signautre/form-signature.module.scss');
|
|
10
|
+
var styles$2 = require('./step-indicator/step-indicator.module.scss');
|
|
10
11
|
|
|
11
12
|
function getDefaultExportFromCjs (x) {
|
|
12
13
|
return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
|
|
@@ -99,7 +100,7 @@ function requireBind () {
|
|
|
99
100
|
var bindExports = requireBind();
|
|
100
101
|
var classNames = /*@__PURE__*/getDefaultExportFromCjs(bindExports);
|
|
101
102
|
|
|
102
|
-
const cx$
|
|
103
|
+
const cx$2 = classNames.bind(styles);
|
|
103
104
|
const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
104
105
|
return /* @__PURE__ */ jsxRuntime.jsxs(
|
|
105
106
|
salesFrontendDesignSystem.Table,
|
|
@@ -114,7 +115,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
|
114
115
|
"th",
|
|
115
116
|
{
|
|
116
117
|
colSpan: header.colSpan,
|
|
117
|
-
className: cx$
|
|
118
|
+
className: cx$2({
|
|
118
119
|
"is-resizing": header.column.getIsResizing()
|
|
119
120
|
// 3. 리사이징 중일 때 클래스 추가
|
|
120
121
|
}),
|
|
@@ -126,7 +127,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
|
126
127
|
{
|
|
127
128
|
onMouseDown: header.getResizeHandler(),
|
|
128
129
|
onTouchStart: header.getResizeHandler(),
|
|
129
|
-
className: cx$
|
|
130
|
+
className: cx$2("resize-handle")
|
|
130
131
|
}
|
|
131
132
|
)
|
|
132
133
|
]
|
|
@@ -147,7 +148,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
|
147
148
|
},
|
|
148
149
|
cell.id
|
|
149
150
|
)) }, row.id)
|
|
150
|
-
)) : /* @__PURE__ */ jsxRuntime.jsx("tr", { children: /* @__PURE__ */ jsxRuntime.jsx("td", { className: cx$
|
|
151
|
+
)) : /* @__PURE__ */ jsxRuntime.jsx("tr", { children: /* @__PURE__ */ jsxRuntime.jsx("td", { className: cx$2("feedback-cell"), colSpan: table.getVisibleLeafColumns().length, children: isLoading ? /* @__PURE__ */ jsxRuntime.jsx("div", { children: "Loading..." }) : /* @__PURE__ */ jsxRuntime.jsx("div", { children: msgText || (isError ? "Error" : "No data") }) }) }) })
|
|
151
152
|
]
|
|
152
153
|
}
|
|
153
154
|
);
|
|
@@ -296,7 +297,398 @@ const FormTextField = ({
|
|
|
296
297
|
);
|
|
297
298
|
};
|
|
298
299
|
|
|
299
|
-
const cx = classNames.bind(styles$1);
|
|
300
|
+
const cx$1 = classNames.bind(styles$1);
|
|
301
|
+
const FormSignature = ({
|
|
302
|
+
label,
|
|
303
|
+
error,
|
|
304
|
+
errorMsg,
|
|
305
|
+
onSubmit,
|
|
306
|
+
onRecentSignatureLoad,
|
|
307
|
+
onChangeSignature,
|
|
308
|
+
onReset,
|
|
309
|
+
isShowRecentSignatureButton,
|
|
310
|
+
isShowResetButton,
|
|
311
|
+
ref,
|
|
312
|
+
canvasWidth = 632,
|
|
313
|
+
canvasHeight = 230
|
|
314
|
+
}) => {
|
|
315
|
+
const canvasRef = React.useRef(null);
|
|
316
|
+
const containerRef = React.useRef(null);
|
|
317
|
+
const [isDrawing, setIsDrawing] = React.useState(false);
|
|
318
|
+
const [isEmpty, setIsEmpty] = React.useState(true);
|
|
319
|
+
const [signatureData, setSignatureData] = React.useState("");
|
|
320
|
+
const [isLoadingSignature, setIsLoadingSignature] = React.useState(false);
|
|
321
|
+
const [isMousePressed, setIsMousePressed] = React.useState(false);
|
|
322
|
+
const [totalLineLength, setTotalLineLength] = React.useState(0);
|
|
323
|
+
const [lastPoint, setLastPoint] = React.useState(null);
|
|
324
|
+
const [strokeLengths, setStrokeLengths] = React.useState([]);
|
|
325
|
+
const [currentStrokeLength, setCurrentStrokeLength] = React.useState(0);
|
|
326
|
+
React.useEffect(() => {
|
|
327
|
+
const canvas = canvasRef.current;
|
|
328
|
+
if (!canvas) return;
|
|
329
|
+
const ctx = canvas.getContext("2d");
|
|
330
|
+
if (!ctx) return;
|
|
331
|
+
ctx.strokeStyle = "#000000";
|
|
332
|
+
ctx.lineWidth = Math.max(1, canvasWidth / 400);
|
|
333
|
+
ctx.lineCap = "round";
|
|
334
|
+
ctx.lineJoin = "round";
|
|
335
|
+
}, [canvasWidth]);
|
|
336
|
+
React.useEffect(() => {
|
|
337
|
+
const handleGlobalMouseUp = () => {
|
|
338
|
+
setIsMousePressed(false);
|
|
339
|
+
if (isDrawing) {
|
|
340
|
+
stopDrawing();
|
|
341
|
+
}
|
|
342
|
+
};
|
|
343
|
+
document.addEventListener("mouseup", handleGlobalMouseUp);
|
|
344
|
+
return () => {
|
|
345
|
+
document.removeEventListener("mouseup", handleGlobalMouseUp);
|
|
346
|
+
};
|
|
347
|
+
}, [isDrawing]);
|
|
348
|
+
const getCanvasCoordinates = React.useCallback((e) => {
|
|
349
|
+
const canvas = canvasRef.current;
|
|
350
|
+
if (!canvas) return { x: 0, y: 0 };
|
|
351
|
+
const rect = canvas.getBoundingClientRect();
|
|
352
|
+
const scaleX = canvas.width / rect.width;
|
|
353
|
+
const scaleY = canvas.height / rect.height;
|
|
354
|
+
return {
|
|
355
|
+
x: (e.clientX - rect.left) * scaleX,
|
|
356
|
+
y: (e.clientY - rect.top) * scaleY
|
|
357
|
+
};
|
|
358
|
+
}, []);
|
|
359
|
+
const calculateDistance = React.useCallback((point1, point2) => {
|
|
360
|
+
return Math.sqrt(Math.pow(point2.x - point1.x, 2) + Math.pow(point2.y - point1.y, 2));
|
|
361
|
+
}, []);
|
|
362
|
+
const startDrawing = React.useCallback(
|
|
363
|
+
(e) => {
|
|
364
|
+
const canvas = canvasRef.current;
|
|
365
|
+
if (!canvas) return;
|
|
366
|
+
const ctx = canvas.getContext("2d");
|
|
367
|
+
if (!ctx) return;
|
|
368
|
+
const { x, y } = getCanvasCoordinates(e);
|
|
369
|
+
setIsDrawing(true);
|
|
370
|
+
setIsMousePressed(true);
|
|
371
|
+
setIsEmpty(false);
|
|
372
|
+
setLastPoint({ x, y });
|
|
373
|
+
setCurrentStrokeLength(0);
|
|
374
|
+
ctx.beginPath();
|
|
375
|
+
ctx.moveTo(x, y);
|
|
376
|
+
},
|
|
377
|
+
[getCanvasCoordinates]
|
|
378
|
+
);
|
|
379
|
+
const draw = React.useCallback(
|
|
380
|
+
(e) => {
|
|
381
|
+
if (!isDrawing) return;
|
|
382
|
+
const canvas = canvasRef.current;
|
|
383
|
+
if (!canvas) return;
|
|
384
|
+
const ctx = canvas.getContext("2d");
|
|
385
|
+
if (!ctx) return;
|
|
386
|
+
const { x, y } = getCanvasCoordinates(e);
|
|
387
|
+
const currentPoint = { x, y };
|
|
388
|
+
if (lastPoint) {
|
|
389
|
+
const distance = calculateDistance(lastPoint, currentPoint);
|
|
390
|
+
if (distance > 0.5) {
|
|
391
|
+
setCurrentStrokeLength((prev) => prev + distance);
|
|
392
|
+
setTotalLineLength((prev) => prev + distance);
|
|
393
|
+
}
|
|
394
|
+
}
|
|
395
|
+
ctx.lineTo(x, y);
|
|
396
|
+
ctx.stroke();
|
|
397
|
+
setLastPoint(currentPoint);
|
|
398
|
+
const data = canvas.toDataURL();
|
|
399
|
+
setSignatureData(data);
|
|
400
|
+
onChangeSignature?.();
|
|
401
|
+
},
|
|
402
|
+
[isDrawing, getCanvasCoordinates, calculateDistance, lastPoint, onChangeSignature]
|
|
403
|
+
);
|
|
404
|
+
const stopDrawing = React.useCallback(() => {
|
|
405
|
+
if (isDrawing && currentStrokeLength > 0) {
|
|
406
|
+
setStrokeLengths((prev) => [...prev, currentStrokeLength]);
|
|
407
|
+
}
|
|
408
|
+
setIsDrawing(false);
|
|
409
|
+
setLastPoint(null);
|
|
410
|
+
}, [isDrawing, currentStrokeLength]);
|
|
411
|
+
const handleMouseEnter = React.useCallback(
|
|
412
|
+
(e) => {
|
|
413
|
+
if (isMousePressed && !isDrawing) {
|
|
414
|
+
const canvas = canvasRef.current;
|
|
415
|
+
if (!canvas) return;
|
|
416
|
+
const ctx = canvas.getContext("2d");
|
|
417
|
+
if (!ctx) return;
|
|
418
|
+
const { x, y } = getCanvasCoordinates(e);
|
|
419
|
+
setIsDrawing(true);
|
|
420
|
+
setLastPoint({ x, y });
|
|
421
|
+
ctx.beginPath();
|
|
422
|
+
ctx.moveTo(x, y);
|
|
423
|
+
}
|
|
424
|
+
},
|
|
425
|
+
[isMousePressed, isDrawing, getCanvasCoordinates]
|
|
426
|
+
);
|
|
427
|
+
const handleMouseLeave = React.useCallback(() => {
|
|
428
|
+
if (isDrawing) {
|
|
429
|
+
setIsDrawing(false);
|
|
430
|
+
setLastPoint(null);
|
|
431
|
+
}
|
|
432
|
+
}, [isDrawing]);
|
|
433
|
+
const clearSignature = React.useCallback(() => {
|
|
434
|
+
const canvas = canvasRef.current;
|
|
435
|
+
if (!canvas) return;
|
|
436
|
+
const ctx = canvas.getContext("2d");
|
|
437
|
+
if (!ctx) return;
|
|
438
|
+
ctx.clearRect(0, 0, canvasWidth, canvasHeight);
|
|
439
|
+
setIsEmpty(true);
|
|
440
|
+
setSignatureData("");
|
|
441
|
+
setTotalLineLength(0);
|
|
442
|
+
setLastPoint(null);
|
|
443
|
+
setStrokeLengths([]);
|
|
444
|
+
setCurrentStrokeLength(0);
|
|
445
|
+
}, [canvasWidth, canvasHeight]);
|
|
446
|
+
const loadSignatureToCanvas = React.useCallback(
|
|
447
|
+
(imageDataURL) => {
|
|
448
|
+
return new Promise((resolve, reject) => {
|
|
449
|
+
const canvas = canvasRef.current;
|
|
450
|
+
if (!canvas) {
|
|
451
|
+
reject(new Error("Canvas not found"));
|
|
452
|
+
return;
|
|
453
|
+
}
|
|
454
|
+
const ctx = canvas.getContext("2d");
|
|
455
|
+
if (!ctx) {
|
|
456
|
+
reject(new Error("Canvas context not found"));
|
|
457
|
+
return;
|
|
458
|
+
}
|
|
459
|
+
const img = new Image();
|
|
460
|
+
img.onload = () => {
|
|
461
|
+
ctx.clearRect(0, 0, canvasWidth, canvasHeight);
|
|
462
|
+
ctx.drawImage(img, 0, 0, canvasWidth, canvasHeight);
|
|
463
|
+
setIsEmpty(false);
|
|
464
|
+
setSignatureData(imageDataURL);
|
|
465
|
+
setTotalLineLength(0);
|
|
466
|
+
setStrokeLengths([]);
|
|
467
|
+
setCurrentStrokeLength(0);
|
|
468
|
+
setLastPoint(null);
|
|
469
|
+
resolve();
|
|
470
|
+
};
|
|
471
|
+
img.onerror = () => {
|
|
472
|
+
reject(new Error("Failed to load signature image"));
|
|
473
|
+
};
|
|
474
|
+
img.src = imageDataURL;
|
|
475
|
+
});
|
|
476
|
+
},
|
|
477
|
+
[canvasWidth, canvasHeight]
|
|
478
|
+
);
|
|
479
|
+
const validateForm = React.useCallback(() => {
|
|
480
|
+
return !(!signatureData || isEmpty);
|
|
481
|
+
}, [signatureData, isEmpty]);
|
|
482
|
+
const getSignatureDataObject = React.useCallback(() => {
|
|
483
|
+
if (!signatureData || isEmpty) {
|
|
484
|
+
return null;
|
|
485
|
+
}
|
|
486
|
+
return {
|
|
487
|
+
signature: signatureData,
|
|
488
|
+
totalLineLength: Math.round(totalLineLength),
|
|
489
|
+
strokeCount: strokeLengths.length,
|
|
490
|
+
strokeLengths: strokeLengths.map((length) => Math.round(length)),
|
|
491
|
+
averageStrokeLength: strokeLengths.length > 0 ? Math.round(totalLineLength / strokeLengths.length) : 0
|
|
492
|
+
};
|
|
493
|
+
}, [signatureData, isEmpty, totalLineLength, strokeLengths]);
|
|
494
|
+
const handleSubmit = React.useCallback(() => {
|
|
495
|
+
if (!validateForm()) return;
|
|
496
|
+
const data = getSignatureDataObject();
|
|
497
|
+
if (data) {
|
|
498
|
+
console.log("\uC81C\uCD9C\uB41C \uB370\uC774\uD130:", data);
|
|
499
|
+
onSubmit(data);
|
|
500
|
+
}
|
|
501
|
+
}, [validateForm, getSignatureDataObject, onSubmit]);
|
|
502
|
+
const handleReset = React.useCallback(() => {
|
|
503
|
+
clearSignature();
|
|
504
|
+
onReset?.();
|
|
505
|
+
}, [clearSignature, onReset]);
|
|
506
|
+
const handleRecentSignatureLoad = React.useCallback(async () => {
|
|
507
|
+
if (!onRecentSignatureLoad) return;
|
|
508
|
+
try {
|
|
509
|
+
setIsLoadingSignature(true);
|
|
510
|
+
const loadedSignature = await onRecentSignatureLoad();
|
|
511
|
+
if (loadedSignature) {
|
|
512
|
+
await loadSignatureToCanvas(loadedSignature);
|
|
513
|
+
console.log("\uCD5C\uADFC \uC11C\uBA85\uC744 \uC131\uACF5\uC801\uC73C\uB85C \uBD88\uB7EC\uC654\uC2B5\uB2C8\uB2E4.");
|
|
514
|
+
} else {
|
|
515
|
+
alert("\uBD88\uB7EC\uC62C \uCD5C\uADFC \uC11C\uBA85\uC774 \uC5C6\uC2B5\uB2C8\uB2E4.");
|
|
516
|
+
}
|
|
517
|
+
} catch (error2) {
|
|
518
|
+
console.error("\uCD5C\uADFC \uC11C\uBA85 \uBD88\uB7EC\uC624\uAE30 \uC2E4\uD328:", error2);
|
|
519
|
+
alert("\uCD5C\uADFC \uC11C\uBA85\uC744 \uBD88\uB7EC\uC624\uB294\uB370 \uC2E4\uD328\uD588\uC2B5\uB2C8\uB2E4.");
|
|
520
|
+
} finally {
|
|
521
|
+
setIsLoadingSignature(false);
|
|
522
|
+
}
|
|
523
|
+
}, [onRecentSignatureLoad, loadSignatureToCanvas]);
|
|
524
|
+
React.useImperativeHandle(
|
|
525
|
+
ref,
|
|
526
|
+
() => ({
|
|
527
|
+
submit: handleSubmit,
|
|
528
|
+
clear: clearSignature,
|
|
529
|
+
getSignatureData: getSignatureDataObject,
|
|
530
|
+
isValid: validateForm
|
|
531
|
+
}),
|
|
532
|
+
[handleSubmit, clearSignature, getSignatureDataObject, validateForm]
|
|
533
|
+
);
|
|
534
|
+
return /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
|
|
535
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: cx$1("form-label"), children: [
|
|
536
|
+
/* @__PURE__ */ jsxRuntime.jsx("label", { className: "typo-subtitle3 text-body_1", children: label }),
|
|
537
|
+
/* @__PURE__ */ jsxRuntime.jsxs("div", { className: cx$1("form-label-button-box"), children: [
|
|
538
|
+
isShowResetButton && /* @__PURE__ */ jsxRuntime.jsx(
|
|
539
|
+
salesFrontendDesignSystem.Button,
|
|
540
|
+
{
|
|
541
|
+
appearance: "filled",
|
|
542
|
+
variant: "neutral",
|
|
543
|
+
size: "xsmall",
|
|
544
|
+
onClick: handleReset,
|
|
545
|
+
className: "text-nowrap-i",
|
|
546
|
+
disabled: isEmpty,
|
|
547
|
+
children: "\uB2E4\uC2DC \uC791\uC131"
|
|
548
|
+
}
|
|
549
|
+
),
|
|
550
|
+
isShowRecentSignatureButton && /* @__PURE__ */ jsxRuntime.jsx(
|
|
551
|
+
salesFrontendDesignSystem.Button,
|
|
552
|
+
{
|
|
553
|
+
appearance: "filled",
|
|
554
|
+
variant: "neutral",
|
|
555
|
+
size: "xsmall",
|
|
556
|
+
className: "text-nowrap-i",
|
|
557
|
+
onClick: handleRecentSignatureLoad,
|
|
558
|
+
disabled: isLoadingSignature,
|
|
559
|
+
children: isLoadingSignature ? "\uBD88\uB7EC\uC624\uB294 \uC911..." : "\uCD5C\uADFC \uC131\uBA85 \uBD88\uB7EC\uC624\uAE30"
|
|
560
|
+
}
|
|
561
|
+
)
|
|
562
|
+
] })
|
|
563
|
+
] }),
|
|
564
|
+
/* @__PURE__ */ jsxRuntime.jsx("div", { className: cx$1("form-signature", error && "is-error"), ref: containerRef, children: /* @__PURE__ */ jsxRuntime.jsx(
|
|
565
|
+
"canvas",
|
|
566
|
+
{
|
|
567
|
+
ref: canvasRef,
|
|
568
|
+
width: canvasWidth,
|
|
569
|
+
height: canvasHeight,
|
|
570
|
+
className: "border border-gray-200 rounded cursor-crosshair bg-white",
|
|
571
|
+
style: { width: "100%", height: `${canvasHeight}px` },
|
|
572
|
+
onMouseDown: startDrawing,
|
|
573
|
+
onMouseMove: draw,
|
|
574
|
+
onMouseUp: stopDrawing,
|
|
575
|
+
onMouseEnter: handleMouseEnter,
|
|
576
|
+
onMouseLeave: handleMouseLeave
|
|
577
|
+
}
|
|
578
|
+
) }),
|
|
579
|
+
error && /* @__PURE__ */ jsxRuntime.jsx("div", { className: cx$1("error-msg-box"), children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: cx$1("error-msg"), children: errorMsg || "\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694." }) })
|
|
580
|
+
] });
|
|
581
|
+
};
|
|
582
|
+
|
|
583
|
+
const useFormSignature = ({ onSubmit, onError, onClear, minLength = 100 } = {}) => {
|
|
584
|
+
const signatureRef = React.useRef(null);
|
|
585
|
+
const [signatureData, setSignatureData] = React.useState(null);
|
|
586
|
+
const [isError, setIsError] = React.useState(false);
|
|
587
|
+
const [errorMessage, setErrorMessage] = React.useState("");
|
|
588
|
+
const [isLoadedFromRecent, setIsLoadedFromRecent] = React.useState(false);
|
|
589
|
+
const handleSignatureSubmit = React.useCallback(
|
|
590
|
+
(data) => {
|
|
591
|
+
setSignatureData(data);
|
|
592
|
+
setIsError(false);
|
|
593
|
+
setErrorMessage("");
|
|
594
|
+
onSubmit?.(data);
|
|
595
|
+
},
|
|
596
|
+
[onSubmit]
|
|
597
|
+
);
|
|
598
|
+
const submit = React.useCallback(() => {
|
|
599
|
+
try {
|
|
600
|
+
if (!signatureRef.current?.isValid()) {
|
|
601
|
+
setIsError(true);
|
|
602
|
+
setErrorMessage("\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694.");
|
|
603
|
+
return false;
|
|
604
|
+
}
|
|
605
|
+
const data = signatureRef.current.getSignatureData();
|
|
606
|
+
if (!data) {
|
|
607
|
+
setIsError(true);
|
|
608
|
+
setErrorMessage("\uC11C\uBA85 \uB370\uC774\uD130\uB97C \uAC00\uC838\uC62C \uC218 \uC5C6\uC2B5\uB2C8\uB2E4.");
|
|
609
|
+
return false;
|
|
610
|
+
}
|
|
611
|
+
if (!isLoadedFromRecent && data.totalLineLength < minLength) {
|
|
612
|
+
setIsError(true);
|
|
613
|
+
setErrorMessage(`\uC11C\uBA85\uC744 \uB2E4\uC2DC \uD655\uC778\uD574 \uC8FC\uC138\uC694.`);
|
|
614
|
+
return false;
|
|
615
|
+
}
|
|
616
|
+
setIsError(false);
|
|
617
|
+
setErrorMessage("");
|
|
618
|
+
signatureRef.current.submit();
|
|
619
|
+
return true;
|
|
620
|
+
} catch (error) {
|
|
621
|
+
const errorObj = error instanceof Error ? error : new Error("\uC54C \uC218 \uC5C6\uB294 \uC624\uB958\uAC00 \uBC1C\uC0DD\uD588\uC2B5\uB2C8\uB2E4.");
|
|
622
|
+
setIsError(true);
|
|
623
|
+
setErrorMessage(errorObj.message);
|
|
624
|
+
onError?.(errorObj);
|
|
625
|
+
return false;
|
|
626
|
+
}
|
|
627
|
+
}, [onError, minLength, isLoadedFromRecent]);
|
|
628
|
+
const clear = React.useCallback(() => {
|
|
629
|
+
signatureRef.current?.clear();
|
|
630
|
+
setSignatureData(null);
|
|
631
|
+
setIsError(false);
|
|
632
|
+
setErrorMessage("");
|
|
633
|
+
setIsLoadedFromRecent(false);
|
|
634
|
+
onClear?.();
|
|
635
|
+
}, [onClear]);
|
|
636
|
+
const loadRecentSignature = React.useCallback(async () => {
|
|
637
|
+
setIsLoadedFromRecent(true);
|
|
638
|
+
setIsError(false);
|
|
639
|
+
setErrorMessage("");
|
|
640
|
+
return "";
|
|
641
|
+
}, []);
|
|
642
|
+
const isValid = React.useCallback(() => {
|
|
643
|
+
const basicValid = signatureRef.current?.isValid() ?? false;
|
|
644
|
+
if (!basicValid) return false;
|
|
645
|
+
const data = signatureRef.current?.getSignatureData();
|
|
646
|
+
if (!data) return false;
|
|
647
|
+
if (isLoadedFromRecent) return true;
|
|
648
|
+
return data.totalLineLength >= minLength;
|
|
649
|
+
}, [minLength, isLoadedFromRecent]);
|
|
650
|
+
const getSignatureData = React.useCallback(() => {
|
|
651
|
+
return signatureRef.current?.getSignatureData() ?? null;
|
|
652
|
+
}, []);
|
|
653
|
+
const handleSignatureChange = React.useCallback(() => {
|
|
654
|
+
setIsError(false);
|
|
655
|
+
setErrorMessage("");
|
|
656
|
+
}, []);
|
|
657
|
+
const handleReset = React.useCallback(() => {
|
|
658
|
+
setSignatureData(null);
|
|
659
|
+
setIsError(false);
|
|
660
|
+
setErrorMessage("");
|
|
661
|
+
setIsLoadedFromRecent(false);
|
|
662
|
+
onClear?.();
|
|
663
|
+
}, [onClear]);
|
|
664
|
+
const getErrorMessage = React.useCallback(() => {
|
|
665
|
+
return errorMessage;
|
|
666
|
+
}, [errorMessage]);
|
|
667
|
+
return {
|
|
668
|
+
// ref
|
|
669
|
+
signatureRef,
|
|
670
|
+
// state
|
|
671
|
+
signatureData,
|
|
672
|
+
isError,
|
|
673
|
+
isLoadedFromRecent,
|
|
674
|
+
// 최근 서명 불러옴 여부 노출
|
|
675
|
+
// actions
|
|
676
|
+
submit,
|
|
677
|
+
clear,
|
|
678
|
+
loadRecentSignature,
|
|
679
|
+
handleSignatureChange,
|
|
680
|
+
// 일반적인 서명 변경
|
|
681
|
+
handleReset,
|
|
682
|
+
// 다시 작성
|
|
683
|
+
handleSignatureSubmit,
|
|
684
|
+
// utilities
|
|
685
|
+
isValid,
|
|
686
|
+
getSignatureData,
|
|
687
|
+
getErrorMessage
|
|
688
|
+
};
|
|
689
|
+
};
|
|
690
|
+
|
|
691
|
+
const cx = classNames.bind(styles$2);
|
|
300
692
|
const StepIndicator = ({
|
|
301
693
|
items,
|
|
302
694
|
onClickItem,
|
|
@@ -331,7 +723,9 @@ const StepIndicator = ({
|
|
|
331
723
|
exports.DataTable = DataTable;
|
|
332
724
|
exports.FormDatePicker = FormDatePicker;
|
|
333
725
|
exports.FormSegmentGroup = FormSegmentGroup;
|
|
726
|
+
exports.FormSignature = FormSignature;
|
|
334
727
|
exports.FormTextField = FormTextField;
|
|
335
728
|
exports.StepIndicator = StepIndicator;
|
|
729
|
+
exports.useFormSignature = useFormSignature;
|
|
336
730
|
exports.useTable = useTable;
|
|
337
731
|
//# sourceMappingURL=index.cjs.js.map
|