@tipp/ui-quill-editor 1.0.1

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.
Files changed (77) hide show
  1. package/README.md +3 -0
  2. package/dist/chunk-5WZL7EM7.js +272 -0
  3. package/dist/chunk-5WZL7EM7.js.map +1 -0
  4. package/dist/chunk-63UMKKIY.js +254 -0
  5. package/dist/chunk-63UMKKIY.js.map +1 -0
  6. package/dist/chunk-6Z3DIHNP.js +253 -0
  7. package/dist/chunk-6Z3DIHNP.js.map +1 -0
  8. package/dist/chunk-ACJWK26B.js +272 -0
  9. package/dist/chunk-ACJWK26B.js.map +1 -0
  10. package/dist/chunk-AEDRFU3N.js +273 -0
  11. package/dist/chunk-AEDRFU3N.js.map +1 -0
  12. package/dist/chunk-CCOIRSYJ.js +258 -0
  13. package/dist/chunk-CCOIRSYJ.js.map +1 -0
  14. package/dist/chunk-EQCXRM42.js +253 -0
  15. package/dist/chunk-EQCXRM42.js.map +1 -0
  16. package/dist/chunk-FFENCIRE.js +254 -0
  17. package/dist/chunk-FFENCIRE.js.map +1 -0
  18. package/dist/chunk-HZN6OVD6.js +253 -0
  19. package/dist/chunk-HZN6OVD6.js.map +1 -0
  20. package/dist/chunk-ICAKLBT5.js +253 -0
  21. package/dist/chunk-ICAKLBT5.js.map +1 -0
  22. package/dist/chunk-IIBNGFLN.mjs +273 -0
  23. package/dist/chunk-IIBNGFLN.mjs.map +1 -0
  24. package/dist/chunk-K4Q6PRQ3.js +272 -0
  25. package/dist/chunk-K4Q6PRQ3.js.map +1 -0
  26. package/dist/chunk-KBRYMAW4.js +254 -0
  27. package/dist/chunk-KBRYMAW4.js.map +1 -0
  28. package/dist/chunk-KYAHTSA6.js +253 -0
  29. package/dist/chunk-KYAHTSA6.js.map +1 -0
  30. package/dist/chunk-NQOX4TPA.js +258 -0
  31. package/dist/chunk-NQOX4TPA.js.map +1 -0
  32. package/dist/chunk-OLRR6W3N.js +272 -0
  33. package/dist/chunk-OLRR6W3N.js.map +1 -0
  34. package/dist/chunk-PSI2XOKW.js +272 -0
  35. package/dist/chunk-PSI2XOKW.js.map +1 -0
  36. package/dist/chunk-PWBZK7KO.js +272 -0
  37. package/dist/chunk-PWBZK7KO.js.map +1 -0
  38. package/dist/chunk-QAD3RXYR.js +253 -0
  39. package/dist/chunk-QAD3RXYR.js.map +1 -0
  40. package/dist/chunk-QGCFWTOT.js +253 -0
  41. package/dist/chunk-QGCFWTOT.js.map +1 -0
  42. package/dist/chunk-QVPYAKUW.js +253 -0
  43. package/dist/chunk-QVPYAKUW.js.map +1 -0
  44. package/dist/chunk-RC5DDS52.js +272 -0
  45. package/dist/chunk-RC5DDS52.js.map +1 -0
  46. package/dist/chunk-VWJC7GOO.js +253 -0
  47. package/dist/chunk-VWJC7GOO.js.map +1 -0
  48. package/dist/chunk-WQQTLSPU.js +253 -0
  49. package/dist/chunk-WQQTLSPU.js.map +1 -0
  50. package/dist/chunk-X5ODDFPU.js +272 -0
  51. package/dist/chunk-X5ODDFPU.js.map +1 -0
  52. package/dist/chunk-XFOLCHPG.js +257 -0
  53. package/dist/chunk-XFOLCHPG.js.map +1 -0
  54. package/dist/chunk-XO4BXXHE.js +258 -0
  55. package/dist/chunk-XO4BXXHE.js.map +1 -0
  56. package/dist/editor.cjs +272 -0
  57. package/dist/editor.cjs.map +1 -0
  58. package/dist/editor.d.cts +30 -0
  59. package/dist/editor.d.mts +30 -0
  60. package/dist/editor.d.ts +30 -0
  61. package/dist/editor.js +7 -0
  62. package/dist/editor.js.map +1 -0
  63. package/dist/editor.mjs +7 -0
  64. package/dist/editor.mjs.map +1 -0
  65. package/dist/index.cjs +274 -0
  66. package/dist/index.cjs.map +1 -0
  67. package/dist/index.css +971 -0
  68. package/dist/index.d.cts +3 -0
  69. package/dist/index.d.mts +3 -0
  70. package/dist/index.d.ts +3 -0
  71. package/dist/index.js +7 -0
  72. package/dist/index.js.map +1 -0
  73. package/dist/index.mjs +7 -0
  74. package/dist/index.mjs.map +1 -0
  75. package/package.json +64 -0
  76. package/src/editor.tsx +236 -0
  77. package/src/index.tsx +1 -0
