@pop-ui/core 0.0.40 → 0.1.0
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/README.md +46 -148
- package/dist/core.css +1 -1
- package/dist/core.js +1872 -1169
- package/dist/core.umd.cjs +26 -24
- package/dist/types/index.d.ts +118 -28
- package/package.json +10 -5
- package/src/components/CalendarDatePicker/README.md +409 -0
- package/LICENSE.md +0 -21
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@pop-ui/core",
|
|
3
3
|
"type": "module",
|
|
4
|
-
"version": "0.0
|
|
4
|
+
"version": "0.1.0",
|
|
5
5
|
"main": "./dist/core.umd.cjs",
|
|
6
6
|
"module": "./dist/core.js",
|
|
7
7
|
"types": "./dist/types/index.d.ts",
|
|
@@ -44,13 +44,18 @@
|
|
|
44
44
|
"preview": "vite preview"
|
|
45
45
|
},
|
|
46
46
|
"dependencies": {
|
|
47
|
+
"@dnd-kit/core": "^6.3.1",
|
|
48
|
+
"@dnd-kit/sortable": "^10.0.0",
|
|
49
|
+
"@dnd-kit/utilities": "^3.2.2",
|
|
47
50
|
"@mantine/core": "^8.3.6",
|
|
48
51
|
"@mantine/dates": "^8.3.6",
|
|
49
52
|
"@mantine/dropzone": "^8.3.6",
|
|
50
53
|
"@mantine/hooks": "^8.3.6",
|
|
51
54
|
"@mantine/notifications": "^8.3.6",
|
|
52
|
-
"@pop-ui/foundation": "
|
|
53
|
-
"dayjs": "^1.11.18"
|
|
55
|
+
"@pop-ui/foundation": "1.1.5",
|
|
56
|
+
"dayjs": "^1.11.18",
|
|
57
|
+
"lottie-react": "^2.4.1",
|
|
58
|
+
"react-easy-crop": "^5.5.3"
|
|
54
59
|
},
|
|
55
60
|
"peerDependencies": {
|
|
56
61
|
"react": "^19.2.0",
|
|
@@ -97,5 +102,5 @@
|
|
|
97
102
|
"typescript": "^5.9.3",
|
|
98
103
|
"vite": "^7.1.12"
|
|
99
104
|
},
|
|
100
|
-
"gitHead": "
|
|
101
|
-
}
|
|
105
|
+
"gitHead": "9a120a33dbe9dfaffabb94219c301d5834968fae"
|
|
106
|
+
}
|
|
@@ -0,0 +1,409 @@
|
|
|
1
|
+
# CalendarDatePicker
|
|
2
|
+
|
|
3
|
+
Mantine DatePicker 기반의 인라인 캘린더 컴포넌트로, 날짜 선택 시 특정 날짜나 요일을 제외할 수 있는 기능을 제공합니다.
|
|
4
|
+
|
|
5
|
+
## 📦 Features
|
|
6
|
+
|
|
7
|
+
- ✅ **날짜 제외**: 특정 날짜 또는 날짜 범위를 비활성화
|
|
8
|
+
- ✅ **요일 제외**: 주말이나 특정 요일을 비활성화
|
|
9
|
+
- ✅ **유연한 API**: 단일 날짜와 범위를 하나의 배열에서 혼합 사용 가능
|
|
10
|
+
- ✅ **외부 날짜 제어**: 현재 달에 포함되지 않는 날짜(이전/다음 달)의 선택 방지
|
|
11
|
+
|
|
12
|
+
## 🚀 Usage
|
|
13
|
+
|
|
14
|
+
### 기본 사용법
|
|
15
|
+
|
|
16
|
+
```tsx
|
|
17
|
+
import { CalendarDatePicker } from '@pop-ui/core';
|
|
18
|
+
|
|
19
|
+
function App() {
|
|
20
|
+
return <CalendarDatePicker type="default" />;
|
|
21
|
+
}
|
|
22
|
+
```
|
|
23
|
+
|
|
24
|
+
### 특정 날짜 제외
|
|
25
|
+
|
|
26
|
+
```tsx
|
|
27
|
+
<CalendarDatePicker
|
|
28
|
+
excludedDates={[
|
|
29
|
+
'2025-11-24', // 단일 날짜
|
|
30
|
+
'2025-12-25', // 크리스마스
|
|
31
|
+
['2025-12-24', '2025-12-26'], // 크리스마스 연휴 (범위)
|
|
32
|
+
]}
|
|
33
|
+
/>
|
|
34
|
+
```
|
|
35
|
+
|
|
36
|
+
### 요일 제외 (주말)
|
|
37
|
+
|
|
38
|
+
```tsx
|
|
39
|
+
<CalendarDatePicker
|
|
40
|
+
excludedDays={[0, 6]} // 0=일요일, 6=토요일
|
|
41
|
+
/>
|
|
42
|
+
```
|
|
43
|
+
|
|
44
|
+
### 복합 사용
|
|
45
|
+
|
|
46
|
+
```tsx
|
|
47
|
+
<CalendarDatePicker
|
|
48
|
+
excludedDays={[0, 6]} // 주말 제외
|
|
49
|
+
excludedDates={[
|
|
50
|
+
'2025-11-28', // 추수감사절
|
|
51
|
+
'2025-12-25', // 크리스마스
|
|
52
|
+
['2025-12-28', '2026-01-03'], // 연말연시 휴가
|
|
53
|
+
]}
|
|
54
|
+
/>
|
|
55
|
+
```
|
|
56
|
+
|
|
57
|
+
## 📋 Props
|
|
58
|
+
|
|
59
|
+
| Prop | Type | Default | Description |
|
|
60
|
+
| ---------------- | ------------------------------------ | ----------- | ------------------------------------------------------------- |
|
|
61
|
+
| `type` | `'default' \| 'multiple' \| 'range'` | `'default'` | 캘린더 선택 모드 |
|
|
62
|
+
| `excludedDates` | `(string \| [string, string])[]` | `[]` | 제외할 날짜 또는 날짜 범위 배열 |
|
|
63
|
+
| `excludedDays` | `number[]` | `[]` | 제외할 요일 배열 (0: 일요일 ~ 6: 토요일) |
|
|
64
|
+
| `highlightToday` | `boolean` | `false` | 오늘 날짜 강조 및 날짜 하단에 '오늘' 텍스트 표시 여부 |
|
|
65
|
+
| `onChange` | `(value: DateValue) => void` | - | 날짜 선택 시 호출되는 콜백 |
|
|
66
|
+
| `...props` | `DatePickerProps` | - | Mantine DatePicker의 나머지 Props (커스텀한 excludeDate 제외) |
|
|
67
|
+
|
|
68
|
+
## 🔍 Detailed Props
|
|
69
|
+
|
|
70
|
+
### `excludedDates`
|
|
71
|
+
|
|
72
|
+
특정 날짜 또는 날짜 범위를 제외합니다.
|
|
73
|
+
|
|
74
|
+
- **타입**: `(string | [string, string])[]`
|
|
75
|
+
- **기본값**: `[]`
|
|
76
|
+
- **형식**:
|
|
77
|
+
- 단일 날짜: `'YYYY-MM-DD'` 형식의 문자열
|
|
78
|
+
- 날짜 범위: `[시작일, 종료일]` 형식의 튜플
|
|
79
|
+
|
|
80
|
+
**예시:**
|
|
81
|
+
|
|
82
|
+
```tsx
|
|
83
|
+
excludedDates={[
|
|
84
|
+
'2025-11-24', // 단일 날짜
|
|
85
|
+
'2025-11-25', // 단일 날짜
|
|
86
|
+
['2025-12-24', '2025-12-26'], // 범위 (3일간)
|
|
87
|
+
['2025-12-28', '2026-01-03'], // 범위 (7일간)
|
|
88
|
+
]}
|
|
89
|
+
```
|
|
90
|
+
|
|
91
|
+
### `excludedDays`
|
|
92
|
+
|
|
93
|
+
특정 요일을 제외합니다.
|
|
94
|
+
|
|
95
|
+
- **타입**: `number[]`
|
|
96
|
+
- **기본값**: `[]`
|
|
97
|
+
- **값**: `0` (일요일) ~ `6` (토요일)
|
|
98
|
+
|
|
99
|
+
**예시:**
|
|
100
|
+
|
|
101
|
+
```tsx
|
|
102
|
+
excludedDays={[0, 6]} // 주말 제외
|
|
103
|
+
excludedDays={[1, 3, 5]} // 월, 수, 금 제외
|
|
104
|
+
```
|
|
105
|
+
|
|
106
|
+
### `type`
|
|
107
|
+
|
|
108
|
+
DatePicker 타입을 지정합니다.
|
|
109
|
+
|
|
110
|
+
- **타입**: `'default' | 'multiple' | 'range'`
|
|
111
|
+
- **기본값**: `'default'`
|
|
112
|
+
|
|
113
|
+
### `onChange`
|
|
114
|
+
|
|
115
|
+
날짜 선택 시 호출되는 콜백 함수입니다.
|
|
116
|
+
|
|
117
|
+
- **타입**: `(value: DateValue) => void`
|
|
118
|
+
- **설명**: `DateValue`는 `type` prop에 따라 다른 타입을 가집니다:
|
|
119
|
+
- `type="default"`: `Date | null`
|
|
120
|
+
- `type="range"`: `[Date | null, Date | null]`
|
|
121
|
+
- `type="multiple"`: `Date[]`
|
|
122
|
+
|
|
123
|
+
**예시:**
|
|
124
|
+
|
|
125
|
+
```tsx
|
|
126
|
+
<CalendarDatePicker
|
|
127
|
+
type="default"
|
|
128
|
+
onChange={(value) => console.log(value)} // value: Date | null
|
|
129
|
+
/>
|
|
130
|
+
|
|
131
|
+
<CalendarDatePicker
|
|
132
|
+
type="range"
|
|
133
|
+
onChange={(value) => console.log(value)} // value: [Date | null, Date | null]
|
|
134
|
+
/>
|
|
135
|
+
```
|
|
136
|
+
|
|
137
|
+
### `highlightToday`
|
|
138
|
+
|
|
139
|
+
오늘 날짜에 "오늘" 텍스트를 표시하고 강조합니다.
|
|
140
|
+
|
|
141
|
+
- **타입**: `boolean`
|
|
142
|
+
- **기본값**: `false`
|
|
143
|
+
- **설명**: Mantine의 `highlightToday` prop과 동일합니다. `true`로 설정하면 오늘 날짜가 강조되고 하단에 "오늘" 텍스트가 표시됩니다.
|
|
144
|
+
|
|
145
|
+
**예시:**
|
|
146
|
+
|
|
147
|
+
```tsx
|
|
148
|
+
<CalendarDatePicker highlightToday />
|
|
149
|
+
```
|
|
150
|
+
|
|
151
|
+
### 기타 Props
|
|
152
|
+
|
|
153
|
+
Mantine의 `DatePicker` 컴포넌트의 나머지 props를 지원합니다.
|
|
154
|
+
|
|
155
|
+
> **주의**:
|
|
156
|
+
>
|
|
157
|
+
> - `excludeDate` prop은 지원하지 않습니다. 날짜 제외 기능을 사용하려면 `excludedDates` 또는 `excludedDays` prop을 사용하세요.
|
|
158
|
+
> - `type`, `locale`, `firstDayOfWeek` 등의 props는 `...props`로 전달하면 덮어쓸 수 있습니다. 컴포넌트의 기본 동작을 변경하려면 명시적으로 prop을 전달하세요.
|
|
159
|
+
|
|
160
|
+
**예시:**
|
|
161
|
+
|
|
162
|
+
```tsx
|
|
163
|
+
// type prop을 덮어쓰기
|
|
164
|
+
<CalendarDatePicker
|
|
165
|
+
type="default" // 기본값
|
|
166
|
+
{...{ type: 'range' }} // range로 덮어써짐
|
|
167
|
+
/>
|
|
168
|
+
|
|
169
|
+
// locale prop을 덮어쓰기
|
|
170
|
+
<CalendarDatePicker
|
|
171
|
+
locale="ko" // 기본값
|
|
172
|
+
{...{ locale: 'en' }} // en으로 덮어써짐
|
|
173
|
+
/>
|
|
174
|
+
```
|
|
175
|
+
|
|
176
|
+
자세한 내용은 [Mantine DatePicker 문서](https://mantine.dev/dates/date-picker/)를 참고하세요.
|
|
177
|
+
|
|
178
|
+
## 🎨 Styling
|
|
179
|
+
|
|
180
|
+
컴포넌트는 `styles.module.scss`를 통해 스타일링됩니다.
|
|
181
|
+
|
|
182
|
+
### classNames prop 사용법
|
|
183
|
+
|
|
184
|
+
`classNames` prop을 사용하여 Mantine DatePicker의 모든 스타일 키를 커스터마이징할 수 있습니다. 기본 클래스와 커스텀 클래스가 자동으로 병합되며, 커스텀 클래스가 우선순위를 가집니다.
|
|
185
|
+
|
|
186
|
+
**특징:**
|
|
187
|
+
|
|
188
|
+
- `DatePickerStylesNames`(Mantine DatePicker Styles API의 selectors)에 속하는 키만 허용됩니다 (타입 안전성 보장)
|
|
189
|
+
- 기본 클래스와 커스텀 클래스가 공백으로 연결되어 병합됩니다
|
|
190
|
+
- `DEFAULT_CLASS_NAMES`에 없는 키도 `DatePickerStylesNames`에 속하면 확장 가능합니다
|
|
191
|
+
- 객체형과 함수형 모두 지원합니다
|
|
192
|
+
|
|
193
|
+
**객체형 예시:**
|
|
194
|
+
|
|
195
|
+
```tsx
|
|
196
|
+
<CalendarDatePicker
|
|
197
|
+
classNames={{
|
|
198
|
+
day: 'my-custom-day',
|
|
199
|
+
calendarHeaderLevel: 'my-custom-header-level',
|
|
200
|
+
// DatePickerStylesNames에 속하는 모든 키 사용 가능
|
|
201
|
+
}}
|
|
202
|
+
/>
|
|
203
|
+
```
|
|
204
|
+
|
|
205
|
+
**함수형 예시:**
|
|
206
|
+
|
|
207
|
+
```tsx
|
|
208
|
+
<CalendarDatePicker
|
|
209
|
+
classNames={(theme, props, ctx) => ({
|
|
210
|
+
day: theme.colorScheme === 'dark' ? 'dark-day' : 'light-day',
|
|
211
|
+
calendarHeader: 'custom-header',
|
|
212
|
+
})}
|
|
213
|
+
/>
|
|
214
|
+
```
|
|
215
|
+
|
|
216
|
+
함수형 `classNames`를 사용하면 테마나 props에 따라 동적으로 스타일을 적용할 수 있습니다. 함수형 `classNames`도 기본 `DEFAULT_CLASS_NAMES`와 자동으로 병합됩니다.
|
|
217
|
+
|
|
218
|
+
### 스타일 오버라이딩 가이드
|
|
219
|
+
|
|
220
|
+
라이브러리 기본 스타일은 CSS `:where()`를 사용하여 **중요도(Specificity)가 0**으로 설정되어 있습니다. 따라서 `!important`나 복잡한 선택자 없이도 커스텀 클래스만으로 쉽게 스타일을 덮어쓸 수 있습니다. 또한, `[data-outside]` 속성이 있는 날짜(현재 월이 아닌 날짜)는 `pointer-events: none`으로 설정되어 클릭이 불가능합니다.
|
|
221
|
+
|
|
222
|
+
#### 1. 모든 상태 덮어쓰기 (Simple Override)
|
|
223
|
+
|
|
224
|
+
상태(in-range, selected 등)에 관계없이 스타일을 적용하고 싶다면 평범한 클래스를 사용하세요.
|
|
225
|
+
|
|
226
|
+
```css
|
|
227
|
+
.myCustomDay {
|
|
228
|
+
/* in-range, weekend 등 모든 상태를 덮어씁니다 */
|
|
229
|
+
background-color: pink;
|
|
230
|
+
border-radius: 4px;
|
|
231
|
+
}
|
|
232
|
+
```
|
|
233
|
+
|
|
234
|
+
#### 2. 특정 상태만 덮어쓰기 (Conditional Override)
|
|
235
|
+
|
|
236
|
+
특정 상태일 때만 스타일을 변경하고 싶다면 `data` 속성을 사용하세요. (Mantine UI의 data 속성 참조)
|
|
237
|
+
|
|
238
|
+
```css
|
|
239
|
+
.myCustomDay {
|
|
240
|
+
/* 기본 스타일 */
|
|
241
|
+
background-color: white;
|
|
242
|
+
|
|
243
|
+
/* 범위 내 날짜일 때만 적용 */
|
|
244
|
+
&[data-in-range] {
|
|
245
|
+
background-color: lightblue;
|
|
246
|
+
}
|
|
247
|
+
|
|
248
|
+
/* 선택된 날짜일 때만 적용 */
|
|
249
|
+
&[data-selected] {
|
|
250
|
+
background-color: blue;
|
|
251
|
+
color: white;
|
|
252
|
+
}
|
|
253
|
+
}
|
|
254
|
+
```
|
|
255
|
+
|
|
256
|
+
### 지원하는 데이터 속성 (Data Attributes)
|
|
257
|
+
|
|
258
|
+
- `[data-selected]`: 선택된 날짜
|
|
259
|
+
- `[data-disabled]`: 비활성화된 날짜
|
|
260
|
+
- `[data-in-range]`: 범위 선택 시 범위 내 날짜
|
|
261
|
+
- `[data-weekend]`: 주말
|
|
262
|
+
- `[data-outside]`: 현재 월 외부의 날짜
|
|
263
|
+
- `[data-today]`: 오늘 날짜 (하이라이트 스타일링을 위해서는 `[data-today][data-highlight-today]` 방식으로 속성을 선택해야 함)
|
|
264
|
+
- 이외에도 Mantine DatePicker의 data attributes를 지원합니다.
|
|
265
|
+
|
|
266
|
+
## 💡 Tips
|
|
267
|
+
|
|
268
|
+
### 1. 날짜 형식 일관성 유지
|
|
269
|
+
|
|
270
|
+
`YYYY-MM-DD` 형식 사용을 권장합니다. 다른 형식도 dayjs가 파싱하지만, 일관성과 명확성을 위해 표준 형식을 사용하세요:
|
|
271
|
+
|
|
272
|
+
```tsx
|
|
273
|
+
// ✅ Good - 권장
|
|
274
|
+
excludedDates={['2025-11-24']}
|
|
275
|
+
excludedDates={['2025-11-24T00:00:00']}
|
|
276
|
+
|
|
277
|
+
// ⚠️ 작동하지만 권장하지 않음
|
|
278
|
+
excludedDates={['11/24/2025']} // 로케일에 따라 해석이 다를 수 있음
|
|
279
|
+
```
|
|
280
|
+
|
|
281
|
+
### 2. 범위 사용 시 주의사항
|
|
282
|
+
|
|
283
|
+
범위는 시작일과 종료일을 **포함**합니다:
|
|
284
|
+
|
|
285
|
+
```tsx
|
|
286
|
+
// 12/24, 12/25, 12/26 모두 제외됨
|
|
287
|
+
excludedDates={[['2025-12-24', '2025-12-26']]}
|
|
288
|
+
```
|
|
289
|
+
|
|
290
|
+
### 3. 성능 최적화
|
|
291
|
+
|
|
292
|
+
많은 날짜를 제외해야 한다면 범위를 사용하는 것이 효율적입니다:
|
|
293
|
+
|
|
294
|
+
```tsx
|
|
295
|
+
// ✅ Good - 범위 사용
|
|
296
|
+
excludedDates={[['2025-12-01', '2025-12-31']]}
|
|
297
|
+
|
|
298
|
+
// ❌ Bad - 개별 날짜 나열
|
|
299
|
+
excludedDates={[
|
|
300
|
+
'2025-12-01', '2025-12-02', '2025-12-03', /* ... */ '2025-12-31'
|
|
301
|
+
]}
|
|
302
|
+
```
|
|
303
|
+
|
|
304
|
+
### 4. 요일과 날짜 조합
|
|
305
|
+
|
|
306
|
+
`excludedDays`와 `excludedDates`를 함께 사용하면 더 세밀한 제어가 가능합니다:
|
|
307
|
+
|
|
308
|
+
```tsx
|
|
309
|
+
<CalendarDatePicker
|
|
310
|
+
excludedDays={[0, 6]} // 기본적으로 주말 제외
|
|
311
|
+
excludedDates={['2025-11-27']}
|
|
312
|
+
/>
|
|
313
|
+
```
|
|
314
|
+
|
|
315
|
+
## 🧪 Storybook
|
|
316
|
+
|
|
317
|
+
Storybook에서 다양한 사용 예시를 확인할 수 있습니다.
|
|
318
|
+
|
|
319
|
+
### 스토리 목록
|
|
320
|
+
|
|
321
|
+
- **DefaultDatePicker**: 기본 캘린더
|
|
322
|
+
- **WithExcludedDays**: 요일 제외 (라디오 버튼으로 선택)
|
|
323
|
+
- **WithExcludedDates**: 날짜/범위 제외
|
|
324
|
+
- **WithCustomStyles**: 스타일 커스터마이징 및 오버라이딩 테스트
|
|
325
|
+
|
|
326
|
+
## 🏗️ Architecture
|
|
327
|
+
|
|
328
|
+
### 파일 구조
|
|
329
|
+
|
|
330
|
+
```
|
|
331
|
+
CalendarDatePicker/
|
|
332
|
+
├── index.tsx # 메인 컴포넌트
|
|
333
|
+
├── types.ts # TypeScript 타입 정의
|
|
334
|
+
├── utils.ts # 유틸리티 함수
|
|
335
|
+
│ ├── createExcludedDateChecker # 날짜 제외 로직
|
|
336
|
+
│ ├── hasExcludedDateInRange # 범위 내 제외 날짜 확인
|
|
337
|
+
│ ├── getEmptyValueForType # 타입별 빈 값 반환
|
|
338
|
+
│ ├── normalizeValueForType # 타입별 값 정규화
|
|
339
|
+
│ └── resolveDatePickerValue # 날짜 값 해석
|
|
340
|
+
├── styles.module.scss # 스타일
|
|
341
|
+
├── CalendarDatePicker.stories.tsx # Storybook 스토리
|
|
342
|
+
└── README.md # 문서 (이 파일)
|
|
343
|
+
```
|
|
344
|
+
|
|
345
|
+
### 핵심 로직: `createExcludedDateChecker`
|
|
346
|
+
|
|
347
|
+
날짜 제외 로직은 `utils.ts`의 `createExcludedDateChecker` 함수에서 처리됩니다.
|
|
348
|
+
|
|
349
|
+
**동작 방식:**
|
|
350
|
+
|
|
351
|
+
1. **입력 처리**: `excludedDates` 배열을 단일 날짜와 범위로 분리
|
|
352
|
+
2. **유효성 검증**: `.isValid()`로 유효하지 않은 날짜 필터링
|
|
353
|
+
3. **정규화**: 날짜를 `YYYY-MM-DD` 형식으로 정규화하고 dayjs 객체로 파싱
|
|
354
|
+
4. **검사**: 주어진 날짜가 다음 조건에 해당하는지 확인
|
|
355
|
+
- 제외된 요일인가?
|
|
356
|
+
- 제외된 단일 날짜인가?
|
|
357
|
+
- 제외된 범위 내에 있는가?
|
|
358
|
+
|
|
359
|
+
**예시:**
|
|
360
|
+
|
|
361
|
+
```typescript
|
|
362
|
+
const isExcluded = createExcludedDateChecker({
|
|
363
|
+
excludedDays: [0, 6], // 주말
|
|
364
|
+
excludedDates: ['2025-12-25', ['2025-12-24', '2025-12-26']],
|
|
365
|
+
});
|
|
366
|
+
|
|
367
|
+
isExcluded(new Date('2025-12-25')); // true (단일 날짜)
|
|
368
|
+
isExcluded(new Date('2025-12-24')); // true (범위 내)
|
|
369
|
+
isExcluded(new Date('2025-12-27')); // false
|
|
370
|
+
```
|
|
371
|
+
|
|
372
|
+
### 성능 최적화: `hasExcludedDateInRange`
|
|
373
|
+
|
|
374
|
+
범위 선택(`type="range"`) 시, 선택된 범위 내에 제외된 날짜가 있는지 확인하는 함수입니다.
|
|
375
|
+
|
|
376
|
+
**범위 겹침 알고리즘 사용:**
|
|
377
|
+
|
|
378
|
+
두 범위가 겹치지 않는 경우는:
|
|
379
|
+
|
|
380
|
+
- 범위 A가 범위 B보다 완전히 앞: `end < exStart`
|
|
381
|
+
- 범위 A가 범위 B보다 완전히 뒤: `start > exEnd`
|
|
382
|
+
|
|
383
|
+
따라서 겹치는 경우: `!(end < exStart || start > exEnd)`
|
|
384
|
+
|
|
385
|
+
```typescript
|
|
386
|
+
// O(m) 복잡도 - m은 제외 범위 개수
|
|
387
|
+
for (const [exStart, exEnd] of excludedRanges) {
|
|
388
|
+
if (!(endDay.isBefore(exStart) || startDay.isAfter(exEnd))) {
|
|
389
|
+
return true; // 범위 겹침 발견!
|
|
390
|
+
}
|
|
391
|
+
}
|
|
392
|
+
```
|
|
393
|
+
|
|
394
|
+
**성능 비교:**
|
|
395
|
+
|
|
396
|
+
- **이전**: O(n) - 범위 내 모든 날짜 순회 (1년 = 365회)
|
|
397
|
+
- **최적화**: O(m) - 제외 범위 개수만큼만 체크 (예: 3회)
|
|
398
|
+
- **결과**: 큰 범위 선택 시 ~100배 빠른 성능 🚀
|
|
399
|
+
|
|
400
|
+
**검사 순서:**
|
|
401
|
+
|
|
402
|
+
1. 제외된 범위와의 겹침 검사 (범위 겹침 알고리즘)
|
|
403
|
+
2. 단일 제외 날짜가 범위 내에 있는지 확인
|
|
404
|
+
3. 제외된 요일이 범위 내에 있는지 순회 검사
|
|
405
|
+
|
|
406
|
+
## 📚 Related
|
|
407
|
+
|
|
408
|
+
- [Mantine DatePicker](https://mantine.dev/dates/date-picker/)
|
|
409
|
+
- [dayjs](https://day.js.org/)
|
package/LICENSE.md
DELETED
|
@@ -1,21 +0,0 @@
|
|
|
1
|
-
MIT License
|
|
2
|
-
|
|
3
|
-
Copyright (c) 2025 (주)텐핑거스
|
|
4
|
-
|
|
5
|
-
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
-
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
-
in the Software without restriction, including without limitation the rights
|
|
8
|
-
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
-
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
-
furnished to do so, subject to the following conditions:
|
|
11
|
-
|
|
12
|
-
The above copyright notice and this permission notice shall be included in all
|
|
13
|
-
copies or substantial portions of the Software.
|
|
14
|
-
|
|
15
|
-
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
-
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
-
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
-
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
-
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
-
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
-
SOFTWARE.
|