sales-frontend-components 0.0.58 → 0.0.60

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,9 +1,13 @@
1
1
  import { jsx, jsxs } from 'react/jsx-runtime';
2
2
  import { useController } from 'react-hook-form';
3
- import { CheckboxButton, Checkbox, DatePicker, DateRangePicker, SegmentGroup, FormField, Icon } from 'sales-frontend-design-system';
4
- import React, { useState, useEffect, useRef, useCallback } from 'react';
3
+ import { CheckboxButton, Checkbox, DatePicker, DateRangePicker, SegmentGroup, FormField, Icon, Button, Modal, List, ListItem, Segment, Select, Accordion } from 'sales-frontend-design-system';
4
+ import React, { useState, useEffect, useRef, useCallback, useMemo } from 'react';
5
5
  import styles from './step-indicator/step-indicator.module.scss';
6
6
  import styles$1 from './camera/camera.module.scss';
7
+ import styles$2 from './debug-tool/debug-tool.module.scss';
8
+ import { isClient } from 'sales-frontend-utils';
9
+ import { useDebugStore } from 'sales-frontend-api';
10
+ import styles$3 from './debug-tool/features/network-log/network-log.module.scss';
7
11
 
8
12
  const FormCheckboxButton = ({
9
13
  name,
@@ -791,5 +795,726 @@ function useCanvasPaint(paintProps = {}) {
791
795
 
792
796
  const testSignatureBase64Data = "";
793
797
 
794
- export { Attachment, FormCheckbox, FormCheckboxButton, FormDatePicker, FormDateRangePicker, FormSegmentGroup, FormTextField, StepIndicator, resize, testSignatureBase64Data, useCamera, useCanvasPaint };
798
+ const FloatingButton = ({ onClick }) => {
799
+ const [position, setPosition] = useState({ x: 0, y: 0 });
800
+ const [isDragging, setIsDragging] = useState(false);
801
+ const dragHappened = useRef(false);
802
+ const dragStartPos = useRef({ x: 0, y: 0 });
803
+ const elementStartPos = useRef({ x: 0, y: 0 });
804
+ const handleMouseDown = (e) => {
805
+ if (e.button !== 0) {
806
+ return;
807
+ }
808
+ dragHappened.current = false;
809
+ setIsDragging(true);
810
+ dragStartPos.current = { x: e.clientX, y: e.clientY };
811
+ elementStartPos.current = { ...position };
812
+ const handleMouseMove = (event) => {
813
+ const dx = event.clientX - dragStartPos.current.x;
814
+ const dy = event.clientY - dragStartPos.current.y;
815
+ if (!dragHappened.current && (Math.abs(dx) > 5 || Math.abs(dy) > 5)) {
816
+ dragHappened.current = true;
817
+ }
818
+ setPosition({
819
+ x: elementStartPos.current.x + dx,
820
+ y: elementStartPos.current.y + dy
821
+ });
822
+ };
823
+ const handleMouseUp = () => {
824
+ window.removeEventListener("mousemove", handleMouseMove);
825
+ window.removeEventListener("mouseup", handleMouseUp);
826
+ setIsDragging(false);
827
+ };
828
+ window.addEventListener("mousemove", handleMouseMove);
829
+ window.addEventListener("mouseup", handleMouseUp);
830
+ };
831
+ const handleClick = (e) => {
832
+ if (dragHappened.current) {
833
+ e.preventDefault();
834
+ e.stopPropagation();
835
+ return;
836
+ }
837
+ onClick();
838
+ };
839
+ return /* @__PURE__ */ jsx(
840
+ Button,
841
+ {
842
+ variant: "secondary",
843
+ appearance: "filled",
844
+ size: "xsmall",
845
+ className: styles$2["debug-floating-button"],
846
+ style: {
847
+ transform: `translate(${position.x}px, ${position.y}px)`,
848
+ cursor: isDragging ? "grabbing" : "grab"
849
+ },
850
+ onMouseDown: handleMouseDown,
851
+ onClick: handleClick,
852
+ children: "DEBUG"
853
+ }
854
+ );
855
+ };
856
+
857
+ const MenuPanel = ({ menuItems, onMenuItemClick, onClose }) => {
858
+ return /* @__PURE__ */ jsxs(Modal.Root, { isOpen: true, onClose, modalSize: "small", children: [
859
+ /* @__PURE__ */ jsx(Modal.Overlay, {}),
860
+ /* @__PURE__ */ jsxs(Modal.Content, { children: [
861
+ /* @__PURE__ */ jsx(Modal.Header, { headerTitle: "Debug Menu", showCloseButton: true }),
862
+ /* @__PURE__ */ jsx(Modal.Body, { children: /* @__PURE__ */ jsx(List, { selectable: true, children: menuItems.map((item) => /* @__PURE__ */ jsx(ListItem, { onClick: () => onMenuItemClick(item.component), children: item.label }, item.id)) }) })
863
+ ] })
864
+ ] });
865
+ };
866
+
867
+ let globalLogs = [];
868
+ let listeners = [];
869
+ const originalConsoleMethods = {};
870
+ let isConsoleOverridden = false;
871
+ const addLog = (type, ...args) => {
872
+ const newLog = {
873
+ id: Date.now() + Math.random(),
874
+ type,
875
+ timestamp: (/* @__PURE__ */ new Date()).toLocaleTimeString(),
876
+ message: args
877
+ };
878
+ globalLogs = [...globalLogs, newLog];
879
+ listeners.forEach((listener) => listener([...globalLogs]));
880
+ };
881
+ const initializeConsoleLogOverride = () => {
882
+ if (!isClient() || isConsoleOverridden) {
883
+ return;
884
+ }
885
+ const logTypes = ["log", "info", "warn", "error", "debug"];
886
+ logTypes.forEach((type) => {
887
+ originalConsoleMethods[type] = console[type];
888
+ console[type] = (...args) => {
889
+ if (originalConsoleMethods[type]) {
890
+ originalConsoleMethods[type]?.apply(console, args);
891
+ }
892
+ addLog(type, ...args);
893
+ };
894
+ });
895
+ isConsoleOverridden = true;
896
+ };
897
+ const restoreConsoleLog = () => {
898
+ if (!isClient() || !isConsoleOverridden) {
899
+ return;
900
+ }
901
+ const logTypes = ["log", "info", "warn", "error", "debug"];
902
+ logTypes.forEach((type) => {
903
+ if (originalConsoleMethods[type]) {
904
+ console[type] = originalConsoleMethods[type];
905
+ }
906
+ });
907
+ isConsoleOverridden = false;
908
+ };
909
+ const useConsoleLog = () => {
910
+ const [logs, setLogs] = useState(globalLogs);
911
+ useEffect(() => {
912
+ listeners.push(setLogs);
913
+ return () => {
914
+ listeners = listeners.filter((l) => l !== setLogs);
915
+ };
916
+ }, []);
917
+ const clearLogs = useCallback(() => {
918
+ globalLogs = [];
919
+ listeners.forEach((listener) => listener([]));
920
+ }, []);
921
+ return { logs, clearLogs };
922
+ };
923
+
924
+ const safeStringify$1 = (obj) => {
925
+ try {
926
+ return JSON.stringify(obj, null, 2);
927
+ } catch (e) {
928
+ return "[Unserializable Object]";
929
+ }
930
+ };
931
+ const LogMessage = ({ entry }) => {
932
+ const messageParts = entry.message.map((part) => {
933
+ if (typeof part === "object" && part !== null) {
934
+ return safeStringify$1(part);
935
+ }
936
+ return String(part);
937
+ });
938
+ return /* @__PURE__ */ jsx("pre", { className: `${styles$2["log-message"]} ${styles$2[`log-${entry.type}`]}`, children: messageParts.join(" ") });
939
+ };
940
+ const ConsoleLogPanel = ({ onClose }) => {
941
+ const { logs, clearLogs } = useConsoleLog();
942
+ const [filter, setFilter] = useState("all");
943
+ const logContainerRef = useRef(null);
944
+ const filteredLogs = useMemo(() => {
945
+ if (filter === "all") {
946
+ return logs;
947
+ }
948
+ return logs.filter((log) => log.type === filter);
949
+ }, [logs, filter]);
950
+ useEffect(() => {
951
+ if (logContainerRef.current) {
952
+ logContainerRef.current.scrollTop = logContainerRef.current.scrollHeight;
953
+ }
954
+ }, [filteredLogs]);
955
+ return /* @__PURE__ */ jsx(Modal.Root, { isOpen: true, onClose, modalSize: "full-screen", children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
956
+ /* @__PURE__ */ jsx(Modal.Header, { headerTitle: "Console Logs", showCloseButton: true }),
957
+ /* @__PURE__ */ jsxs(Modal.Body, { children: [
958
+ /* @__PURE__ */ jsxs("div", { className: styles$2["panel-toolbar"], children: [
959
+ /* @__PURE__ */ jsxs(
960
+ Segment.Root,
961
+ {
962
+ size: "xsmall",
963
+ appearance: "outline",
964
+ align: "left",
965
+ defaultValue: filter,
966
+ onValueChange: (value) => setFilter(value),
967
+ children: [
968
+ /* @__PURE__ */ jsx(Segment.Item, { value: "all", children: "All" }),
969
+ /* @__PURE__ */ jsx(Segment.Item, { value: "log", children: "Log" }),
970
+ /* @__PURE__ */ jsx(Segment.Item, { value: "warn", children: "Warn" }),
971
+ /* @__PURE__ */ jsx(Segment.Item, { value: "error", children: "Error" })
972
+ ]
973
+ }
974
+ ),
975
+ /* @__PURE__ */ jsx(
976
+ Button,
977
+ {
978
+ variant: "neutral",
979
+ appearance: "outline",
980
+ size: "xsmall",
981
+ onClick: clearLogs,
982
+ style: { marginLeft: "auto" },
983
+ children: "Clear"
984
+ }
985
+ )
986
+ ] }),
987
+ /* @__PURE__ */ jsx("div", { className: styles$2["log-list-container"], ref: logContainerRef, children: filteredLogs.map((log) => /* @__PURE__ */ jsxs("div", { className: styles$2["log-item"], children: [
988
+ /* @__PURE__ */ jsx("span", { className: styles$2["log-timestamp"], children: log.timestamp }),
989
+ /* @__PURE__ */ jsx(LogMessage, { entry: log })
990
+ ] }, log.id)) })
991
+ ] })
992
+ ] }) });
993
+ };
994
+
995
+ const LoginPanel = ({ onClose, onLogin }) => {
996
+ const [selectedTypeId, setSelectedTypeId] = useState(loginTypes[0]?.id || "");
997
+ const [formData, setFormData] = useState({});
998
+ const selectedType = useMemo(() => loginTypes.find((t) => t.id === selectedTypeId), [selectedTypeId]);
999
+ const handleInputChange = (e) => {
1000
+ const { name, value } = e.target;
1001
+ setFormData((prev) => ({ ...prev, [name]: value }));
1002
+ };
1003
+ const handleSubmit = (e) => {
1004
+ e.preventDefault();
1005
+ if (selectedType) {
1006
+ onLogin(selectedType.id, formData);
1007
+ onClose();
1008
+ }
1009
+ };
1010
+ useEffect(() => {
1011
+ setFormData({});
1012
+ }, [selectedType]);
1013
+ return /* @__PURE__ */ jsx(Modal.Root, { isOpen: true, onClose, modalSize: "full-screen", children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
1014
+ /* @__PURE__ */ jsx(Modal.Header, { headerTitle: "\uAC04\uD3B8 \uB85C\uADF8\uC778", showCloseButton: true }),
1015
+ /* @__PURE__ */ jsx(Modal.Body, { children: /* @__PURE__ */ jsxs("form", { className: styles$2["login-form-container"], onSubmit: handleSubmit, children: [
1016
+ /* @__PURE__ */ jsxs(FormField.FieldControl, { children: [
1017
+ /* @__PURE__ */ jsx(FormField.Label, { htmlFor: "login-type-select", children: "\uB85C\uADF8\uC778 \uC720\uD615" }),
1018
+ /* @__PURE__ */ jsx(
1019
+ Select,
1020
+ {
1021
+ value: selectedTypeId,
1022
+ onChange: setSelectedTypeId,
1023
+ size: "xsmall",
1024
+ rootProps: { style: { width: "100%" } },
1025
+ children: loginTypes.map((type) => /* @__PURE__ */ jsx(Select.Option, { value: type.id, children: type.label }, type.id))
1026
+ }
1027
+ )
1028
+ ] }),
1029
+ selectedType?.fields.map((field) => /* @__PURE__ */ jsxs(FormField.FieldControl, { children: [
1030
+ /* @__PURE__ */ jsx(FormField.Label, { htmlFor: field.name, children: field.label }),
1031
+ /* @__PURE__ */ jsx(
1032
+ FormField.TextField,
1033
+ {
1034
+ size: "xsmall",
1035
+ type: field.type,
1036
+ id: field.name,
1037
+ name: field.name,
1038
+ placeholder: field.placeholder,
1039
+ value: formData[field.name] || "",
1040
+ onChange: handleInputChange,
1041
+ required: true
1042
+ }
1043
+ )
1044
+ ] }, field.name)),
1045
+ /* @__PURE__ */ jsx("div", { className: styles$2["form-actions"], children: /* @__PURE__ */ jsx(Button, { type: "submit", variant: "primary", appearance: "filled", size: "xsmall", children: "\uB85C\uADF8\uC778" }) })
1046
+ ] }) })
1047
+ ] }) });
1048
+ };
1049
+
1050
+ const SEARCH_SECTIONS = [
1051
+ { value: "all", label: "\uC804\uCCB4" },
1052
+ { value: "url", label: "\uC8FC\uC18C" },
1053
+ { value: "headers", label: "\uD5E4\uB354" },
1054
+ { value: "params", label: "\uD30C\uB77C\uBBF8\uD130" },
1055
+ { value: "data", label: "\uC694\uCCAD" },
1056
+ { value: "response", label: "\uC751\uB2F5" },
1057
+ { value: "error", label: "\uC5D0\uB7EC" }
1058
+ ];
1059
+ const NetworkLog = ({ onClose }) => {
1060
+ const { requests, responses, errors, clear } = useDebugStore();
1061
+ const [searchTerm, setSearchTerm] = useState("");
1062
+ const [searchSection, setSearchSection] = useState(SEARCH_SECTIONS[0]?.value);
1063
+ const allLogs = useMemo(() => {
1064
+ const logs = [];
1065
+ requests.forEach((req) => logs.push({ ...req, type: "request" }));
1066
+ responses.forEach((res) => logs.push({ ...res, type: "response" }));
1067
+ errors.forEach((err) => logs.push({ ...err, type: "error" }));
1068
+ logs.sort((a, b) => {
1069
+ const aTime = "startTime" in a ? a.startTime : 0;
1070
+ const bTime = "startTime" in b ? b.startTime : 0;
1071
+ return bTime - aTime;
1072
+ });
1073
+ return logs;
1074
+ }, [requests, responses, errors]);
1075
+ const filteredLogs = useMemo(() => {
1076
+ if (!searchTerm) {
1077
+ return allLogs;
1078
+ }
1079
+ const lowercasedTerm = searchTerm.toLowerCase();
1080
+ return allLogs.filter((log) => {
1081
+ switch (searchSection) {
1082
+ case "url":
1083
+ return log.url?.toLowerCase().includes(lowercasedTerm);
1084
+ case "headers":
1085
+ return "headers" in log && log.headers && JSON.stringify(log.headers)?.toLowerCase().includes(lowercasedTerm);
1086
+ case "params":
1087
+ return "params" in log && log.params && JSON.stringify(log.params)?.toLowerCase().includes(lowercasedTerm);
1088
+ case "data":
1089
+ return "data" in log && log.data && log.type === "request" && JSON.stringify(log.data)?.toLowerCase().includes(lowercasedTerm);
1090
+ case "response":
1091
+ return "data" in log && log.data && log.type === "response" && JSON.stringify(log.data)?.toLowerCase().includes(lowercasedTerm);
1092
+ case "error":
1093
+ return log.type === "error" && JSON.stringify(log)?.toLowerCase().includes(lowercasedTerm);
1094
+ case "all":
1095
+ default:
1096
+ return JSON.stringify(log).toLowerCase().includes(lowercasedTerm);
1097
+ }
1098
+ });
1099
+ }, [allLogs, searchTerm, searchSection]);
1100
+ const renderLogSummary = (log) => {
1101
+ if (log.type === "request") {
1102
+ return `[REQ] ${log.method.toUpperCase()} ${log.url}`;
1103
+ }
1104
+ if (log.type === "response" && "status" in log) {
1105
+ return `[RES] ${log.method.toUpperCase()} ${log.url} - ${log.status}`;
1106
+ }
1107
+ if (log.type === "error" && "message" in log) {
1108
+ return `[ERR] ${log.method.toUpperCase()} ${log.url} - ${log.message}`;
1109
+ }
1110
+ return "Unknown Log";
1111
+ };
1112
+ const renderLogContent = (log) => {
1113
+ return /* @__PURE__ */ jsx("pre", { children: JSON.stringify(log, null, 2) });
1114
+ };
1115
+ return /* @__PURE__ */ jsxs(Modal.Root, { isOpen: true, onClose, modalSize: "full-screen", children: [
1116
+ /* @__PURE__ */ jsx(Modal.Overlay, {}),
1117
+ /* @__PURE__ */ jsxs(Modal.Content, { children: [
1118
+ /* @__PURE__ */ jsx(Modal.Header, { headerTitle: "\uB124\uD2B8\uC6CC\uD06C \uB85C\uADF8", showCloseButton: true }),
1119
+ /* @__PURE__ */ jsxs(Modal.Body, { children: [
1120
+ /* @__PURE__ */ jsxs("div", { className: styles$3["search-container"], children: [
1121
+ /* @__PURE__ */ jsx(Select, { value: searchSection, onChange: (value) => setSearchSection(value), size: "xsmall", children: SEARCH_SECTIONS.map((option) => /* @__PURE__ */ jsx(Select.Option, { value: option.value, children: option.label }, option.value)) }),
1122
+ /* @__PURE__ */ jsx(
1123
+ FormField.TextField,
1124
+ {
1125
+ placeholder: "\uAC80\uC0C9...",
1126
+ value: searchTerm,
1127
+ size: "xsmall",
1128
+ onChange: (e) => setSearchTerm(e.target.value)
1129
+ }
1130
+ ),
1131
+ /* @__PURE__ */ jsx(Button, { variant: "neutral", appearance: "outline", size: "xsmall", onClick: clear, children: "clear" })
1132
+ ] }),
1133
+ /* @__PURE__ */ jsx("div", { className: styles$3["log-list"], children: filteredLogs.length === 0 ? /* @__PURE__ */ jsx("p", { children: "\uD45C\uC2DC\uD560 \uB85C\uADF8\uAC00 \uC5C6\uC2B5\uB2C8\uB2E4." }) : /* @__PURE__ */ jsx(Accordion.Root, { children: filteredLogs.map((log, index) => /* @__PURE__ */ jsxs(Accordion.Item, { children: [
1134
+ /* @__PURE__ */ jsx(Accordion.HeaderButton, { children: /* @__PURE__ */ jsx("p", { className: styles$3["log-summary"], children: renderLogSummary(log) }) }),
1135
+ /* @__PURE__ */ jsx(Accordion.Content, { variant: "text", children: /* @__PURE__ */ jsx("div", { className: styles$3["log-content"], children: renderLogContent(log) }) })
1136
+ ] }, index)) }) })
1137
+ ] })
1138
+ ] })
1139
+ ] });
1140
+ };
1141
+
1142
+ const PageNavigationPanel = ({ onClose }) => {
1143
+ const [url, setUrl] = useState("");
1144
+ const handleNavigate = () => {
1145
+ if (url) {
1146
+ self.location.href = url;
1147
+ }
1148
+ };
1149
+ return /* @__PURE__ */ jsx(Modal.Root, { isOpen: true, onClose, modalSize: "full-screen", children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
1150
+ /* @__PURE__ */ jsx(Modal.Header, { headerTitle: "\uD398\uC774\uC9C0 \uC774\uB3D9", showCloseButton: true }),
1151
+ /* @__PURE__ */ jsx(Modal.Body, { children: /* @__PURE__ */ jsxs("div", { className: styles$2["page-navigation-content"], children: [
1152
+ /* @__PURE__ */ jsx(
1153
+ FormField.TextField,
1154
+ {
1155
+ size: "xsmall",
1156
+ type: "text",
1157
+ value: url,
1158
+ onChange: (e) => setUrl(e.target.value),
1159
+ placeholder: "\uC774\uB3D9\uD560 URL\uC744 \uC785\uB825\uD558\uC138\uC694 (\uC608: /main)",
1160
+ rootProps: { style: { flex: 1 } }
1161
+ }
1162
+ ),
1163
+ /* @__PURE__ */ jsx(Button, { variant: "primary", appearance: "filled", size: "xsmall", onClick: handleNavigate, children: "\uC774\uB3D9" })
1164
+ ] }) })
1165
+ ] }) });
1166
+ };
1167
+
1168
+ const safeStringify = (obj) => {
1169
+ try {
1170
+ const replacer = (key, value) => typeof value === "bigint" ? value.toString() : value;
1171
+ return JSON.stringify(obj, replacer, 2);
1172
+ } catch (e) {
1173
+ return "[Unserializable Object]";
1174
+ }
1175
+ };
1176
+ const ResultMessage = ({ log }) => {
1177
+ const messageParts = log.message.map((part) => {
1178
+ if (typeof part === "object" && part !== null) {
1179
+ return safeStringify(part);
1180
+ }
1181
+ return String(part);
1182
+ });
1183
+ const prefix = log.type === "return" ? "\u21A9 " : "";
1184
+ return /* @__PURE__ */ jsxs("pre", { className: `${styles$2["log-message"]} ${styles$2[`log-${log.type}`]}`, children: [
1185
+ prefix,
1186
+ messageParts.join(" ")
1187
+ ] });
1188
+ };
1189
+ const ScriptExecutorPanel = ({ onClose }) => {
1190
+ const [code, setCode] = useState('// \uC5EC\uAE30\uC5D0 \uC2E4\uD589\uD560 \uCF54\uB4DC\uB97C \uC785\uB825\uD558\uC138\uC694\nconsole.log("Hello, World!");');
1191
+ const [results, setResults] = useState([]);
1192
+ const addResult = (type, ...args) => {
1193
+ setResults((prevResults) => [
1194
+ ...prevResults,
1195
+ {
1196
+ id: Date.now() + Math.random(),
1197
+ type,
1198
+ timestamp: (/* @__PURE__ */ new Date()).toLocaleTimeString(),
1199
+ message: args
1200
+ }
1201
+ ]);
1202
+ };
1203
+ const handleExecute = () => {
1204
+ const originalConsole = { ...console };
1205
+ const newConsole = {
1206
+ log: (...args) => {
1207
+ originalConsole.log(...args);
1208
+ addResult("log", ...args);
1209
+ },
1210
+ warn: (...args) => {
1211
+ originalConsole.warn(...args);
1212
+ addResult("warn", ...args);
1213
+ },
1214
+ error: (...args) => {
1215
+ originalConsole.error(...args);
1216
+ addResult("error", ...args);
1217
+ }
1218
+ };
1219
+ try {
1220
+ Object.assign(console, newConsole);
1221
+ const result = new Function(code)();
1222
+ if (result !== void 0) {
1223
+ addResult("return", result);
1224
+ }
1225
+ } catch (e) {
1226
+ addResult("error", e.name, e.message);
1227
+ } finally {
1228
+ Object.assign(console, originalConsole);
1229
+ }
1230
+ };
1231
+ const handleClear = () => {
1232
+ setResults([]);
1233
+ };
1234
+ return /* @__PURE__ */ jsx(Modal.Root, { isOpen: true, onClose, modalSize: "full-screen", children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
1235
+ /* @__PURE__ */ jsx(Modal.Header, { headerTitle: "\uC2A4\uD06C\uB9BD\uD2B8 \uC2E4\uD589", showCloseButton: true }),
1236
+ /* @__PURE__ */ jsx(Modal.Body, { children: /* @__PURE__ */ jsx("div", { className: `${styles$2["debug-feature-panel"]} ${styles$2["script-executor-panel"]}`, children: /* @__PURE__ */ jsxs("div", { className: styles$2["script-executor-content"], children: [
1237
+ /* @__PURE__ */ jsxs("div", { className: styles$2["script-input-section"], children: [
1238
+ /* @__PURE__ */ jsx(
1239
+ "textarea",
1240
+ {
1241
+ className: styles$2["script-textarea"],
1242
+ value: code,
1243
+ onChange: (e) => setCode(e.target.value),
1244
+ spellCheck: "false"
1245
+ }
1246
+ ),
1247
+ /* @__PURE__ */ jsxs("div", { className: styles$2["script-actions"], children: [
1248
+ /* @__PURE__ */ jsx(Button, { variant: "neutral", appearance: "outline", size: "xsmall", onClick: handleExecute, children: "\uC2E4\uD589 (\u25B6)" }),
1249
+ /* @__PURE__ */ jsx(Button, { variant: "neutral", appearance: "outline", size: "xsmall", onClick: handleClear, children: "\uACB0\uACFC \uC9C0\uC6B0\uAE30" })
1250
+ ] })
1251
+ ] }),
1252
+ /* @__PURE__ */ jsxs("div", { className: styles$2["script-output-section"], children: [
1253
+ /* @__PURE__ */ jsx("div", { className: styles$2["panel-header"], children: /* @__PURE__ */ jsx("h3", { children: "\uACB0\uACFC" }) }),
1254
+ /* @__PURE__ */ jsx("div", { className: styles$2["log-list-container"], children: results.map((log) => /* @__PURE__ */ jsxs("div", { className: styles$2["log-item"], children: [
1255
+ /* @__PURE__ */ jsx("span", { className: styles$2["log-timestamp"], children: log.timestamp }),
1256
+ /* @__PURE__ */ jsx(ResultMessage, { log })
1257
+ ] }, log.id)) })
1258
+ ] })
1259
+ ] }) }) })
1260
+ ] }) });
1261
+ };
1262
+
1263
+ const setCookie = (name, value) => {
1264
+ document.cookie = `${name}=${encodeURIComponent(value)};path=/`;
1265
+ };
1266
+ const removeCookie = (name) => {
1267
+ document.cookie = `${name}=;path=/;expires=Thu, 01 Jan 1970 00:00:01 GMT;`;
1268
+ };
1269
+ const useStorage = (storageType) => {
1270
+ const [items, setItems] = useState([]);
1271
+ const loadItems = useCallback(() => {
1272
+ const newItems = [];
1273
+ if (storageType === "localStorage") {
1274
+ for (let i = 0; i < localStorage.length; i++) {
1275
+ const key = localStorage.key(i);
1276
+ if (key) {
1277
+ newItems.push({ key, value: localStorage.getItem(key) || "" });
1278
+ }
1279
+ }
1280
+ } else {
1281
+ const cookies = document.cookie.split(";");
1282
+ cookies.forEach((cookie) => {
1283
+ const parts = cookie.split("=");
1284
+ const key = parts.shift()?.trim();
1285
+ if (key) {
1286
+ newItems.push({ key, value: decodeURIComponent(parts.join("=")) });
1287
+ }
1288
+ });
1289
+ }
1290
+ setItems(newItems);
1291
+ }, [storageType]);
1292
+ useEffect(() => {
1293
+ loadItems();
1294
+ }, [loadItems]);
1295
+ const setItem = useCallback(
1296
+ (key, value) => {
1297
+ if (storageType === "localStorage") {
1298
+ localStorage.setItem(key, value);
1299
+ } else {
1300
+ setCookie(key, value);
1301
+ }
1302
+ loadItems();
1303
+ },
1304
+ [storageType, loadItems]
1305
+ );
1306
+ const removeItem = useCallback(
1307
+ (key) => {
1308
+ if (storageType === "localStorage") {
1309
+ localStorage.removeItem(key);
1310
+ } else {
1311
+ removeCookie(key);
1312
+ }
1313
+ loadItems();
1314
+ },
1315
+ [storageType, loadItems]
1316
+ );
1317
+ return { items, setItem, removeItem, loadItems };
1318
+ };
1319
+
1320
+ const ValueDisplay = ({ value }) => {
1321
+ try {
1322
+ const parsed = JSON.parse(value);
1323
+ if (typeof parsed === "object" && parsed !== null) {
1324
+ return /* @__PURE__ */ jsx("pre", { className: styles$2["log-content"], children: JSON.stringify(parsed, null, 2) });
1325
+ }
1326
+ } catch (e) {
1327
+ }
1328
+ return /* @__PURE__ */ jsx("div", { className: styles$2["log-content"], children: value });
1329
+ };
1330
+ const EditableRow = ({ item, onSave, onRemove }) => {
1331
+ const [isEditing, setIsEditing] = useState(false);
1332
+ const [value, setValue] = useState(item.value);
1333
+ const handleSave = () => {
1334
+ onSave(item.key, value);
1335
+ setIsEditing(false);
1336
+ };
1337
+ const handleEdit = () => {
1338
+ setValue(item.value);
1339
+ setIsEditing(true);
1340
+ };
1341
+ return /* @__PURE__ */ jsx(Accordion.Root, { children: /* @__PURE__ */ jsxs(Accordion.Item, { children: [
1342
+ /* @__PURE__ */ jsxs(Accordion.HeaderButton, { children: [
1343
+ " ",
1344
+ item.key
1345
+ ] }),
1346
+ /* @__PURE__ */ jsxs(Accordion.Content, { variant: "text", style: { flex: 1, display: "flex", flexDirection: "column" }, children: [
1347
+ isEditing ? /* @__PURE__ */ jsx(Button, { variant: "neutral", appearance: "outline", size: "xsmall", onClick: handleSave, children: "Save" }) : /* @__PURE__ */ jsx(Button, { variant: "neutral", appearance: "outline", size: "xsmall", onClick: handleEdit, children: "Edit" }),
1348
+ /* @__PURE__ */ jsx(Button, { variant: "neutral", appearance: "outline", size: "xsmall", onClick: () => onRemove(item.key), children: "Remove" }),
1349
+ isEditing ? /* @__PURE__ */ jsx("textarea", { value, onChange: (e) => setValue(e.target.value) }) : /* @__PURE__ */ jsx(ValueDisplay, { value: item.value })
1350
+ ] })
1351
+ ] }) });
1352
+ };
1353
+ const StoragePanel = ({ onClose, storageType }) => {
1354
+ const { items, setItem, removeItem } = useStorage(storageType);
1355
+ const [filter, setFilter] = useState("");
1356
+ const filteredItems = useMemo(() => {
1357
+ if (!filter) {
1358
+ return items;
1359
+ }
1360
+ return items.filter(
1361
+ (item) => item.key.toLowerCase().includes(filter.toLowerCase()) || item.value.toLowerCase().includes(filter.toLowerCase())
1362
+ );
1363
+ }, [items, filter]);
1364
+ return /* @__PURE__ */ jsx(Modal.Root, { isOpen: true, onClose, modalSize: "full-screen", children: /* @__PURE__ */ jsxs(Modal.Content, { children: [
1365
+ /* @__PURE__ */ jsx(Modal.Header, { headerTitle: storageType === "localStorage" ? "Local Storage" : "Cookies", showCloseButton: true }),
1366
+ /* @__PURE__ */ jsxs(Modal.Body, { children: [
1367
+ /* @__PURE__ */ jsx("div", { className: styles$2["panel-toolbar"], children: /* @__PURE__ */ jsx(
1368
+ FormField.TextField,
1369
+ {
1370
+ placeholder: "Search key or value...",
1371
+ size: "xsmall",
1372
+ value: filter,
1373
+ onChange: (e) => setFilter(e.target.value),
1374
+ rootProps: { style: { width: "100%" } }
1375
+ }
1376
+ ) }),
1377
+ /* @__PURE__ */ jsx("ul", { children: filteredItems.map((item) => /* @__PURE__ */ jsx("li", { children: /* @__PURE__ */ jsx(EditableRow, { item, onSave: setItem, onRemove: removeItem }) }, item.key)) })
1378
+ ] })
1379
+ ] }) });
1380
+ };
1381
+
1382
+ const loginTypes = [
1383
+ {
1384
+ id: "fp",
1385
+ label: "FP \uB85C\uADF8\uC778",
1386
+ fields: [{ name: "employeeId", label: "\uC0AC\uBC88", type: "text", placeholder: "\uC0AC\uBC88\uC744 \uC785\uB825\uD558\uC138\uC694" }]
1387
+ },
1388
+ {
1389
+ id: "customer",
1390
+ label: "\uACE0\uAC1D \uB85C\uADF8\uC778",
1391
+ fields: [
1392
+ { name: "name", label: "\uC774\uB984", type: "text", placeholder: "\uC774\uB984" },
1393
+ { name: "phone", label: "\uC804\uD654\uBC88\uD638", type: "tel", placeholder: "\uC804\uD654\uBC88\uD638" },
1394
+ { name: "residentNumber", label: "\uC8FC\uBBFC\uBC88\uD638", type: "text", placeholder: "\uC8FC\uBBFC\uBC88\uD638 \uC55E 7\uC790\uB9AC" },
1395
+ { name: "userId", label: "\uC544\uC774\uB514", type: "text", placeholder: "\uC544\uC774\uB514" },
1396
+ { name: "password", label: "\uBE44\uBC00\uBC88\uD638", type: "password", placeholder: "\uBE44\uBC00\uBC88\uD638" }
1397
+ ]
1398
+ }
1399
+ ];
1400
+ const menuItems = [
1401
+ {
1402
+ id: "network-log",
1403
+ label: "\uB124\uD2B8\uC6CC\uD06C \uB85C\uADF8",
1404
+ component: NetworkLog,
1405
+ display: true
1406
+ },
1407
+ {
1408
+ id: "console-log",
1409
+ label: "\uCF58\uC194 \uB85C\uADF8 \uD655\uC778",
1410
+ component: ConsoleLogPanel,
1411
+ display: true
1412
+ },
1413
+ {
1414
+ id: "local-storage",
1415
+ label: "\uB85C\uCEEC \uC2A4\uD1A0\uB9AC\uC9C0 \uAD00\uB9AC",
1416
+ // StoragePanel은 재사용 가능한 컴포넌트이므로, props를 통해 어떤 스토리지를 다룰지 지정합니다.
1417
+ component: (props) => /* @__PURE__ */ jsx(StoragePanel, { ...props, storageType: "localStorage" }),
1418
+ display: true
1419
+ },
1420
+ {
1421
+ id: "cookie-management",
1422
+ label: "\uCFE0\uD0A4 \uAD00\uB9AC",
1423
+ component: (props) => /* @__PURE__ */ jsx(StoragePanel, { ...props, storageType: "cookie" }),
1424
+ display: true
1425
+ },
1426
+ {
1427
+ id: "quick-login",
1428
+ label: "\uAC04\uD3B8 \uB85C\uADF8\uC778",
1429
+ component: LoginPanel,
1430
+ display: true
1431
+ // 실제 표시는 DebugTool.tsx에서 onLogin prop 존재 여부에 따라 결정됩니다.
1432
+ },
1433
+ {
1434
+ id: "script-executor",
1435
+ label: "\uC2A4\uD06C\uB9BD\uD2B8 \uC2E4\uD589",
1436
+ component: ScriptExecutorPanel,
1437
+ display: true
1438
+ },
1439
+ {
1440
+ id: "page-navigation",
1441
+ label: "\uD398\uC774\uC9C0 \uC774\uB3D9",
1442
+ component: PageNavigationPanel,
1443
+ display: true
1444
+ }
1445
+ ];
1446
+
1447
+ const useEnvironment = () => {
1448
+ const [envInfo, setEnvInfo] = useState({
1449
+ env: "unknown"
1450
+ });
1451
+ useEffect(() => {
1452
+ const { hostname } = window.location;
1453
+ let currentEnv = "unknown";
1454
+ if (hostname.includes("localhost") || hostname.includes("127.0.0.1")) {
1455
+ currentEnv = "local";
1456
+ } else if (hostname.includes("dev.")) {
1457
+ currentEnv = "dev";
1458
+ } else if (hostname.includes("stg.")) {
1459
+ currentEnv = "stage";
1460
+ } else {
1461
+ currentEnv = "prod";
1462
+ }
1463
+ setEnvInfo({ env: currentEnv });
1464
+ }, []);
1465
+ return envInfo;
1466
+ };
1467
+
1468
+ const DebugTool = ({ onLogin, envOverride, menuItemsOverride }) => {
1469
+ useEffect(() => {
1470
+ initializeConsoleLogOverride();
1471
+ return () => {
1472
+ restoreConsoleLog();
1473
+ };
1474
+ }, []);
1475
+ const environment = useEnvironment();
1476
+ const env = envOverride || environment.env;
1477
+ const menuItems$1 = menuItemsOverride || menuItems;
1478
+ const [isMenuOpen, setMenuOpen] = useState(false);
1479
+ const [activeFeature, setActiveFeature] = useState(null);
1480
+ const isVisible = ["local", "dev", "stage"].includes(env);
1481
+ const handleMenuClick = (itemComponent) => {
1482
+ setActiveFeature(() => itemComponent);
1483
+ setMenuOpen(false);
1484
+ };
1485
+ const handleCloseFeature = () => {
1486
+ setActiveFeature(null);
1487
+ };
1488
+ if (!isVisible) {
1489
+ return null;
1490
+ }
1491
+ const toggleMenu = () => {
1492
+ setMenuOpen((prev) => !prev);
1493
+ if (!isMenuOpen) {
1494
+ setActiveFeature(null);
1495
+ }
1496
+ };
1497
+ const availableMenuItems = menuItems$1.filter(
1498
+ (item) => item.display && // config에서 display가 true인 항목만 필터링합니다.
1499
+ // 'quick-login' 기능은 onLogin prop이 제공된 경우에만 활성화합니다.
1500
+ (item.id !== "quick-login" || item.id === "quick-login" && onLogin)
1501
+ );
1502
+ if (activeFeature) {
1503
+ const FeatureComponent = activeFeature;
1504
+ return /* @__PURE__ */ jsx(FeatureComponent, { onClose: handleCloseFeature, onLogin });
1505
+ }
1506
+ return /* @__PURE__ */ jsxs("div", { className: "debug-tool-container", children: [
1507
+ /* @__PURE__ */ jsx(FloatingButton, { onClick: toggleMenu }),
1508
+ isMenuOpen && /* @__PURE__ */ jsx(
1509
+ MenuPanel,
1510
+ {
1511
+ menuItems: availableMenuItems,
1512
+ onMenuItemClick: handleMenuClick,
1513
+ onClose: () => setMenuOpen(false)
1514
+ }
1515
+ )
1516
+ ] });
1517
+ };
1518
+
1519
+ export { Attachment, DebugTool, FormCheckbox, FormCheckboxButton, FormDatePicker, FormDateRangePicker, FormSegmentGroup, FormTextField, StepIndicator, resize, testSignatureBase64Data, useCamera, useCanvasPaint };
795
1520
  //# sourceMappingURL=index.esm.js.map