sales-frontend-components 0.0.34 → 0.0.35

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