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/index.cjs.js CHANGED
@@ -6,9 +6,8 @@ var salesFrontendDesignSystem = require('sales-frontend-design-system');
6
6
  var styles = require('./data-table/data-table.module.scss');
7
7
  var React = require('react');
8
8
  var reactHookForm = require('react-hook-form');
9
- var styles$1 = require('./form/form-signautre/form-signature.module.scss');
10
- var styles$2 = require('./step-indicator/step-indicator.module.scss');
11
- var styles$3 = require('./camera/camera.module.scss');
9
+ var styles$1 = require('./step-indicator/step-indicator.module.scss');
10
+ var styles$2 = require('./camera/camera.module.scss');
12
11
 
13
12
  function getDefaultExportFromCjs (x) {
14
13
  return x && x.__esModule && Object.prototype.hasOwnProperty.call(x, 'default') ? x['default'] : x;
@@ -101,7 +100,7 @@ function requireBind () {
101
100
  var bindExports = requireBind();
102
101
  var classNames = /*@__PURE__*/getDefaultExportFromCjs(bindExports);
103
102
 
104
- const cx$3 = classNames.bind(styles);
103
+ const cx$2 = classNames.bind(styles);
105
104
  const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
106
105
  return /* @__PURE__ */ jsxRuntime.jsxs(
107
106
  salesFrontendDesignSystem.Table,
@@ -116,7 +115,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
116
115
  "th",
117
116
  {
118
117
  colSpan: header.colSpan,
119
- className: cx$3({
118
+ className: cx$2({
120
119
  "is-resizing": header.column.getIsResizing()
121
120
  // 3. 리사이징 중일 때 클래스 추가
122
121
  }),
@@ -128,7 +127,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
128
127
  {
129
128
  onMouseDown: header.getResizeHandler(),
130
129
  onTouchStart: header.getResizeHandler(),
131
- className: cx$3("resize-handle")
130
+ className: cx$2("resize-handle")
132
131
  }
133
132
  )
134
133
  ]
@@ -149,7 +148,7 @@ const DataTable = ({ table, isError, isLoading, msgText, ...props }) => {
149
148
  },
150
149
  cell.id
151
150
  )) }, row.id)
