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.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, Button, Icon } from 'sales-frontend-design-system';
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, useRef, useEffect, useCallback, useImperativeHandle } from 'react';
5
+ import React, { useState, useEffect, useRef, useCallback } from 'react';
6
6
  import { useController } from 'react-hook-form';
7
- import styles$1 from './form/form-signautre/form-signature.module.scss';
8
- import styles$2 from './step-indicator/step-indicator.module.scss';
9
- import styles$3 from './camera/camera.module.scss';
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$3 = classNames.bind(styles);
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$3({
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$3("resize-handle")
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$3("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") }) }) }) })
151
150
  ]
152
151
  }
153
152
  );
@@ -296,432 +295,7 @@ const FormTextField = ({
296
295
  );
297
296
  };
298
297
 
299
- const cx$2 = classNames.bind(styles$1);
300
- const FormSignature = ({
301
- label,
302
- error,
303
- errorMsg,
304
- onSubmit,
305
- onRecentSignatureLoad,
306
- onChangeSignature,
307
- onReset,
308
- isShowRecentSignatureButton,
309
- isShowResetButton,
310
- ref,
311
- canvasWidth = 632,
312
- canvasHeight = 230
313
- }) => {
314
- const canvasRef = useRef(null);
315
- const containerRef = useRef(null);
316
- const [isDrawing, setIsDrawing] = useState(false);
317
- const [isEmpty, setIsEmpty] = useState(true);
318
- const [signatureData, setSignatureData] = useState("");
319
- const [isLoadingSignature, setIsLoadingSignature] = useState(false);
320
- const [isMousePressed, setIsMousePressed] = useState(false);
321
- const [totalLineLength, setTotalLineLength] = useState(0);
322
- const [lastPoint, setLastPoint] = useState(null);
323
- const [strokeLengths, setStrokeLengths] = useState([]);
324
- const [currentStrokeLength, setCurrentStrokeLength] = useState(0);
325
- useEffect(() => {
326
- const canvas = canvasRef.current;
327
- if (!canvas) {
328
- return;
329
- }
330
- const ctx = canvas.getContext("2d");
331
- if (!ctx) {
332
- return;
333
- }
334
- ctx.strokeStyle = "#000000";
335
- ctx.lineWidth = Math.max(1, canvasWidth / 400);
336
- ctx.lineCap = "round";
337
- ctx.lineJoin = "round";
338
- }, [canvasWidth]);
339
- useEffect(() => {
340
- const handleGlobalMouseUp = () => {
341
- setIsMousePressed(false);
342
- if (isDrawing) {
343
- stopDrawing();
344
- }
345
- };
346
- document.addEventListener("mouseup", handleGlobalMouseUp);
347
- return () => {
348
- document.removeEventListener("mouseup", handleGlobalMouseUp);
349
- };
350
- }, [isDrawing]);
351
- const getCanvasCoordinates = useCallback((e) => {
352
- const canvas = canvasRef.current;
353
- if (!canvas) {
354
- return { x: 0, y: 0 };
355
- }
356
- const rect = canvas.getBoundingClientRect();
357
- const scaleX = canvas.width / rect.width;
358
- const scaleY = canvas.height / rect.height;
359
- return {
360
- x: (e.clientX - rect.left) * scaleX,
361
- y: (e.clientY - rect.top) * scaleY
362
- };
363
- }, []);
364
- const calculateDistance = useCallback((point1, point2) => {
365
- return Math.sqrt(Math.pow(point2.x - point1.x, 2) + Math.pow(point2.y - point1.y, 2));
366
- }, []);
367
- const startDrawing = useCallback(
368
- (e) => {
369
- const canvas = canvasRef.current;
370
- if (!canvas) {
371
- return;
372
- }
373
- const ctx = canvas.getContext("2d");
374
- if (!ctx) {
375
- return;
376
- }
377
- const { x, y } = getCanvasCoordinates(e);
378
- setIsDrawing(true);
379
- setIsMousePressed(true);
380
- setIsEmpty(false);
381
- setLastPoint({ x, y });
382
- setCurrentStrokeLength(0);
383
- ctx.beginPath();
384
- ctx.moveTo(x, y);
385
- },
386
- [getCanvasCoordinates]
387
- );
388
- const draw = useCallback(
389
- (e) => {
390
- if (!isDrawing) {
391
- return;
392
- }
393
- const canvas = canvasRef.current;
394
- if (!canvas) {
395
- return;
396
- }
397
- const ctx = canvas.getContext("2d");
398
- if (!ctx) {
399
- return;
400
- }
401
- const { x, y } = getCanvasCoordinates(e);
402
- const currentPoint = { x, y };
403
- if (lastPoint) {
404
- const distance = calculateDistance(lastPoint, currentPoint);
405
- if (distance > 0.5) {
406
- setCurrentStrokeLength((prev) => prev + distance);
407
- setTotalLineLength((prev) => prev + distance);
408
- }
409
- }
410
- ctx.lineTo(x, y);
411
- ctx.stroke();
412
- setLastPoint(currentPoint);
413
- const data = canvas.toDataURL();
414
- setSignatureData(data);
415
- onChangeSignature?.();
416
- },
417
- [isDrawing, getCanvasCoordinates, calculateDistance, lastPoint, onChangeSignature]
418
- );
419
- const stopDrawing = useCallback(() => {
420
- if (isDrawing && currentStrokeLength > 0) {
421
- setStrokeLengths((prev) => [...prev, currentStrokeLength]);
422
- }
423
- setIsDrawing(false);
424
- setLastPoint(null);
425
- }, [isDrawing, currentStrokeLength]);
426
- const handleMouseEnter = useCallback(
427
- (e) => {
428
- if (isMousePressed && !isDrawing) {
429
- const canvas = canvasRef.current;
430
- if (!canvas) {
431
- return;
432
- }
433
- const ctx = canvas.getContext("2d");
434
- if (!ctx) {
435
- return;
436
- }
437
- const { x, y } = getCanvasCoordinates(e);
438
- setIsDrawing(true);
439
- setLastPoint({ x, y });
440
- ctx.beginPath();
441
- ctx.moveTo(x, y);
442
- }
443
- },
444
- [isMousePressed, isDrawing, getCanvasCoordinates]
445
- );
446
- const handleMouseLeave = useCallback(() => {
447
- if (isDrawing) {
448
- setIsDrawing(false);
449
- setLastPoint(null);
450
- }
451
- }, [isDrawing]);
452
- const clearSignature = useCallback(() => {
453
- const canvas = canvasRef.current;
454
- if (!canvas) {
455
- return;
456
- }
457
- const ctx = canvas.getContext("2d");
458
- if (!ctx) {
459
- return;
460
- }
461
- ctx.clearRect(0, 0, canvasWidth, canvasHeight);
462
- setIsEmpty(true);
463
- setSignatureData("");
464
- setTotalLineLength(0);
465
- setLastPoint(null);
466
- setStrokeLengths([]);
467
- setCurrentStrokeLength(0);
468
- }, [canvasWidth, canvasHeight]);
469
- const loadSignatureToCanvas = useCallback(
470
- (imageDataURL) => {
471
- return new Promise((resolve, reject) => {
472
- const canvas = canvasRef.current;
473
- if (!canvas) {
474
- reject(new Error("Canvas not found"));
475
- return;
476
- }
477
- const ctx = canvas.getContext("2d");
478
- if (!ctx) {
479
- reject(new Error("Canvas context not found"));
480
- return;
481
- }
482
- const img = new Image();
483
- img.onload = () => {
484
- ctx.clearRect(0, 0, canvasWidth, canvasHeight);
485
- ctx.drawImage(img, 0, 0, canvasWidth, canvasHeight);
486
- setIsEmpty(false);
487
- setSignatureData(imageDataURL);
488
- setTotalLineLength(0);
489
- setStrokeLengths([]);
490
- setCurrentStrokeLength(0);
491
- setLastPoint(null);
492
- resolve();
493
- };
494
- img.onerror = () => {
495
- reject(new Error("Failed to load signature image"));
496
- };
497
- img.src = imageDataURL;
498
- });
499
- },
500
- [canvasWidth, canvasHeight]
501
- );
502
- const validateForm = useCallback(() => {
503
- return !(!signatureData || isEmpty);
504
- }, [signatureData, isEmpty]);
505
- const getSignatureDataObject = useCallback(() => {
506
- if (!signatureData || isEmpty) {
507
- return null;
508
- }
509
- return {
510
- signature: signatureData,
511
- totalLineLength: Math.round(totalLineLength),
512
- strokeCount: strokeLengths.length,
513
- strokeLengths: strokeLengths.map((length) => Math.round(length)),
514
- averageStrokeLength: strokeLengths.length > 0 ? Math.round(totalLineLength / strokeLengths.length) : 0
515
- };
516
- }, [signatureData, isEmpty, totalLineLength, strokeLengths]);
517
- const handleSubmit = useCallback(() => {
518
- if (!validateForm()) {
519
- return;
520
- }
521
- const data = getSignatureDataObject();
522
- if (data) {
523
- console.log("\uC81C\uCD9C\uB41C \uB370\uC774\uD130:", data);
524
- onSubmit(data);
525
- }
526
- }, [validateForm, getSignatureDataObject, onSubmit]);
527
- const handleReset = useCallback(() => {
528
- clearSignature();
529
- onReset?.();
530
- }, [clearSignature, onReset]);
531
- const handleRecentSignatureLoad = useCallback(async () => {
532
- if (!onRecentSignatureLoad) {
533
- return;
534
- }
535
- try {
536
- setIsLoadingSignature(true);
537
- const loadedSignature = await onRecentSignatureLoad();
538
- if (loadedSignature) {
539
- await loadSignatureToCanvas(loadedSignature);
540
- console.log("\uCD5C\uADFC \uC11C\uBA85\uC744 \uC131\uACF5\uC801\uC73C\uB85C \uBD88\uB7EC\uC654\uC2B5\uB2C8\uB2E4.");
541
- } else {
542
- alert("\uBD88\uB7EC\uC62C \uCD5C\uADFC \uC11C\uBA85\uC774 \uC5C6\uC2B5\uB2C8\uB2E4.");
543
- }
544
- } catch (error2) {
545
- console.error("\uCD5C\uADFC \uC11C\uBA85 \uBD88\uB7EC\uC624\uAE30 \uC2E4\uD328:", error2);
546
- alert("\uCD5C\uADFC \uC11C\uBA85\uC744 \uBD88\uB7EC\uC624\uB294\uB370 \uC2E4\uD328\uD588\uC2B5\uB2C8\uB2E4.");
547
- } finally {
548
- setIsLoadingSignature(false);
549
- }
550
- }, [onRecentSignatureLoad, loadSignatureToCanvas]);
551
- useImperativeHandle(
552
- ref,
553
- () => ({
554
- submit: handleSubmit,
555
- clear: clearSignature,
556
- getSignatureData: getSignatureDataObject,
557
- isValid: validateForm
558
- }),
559
- [handleSubmit, clearSignature, getSignatureDataObject, validateForm]
560
- );
561
- return /* @__PURE__ */ jsxs("div", { children: [
562
- /* @__PURE__ */ jsxs("div", { className: cx$2("form-label"), children: [
563
- /* @__PURE__ */ jsx("label", { className: "typo-subtitle3 text-body_1", children: label }),
564
- /* @__PURE__ */ jsxs("div", { className: cx$2("form-label-button-box"), children: [
565
- isShowResetButton && /* @__PURE__ */ jsx(
566
- Button,
567
- {
568
- appearance: "filled",
569
- variant: "neutral",
570
- size: "xsmall",
571
- onClick: handleReset,
572
- className: "text-nowrap-i",
573
- disabled: isEmpty,
574
- children: "\uB2E4\uC2DC \uC791\uC131"
575
- }
576
- ),
577
- isShowRecentSignatureButton && /* @__PURE__ */ jsx(
578
- Button,
579
- {
580
- appearance: "filled",
581
- variant: "neutral",
582
- size: "xsmall",
583
- className: "text-nowrap-i",
584
- onClick: handleRecentSignatureLoad,
585
- disabled: isLoadingSignature,
586
- children: isLoadingSignature ? "\uBD88\uB7EC\uC624\uB294 \uC911..." : "\uCD5C\uADFC \uC131\uBA85 \uBD88\uB7EC\uC624\uAE30"
587
- }
588
- )
589
- ] })
590
- ] }),
591
- /* @__PURE__ */ jsx("div", { className: cx$2("form-signature", error && "is-error"), ref: containerRef, children: /* @__PURE__ */ jsx(
592
- "canvas",
593
- {
594
- ref: canvasRef,
595
- width: canvasWidth,
596
- height: canvasHeight,
597
- className: "border border-gray-200 rounded cursor-crosshair bg-white",
598
- style: { width: "100%", height: `${canvasHeight}px` },
599
- onMouseDown: startDrawing,
600
- onMouseMove: draw,
601
- onMouseUp: stopDrawing,
602
- onMouseEnter: handleMouseEnter,
603
- onMouseLeave: handleMouseLeave
604
- }
605
- ) }),
606
- error && /* @__PURE__ */ jsx("div", { className: cx$2("error-msg-box"), children: /* @__PURE__ */ jsx("span", { className: cx$2("error-msg"), children: errorMsg || "\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694." }) })
607
- ] });
608
- };
609
-
610
- const useFormSignature = ({ onSubmit, onError, onClear, minLength = 100 } = {}) => {
611
- const signatureRef = useRef(null);
612
- const [signatureData, setSignatureData] = useState(null);
613
- const [isError, setIsError] = useState(false);
614
- const [errorMessage, setErrorMessage] = useState("");
615
- const [isLoadedFromRecent, setIsLoadedFromRecent] = useState(false);
616
- const handleSignatureSubmit = useCallback(
617
- (data) => {
618
- setSignatureData(data);
619
- setIsError(false);
620
- setErrorMessage("");
621
- onSubmit?.(data);
622
- },
623
- [onSubmit]
624
- );
625
- const submit = useCallback(() => {
626
- try {
627
- if (!signatureRef.current?.isValid()) {
628
- setIsError(true);
629
- setErrorMessage("\uC11C\uBA85\uC744 \uC785\uB825\uD574 \uC8FC\uC138\uC694.");
630
- return false;
631
- }
632
- const data = signatureRef.current.getSignatureData();
633
- if (!data) {
634
- setIsError(true);
635
- setErrorMessage("\uC11C\uBA85 \uB370\uC774\uD130\uB97C \uAC00\uC838\uC62C \uC218 \uC5C6\uC2B5\uB2C8\uB2E4.");
636
- return false;
637
- }
638
- if (!isLoadedFromRecent && data.totalLineLength < minLength) {
639
- setIsError(true);
640
- setErrorMessage(`\uC11C\uBA85\uC744 \uB2E4\uC2DC \uD655\uC778\uD574 \uC8FC\uC138\uC694.`);
641
- return false;
642
- }
643
- setIsError(false);
644
- setErrorMessage("");
645
- signatureRef.current.submit();
646
- return true;
647
- } catch (error) {
648
- const errorObj = error instanceof Error ? error : new Error("\uC54C \uC218 \uC5C6\uB294 \uC624\uB958\uAC00 \uBC1C\uC0DD\uD588\uC2B5\uB2C8\uB2E4.");
649
- setIsError(true);
650
- setErrorMessage(errorObj.message);
651
- onError?.(errorObj);
652
- return false;
653
- }
654
- }, [onError, minLength, isLoadedFromRecent]);
655
- const clear = useCallback(() => {
656
- signatureRef.current?.clear();
657
- setSignatureData(null);
658
- setIsError(false);
659
- setErrorMessage("");
660
- setIsLoadedFromRecent(false);
661
- onClear?.();
662
- }, [onClear]);
663
- const loadRecentSignature = useCallback(async () => {
664
- setIsLoadedFromRecent(true);
665
- setIsError(false);
666
- setErrorMessage("");
667
- return "";
668
- }, []);
669
- const isValid = useCallback(() => {
670
- const basicValid = signatureRef.current?.isValid() ?? false;
671
- if (!basicValid) {
672
- return false;
673
- }
674
- const data = signatureRef.current?.getSignatureData();
675
- if (!data) {
676
- return false;
677
- }
678
- if (isLoadedFromRecent) {
679
- return true;
680
- }
681
- return data.totalLineLength >= minLength;
682
- }, [minLength, isLoadedFromRecent]);
683
- const getSignatureData = useCallback(() => {
684
- return signatureRef.current?.getSignatureData() ?? null;
685
- }, []);
686
- const handleSignatureChange = useCallback(() => {
687
- setIsError(false);
688
- setErrorMessage("");
689
- }, []);
690
- const handleReset = useCallback(() => {
691
- setSignatureData(null);
692
- setIsError(false);
693
- setErrorMessage("");
694
- setIsLoadedFromRecent(false);
695
- onClear?.();
696
- }, [onClear]);
697
- const getErrorMessage = useCallback(() => {
698
- return errorMessage;
699
- }, [errorMessage]);
700
- return {
701
- // ref
702
- signatureRef,
703
- // state
704
- signatureData,
705
- isError,
706
- isLoadedFromRecent,
707
- // 최근 서명 불러옴 여부 노출
708
- // actions
709
- submit,
710
- clear,
711
- loadRecentSignature,
712
- handleSignatureChange,
713
- // 일반적인 서명 변경
714
- handleReset,
715
- // 다시 작성
716
- handleSignatureSubmit,
717
- // utilities
718
- isValid,
719
- getSignatureData,
720
- getErrorMessage
721
- };
722
- };
723
-
724
- const cx$1 = classNames.bind(styles$2);
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$3);
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 usePaint(paintProps = {}) {
570
+ function useCanvasPaint(paintProps = {}) {
997
571
  const { pen = { strokeWeight: 5, strokeColor: "black" }, onChange, onStart, onEnd } = paintProps;
998
- const canvasRef = useRef(null);
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
- const canvas = canvasRef.current;
1005
- if (!canvas) {
578
+ if (!canvasRefState) {
1006
579
  return;
1007
580
  }
1008
- const dataUrl = canvas.toDataURL();
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 (!canvas || !context || !history[index]) {
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, canvas.width, canvas.height);
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
- const canvas = canvasRef.current;
1038
- if (!canvas) {
609
+ if (!canvasRefState) {
1039
610
  return;
1040
611
  }
1041
- const context = canvas.getContext("2d", { willReadFrequently: true });
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 = canvas.toDataURL();
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 } = event;
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 { offsetX, offsetY } = event;
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
- const canvas = canvasRef.current;
1102
- if (!canvas) {
703
+ if (!canvasRefState) {
1103
704
  return;
1104
705
  }
1105
- canvas.addEventListener("mousedown", startPainting);
1106
- canvas.addEventListener("mouseup", stopPainting);
1107
- canvas.addEventListener("mousemove", paint);
1108
- canvas.addEventListener("mouseleave", stopPainting);
1109
- canvas.addEventListener("mouseenter", handleMouseEnter);
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
- canvas.removeEventListener("mousedown", startPainting);
1112
- canvas.removeEventListener("mouseup", stopPainting);
1113
- canvas.removeEventListener("mousemove", paint);
1114
- canvas.removeEventListener("mouseleave", stopPainting);
1115
- canvas.removeEventListener("mouseenter", handleMouseEnter);
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, handleMouseEnter]);
724
+ }, [startPainting, stopPainting, paint, canvasRefState]);
1118
725
  const clear = () => {
1119
- const canvas = canvasRef.current;
1120
726
  const context = contextRef.current;
1121
- if (canvas && context) {
1122
- context.clearRect(0, 0, canvas.width, canvas.height);
1123
- const initialDataUrl = canvas.toDataURL();
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 (!canvas || !context) {
751
+ if (!canvasRefState || !context) {
1147
752
  return;
1148
753
  }
1149
754
  const img = new Image();
1150
755
  img.onload = () => {
1151
- context.clearRect(0, 0, canvas.width, canvas.height);
1152
- context.drawImage(img, 0, 0, canvas.width, canvas.height);
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 = canvasRef.current;
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 = canvasRef.current;
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
- canvasRef,
1207
- getBase64String
830
+ getBase64String,
831
+ isCanvasBlank,
832
+ setCanvasRefState,
833
+ canvasRefState
1208
834
  };
1209
835
  }
1210
836
 
1211
- export { Attachment, DataTable, FormDatePicker, FormSegmentGroup, FormSignature, FormTextField, StepIndicator, resize, useCamera, useFormSignature, usePaint, useTable };
837
+ const testSignatureBase64Data = "";
838
+
839
+ export { Attachment, DataTable, FormDatePicker, FormSegmentGroup, FormTextField, StepIndicator, resize, testSignatureBase64Data, useCamera, useCanvasPaint, useTable };
1212
840
  //# sourceMappingURL=index.esm.js.map