๐ค ์๋ก
ํ์ฌ ํ๋ก์ ํธ์์ ์ฌ์ฉํ ์๋ํฐ๊ฐ ํ์ํ๋ค. ์ด์ ์ CK์๋ํฐ๋ฅผ ์ฌ์ฉํด๋ณธ ์ ์ด ์๋๋ฐ, ์ด์ํ๊ฒ๋ ๋ฒ์ ๋ฌธ์ ์ธ์ง CK์๋ํฐ๊ฐ ์๋ํ์ง ์์๊ณ , ์ฐจ์ ์ฑ ์ผ๋ก quill์ ์ฌ์ฉํด๋ดค์ผ๋ ๊ตฌํ ํ ๋น๋๋ฅผ ํด๋ณด๋ import์๋ฌ๊ฐ ๋ฐ์ํด ์ฌ์ฉํ ์ ์์๋ค. ๊ทธ๋์ ์ต์ข ์ ์ผ๋ก๋ toast-ui editor๋ฅผ ์ฌ์ฉํ๊ณ , ์ง์ ํด๋ฐ์ ์์ดํ ์ ์์ ๋กญ๊ฒ ์ปค์คํ ํ ์ ์๋ ๋งํผ ์๋ ๊ธฐ๋ฅ์ด ๋ง์ ๊ณ ์์ ํ๊ธฐ์ ์ด ๊ณผ์ ์ ๊ธ๋ก ๋จ๊ธด๋ค.
๐ฅ ๋ชฉํ
์ด๋ฏธ์ง ์ ์ฅ, youtube/viemo ๋ ธ์ถ(iframe)์ด ๊ฐ๋ฅํ toast-ui editor๋ก ์ปค์คํ ํ๊ธฐ
๐ฉ๐ปโ๐ป ๊ตฌํ
* {{}} ์ด๋ ๊ฒ ๊ฐ์ผ ๋ถ๋ถ์ ์ง์ ๋ฃ์ด์ผ ํ๋ค.
๊ฒฐ๊ณผ

