@pop-ui/core 0.0.41 → 0.1.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.
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@pop-ui/core",
3
3
  "type": "module",
4
- "version": "0.0.41",
4
+ "version": "0.1.1",
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": "0.0.35",
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": "2d032b939907ec4360e4f8e3ecb97a88311df6eb"
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.