152
- )) : /* @__PURE__ */ jsxRuntime.jsx("tr", { children: /* @__PURE__ */ jsxRuntime.jsx("td", { className: cx$3("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
+ )) : /* @__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") }) }) }) })
153
152
  ]
154
153
  }
155
154
  );
@@ -298,432 +297,7 @@ const FormTextField = ({
298
297
  );
299
298
  };
300
299
 
301
- const cx$2 = classNames.bind(styles$1);
302
- const FormSignature = ({
303
- label,
304
- error,
305
- errorMsg,
306
- onSubmit,
307
- onRecentSignatureLoad,
308
- onChangeSignature,
309
- onReset,
310
- isShowRecentSignatureButton,
311
- isShowResetButton,
312
- ref,
313
- canvasWidth = 632,
314
- canvasHeight = 230
315
- }) => {
316
- const canvasRef = React.useRef(null);
317
- const containerRef = React.useRef(null);
318
- const [isDrawing, setIsDrawing] = React.useState(false);
319
- const [isEmpty, setIsEmpty] = React.useState(true);
320
- const [signatureData, setSignatureData] = React.useState("");
321
- const [isLoadingSignature, setIsLoadingSignature] = React.useState(false);
322
- const [isMousePressed, setIsMousePressed] = React.useState(false);
323
- const [totalLineLength, setTotalLineLength] = React.useState(0);
324
- const [lastPoint, setLastPoint] = React.useState(null);
325
- const [strokeLengths, setStrokeLengths] = React.useState([]);
326
- const [currentStrokeLength, setCurrentStrokeLength] = React.useState(0);
327
- React.useEffect(() => {
328
- const canvas = canvasRef.current;
329
- if (!canvas) {
330
- return;
331
- }
332
- const ctx = canvas.getContext("2d");
333
- if (!ctx) {
334
- return;
335
- }
336
- ctx.strokeStyle = "#000000";
337
- ctx.lineWidth = Math.max(1, canvasWidth / 400);
338
- ctx.lineCap = "round";
339
- ctx.lineJoin = "round";
340
- }, [canvasWidth]);
341
- React.useEffect(() => {
342
- const handleGlobalMouseUp = () => {
343
- setIsMousePressed(false);
344
- if (isDrawing) {
345
- stopDrawing();
346
- }
347
- };
348
- document.addEventListener("mouseup", handleGlobalMouseUp);
349
- return () => {
350
- document.removeEventListener("mouseup", handleGlobalMouseUp);
351
- };
352
- }, [isDrawing]);
353
- const getCanvasCoordinates = React.useCallback((e) => {
354
- const canvas = canvasRef.current;
355
- if (!canvas) {
356
- return { x: 0, y: 0 };
357
- }
358
- const rect = canvas.getBoundingClientRect();
359
- const scaleX = canvas.width / rect.width;
360
- const scaleY = canvas.height / rect.height;
361
- return {
362
- x: (e.clientX - rect.left) * scaleX,
363
- y: (e.clientY - rect.top) * scaleY
364
- };
365
- }, []);
366
- const calculateDistance = React.useCallback((point1, point2) => {
367
- return Math.sqrt(Math.pow(point2.x - point1.x, 2) + Math.pow(point2.y - point1.y, 2));
368
- }, []);
369
- const startDrawing = React.useCallback(
370
- (e) => {
371
- const canvas = canvasRef.current;
372
- if (!canvas) {
373
- return;
374
- }
375
- const ctx = canvas.getContext("2d");
376
- if (!ctx) {
377
- return;
378
- }
379
- const { x, y } = getCanvasCoordinates(e);
380
- setIsDrawing(true);
381
- setIsMousePressed(true);
382
- setIsEmpty(false);
383
- setLastPoint({ x, y });
384
- setCurrentStrokeLength(0);
385
- ctx.beginPath();
386
- ctx.moveTo(x, y);
387
- },
388
- [getCanvasCoordinates]
389
- );
390
- const draw = React.useCallback(
391
- (e) => {
392
- if (!isDrawing) {
393
- return;
394
- }
395
- const canvas = canvasRef.current;
396
- if (!canvas) {
397
- return;
398
- }
399
- const ctx = canvas.getContext("2d");
400
- if (!ctx) {
401
- return;
402
- }
403
- const { x, y } = getCanvasCoordinates(e);
404
- const currentPoint = { x, y };
405
- if (lastPoint) {
406
- const distance = calculateDistance(lastPoint, currentPoint);
407
- if (distance > 0.5) {
408
- setCurrentStrokeLength((prev) => prev + distance);
409
- setTotalLineLength((prev) => prev + distance);
410
- }
411
- }
412
- ctx.lineTo(x, y);
413
- ctx.stroke();
414
- setLastPoint(currentPoint);
415
- const data = canvas.toDataURL();
416
- setSignatureData(data);
417
- onChangeSignature?.();
418
- },
419
- [isDrawing, getCanvasCoordinates, calculateDistance, lastPoint, onChangeSignature]
420
- );
421
- const stopDrawing = React.useCallback(() => {
422
- if (isDrawing && currentStrokeLength > 0) {
423
- setStrokeLengths((prev) => [...prev, currentStrokeLength]);
424
- }
425
- setIsDrawing(false);
426
- setLastPoint(null);
427
- }, [isDrawing, currentStrokeLength]);
428
- const handleMouseEnter = React.useCallback(
429
- (e) => {
430
- if (isMousePressed && !isDrawing) {
431
- const canvas = canvasRef.current;
432
- if (!canvas) {
433
- return;
434
- }
435
- const ctx = canvas.getContext("2d");
436
- if (!ctx) {
437
- return;
438
- }
439
- const { x, y } = getCanvasCoordinates(e);
440
- setIsDrawing(true);
441
- setLastPoint({ x, y });
442
- ctx.beginPath();
443
- ctx.moveTo(x, y);
444
- }
445
- },
446
- [isMousePressed, isDrawing, getCanvasCoordinates]
447
- );
448
- const handleMouseLeave = React.useCallback(() => {
449
- if (isDrawing) {
450
- setIsDrawing(false);
451
- setLastPoint(null);
452
- }
453
- }, [isDrawing]);
454
- const clearSignature = React.useCallback(() => {
455
- const canvas = canvasRef.current;
456
- if (!canvas) {
457
- return;
458
- }
459
- const ctx = canvas.getContext("2d");
460
- if (!ctx) {
461
- return;
462
- }
463
- ctx.clearRect(0, 0, canvasWidth, canvasHeight);
464
- setIsEmpty(true);
465
- setSignatureData("");
466
- setTotalLineLength(0);
467
- setLastPoint(null);
468
- setStrokeLengths([]);
469
- setCurrentStrokeLength(0);
470
- }, [canvasWidth, canvasHeight]);
471
- const loadSignatureToCanvas = React.useCallback(
472
- (imageDataURL) => {
473
- return new Promise((resolve, reject) => {
474
- const canvas = canvasRef.current;
475
- if (!canvas) {
476
- reject(new Error("Canvas not found"));
477
- return;
478
- }
479
- const ctx = canvas.getContext("2d");
480
- if (!ctx) {
481
- reject(new Error("Canvas context not found"));
482
- return;
483
- }
484
- const img = new Image();
485
- img.onload = () => {
486
- ctx.clearRect(0, 0, canvasWidth, canvasHeight);
487
- ctx.drawImage(img, 0, 0, canvasWidth, canvasHeight);
488
- setIsEmpty(false);
489
- setSignatureData(imageDataURL);
490
- setTotalLineLength(0);
491
- setStrokeLengths([]);
492
- setCurrentStrokeLength(0);
493
- setLastPoint(null);
494
- resolve();
495
- };
496
- img.onerror = () => {
497
- reject(new Error("Failed to load signature image"));
498
- };
499
- img.src = imageDataURL;
500
- });
501
- },
502
- [canvasWidth, canvasHeight]
503
- );
504
- const validateForm = React.useCallback(() => {
505
- return !(!signatureData || isEmpty);
506
- }, [signatureData, isEmpty]);
507
- const getSignatureDataObject = React.useCallback(() => {
508
- if (!signatureData || isEmpty) {
509
- return null;
510
- }
511
- return {
512
- signature: signatureData,
513
- totalLineLength: Math.round(totalLineLength),
514
- strokeCount: strokeLengths.length,
515
- strokeLengths: strokeLengths.map((length) => Math.round(length)),
516
- averageStrokeLength: strokeLengths.length > 0 ? Math.round(totalLineLength / strokeLengths.length) : 0
517
- };
518
- }, [signatureData, isEmpty, totalLineLength, strokeLengths]);
519
- const handleSubmit = React.useCallback(() => {
520
- if (!validateForm()) {
521
- return;
522
- }
523
- const data = getSignatureDataObject();
524
- if (data) {
525
- console.log("\uC81C\uCD9C\uB41C \uB370\uC774\uD130:", data);
526
- onSubmit(data);
527
- }
528
- }, [validateForm, getSignatureDataObject, onSubmit]);
529
- const handleReset = React.useCallback(() => {
530
- clearSignature();
531
- onReset?.();
532
- }, [clearSignature, onReset]);
533
- const handleRecentSignatureLoad = React.useCallback(async () => {
534
- if (!onRecentSignatureLoad) {
535
- return;
536
- }
537
- try {
538
- setIsLoadingSignature(true);
539
- const loadedSignature = await onRecentSignatureLoad();
540
- if (loadedSignature) {
541
- await loadSignatureToCanvas(loadedSignature);
542
- console.log("\uCD5C\uADFC \uC11C\uBA85\uC744 \uC131\uACF5\uC801\uC73C\uB85C \uBD88\uB7EC\uC654\uC2B5\uB2C8\uB2E4.");
543
- } else {
544
- alert("\uBD88\uB7EC\uC62C \uCD5C\uADFC \uC11C\uBA85\uC774 \uC5C6\uC2B5\uB2C8\uB2E4.");
545
- }
546
- } catch (error2) {
547
- console.error("\uCD5C\uADFC \uC11C\uBA85 \uBD88\uB7EC\uC624\uAE30 \uC2E4\uD328:", error2);
548
- alert("\uCD5C\uADFC \uC11C\uBA85\uC744 \uBD88\uB7EC\uC624\uB294\uB370 \uC2E4\uD328\uD588\uC2B5\uB2C8\uB2E4.");
549
- } finally {
550
- setIsLoadingSignature(false);
551
- }
552
- }, [onRecentSignatureLoad, loadSignatureToCanvas]);
553
- React.useImperativeHandle(
554
- ref,
555
- () => ({
556
- submit: handleSubmit,
557
- clear: clearSignature,
558
- getSignatureData: getSignatureDataObject,
559
- isValid: validateForm
560
- }),
561
- [handleSubmit, clearSignature, getSignatureDataObject, validateForm]
562
- );
563
- return /* @__PURE__ */ jsxRuntime.jsxs("div", { children: [
564
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: cx$2("form-label"), children: [
565
- /* @__PURE__ */ jsxRuntime.jsx("label", { className: "typo-subtitle3 text-body_1", children: label }),
566
- /* @__PURE__ */ jsxRuntime.jsxs("div", { className: cx$2("form-label-button-box"), children: [
567
- isShowResetButton && /* @__PURE__ */ jsxRuntime.jsx(
568
- salesFrontendDesignSystem.Button,
569
- {
570
- appearance: "filled",
571
- variant: "neutral",
572
- size: "xsmall",
573
- onClick: handleReset,
574
- className: "text-nowrap-i",
575
- disabled: isEmpty,
576
- children: "\uB2E4\uC2DC \uC791\uC131"
577
- }
578
- ),
579
- isShowRecentSignatureButton && /* @__PURE__ */ jsxRuntime.jsx(
580
- salesFrontendDesignSystem.Button,
581
- {
582
- appearance: "filled",
583
- variant: "neutral",
584
- size: "xsmall",
585
- className: "text-nowrap-i",
586
- onClick: handleRecentSignatureLoad,
587
- disabled: isLoadingSignature,
588
- children: isLoadingSignature ? "\uBD88\uB7EC\uC624\uB294 \uC911..." : "\uCD5C\uADFC \uC131\uBA85 \uBD88\uB7EC\uC624\uAE30"
589
- }
590
- )
591
- ] })
592
- ] }),
593
- /* @__PURE__ */ jsxRuntime.jsx("div", { className: cx$2("form-signature", error && "is-error"), ref: containerRef, children: /* @__PURE__ */ jsxRuntime.jsx(
594
- "canvas",
595
- {
596
- ref: canvasRef,
597
- width: canvasWidth,
598
- height: canvasHeight,
599
- className: "border border-gray-200 rounded cursor-crosshair bg-white",
600
- style: { width: "100%", height: `${canvasHeight}px` },
601
- onMouseDown: startDrawing,
602
- onMouseMove: draw,
603
- onMouseUp: stopDrawing,
604
- onMouseEnter: handleMouseEnter,
605
- onMouseLeave: handleMouseLeave
606
- }
607
- ) }),
608
- error && /* @__PURE__ */ jsxRuntime.jsx("div", { className: cx$2("error-msg-box"), children: /* @__PURE__ */ jsxRuntime.jsx("span", { className: cx$2("error-msg"), children: errorMsg || "\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694." }) })
609
- ] });
610
- };
611
-
612
- const useFormSignature = ({ onSubmit, onError, onClear, minLength = 100 } = {}) => {
613
- const signatureRef = React.useRef(null);
614
- const [signatureData, setSignatureData] = React.useState(null);
615
- const [isError, setIsError] = React.useState(false);
616
- const [errorMessage, setErrorMessage] = React.useState("");
617
- const [isLoadedFromRecent, setIsLoadedFromRecent] = React.useState(false);
618
- const handleSignatureSubmit = React.useCallback(
619
- (data) => {
620
- setSignatureData(data);
621
- setIsError(false);
622
- setErrorMessage("");
623
- onSubmit?.(data);
624
- },
625
- [onSubmit]
626
- );
627
- const submit = React.useCallback(() => {
628
- try {
629
- if (!signatureRef.current?.isValid()) {
630
- setIsError(true);
631
- setErrorMessage("\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694.");
632
- return false;
633
- }
634
- const data = signatureRef.current.getSignatureData();
635
- if (!data) {
636
- setIsError(true);
637
- setErrorMessage("\uC11C\uBA85 \uB370\uC774\uD130\uB97C \uAC00\uC838\uC62C \uC218 \uC5C6\uC2B5\uB2C8\uB2E4.");
638
- return false;
639
- }
640
- if (!isLoadedFromRecent && data.totalLineLength < minLength) {
641
- setIsError(true);
642
- setErrorMessage(`\uC11C\uBA85\uC744 \uB2E4\uC2DC \uD655\uC778\uD574 \uC8FC\uC138\uC694.`);
643
- return false;
644
- }
645
- setIsError(false);
646
- setErrorMessage("");
647
- signatureRef.current.submit();
648
- return true;
649
- } catch (error) {
650
- const errorObj = error instanceof Error ? error : new Error("\uC54C \uC218 \uC5C6\uB294 \uC624\uB958\uAC00 \uBC1C\uC0DD\uD588\uC2B5\uB2C8\uB2E4.");
651
- setIsError(true);
652
- setErrorMessage(errorObj.message);
653
- onError?.(errorObj);
654
- return false;
655
- }
656
- }, [onError, minLength, isLoadedFromRecent]);
657
- const clear = React.useCallback(() => {
658
- signatureRef.current?.clear();
659
- setSignatureData(null);
660
- setIsError(false);
661
- setErrorMessage("");
662
- setIsLoadedFromRecent(false);
663
- onClear?.();
664
- }, [onClear]);
665
- const loadRecentSignature = React.useCallback(async () => {
666
- setIsLoadedFromRecent(true);
667
- setIsError(false);
668
- setErrorMessage("");
669
- return "";
670
- }, []);
671
- const isValid = React.useCallback(() => {
672
- const basicValid = signatureRef.current?.isValid() ?? false;
673
- if (!basicValid) {
674
- return false;
675
- }
676
- const data = signatureRef.current?.getSignatureData();
677
- if (!data) {
678
- return false;
679
- }
680
- if (isLoadedFromRecent) {
681
- return true;
682
- }
683
- return data.totalLineLength >= minLength;
684
- }, [minLength, isLoadedFromRecent]);
685
- const getSignatureData = React.useCallback(() => {
686
- return signatureRef.current?.getSignatureData() ?? null;
687
- }, []);
688
- const handleSignatureChange = React.useCallback(() => {
689
- setIsError(false);
690
- setErrorMessage("");
691
- }, []);
692
- const handleReset = React.useCallback(() => {
693
- setSignatureData(null);
694
- setIsError(false);
695
- setErrorMessage("");
696
- setIsLoadedFromRecent(false);
697
- onClear?.();
698
- }, [onClear]);
699
- const getErrorMessage = React.useCallback(() => {
700
- return errorMessage;
701
- }, [errorMessage]);
702
- return {
703
- // ref
704
- signatureRef,
705
- // state
706
- signatureData,
707
- isError,
708
- isLoadedFromRecent,
709
- // 최근 서명 불러옴 여부 노출
710
- // actions
711
- submit,
712
- clear,
713
- loadRecentSignature,
714
- handleSignatureChange,
715
- // 일반적인 서명 변경
716
- handleReset,
717
- // 다시 작성
718
- handleSignatureSubmit,
719
- // utilities
720
- isValid,
721
- getSignatureData,
722
- getErrorMessage
723
- };
724
- };
725
-
726
- const cx$1 = classNames.bind(styles$2);
300
+ const cx$1 = classNames.bind(styles$1);
727
301
  const StepIndicator = ({
728
302
  items,
729
303
  onClickItem,
@@ -755,7 +329,7 @@ const StepIndicator = ({
755
329
  ] });
756
330
  };