์ ์ฒด ์ฝ๋(์๋ํฐ)
import React, { useEffect, useRef, useState } from 'react';
import styled from '@emotion/styled';
import axios from 'axios';
/* Toast-UI */
import colorSyntax from '@toast-ui/editor-plugin-color-syntax';
import fontSize from 'tui-editor-plugin-font-size';
import '@toast-ui/editor/dist/i18n/ko-kr';
import '@toast-ui/editor-plugin-color-syntax/dist/toastui-editor-plugin-color-syntax.css';
import '@toast-ui/editor/dist/toastui-editor.css';
import 'tui-color-picker/dist/tui-color-picker.css';
import { Editor } from '@toast-ui/react-editor';
import { theme } from '@/styles/theme';
interface CustomEditorProps {
data: string;
setData: React.Dispatch<React.SetStateAction<string>>;
reset?: boolean;
setReset?: React.Dispatch<React.SetStateAction<boolean>>;
}
const CustomEditor: React.FC<CustomEditorProps> = ({ data, setData, reset, setReset }) => {
const editorRef = useRef<Editor>(null);
const [isSet, setIsSet] = useState<boolean>(false);
const toolbarItems = [['heading'], ['bold', 'italic', 'strike'], ['ul', 'ol'], ['image', 'link']];
const handleSave = () => {
let htmlContent = editorRef.current && editorRef.current.getInstance().getHTML();
if (htmlContent) {
setData(htmlContent);
}
};
// ๋ฆฌ์
useEffect(() => {
if (editorRef.current && reset && setReset) {
editorRef.current.getInstance().setMarkdown('');
setReset(false);
}
}, [reset]);
// ์ด๊ธฐ๊ฐ ์ธํ
useEffect(() => {
if (!isSet && editorRef.current) {
editorRef.current.getInstance().setMarkdown(data ? data : '');
setIsSet(true);
}
}, []);
const [load, setLoad] = useState<number>(0);
useEffect(() => {
if (load < 2 && data && data != '<p><br></p>' && isSet && editorRef.current) {
editorRef.current.getInstance().setMarkdown(data);
setLoad((prev) => prev + 1);
}
}, [isSet, data]);
return (
<StEditor>
<Editor
initialEditType="wysiwyg"
initialValue={data}
previewStyle="vertical"
onChange={handleSave}
ref={editorRef}
toolbarItems={toolbarItems}
hideModeSwitch={true}
plugins={[colorSyntax, fontSize]}
placeholder="๋ด์ฉ์ ์
๋ ฅํด์ฃผ์ธ์"
language="ko-KR"
hooks={{
addImageBlobHook: (blob: any, callback: any) => {
const formData = new FormData();
formData.append('upload', blob);
// ์ด๋ฏธ์ง๋ฅผ ์๋ฒ์ ์
๋ก๋ํ๋ ์์ฒญ
axios
.post({{'์์ฒญ URL'}}, formData, {
headers: {
'Content-Type': 'multipart/form-data',
authorization: {{ํ ํฐ}},
ClientSecret: {{ํด๋ผ์ด์ธํธ ์ํฌ๋ฆฟ}},
},
})
.then((response) => {
// ์ด๋ฏธ์ง ์
๋ก๋ ์ฑ๊ณต ์ ์ด๋ฏธ์ง URL์ ์ฝ๋ฐฑ ํจ์์ ์ ๋ฌํ์ฌ ์๋ํฐ์ ํ์
const imageUrl = response.data.url;
callback(imageUrl, '์ฌ์ง ๋์ฒด ํ
์คํธ ์
๋ ฅ');
})
.catch((error) => {
// ์ด๋ฏธ์ง ์
๋ก๋ ์คํจ ์ ์๋ํฐ์ ์คํจ ๋ฉ์์ง๋ฅผ ํ์
console.log(error);
callback('image_load_fail', '์ฌ์ง ๋์ฒด ํ
์คํธ ์
๋ ฅ');
});
},
}}
/>
</StEditor>
);
};
export default CustomEditor;
const StEditor = styled.div`
& .toastui-editor-contents {
z-index: 0;
}
& p {
font-size: 16px;
}
& h1,
h2,
h3,
h4,
h5,
h6 {
border: none;
}
// ํฐํธ ์ฌ์ด์ฆ ์
๋ ฅ์ฐฝ
& .size-input {
width: 100%;
}
& .drop-down-item {
font-size: 0.6rem;
padding: 0.5rem 0.2rem;
cursor: pointer;
}
`;
Props
๋ฐ์์ค๋ ๊ฐ์ data, setData, reset, setRest ์ด๋ ๊ฒ 4๊ฐ์ง์ด๋ค.
- data: ์๋ํฐ ์์ ์ ํ๋ ๋ด์ฉ
- setData: ์๋ํฐ ์์ ๋ด์ฉ์ ๋ณ๊ฒฝํ๋ setState
- reset(optional): ์ด๊ธฐํ ํธ๋ฆฌ๊ฑฐ, default ๊ฐ์ false
- setReset(optional): ์ด๊ธฐํ ํธ๋ฆฌ๊ฑฐ๋ฅผ ํ์ฑํํ๊ธฐ ์ํ setState
reset๊ณผ setReset์ toast-ui editor์์ ์ด๊ธฐ๊ฐ์ ์ค์ ํ ๋ ์ด์ ๊ฐ์ด ๋จ์์๋ ์ด์๊ฐ ์์ด์ ์์๋ก ๋ฆฌ์ ์ํค๊ธฐ ์ํด ์ถ๊ฐํ๋ค. ๋ง์ฝ ์ด๊ธฐ๊ฐ ์ค์ ์ด ์ ๋๋ค๋ฉด ์ฌ์ฉํ ํ์๊ฐ ์๋ค.
ํด๋ฐ
ํด๋ฐ์๋ ๋๊ฐ์ง ์์๋ฅผ ์ถ๊ฐํ๋๋ฐ ๊ธ์์ ๋ณ๊ฒฝ๊ณผ ํฐํธ ํฌ๊ธฐ ๋ณ๊ฒฝ์ด๋ค. colorSyntax๋ toast-ui์์ ์ ๊ณตํ๋ ํ๋ฌ๊ทธ์ธ์ด๊ณ , fontSize๋ ๊นํ์์ ๋ค๋ฅธ ๋ถ์ด ๋ง๋์ ๊ฑธ ์ฌ์ฉํ๋ค. ์ง์ ๋ง๋ค์ด๋ ๋์ง๋ง ํ์๋ก ์์ฒญ๋ ์ฌํญ์ด ์๋์๊ธฐ์ ๊ฐํธํ๊ฒ ์ฌ์ฉํ๊ณ ์ถ์๋ค. (toast-ui editor์์ ํ๋ฌ๊ทธ์ธ์ผ๋ก ์ ๊ณตํ๋ฉด ๋ ์ข๊ฒ ์ง๋ง)


์ด๊ธฐ๊ฐ ์ธํ
useEffect(() => {
if (!isSet && editorRef.current) {
editorRef.current.getInstance().setMarkdown(data ? data : '');
setIsSet(true);
}
}, []);
const [load, setLoad] = useState<number>(0);
useEffect(() => {
if (load < 2 && data && data != '<p><br></p>' && isSet && editorRef.current) {
editorRef.current.getInstance().setMarkdown(data);
setLoad((prev) => prev + 1);
}
}, [isSet, data]);
ํด๋น ๋ถ๋ถ์ ์ด๊ธฐ๊ฐ์ ์ธํ ํ๋ ๋ถ๋ถ์ธ๋ฐ ๋ ์ข์ ์ฝ๋๊ฐ ์์๊ฑฐ๋ผ ์๊ฐํ๋ค. ์ฒ์ ๋ ๋๋ง๋์ ๋๋ isSet์ ๊ฐ์ด false์ด๋ค. ์ด๋ ๋ฐ์ดํฐ๊ฐ ์์ผ๋ฉด ๋ฐ์ดํฐ๋ก ์ธํ ์, ์๋๋ฉด ๋น ์คํธ๋ง์ผ๋ก ์ธํ ์ ํ๋ฉด์ isSet์ true๋ก ๋ง๋ ๋ค.
์๋ useEffect๋ ์ฒ์ ์๋ํฐ ์ปดํฌ๋ํธ๊ฐ ๋ ๋๋ง๋ ๋ ๋ฐ์ดํฐ๊ฐ 2๋ฒ ๋ณ๊ฒฝ๋๋ ์ด์ ๋๋ฌธ์ ํ์คํ๋๋ก ์ด๊ธฐ๊ฐ data๋ฅผ ๋ ๋ฒ ์ธํ ํ๋ ค๊ณ ๋ง๋ ๋ถ๋ถ์ด๋ค. ๋นํจ์จ์ ์ผ๋ก ๋ณด์ผ ์ ์์ง๋ง ์ด๊ธฐ๊ฐ ์ธํ ์ด ์ ๋์ง ์๋ ์ด์ ๋๋ฌธ์ ์์ ์ 5๋ฒ์ ๋๋ ํ ๊ฒ ๊ฐ๋ค.
์ด๋ฏธ์ง ์ ๋ก๋
hooks={{
addImageBlobHook: (blob: any, callback: any) => {
const formData = new FormData();
formData.append('upload', blob);
// ์ด๋ฏธ์ง๋ฅผ ์๋ฒ์ ์
๋ก๋ํ๋ ์์ฒญ
axios
.post({{'์์ฒญ URL'}}, formData, {
headers: {
'Content-Type': 'multipart/form-data',
authorization: {{ํ ํฐ}},
ClientSecret: {{ํด๋ผ์ด์ธํธ ์ํฌ๋ฆฟ}},
},
})
.then((response) => {
// ์ด๋ฏธ์ง ์
๋ก๋ ์ฑ๊ณต ์ ์ด๋ฏธ์ง URL์ ์ฝ๋ฐฑ ํจ์์ ์ ๋ฌํ์ฌ ์๋ํฐ์ ํ์
const imageUrl = response.data.url;
callback(imageUrl, '์ฌ์ง ๋์ฒด ํ
์คํธ ์
๋ ฅ');
})
.catch((error) => {
// ์ด๋ฏธ์ง ์
๋ก๋ ์คํจ ์ ์๋ํฐ์ ์คํจ ๋ฉ์์ง๋ฅผ ํ์
callback('image_load_fail', '์ฌ์ง ๋์ฒด ํ
์คํธ ์
๋ ฅ');
});
},
}}
์ด๋ฏธ์ง ์ ๋ก๋๋ ๊ฒ์ํ์ ๋ ๋ง์ ์๋ฃ๊ฐ ๋์ฌ ๊ฒ์ด๋ค. ์ฐ๋ฆฌ๋ ์๋ํฐ์ ์ด๋ฏธ์ง๋ฅผ S3์ ์ ๋ก๋ํ๋ ๋ฐฉ์์ ์ฌ์ฉํ๊ธฐ ๋๋ฌธ์ axios๋ก S3์ ์ ๋ก๋ ํ ์ด๋ฏธ์ง url์ ๋ฐ์๋ค.
์ ์ฒด์ฝ๋(์๋ํฐ ๋ด์ฉ ๋ถ๋ฌ์ค๊ธฐ)
import '@/styles/Watermark.styles.css';
export const convertLink = (data: string | undefined) => {
if (data) {
const newData = data.replace(/<a\b[^>]*>(.*?)<\/a>/gi, (match: any) => {
// <a> ํ๊ทธ์ href ์์ฑ์ ๊ฐ์ ธ์ต๋๋ค.
const hrefMatch = match.match(/href="([^"]*)"/);
const href = hrefMatch ? hrefMatch[1] : '';
// href๊ฐ Vimeo ๋์์์ ๋งํฌ์ธ์ง ํ์ธ
const isVimeoLink = href.includes('vimeo.com');
const isYoutubeLinkShare = href.includes('youtu.be');
const isYoutubeLink = href.includes('youtube.com/watch');
if (isVimeoLink) {
const videoIdMatch = href.match(/vimeo\.com\/(\d+)/);
const videoId = videoIdMatch ? videoIdMatch[1] : '';
const nickName = localStorage.getItem('nickName');
return `
<div style="position: relative;">
<div class="watermark">${nickName}
${videoId}" class="vimeo_vid" width="100%" frameborder="0"></iframe>
</div>
`;
} else if (isYoutubeLinkShare || isYoutubeLink) {
const videoIdMatch = isYoutubeLinkShare
? href.match(/youtu\.be\/([^?&]+)/)
: href.match(/[?&]v=([^?&]+)/);
const videoId = videoIdMatch ? videoIdMatch[1] : '';
return `${videoId}" width="640" height="360" frameborder="0" allow="autoplay; fullscreen" allowfullscreen></iframe>`;
} else {
// Vimeo, Youtube ๋์์ ๋งํฌ๊ฐ ์๋ ๊ฒฝ์ฐ์๋ <a> ํ๊ทธ์ target="_blank"๋ฅผ ์ถ๊ฐํ์ฌ ๋ฐํ
return match.replace(/<a\b/gi, '<a target="_blank"');
}
});
return newData;
}
};
toast-ui editor์์๋ ์ฝ์ ํ link๋ฅผ aํ๊ทธ์ ํํ๋ก ์ ์ฅํ๋ค. ๊ทธ๋ ๊ธฐ ๋๋ฌธ์ ๋ถ๋ฌ์ฌ ๋ iframe์ผ๋ก ๋ณด์ฌ์ฃผ๊ณ ์ถ์๋ ๋๋ก์๋ aํ๊ทธ๋ฅผ iframe์ผ๋ก ๋ณํํ๋ ๊ณผ์ ์ด ํ์ํ๋ค. youtube, viemo ๋ ๊ฐ์ง์ ์์ ๋งํฌ๋ง ๋ณํ์ ํ๋ฉด ๋๊ธฐ ๋๋ฌธ์ a๋งํฌ์์ youtube, viemo url์ ํค์๋๊ฐ ์ถ์ถ๋๋์ง ํ์ธํ๊ณ ๋ณํ์ ํ๋ค. (์ ํ๋ธ์ ๊ฒฝ์ฐ url์ ๋ ฅ ํ๋์์ ๋ณต์ฌํ ์ ์๋ url๊ณผ ๊ณต์ ํ๊ธฐ ๋ฒํผ์ url์ด ๋ฌ๋ผ ๋ ๊ฐ์ง ๋ชจ๋ ํ์ธ์ ํด์ฃผ์๋ค.


<div
dangerouslySetInnerHTML={{
__html:convertLink(Dompurify.sanitize({{"๋ด์ฉ"}})) || '',
}}
/>
์๋ํฐ์ ๋ด์ฉ์ html ํํ์ด๊ธฐ ๋๋ฌธ์ ๋ถ๋ฌ์ฌ ๋๋ dangerouslySetInnerHTML์ ์ฌ์ฉํด์ ๋ถ๋ฌ์๋ค.
๐ค ์ Dompurify.sanitize๋ฅผ ์ฌ์ฉํ ๊น?
1. XSS๋ ์ ์ฑ ์ฌ์ฉ์๊ฐ ์น ํ์ด์ง์ ์ ์ฑ ์คํฌ๋ฆฝํธ๋ฅผ ์ฝ์ ํ์ฌ ๋ค๋ฅธ ์ฌ์ฉ์์๊ฒ ์คํ๋๋๋ก ํ๋ ๊ณต๊ฒฉ์ด๋ค. ์ด๋ฅผ ํตํด ๊ณต๊ฒฉ์๋ ์ฌ์ฉ์์ ์ธ์ ์ ํ์ทจํ๊ฑฐ๋, ์ ์ฑ ์ฝ๋๋ฅผ ์คํ์ํค๊ฑฐ๋, ๋ฏผ๊ฐํ ๋ฐ์ดํฐ๋ฅผ ํ์ทจํ ์ ์๋ค.
DOMPurify๋ ์ฌ์ฉ์ ์ ๋ ฅ ํน์ ์ธ๋ถ ๋ฐ์ดํฐ์์ ์ ์ฌ์ ์ผ๋ก ์ํํ ์คํฌ๋ฆฝํธ ํ๊ทธ ๋ฐ ์์ฑ์ ์ ๊ฑฐํ์ฌ XSS ๊ณต๊ฒฉ์ ๋ฐฉ์งํ๋ค.
์๋ฅผ ๋ค์ด, <script> ํ๊ทธ๋ onmouseover์ ๊ฐ์ ์ด๋ฒคํธ ํธ๋ค๋ฌ ์์ฑ ๋ฑ์ ์ ๊ฑฐํฉ๋๋ค.
2. ์น ์ ํ๋ฆฌ์ผ์ด์ ์ ์ข ์ข ์ธ๋ถ API๋ ์ฌ์ฉ์ ์ ๋ ฅ์ ํตํด ๋ฐ์ดํฐ๋ฅผ ์์งํ๋ค. ์ด๋ฌํ ๋ฐ์ดํฐ๋ ์ ๋ขฐํ ์ ์์ผ๋ฉฐ, ์ ์์ ์ธ ๋ด์ฉ์ด ํฌํจ๋ ์ ์์ต๋๋ค.
DOMPurify๋ ์ด๋ฌํ ์ ๋ขฐํ ์ ์๋ ๋ฐ์ดํฐ๋ฅผ ์ ํํ์ฌ, ์ ์ฑ ์ฝ๋๊ฐ ํฌํจ๋ HTML์ด ์น ํ์ด์ง์ ์ฝ์ ๋๋ ๊ฒ์ ๋ฐฉ์งํ๋ค. ์ด๋ฅผ ํตํด ์ธ๋ถ ๋ฐ์ดํฐ ์์ค๋ก๋ถํฐ ์ค๋ ๋ณด์ ์ํ์ ์ค์ผ ์ ์๋ค.
'L > Javascript' ์นดํ ๊ณ ๋ฆฌ์ ๋ค๋ฅธ ๊ธ
[react-datepicker] datepicker ์ปค์คํ ํ๊ธฐ (+ timepicker) (0) | 2024.07.18 |
---|