sales-frontend-components 0.0.50 → 0.0.51
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/canvas-paint/canvas-paint.module.scss +7 -0
- package/dist/index.cjs.js +116 -489
- package/dist/index.cjs.js.map +1 -1
- package/dist/index.d.ts +12 -57
- package/dist/index.esm.js +117 -489
- package/dist/index.esm.js.map +1 -1
- package/package.json +9 -9
package/dist/index.esm.js
CHANGED
|
@@ -1,12 +1,11 @@
|
|
|
1
1
|
import { jsxs, jsx } from 'react/jsx-runtime';
|
|
2
2
|
import { flexRender, useReactTable, getSortedRowModel, getCoreRowModel, getPaginationRowModel } from '@tanstack/react-table';
|
|
3
|
-
import { Table, DatePicker, SegmentGroup, FormField,
|
|
3
|
+
import { Table, DatePicker, SegmentGroup, FormField, Icon } from 'sales-frontend-design-system';
|
|
4
4
|
import styles from './data-table/data-table.module.scss';
|
|
5
|
-
import React, { useState,
|
|
5
|
+
import React, { useState, useEffect, useRef, useCallback } from 'react';
|
|
6
6
|
import { useController } from 'react-hook-form';
|
|
7
|
-
import styles$1 from './
|
|
8
|
-
import styles$2 from './
|
|
9
|
-
import styles$3 from './camera/camera.module.scss';
|
|
7
|
+
import styles$1 from './step-indicator/step-indicator.module.scss';
|
|
8
|
+
import styles$2 from './camera/camera.module.scss';
|
|
10
9
|
|
|
11
10
|
function getDefaultExportFromCjs (x) {
|
|
12
11
|
return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
|
|
@@ -99,7 +98,7 @@ function requireBind () {
|
|
|
99
98
|
var bindExports = requireBind();
|
|
100
99
|
var classNames = /*@__PURE__*/getDefaultExportFromCjs(bindExports);
|
|
101
100
|
|
|
102
|
-
const cx$
|
|
101
|
+
const cx$2 = classNames.bind(styles);
|
|
103
102
|
const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
104
103
|
return /* @__PURE__ */ jsxs(
|
|
105
104
|
Table,
|
|
@@ -114,7 +113,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
|
114
113
|
"th",
|
|
115
114
|
{
|
|
116
115
|
colSpan: header.colSpan,
|
|
117
|
-
className: cx$
|
|
116
|
+
className: cx$2({
|
|
118
117
|
"is-resizing": header.column.getIsResizing()
|
|
119
118
|
// 3. 리사이징 중일 때 클래스 추가
|
|
120
119
|
}),
|
|
@@ -126,7 +125,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
|
126
125
|
{
|
|
127
126
|
onMouseDown: header.getResizeHandler(),
|
|
128
127
|
onTouchStart: header.getResizeHandler(),
|
|
129
|
-
className: cx$
|
|
128
|
+
className: cx$2("resize-handle")
|
|
130
129
|
}
|
|
131
130
|
)
|
|
132
131
|
]
|
|
@@ -147,7 +146,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
|
|
|
147
146
|
},
|
|
148
147
|
cell.id
|
|
149
148
|
)) }, row.id)
|
|
150
|
-
)) : /* @__PURE__ */ jsx("tr", { children: /* @__PURE__ */ jsx("td", { className: cx$
|
|
149
|
+
)) : /* @__PURE__ */ jsx("tr", { children: /* @__PURE__ */ jsx("td", { className: cx$2("feedback-cell"), colSpan: table.getVisibleLeafColumns().length, children: isLoading ? /* @__PURE__ */ jsx("div", { children: "Loading..." }) : /* @__PURE__ */ jsx("div", { children: msgText || (isError ? "Error" : "No data") }) }) }) })
|
|
151
150
|
]
|
|
152
151
|
}
|
|
153
152
|
);
|
|
@@ -296,432 +295,7 @@ const FormTextField = ({
|
|
|
296
295
|
);
|
|
297
296
|
};
|
|
298
297
|
|
|
299
|
-
const cx$
|
|
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);
|
|
298
|
+
const cx$1 = classNames.bind(styles$1);
|
|
725
299
|
const StepIndicator = ({
|
|
726
300
|
items,
|
|
727
301
|
onClickItem,
|
|
@@ -753,7 +327,7 @@ const StepIndicator = ({
|
|
|
753
327
|
] });
|
|
754
328
|
};
|
|
755
329
|
|
|
756
|
-
const cx = classNames.bind(styles$
|
|
330
|
+
const cx = classNames.bind(styles$2);
|
|
757
331
|
function Attachment({
|
|
758
332
|
photos,
|
|
759
333
|
onAddPhoto,
|
|
@@ -993,19 +567,18 @@ const DEFAULT_DOWNLOAD_PROPS = {
|
|
|
993
567
|
width: 500,
|
|
994
568
|
height: 500
|
|
995
569
|
};
|
|
996
|
-
function
|
|
570
|
+
function useCanvasPaint(paintProps = {}) {
|
|
997
571
|
const { pen = { strokeWeight: 5, strokeColor: "black" }, onChange, onStart, onEnd } = paintProps;
|
|
998
|
-
const
|
|
572
|
+
const [canvasRefState, setCanvasRefState] = useState(null);
|
|
999
573
|
const contextRef = useRef(null);
|
|
1000
574
|
const [isPainting, setIsPainting] = useState(false);
|
|
1001
575
|
const [history, setHistory] = useState([]);
|
|
1002
576
|
const [historyIndex, setHistoryIndex] = useState(-1);
|
|
1003
577
|
const saveState = useCallback(() => {
|
|
1004
|
-
|
|
1005
|
-
if (!canvas) {
|
|
578
|
+
if (!canvasRefState) {
|
|
1006
579
|
return;
|
|
1007
580
|
}
|
|
1008
|
-
const dataUrl =
|
|
581
|
+
const dataUrl = canvasRefState.toDataURL();
|
|
1009
582
|
const newHistory = history.slice(0, historyIndex + 1);
|
|
1010
583
|
if (newHistory.length >= HISTORY_SIZE) {
|
|
1011
584
|
newHistory.shift();
|
|
@@ -1014,39 +587,37 @@ function usePaint(paintProps = {}) {
|
|
|
1014
587
|
setHistory(newHistory);
|
|
1015
588
|
setHistoryIndex(newHistory.length - 1);
|
|
1016
589
|
onChange && onChange();
|
|
1017
|
-
}, [history, historyIndex, onChange]);
|
|
590
|
+
}, [history, historyIndex, onChange, canvasRefState]);
|
|
1018
591
|
const restoreState = useCallback(
|
|
1019
592
|
(index) => {
|
|
1020
|
-
const canvas = canvasRef.current;
|
|
1021
593
|
const context = contextRef.current;
|
|
1022
|
-
if (!
|
|
594
|
+
if (!canvasRefState || !context || !history[index]) {
|
|
1023
595
|
return;
|
|
1024
596
|
}
|
|
1025
597
|
const dataUrl = history[index];
|
|
1026
598
|
const img = new Image();
|
|
1027
599
|
img.onload = () => {
|
|
1028
|
-
context.clearRect(0, 0,
|
|
600
|
+
context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
|
|
1029
601
|
context.drawImage(img, 0, 0);
|
|
1030
602
|
};
|
|
1031
603
|
img.src = dataUrl;
|
|
1032
604
|
onChange && onChange();
|
|
1033
605
|
},
|
|
1034
|
-
[history, onChange]
|
|
606
|
+
[history, onChange, canvasRefState]
|
|
1035
607
|
);
|
|
1036
608
|
useEffect(() => {
|
|
1037
|
-
|
|
1038
|
-
if (!canvas) {
|
|
609
|
+
if (!canvasRefState) {
|
|
1039
610
|
return;
|
|
1040
611
|
}
|
|
1041
|
-
const context =
|
|
612
|
+
const context = canvasRefState.getContext("2d", { willReadFrequently: true });
|
|
1042
613
|
if (!context) {
|
|
1043
614
|
return;
|
|
1044
615
|
}
|
|
1045
616
|
contextRef.current = context;
|
|
1046
|
-
const initialDataUrl =
|
|
617
|
+
const initialDataUrl = canvasRefState.toDataURL();
|
|
1047
618
|
setHistory([initialDataUrl]);
|
|
1048
619
|
setHistoryIndex(0);
|
|
1049
|
-
}, []);
|
|
620
|
+
}, [canvasRefState]);
|
|
1050
621
|
useEffect(() => {
|
|
1051
622
|
const context = contextRef.current;
|
|
1052
623
|
if (context) {
|
|
@@ -1054,19 +625,53 @@ function usePaint(paintProps = {}) {
|
|
|
1054
625
|
context.strokeStyle = pen.strokeColor;
|
|
1055
626
|
}
|
|
1056
627
|
}, [pen]);
|
|
628
|
+
const getEventPosition = useCallback(
|
|
629
|
+
(event) => {
|
|
630
|
+
if (!canvasRefState) {
|
|
631
|
+
return null;
|
|
632
|
+
}
|
|
633
|
+
const rect = canvasRefState.getBoundingClientRect();
|
|
634
|
+
const scaleX = canvasRefState.width / rect.width;
|
|
635
|
+
const scaleY = canvasRefState.height / rect.height;
|
|
636
|
+
let clientX;
|
|
637
|
+
let clientY;
|
|
638
|
+
if (event instanceof MouseEvent) {
|
|
639
|
+
clientX = event.clientX;
|
|
640
|
+
clientY = event.clientY;
|
|
641
|
+
} else if (event instanceof TouchEvent) {
|
|
642
|
+
const touch = event.touches[0];
|
|
643
|
+
if (!touch) {
|
|
644
|
+
return null;
|
|
645
|
+
}
|
|
646
|
+
clientX = touch.clientX;
|
|
647
|
+
clientY = touch.clientY;
|
|
648
|
+
} else {
|
|
649
|
+
return null;
|
|
650
|
+
}
|
|
651
|
+
const offsetX = (clientX - rect.left) * scaleX;
|
|
652
|
+
const offsetY = (clientY - rect.top) * scaleY;
|
|
653
|
+
return { offsetX, offsetY };
|
|
654
|
+
},
|
|
655
|
+
[canvasRefState]
|
|
656
|
+
);
|
|
1057
657
|
const startPainting = useCallback(
|
|
1058
658
|
(event) => {
|
|
659
|
+
event.preventDefault();
|
|
1059
660
|
const context = contextRef.current;
|
|
1060
661
|
if (!context) {
|
|
1061
662
|
return;
|
|
1062
663
|
}
|
|
664
|
+
const pos = getEventPosition(event);
|
|
665
|
+
if (!pos) {
|
|
666
|
+
return;
|
|
667
|
+
}
|
|
1063
668
|
onStart && onStart();
|
|
1064
|
-
const { offsetX, offsetY } =
|
|
669
|
+
const { offsetX, offsetY } = pos;
|
|
1065
670
|
context.beginPath();
|
|
1066
671
|
context.moveTo(offsetX, offsetY);
|
|
1067
672
|
setIsPainting(true);
|
|
1068
673
|
},
|
|
1069
|
-
[onStart]
|
|
674
|
+
[onStart, getEventPosition]
|
|
1070
675
|
);
|
|
1071
676
|
const stopPainting = useCallback(() => {
|
|
1072
677
|
const context = contextRef.current;
|
|
@@ -1080,47 +685,48 @@ function usePaint(paintProps = {}) {
|
|
|
1080
685
|
}, [isPainting, onEnd, saveState]);
|
|
1081
686
|
const paint = useCallback(
|
|
1082
687
|
(event) => {
|
|
688
|
+
event.preventDefault();
|
|
1083
689
|
if (!isPainting || !contextRef.current) {
|
|
1084
690
|
return;
|
|
1085
691
|
}
|
|
1086
|
-
const
|
|
692
|
+
const pos = getEventPosition(event);
|
|
693
|
+
if (!pos) {
|
|
694
|
+
return;
|
|
695
|
+
}
|
|
696
|
+
const { offsetX, offsetY } = pos;
|
|
1087
697
|
contextRef.current.lineTo(offsetX, offsetY);
|
|
1088
698
|
contextRef.current.stroke();
|
|
1089
699
|
},
|
|
1090
|
-
[isPainting]
|
|
1091
|
-
);
|
|
1092
|
-
const handleMouseEnter = useCallback(
|
|
1093
|
-
(event) => {
|
|
1094
|
-
if (event.buttons === 1 && !isPainting) {
|
|
1095
|
-
startPainting(event);
|
|
1096
|
-
}
|
|
1097
|
-
},
|
|
1098
|
-
[isPainting, startPainting]
|
|
700
|
+
[isPainting, getEventPosition]
|
|
1099
701
|
);
|
|
1100
702
|
useEffect(() => {
|
|
1101
|
-
|
|
1102
|
-
if (!canvas) {
|
|
703
|
+
if (!canvasRefState) {
|
|
1103
704
|
return;
|
|
1104
705
|
}
|
|
1105
|
-
|
|
1106
|
-
|
|
1107
|
-
|
|
1108
|
-
|
|
1109
|
-
|
|
706
|
+
canvasRefState.addEventListener("mousedown", startPainting);
|
|
707
|
+
canvasRefState.addEventListener("mouseup", stopPainting);
|
|
708
|
+
canvasRefState.addEventListener("mousemove", paint);
|
|
709
|
+
canvasRefState.addEventListener("mouseleave", stopPainting);
|
|
710
|
+
canvasRefState.addEventListener("touchstart", startPainting);
|
|
711
|
+
canvasRefState.addEventListener("touchend", stopPainting);
|
|
712
|
+
canvasRefState.addEventListener("touchmove", paint);
|
|
713
|
+
canvasRefState.addEventListener("touchcancel", stopPainting);
|
|
1110
714
|
return () => {
|
|
1111
|
-
|
|
1112
|
-
|
|
1113
|
-
|
|
1114
|
-
|
|
1115
|
-
|
|
715
|
+
canvasRefState.removeEventListener("mousedown", startPainting);
|
|
716
|
+
canvasRefState.removeEventListener("mouseup", stopPainting);
|
|
717
|
+
canvasRefState.removeEventListener("mousemove", paint);
|
|
718
|
+
canvasRefState.removeEventListener("mouseleave", stopPainting);
|
|
719
|
+
canvasRefState.removeEventListener("touchstart", startPainting);
|
|
720
|
+
canvasRefState.removeEventListener("touchend", stopPainting);
|
|
721
|
+
canvasRefState.removeEventListener("touchmove", paint);
|
|
722
|
+
canvasRefState.removeEventListener("touchcancel", stopPainting);
|
|
1116
723
|
};
|
|
1117
|
-
}, [startPainting, stopPainting, paint,
|
|
724
|
+
}, [startPainting, stopPainting, paint, canvasRefState]);
|
|
1118
725
|
const clear = () => {
|
|
1119
|
-
const canvas = canvasRef.current;
|
|
1120
726
|
const context = contextRef.current;
|
|
1121
|
-
if (
|
|
1122
|
-
context.clearRect(0, 0,
|
|
1123
|
-
const initialDataUrl =
|
|
727
|
+
if (canvasRefState && context) {
|
|
728
|
+
context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
|
|
729
|
+
const initialDataUrl = canvasRefState.toDataURL();
|
|
1124
730
|
setHistory([initialDataUrl]);
|
|
1125
731
|
setHistoryIndex(0);
|
|
1126
732
|
onChange && onChange();
|
|
@@ -1141,15 +747,14 @@ function usePaint(paintProps = {}) {
|
|
|
1141
747
|
}
|
|
1142
748
|
};
|
|
1143
749
|
const loadImage = (base64String) => {
|
|
1144
|
-
const canvas = canvasRef.current;
|
|
1145
750
|
const context = contextRef.current;
|
|
1146
|
-
if (!
|
|
751
|
+
if (!canvasRefState || !context) {
|
|
1147
752
|
return;
|
|
1148
753
|
}
|
|
1149
754
|
const img = new Image();
|
|
1150
755
|
img.onload = () => {
|
|
1151
|
-
context.clearRect(0, 0,
|
|
1152
|
-
context.drawImage(img, 0, 0,
|
|
756
|
+
context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
|
|
757
|
+
context.drawImage(img, 0, 0, canvasRefState.width, canvasRefState.height);
|
|
1153
758
|
saveState();
|
|
1154
759
|
};
|
|
1155
760
|
img.src = base64String;
|
|
@@ -1157,7 +762,7 @@ function usePaint(paintProps = {}) {
|
|
|
1157
762
|
const getBase64String = (imageProps) => {
|
|
1158
763
|
const props = { ...DEFAULT_DOWNLOAD_PROPS, ...imageProps };
|
|
1159
764
|
const { fileExtendsion, transparent, backgroundColor, width, height } = props;
|
|
1160
|
-
const originalCanvas =
|
|
765
|
+
const originalCanvas = canvasRefState;
|
|
1161
766
|
if (!originalCanvas) {
|
|
1162
767
|
return;
|
|
1163
768
|
}
|
|
@@ -1177,7 +782,7 @@ function usePaint(paintProps = {}) {
|
|
|
1177
782
|
const download = (downloadProps) => {
|
|
1178
783
|
const props = { ...DEFAULT_DOWNLOAD_PROPS, ...downloadProps };
|
|
1179
784
|
const { fileName, fileExtendsion, transparent, backgroundColor, width, height } = props;
|
|
1180
|
-
const originalCanvas =
|
|
785
|
+
const originalCanvas = canvasRefState;
|
|
1181
786
|
if (!originalCanvas) {
|
|
1182
787
|
return;
|
|
1183
788
|
}
|
|
@@ -1197,16 +802,39 @@ function usePaint(paintProps = {}) {
|
|
|
1197
802
|
link.click();
|
|
1198
803
|
}
|
|
1199
804
|
};
|
|
805
|
+
const isCanvasBlank = () => {
|
|
806
|
+
if (!canvasRefState) {
|
|
807
|
+
return true;
|
|
808
|
+
}
|
|
809
|
+
const { width, height } = canvasRefState;
|
|
810
|
+
const ctx = canvasRefState.getContext("2d");
|
|
811
|
+
if (!ctx) {
|
|
812
|
+
console.error("2D context not available");
|
|
813
|
+
return true;
|
|
814
|
+
}
|
|
815
|
+
const imageData = ctx.getImageData(0, 0, width, height);
|
|
816
|
+
const { data } = imageData;
|
|
817
|
+
for (let i = 0; i < data.length; i += 4) {
|
|
818
|
+
if (data[i + 3] !== 0) {
|
|
819
|
+
return false;
|
|
820
|
+
}
|
|
821
|
+
}
|
|
822
|
+
return true;
|
|
823
|
+
};
|
|
1200
824
|
return {
|
|
1201
825
|
clear,
|
|
1202
826
|
undo,
|
|
1203
827
|
redo,
|
|
1204
828
|
loadImage,
|
|
1205
829
|
download,
|
|
1206
|
-
|
|
1207
|
-
|
|
830
|
+
getBase64String,
|
|
831
|
+
isCanvasBlank,
|
|
832
|
+
setCanvasRefState,
|
|
833
|
+
canvasRefState
|
|
1208
834
|
};
|
|
1209
835
|
}
|
|
1210
836
|
|
|
1211
|
-
|
|
837
|
+
const testSignatureBase64Data = "";
|
|
838
|
+
|
|
839
|
+
export { Attachment, DataTable, FormDatePicker, FormSegmentGroup, FormTextField, StepIndicator, resize, testSignatureBase64Data, useCamera, useCanvasPaint, useTable };
|
|
1212
840
|
//# sourceMappingURL=index.esm.js.map
|