package/dist/editor.js ADDED
@@ -0,0 +1,7 @@
1
+ import {
2
+ Editor
3
+ } from "./chunk-ICAKLBT5.js";
4
+ export {
5
+ Editor
6
+ };
7
+ //# sourceMappingURL=editor.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
@@ -0,0 +1,7 @@
1
+ import {
2
+ Editor
3
+ } from "./chunk-IIBNGFLN.mjs";
4
+ export {
5
+ Editor
6
+ };
7
+ //# sourceMappingURL=editor.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":[],"sourcesContent":[],"mappings":"","names":[]}
package/dist/index.cjs ADDED
@@ -0,0 +1,274 @@
1
+ "use strict";
2
+ var __create = Object.create;
3
+ var __defProp = Object.defineProperty;
4
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
5
+ var __getOwnPropNames = Object.getOwnPropertyNames;
6
+ var __getOwnPropSymbols = Object.getOwnPropertySymbols;
7
+ var __getProtoOf = Object.getPrototypeOf;
8
+ var __hasOwnProp = Object.prototype.hasOwnProperty;
9
+ var __propIsEnum = Object.prototype.propertyIsEnumerable;
10
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
11
+ var __spreadValues = (a, b) => {
12
+ for (var prop in b || (b = {}))
13
+ if (__hasOwnProp.call(b, prop))
14
+ __defNormalProp(a, prop, b[prop]);
15
+ if (__getOwnPropSymbols)
16
+ for (var prop of __getOwnPropSymbols(b)) {
17
+ if (__propIsEnum.call(b, prop))
18
+ __defNormalProp(a, prop, b[prop]);
19
+ }
20
+ return a;
21
+ };
22
+ var __objRest = (source, exclude) => {
23
+ var target = {};
24
+ for (var prop in source)
25
+ if (__hasOwnProp.call(source, prop) && exclude.indexOf(prop) < 0)
26
+ target[prop] = source[prop];
27
+ if (source != null && __getOwnPropSymbols)
28
+ for (var prop of __getOwnPropSymbols(source)) {
29
+ if (exclude.indexOf(prop) < 0 && __propIsEnum.call(source, prop))
30
+ target[prop] = source[prop];
31
+ }
32
+ return target;
33
+ };
34
+ var __export = (target, all) => {
35
+ for (var name in all)
36
+ __defProp(target, name, { get: all[name], enumerable: true });
37
+ };
38
+ var __copyProps = (to, from, except, desc) => {
39
+ if (from && typeof from === "object" || typeof from === "function") {
40
+ for (let key of __getOwnPropNames(from))
41
+ if (!__hasOwnProp.call(to, key) && key !== except)
42
+ __defProp(to, key, { get: () => from[key], enumerable: !(desc = __getOwnPropDesc(from, key)) || desc.enumerable });
43
+ }
44
+ return to;
45
+ };
46
+ var __toESM = (mod, isNodeMode, target) => (target = mod != null ? __create(__getProtoOf(mod)) : {}, __copyProps(
47
+ // If the importer is in node compatibility mode or this is not an ESM
48
+ // file that has been converted to a CommonJS file using a Babel-
49
+ // compatible transform (i.e. "__esModule" has not been set), then set
50
+ // "default" to the CommonJS "module.exports" for node compatibility.
51
+ isNodeMode || !mod || !mod.__esModule ? __defProp(target, "default", { value: mod, enumerable: true }) : target,
52
+ mod
53
+ ));
54
+ var __toCommonJS = (mod) => __copyProps(__defProp({}, "__esModule", { value: true }), mod);
55
+ var __async = (__this, __arguments, generator) => {
56
+ return new Promise((resolve, reject) => {
57
+ var fulfilled = (value) => {
58
+ try {
59
+ step(generator.next(value));
60
+ } catch (e) {
61
+ reject(e);
62
+ }
63
+ };
64
+ var rejected = (value) => {
65
+ try {
66
+ step(generator.throw(value));
67
+ } catch (e) {
68
+ reject(e);
69
+ }
70
+ };
71
+ var step = (x) => x.done ? resolve(x.value) : Promise.resolve(x.value).then(fulfilled, rejected);
72
+ step((generator = generator.apply(__this, __arguments)).next());
73
+ });
74
+ };
75
+
76
+ // src/index.tsx
77
+ var src_exports = {};
78
+ __export(src_exports, {
79
+ Editor: () => Editor
80
+ });
81
+ module.exports = __toCommonJS(src_exports);
82
+
83
+ // src/editor.tsx
84
+ var import_react = require("react");
85
+ var import_ui = require("@tipp/ui");
86
+ var import_react_quill = __toESM(require("react-quill"), 1);
87
+ var import_jsx_runtime = require("react/jsx-runtime");
88
+ var Editor = (0, import_react.forwardRef)(
89
+ (props, ref) => {
90
+ const _a = props, {
91
+ defaultAttachedFiles,
92
+ defaultTitle,
93
+ defaultValue,
94
+ onClickSave,
95
+ uploadFile,
96
+ deleteFile,
97
+ isLoading,
98
+ SecondaryButton,
99
+ clearEditor
100
+ } = _a, quillProps = __objRest(_a, [
101
+ "defaultAttachedFiles",
102
+ "defaultTitle",
103
+ "defaultValue",
104
+ "onClickSave",
105
+ "uploadFile",
106
+ "deleteFile",
107
+ "isLoading",
108
+ "SecondaryButton",
109
+ "clearEditor"
110
+ ]);
111
+ const defaultRef = (0, import_react.useRef)(null);
112
+ const editorRef = ref || defaultRef;
113
+ const [attachedFiles, setAttachedFiles] = (0, import_react.useState)(
114
+ defaultAttachedFiles || []
115
+ );
116
+ const [fileDeleteLoading, setFileDeleteLoading] = (0, import_react.useState)(/* @__PURE__ */ new Set());
117
+ const [title, setTitle] = (0, import_react.useState)(defaultTitle || "");
118
+ const [content, setContent] = (0, import_react.useState)(defaultValue || "");
119
+ const handleOnChangeContent = (0, import_react.useCallback)((value) => {
120
+ setContent(value);
121
+ }, []);
122
+ const handleButtonClick = (0, import_react.useCallback)(() => {
123
+ let input = document.createElement("input");
124
+ input.type = "file";
125
+ input.onchange = (event) => __async(void 0, null, function* () {
126
+ var _a2;
127
+ const file = (_a2 = event.target.files) == null ? void 0 : _a2[0];
128
+ if (!file) {
129
+ import_ui.toast.error("\uD30C\uC77C\uC744 \uC120\uD0DD\uD574\uC8FC\uC138\uC694.");
130
+ return;
131
+ }
132
+ const fileName = file.name;
133
+ const attachment = yield uploadFile == null ? void 0 : uploadFile(file, `hr-notes/${fileName}`);
134
+ if (attachment) {
135
+ setAttachedFiles((prev) => [...prev, attachment]);
136
+ }
137
+ input = null;
138
+ });
139
+ input.click();
140
+ }, [uploadFile]);
141
+ const handleDeleteFile = (0, import_react.useCallback)(
142
+ (fileUrl) => __async(void 0, null, function* () {
143
+ try {
144
+ setFileDeleteLoading((p) => p.add(fileUrl));
145
+ yield deleteFile == null ? void 0 : deleteFile(fileUrl);
146
+ setAttachedFiles(
147
+ (currentFiles) => currentFiles.filter((item) => item.url !== fileUrl)
148
+ );
149
+ } catch (err) {
150
+ import_ui.toast.error("\uD30C\uC77C \uC0AD\uC81C\uC5D0 \uC2E4\uD328\uD588\uC2B5\uB2C8\uB2E4.");
151
+ } finally {
152
+ setFileDeleteLoading((p) => {
153
+ p.delete(fileUrl);
154
+ return p;
155
+ });
156
+ }
157
+ }),
158
+ [deleteFile]
159
+ );
160
+ const renderAttachedFiles = (0, import_react.useCallback)(() => {
161
+ return /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Box, { width: "100%", children: attachedFiles.map((file) => {
162
+ return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_jsx_runtime.Fragment, { children: [
163
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Separator, { size: "4" }),
164
+ /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(
165
+ import_ui.Flex,
166
+ {
167
+ align: "center",
168
+ justify: "between",
169
+ p: "4",
170
+ width: "100%",
171
+ children: [
172
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Link, { href: file.url, size: "2", children: /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_ui.Flex, { align: "center", gap: "3", children: [
173
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.FileIcon, {}),
174
+ file.fileName
175
+ ] }) }),
176
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(
177
+ import_ui.Button,
178
+ {
179
+ loading: fileDeleteLoading.has(file.url),
180
+ onClick: () => {
181
+ void handleDeleteFile(file.url);
182
+ },
183
+ variant: "ghost",
184
+ children: "\uCCA8\uBD80 \uD30C\uC77C \uC0AD\uC81C"
185
+ }
186
+ )
187
+ ]
188
+ },
189
+ `${file.url}_${file.fileName}`
190
+ )
191
+ ] });
192
+ }) });
193
+ }, [attachedFiles, fileDeleteLoading, handleDeleteFile]);
194
+ const handleOnChangeTitle = (0, import_react.useCallback)((e) => {
195
+ setTitle(e.target.value);
196
+ }, []);
197
+ const clearEditorState = (0, import_react.useCallback)(() => {
198
+ setContent("");
199
+ setAttachedFiles([]);
200
+ setTitle("");
201
+ }, []);
202
+ const handleSaveClick = (0, import_react.useCallback)(() => {
203
+ onClickSave == null ? void 0 : onClickSave({
204
+ title,
205
+ content,
206
+ files: attachedFiles
207
+ });
208
+ }, [onClickSave, title, content, attachedFiles]);
209
+ if (props.clearEditor) {
210
+ props.clearEditor.current = clearEditorState;
211
+ }
212
+ return /* @__PURE__ */ (0, import_jsx_runtime.jsxs)("div", { className: "tipp-ql-wrapper", children: [
213
+ /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(
214
+ import_ui.Grid,
215
+ {
216
+ align: "center",
217
+ columns: "auto auto 1fr",
218
+ gap: "2",
219
+ height: "42px",
220
+ pl: "2",
221
+ pr: "3",
222
+ width: "100%",
223
+ children: [
224
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Box, { pl: "3", pr: "3", children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Typo, { children: "\uC81C\uBAA9" }) }),
225
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Separator, { orientation: "vertical", style: { height: "100%" } }),
226
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(
227
+ import_ui.TextField.Root,
228
+ {
229
+ className: "editor-title-text-field",
230
+ onChange: handleOnChangeTitle,
231
+ placeholder: "\uC81C\uBAA9\uC744 \uC785\uB825\uD574\uC8FC\uC138\uC694",
232
+ value: title
233
+ }
234
+ )
235
+ ]
236
+ }
237
+ ),
238
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(
239
+ import_react_quill.default,
240
+ __spreadValues({
241
+ className: "tipp-ql-editor write-mode",
242
+ onChange: handleOnChangeContent,
243
+ ref: editorRef,
244
+ theme: "snow",
245
+ value: content
246
+ }, quillProps)
247
+ ),
248
+ renderAttachedFiles(),
249
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Separator, { size: "4" }),
250
+ /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_ui.Flex, { align: "center", justify: "between", p: "2", pl: "4", pr: "4", width: "100%", children: [
251
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(
252
+ import_ui.Button,
253
+ {
254
+ color: "gray",
255
+ onClick: handleButtonClick,
256
+ variant: "transparent",
257
+ children: /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Link2Icon, { height: 20, width: 20 })
258
+ }
259
+ ),
260
+ /* @__PURE__ */ (0, import_jsx_runtime.jsxs)(import_ui.Flex, { gap: "2", children: [
261
+ clearEditor ? /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Button, { color: "gray", onClick: clearEditorState, variant: "outline", children: "\uCD08\uAE30\uD654" }) : null,
262
+ SecondaryButton ? SecondaryButton : null,
263
+ /* @__PURE__ */ (0, import_jsx_runtime.jsx)(import_ui.Button, { disabled: isLoading, onClick: handleSaveClick, children: "\uC800\uC7A5" })
264
+ ] })
265
+ ] })
266
+ ] });
267
+ }
268
+ );
269
+ Editor.displayName = "TIPP-Quill-Editor";
270
+ // Annotate the CommonJS export names for ESM import in node:
271
+ 0 && (module.exports = {
272
+ Editor
273
+ });
274
+ //# sourceMappingURL=index.cjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/index.tsx","../src/editor.tsx"],"sourcesContent":["export { Editor } from './editor';\n","import React, { forwardRef, useCallback, useRef, useState } from 'react';\nimport {\n Box,\n Button,\n Flex,\n Grid,\n Link,\n Separator,\n TextField,\n Typo,\n Link2Icon,\n toast,\n FileIcon,\n} from '@tipp/ui';\nimport type { ReactQuillProps } from 'react-quill';\nimport ReactQuill from 'react-quill';\n\nexport interface Attachment {\n fileName: string;\n url: string;\n}\n\nexport interface TippEditorProps extends ReactQuillProps {\n defaultTitle?: string;\n defaultValue?: string;\n defaultAttachedFiles?: Attachment[];\n /** 저장하기 버튼 클릭 시 실행 */\n onClickSave?: (values: {\n title: string;\n content: string;\n files: Attachment[];\n }) => void;\n /** 파일 업로드 버튼 클릭 시 실행 */\n uploadFile?: (\n file: File,\n destination: string\n ) => Promise<Attachment | undefined>;\n deleteFile?: (fileUrl: string) => Promise<void>;\n /** 외부에서 Editor를 빈 상태로 초기화 시켜야 할 때 사용 */\n clearEditor?: React.MutableRefObject<(() => void) | undefined>;\n /** 초기화 버튼말고 다른 버튼 추가시 */\n SecondaryButton?: React.ReactNode;\n /** true인 경우 저장하기 버튼이 비활성 화 됨. 연타 방지 */\n isLoading?: boolean;\n}\n\nexport const Editor = forwardRef<ReactQuill, TippEditorProps>(\n (props, ref): React.ReactNode => {\n const {\n defaultAttachedFiles,\n defaultTitle,\n defaultValue,\n onClickSave,\n uploadFile,\n deleteFile,\n isLoading,\n SecondaryButton,\n clearEditor,\n ...quillProps\n } = props;\n const defaultRef = useRef<ReactQuill>(null);\n const editorRef = ref || defaultRef;\n const [attachedFiles, setAttachedFiles] = useState<Attachment[]>(\n defaultAttachedFiles || []\n );\n const [fileDeleteLoading, setFileDeleteLoading] = useState(new Set());\n\n const [title, setTitle] = useState(defaultTitle || '');\n const [content, setContent] = useState(defaultValue || '');\n\n const handleOnChangeContent = useCallback((value: string) => {\n setContent(value);\n }, []);\n\n const handleButtonClick = useCallback(() => {\n let input: HTMLInputElement | null = document.createElement('input');\n input.type = 'file';\n input.onchange = async (event) => {\n const file = (event.target as HTMLInputElement).files?.[0];\n if (!file) {\n // console.log('DEBUG: no file');\n toast.error('파일을 선택해주세요.');\n return;\n }\n\n const fileName = file.name;\n const attachment = await uploadFile?.(file, `hr-notes/${fileName}`);\n if (attachment) {\n setAttachedFiles((prev) => [...prev, attachment]);\n }\n input = null;\n };\n input.click();\n }, [uploadFile]);\n\n const handleDeleteFile = useCallback(\n async (fileUrl: string) => {\n try {\n setFileDeleteLoading((p) => p.add(fileUrl));\n await deleteFile?.(fileUrl);\n setAttachedFiles((currentFiles) =>\n currentFiles.filter((item) => item.url !== fileUrl)\n );\n } catch (err) {\n toast.error('파일 삭제에 실패했습니다.');\n } finally {\n setFileDeleteLoading((p) => {\n p.delete(fileUrl);\n return p;\n });\n }\n },\n [deleteFile]\n );\n\n const renderAttachedFiles = useCallback(() => {\n return (\n <Box width=\"100%\">\n {attachedFiles.map((file) => {\n return (\n <>\n <Separator size=\"4\" />\n <Flex\n align=\"center\"\n justify=\"between\"\n key={`${file.url}_${file.fileName}`}\n p=\"4\"\n width=\"100%\"\n >\n <Link href={file.url} size=\"2\">\n <Flex align=\"center\" gap=\"3\">\n <FileIcon />\n {file.fileName}\n </Flex>\n </Link>\n <Button\n loading={fileDeleteLoading.has(file.url)}\n onClick={() => {\n void handleDeleteFile(file.url);\n }}\n variant=\"ghost\"\n >\n 첨부 파일 삭제\n </Button>\n </Flex>\n </>\n );\n })}\n </Box>\n );\n }, [attachedFiles, fileDeleteLoading, handleDeleteFile]);\n\n const handleOnChangeTitle = useCallback<\n React.ChangeEventHandler<HTMLInputElement | HTMLTextAreaElement>\n >((e) => {\n setTitle(e.target.value);\n }, []);\n\n const clearEditorState = useCallback(() => {\n setContent('');\n setAttachedFiles([]);\n setTitle('');\n }, []);\n\n const handleSaveClick = useCallback(() => {\n onClickSave?.({\n title,\n content,\n files: attachedFiles,\n });\n }, [onClickSave, title, content, attachedFiles]);\n\n if (props.clearEditor) {\n props.clearEditor.current = clearEditorState;\n }\n\n return (\n <div className=\"tipp-ql-wrapper\">\n {/* 제목 입력창 */}\n <Grid\n align=\"center\"\n columns=\"auto auto 1fr\"\n gap=\"2\"\n height=\"42px\"\n pl=\"2\"\n pr=\"3\"\n width=\"100%\"\n >\n <Box pl=\"3\" pr=\"3\">\n <Typo>제목</Typo>\n </Box>\n <Separator orientation=\"vertical\" style={{ height: '100%' }} />\n <TextField.Root\n className=\"editor-title-text-field\"\n onChange={handleOnChangeTitle}\n placeholder=\"제목을 입력해주세요\"\n value={title}\n />\n </Grid>\n <ReactQuill\n className=\"tipp-ql-editor write-mode\"\n onChange={handleOnChangeContent}\n ref={editorRef}\n theme=\"snow\"\n value={content}\n {...quillProps}\n />\n {renderAttachedFiles()}\n <Separator size=\"4\" />\n <Flex align=\"center\" justify=\"between\" p=\"2\" pl=\"4\" pr=\"4\" width=\"100%\">\n <Button\n color=\"gray\"\n onClick={handleButtonClick}\n variant=\"transparent\"\n >\n <Link2Icon height={20} width={20} />\n </Button>\n\n <Flex gap=\"2\">\n {clearEditor ? (\n <Button color=\"gray\" onClick={clearEditorState} variant=\"outline\">\n 초기화\n </Button>\n ) : null}\n {SecondaryButton ? SecondaryButton : null}\n <Button disabled={isLoading} onClick={handleSaveClick}>\n 저장\n </Button>\n </Flex>\n </Flex>\n </div>\n );\n }\n);\n\nEditor.displayName = 'TIPP-Quill-Editor';\n"],"mappings":"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","names":["_a","ReactQuill"]}