Skip to content

Commit ffcfefa

Browse files
committed
[FEAT]: 괄호 닫기 기능 구현
1 parent 59a348c commit ffcfefa

File tree

2 files changed

+57
-8
lines changed

2 files changed

+57
-8
lines changed

client/src/components/Editor.jsx

+16-8
Original file line numberDiff line numberDiff line change
@@ -21,12 +21,15 @@ export default function Editor({
2121
const [textSplit, setTextSplit] = useState([]);
2222
const [isEventMode, setIsEventMode] = useState(false);
2323

24+
// 백스페이스로 인해 괄호 닫기 지워지지 않는 현상 해결
25+
const [deletionMode, setDeletionMode] = useState(false);
26+
2427
// 이벤트 모드 체크를 위한 useEffect 추가
25-
useEffect(() => {
26-
const queryParams = new URLSearchParams(window.location.search);
27-
const eventParam = queryParams.get("event");
28-
setIsEventMode(eventParam?.toLowerCase() === "bisc");
29-
}, []);
28+
// useEffect(() => {
29+
// const queryParams = new URLSearchParams(window.location.search);
30+
// const eventParam = queryParams.get("event");
31+
// setIsEventMode(eventParam?.toLowerCase() === "bisc");
32+
// }, []);
3033

3134
useEffect(() => {
3235
setTextSplit(getFilecontents(file).content);
@@ -54,11 +57,15 @@ export default function Editor({
5457
}, [userInput]);
5558

5659
const userInputTabHandler = (event) => {
57-
if (isEventMode && (event.key === "Backspace" || event.key === "Delete")) {
58-
event.preventDefault();
60+
setDeletionMode(false);
61+
// if (isEventMode && (event.key === "Backspace" || event.key === "Delete")) {
62+
// event.preventDefault();
63+
// return;
64+
// }
65+
if (event.key === "Backspace") {
66+
setDeletionMode(true);
5967
return;
6068
}
61-
6269
// todo: 조건식 최적화
6370
if (event.key === "Escape") {
6471
setUseAutoComplete(false);
@@ -171,6 +178,7 @@ export default function Editor({
171178
file={file}
172179
setFileLength={setFileLength}
173180
daynight={daynight}
181+
deletionMode={deletionMode}
174182
/>
175183
</div>
176184
) : (

client/src/components/Text.jsx

+41
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,10 @@ import AutoCompletion from "./AutoCompletion";
88

99
function Text(props) {
1010
//const colormap = file.colormap
11+
const bracketClosed = [")", "}", "]", '"', "'", "`"];
1112
const [textSplit, setTextSplit] = useState([]);
1213
const [themeColor, setThemeColor] = useState("");
14+
const [quotePositions, setQuotePositions] = useState([]); // 짝수 번째 따옴표 위치 기록
1315
let wrong = 0,
1416
correct = 0,
1517
space = 0;
@@ -26,6 +28,45 @@ function Text(props) {
2628
// eslint-disable-next-line react-hooks/exhaustive-deps
2729
}, [textSplit]);
2830

31+
useEffect(() => {
32+
const content = getFilecontents(props.file).content;
33+
setTextSplit(content);
34+
35+
// 따옴표 위치 미리 계산
36+
const positions = [];
37+
let quoteCount = 0;
38+
for (let i = 0; i < content.length; i++) {
39+
if (['"', "'", "`"].includes(content[i])) {
40+
quoteCount++;
41+
if (quoteCount % 2 === 0) {
42+
positions.push(i); // 짝수 번째 따옴표 위치 기록
43+
}
44+
}
45+
}
46+
setQuotePositions(positions);
47+
}, [props.file]);
48+
49+
// 자동 괄호 입력 로직
50+
useEffect(() => {
51+
if (props.deletionMode) return; // 삭제 모드인 경우 실행하지 않음
52+
53+
const currentIndex = user.length;
54+
const nextChar = textSplit[currentIndex]; // 다음에 입력할 위치의 문자
55+
56+
if (bracketClosed.includes(nextChar)) {
57+
// 닫히는 괄호인 경우
58+
if (['"', "'", "`"].includes(nextChar)) {
59+
// 따옴표인 경우, 짝수 번째 위치인지 확인
60+
if (quotePositions.includes(currentIndex)) {
61+
props.setUserInput(user + nextChar);
62+
}
63+
} else {
64+
// 다른 닫히는 괄호인 경우 자동으로 입력
65+
props.setUserInput(user + nextChar);
66+
}
67+
}
68+
}, [user, textSplit, quotePositions]); // user, textSplit, quotePositions가 변경될 때마다 실행
69+
2970
useEffect(() => {
3071
if (props.daynight % 2 === 1) setThemeColor("#585858");
3172
else setThemeColor("#BEBEBE");

0 commit comments

Comments
 (0)