sales-frontend-components 0.0.58 → 0.0.59

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