757
331
 
758
- const cx = classNames.bind(styles$3);
332
+ const cx = classNames.bind(styles$2);
759
333
  function Attachment({
760
334
  photos,
761
335
  onAddPhoto,
@@ -995,19 +569,18 @@ const DEFAULT_DOWNLOAD_PROPS = {
995
569
  width: 500,
996
570
  height: 500
997
571
  };
998
- function usePaint(paintProps = {}) {
572
+ function useCanvasPaint(paintProps = {}) {
999
573
  const { pen = { strokeWeight: 5, strokeColor: "black" }, onChange, onStart, onEnd } = paintProps;
1000
- const canvasRef = React.useRef(null);
574
+ const [canvasRefState, setCanvasRefState] = React.useState(null);
1001
575
  const contextRef = React.useRef(null);
1002
576
  const [isPainting, setIsPainting] = React.useState(false);
1003
577
  const [history, setHistory] = React.useState([]);
1004
578
  const [historyIndex, setHistoryIndex] = React.useState(-1);
1005
579
  const saveState = React.useCallback(() => {
1006
- const canvas = canvasRef.current;
1007
- if (!canvas) {
580
+ if (!canvasRefState) {
1008
581
  return;
1009
582
  }
1010
- const dataUrl = canvas.toDataURL();
583
+ const dataUrl = canvasRefState.toDataURL();
1011
584
  const newHistory = history.slice(0, historyIndex + 1);
1012
585
  if (newHistory.length >= HISTORY_SIZE) {
1013
586
  newHistory.shift();
@@ -1016,39 +589,37 @@ function usePaint(paintProps = {}) {
1016
589
  setHistory(newHistory);
1017
590
  setHistoryIndex(newHistory.length - 1);
1018
591
  onChange && onChange();
1019
- }, [history, historyIndex, onChange]);
592
+ }, [history, historyIndex, onChange, canvasRefState]);
1020
593
  const restoreState = React.useCallback(
1021
594
  (index) => {
1022
- const canvas = canvasRef.current;
1023
595
  const context = contextRef.current;
1024
- if (!canvas || !context || !history[index]) {
596
+ if (!canvasRefState || !context || !history[index]) {
1025
597
  return;
1026
598
  }
1027
599
  const dataUrl = history[index];
1028
600
  const img = new Image();
1029
601
  img.onload = () => {
1030
- context.clearRect(0, 0, canvas.width, canvas.height);
602
+ context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
1031
603
  context.drawImage(img, 0, 0);
1032
604
  };
1033
605
  img.src = dataUrl;
1034
606
  onChange && onChange();
1035
607
  },
1036
- [history, onChange]
608
+ [history, onChange, canvasRefState]
1037
609
  );
1038
610
  React.useEffect(() => {
1039
- const canvas = canvasRef.current;
1040
- if (!canvas) {
611
+ if (!canvasRefState) {
1041
612
  return;
1042
613
  }
1043
- const context = canvas.getContext("2d", { willReadFrequently: true });
614
+ const context = canvasRefState.getContext("2d", { willReadFrequently: true });
1044
615
  if (!context) {
1045
616
  return;
1046
617
  }
1047
618
  contextRef.current = context;
1048
- const initialDataUrl = canvas.toDataURL();
619
+ const initialDataUrl = canvasRefState.toDataURL();
1049
620
  setHistory([initialDataUrl]);
1050
621
  setHistoryIndex(0);
1051
- }, []);
622
+ }, [canvasRefState]);
1052
623
  React.useEffect(() => {
1053
624
  const context = contextRef.current;
1054
625
  if (context) {
@@ -1056,19 +627,53 @@ function usePaint(paintProps = {}) {
1056
627
  context.strokeStyle = pen.strokeColor;
1057
628
  }
1058
629
  }, [pen]);
630
+ const getEventPosition = React.useCallback(
631
+ (event) => {
632
+ if (!canvasRefState) {
633
+ return null;
634
+ }
635
+ const rect = canvasRefState.getBoundingClientRect();
636
+ const scaleX = canvasRefState.width / rect.width;
637
+ const scaleY = canvasRefState.height / rect.height;
638
+ let clientX;
639
+ let clientY;
640
+ if (event instanceof MouseEvent) {
641
+ clientX = event.clientX;
642
+ clientY = event.clientY;
643
+ } else if (event instanceof TouchEvent) {
644
+ const touch = event.touches[0];
645
+ if (!touch) {
646
+ return null;
647
+ }
648
+ clientX = touch.clientX;
649
+ clientY = touch.clientY;
650
+ } else {
651
+ return null;
652
+ }
653
+ const offsetX = (clientX - rect.left) * scaleX;
654
+ const offsetY = (clientY - rect.top) * scaleY;
655
+ return { offsetX, offsetY };
656
+ },
657
+ [canvasRefState]
658
+ );
1059
659
  const startPainting = React.useCallback(
1060
660
  (event) => {
661
+ event.preventDefault();
1061
662
  const context = contextRef.current;
1062
663
  if (!context) {
1063
664
  return;
1064
665
  }
666
+ const pos = getEventPosition(event);
667
+ if (!pos) {
668
+ return;
669
+ }
1065
670
  onStart && onStart();
1066
- const { offsetX, offsetY } = event;
671
+ const { offsetX, offsetY } = pos;
1067
672
  context.beginPath();
1068
673
  context.moveTo(offsetX, offsetY);
1069
674
  setIsPainting(true);
1070
675
  },
1071
- [onStart]
676
+ [onStart, getEventPosition]
1072
677
  );
1073
678
  const stopPainting = React.useCallback(() => {
1074
679
  const context = contextRef.current;
@@ -1082,47 +687,48 @@ function usePaint(paintProps = {}) {
1082
687
  }, [isPainting, onEnd, saveState]);
1083
688
  const paint = React.useCallback(
1084
689
  (event) => {
690
+ event.preventDefault();
1085
691
  if (!isPainting || !contextRef.current) {
1086
692
  return;
1087
693
  }
1088
- const { offsetX, offsetY } = event;
694
+ const pos = getEventPosition(event);
695
+ if (!pos) {
696
+ return;
697
+ }
698
+ const { offsetX, offsetY } = pos;
1089
699
  contextRef.current.lineTo(offsetX, offsetY);
1090
700
  contextRef.current.stroke();
1091
701
  },
1092
- [isPainting]
1093
- );
1094
- const handleMouseEnter = React.useCallback(
1095
- (event) => {
1096
- if (event.buttons === 1 && !isPainting) {
1097
- startPainting(event);
1098
- }
1099
- },
1100
- [isPainting, startPainting]
702
+ [isPainting, getEventPosition]
1101
703
  );
1102
704
  React.useEffect(() => {
1103
- const canvas = canvasRef.current;
1104
- if (!canvas) {
705
+ if (!canvasRefState) {
1105
706
  return;
1106
707
  }
1107
- canvas.addEventListener("mousedown", startPainting);
1108
- canvas.addEventListener("mouseup", stopPainting);
1109
- canvas.addEventListener("mousemove", paint);
1110
- canvas.addEventListener("mouseleave", stopPainting);
1111
- canvas.addEventListener("mouseenter", handleMouseEnter);
708
+ canvasRefState.addEventListener("mousedown", startPainting);
709
+ canvasRefState.addEventListener("mouseup", stopPainting);
710
+ canvasRefState.addEventListener("mousemove", paint);
711
+ canvasRefState.addEventListener("mouseleave", stopPainting);
712
+ canvasRefState.addEventListener("touchstart", startPainting);
713
+ canvasRefState.addEventListener("touchend", stopPainting);
714
+ canvasRefState.addEventListener("touchmove", paint);
715
+ canvasRefState.addEventListener("touchcancel", stopPainting);
1112
716
  return () => {
1113
- canvas.removeEventListener("mousedown", startPainting);
1114
- canvas.removeEventListener("mouseup", stopPainting);
1115
- canvas.removeEventListener("mousemove", paint);
1116
- canvas.removeEventListener("mouseleave", stopPainting);
1117
- canvas.removeEventListener("mouseenter", handleMouseEnter);
717
+ canvasRefState.removeEventListener("mousedown", startPainting);
718
+ canvasRefState.removeEventListener("mouseup", stopPainting);
719
+ canvasRefState.removeEventListener("mousemove", paint);
720
+ canvasRefState.removeEventListener("mouseleave", stopPainting);
721
+ canvasRefState.removeEventListener("touchstart", startPainting);
722
+ canvasRefState.removeEventListener("touchend", stopPainting);
723
+ canvasRefState.removeEventListener("touchmove", paint);
724
+ canvasRefState.removeEventListener("touchcancel", stopPainting);
1118
725
  };
1119
- }, [startPainting, stopPainting, paint, handleMouseEnter]);
726
+ }, [startPainting, stopPainting, paint, canvasRefState]);
1120
727
  const clear = () => {
1121
- const canvas = canvasRef.current;
1122
728
  const context = contextRef.current;
1123
- if (canvas && context) {
1124
- context.clearRect(0, 0, canvas.width, canvas.height);
1125
- const initialDataUrl = canvas.toDataURL();
729
+ if (canvasRefState && context) {
730
+ context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
731
+ const initialDataUrl = canvasRefState.toDataURL();
1126
732
  setHistory([initialDataUrl]);
1127
733
  setHistoryIndex(0);
1128
734
  onChange && onChange();
@@ -1143,15 +749,14 @@ function usePaint(paintProps = {}) {
1143
749
  }
1144
750
  };
1145
751
  const loadImage = (base64String) => {
1146
- const canvas = canvasRef.current;
1147
752
  const context = contextRef.current;
1148
- if (!canvas || !context) {
753
+ if (!canvasRefState || !context) {
1149
754
  return;
1150
755
  }
1151
756
  const img = new Image();
1152
757
  img.onload = () => {
1153
- context.clearRect(0, 0, canvas.width, canvas.height);
1154
- context.drawImage(img, 0, 0, canvas.width, canvas.height);
758
+ context.clearRect(0, 0, canvasRefState.width, canvasRefState.height);
759
+ context.drawImage(img, 0, 0, canvasRefState.width, canvasRefState.height);
1155
760
  saveState();
1156
761
  };
1157
762
  img.src = base64String;
@@ -1159,7 +764,7 @@ function usePaint(paintProps = {}) {
1159
764
  const getBase64String = (imageProps) => {
1160
765
  const props = { ...DEFAULT_DOWNLOAD_PROPS, ...imageProps };
1161
766
  const { fileExtendsion, transparent, backgroundColor, width, height } = props;
1162
- const originalCanvas = canvasRef.current;
767
+ const originalCanvas = canvasRefState;
1163
768
  if (!originalCanvas) {
1164
769
  return;
1165
770
  }
@@ -1179,7 +784,7 @@ function usePaint(paintProps = {}) {
1179
784
  const download = (downloadProps) => {
1180
785
  const props = { ...DEFAULT_DOWNLOAD_PROPS, ...downloadProps };
1181
786
  const { fileName, fileExtendsion, transparent, backgroundColor, width, height } = props;
1182
- const originalCanvas = canvasRef.current;
787
+ const originalCanvas = canvasRefState;
1183
788
  if (!originalCanvas) {
1184
789
  return;
1185
790
  }
@@ -1199,27 +804,49 @@ function usePaint(paintProps = {}) {
1199
804
  link.click();
1200
805
  }
1201
806
  };
807
+ const isCanvasBlank = () => {
808
+ if (!canvasRefState) {
809
+ return true;
810
+ }
811
+ const { width, height } = canvasRefState;
812
+ const ctx = canvasRefState.getContext("2d");
813
+ if (!ctx) {
814
+ console.error("2D context not available");
815
+ return true;
816
+ }
817
+ const imageData = ctx.getImageData(0, 0, width, height);
818
+ const { data } = imageData;
819
+ for (let i = 0; i < data.length; i += 4) {
820
+ if (data[i + 3] !== 0) {
821
+ return false;
822
+ }
823
+ }
824
+ return true;
825
+ };
1202
826
  return {
1203
827
  clear,
1204
828
  undo,
1205
829
  redo,
1206
830
  loadImage,
1207
831
  download,
1208
- canvasRef,
1209
- getBase64String
832
+ getBase64String,
833
+ isCanvasBlank,
834
+ setCanvasRefState,
835
+ canvasRefState
1210
836
  };
1211
837
  }
1212
838
 
839
+ const testSignatureBase64Data = "";
840
+
1213
841
  exports.Attachment = Attachment;
1214
842
  exports.DataTable = DataTable;
1215
843
  exports.FormDatePicker = FormDatePicker;
1216
844
  exports.FormSegmentGroup = FormSegmentGroup;
1217
- exports.FormSignature = FormSignature;
1218
845
  exports.FormTextField = FormTextField;
1219
846
  exports.StepIndicator = StepIndicator;
1220
847
  exports.resize = resize;
848
+ exports.testSignatureBase64Data = testSignatureBase64Data;
1221
849
  exports.useCamera = useCamera;
1222
- exports.useFormSignature = useFormSignature;
1223
- exports.usePaint = usePaint;
850
+ exports.useCanvasPaint = useCanvasPaint;
1224
851
  exports.useTable = useTable;
1225
852
  //# sourceMappingURL=index.cjs.js.map