@weing-dev/ui-kit-primitive 0.0.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.
Files changed (328) hide show
  1. package/README.md +50 -0
  2. package/eslint.config.js +29 -0
  3. package/index.html +13 -0
  4. package/package.json +75 -0
  5. package/public/static/icon/Add.svg +1 -0
  6. package/public/static/icon/ApprovalInactive.svg +1 -0
  7. package/public/static/icon/ArrowRight.svg +5 -0
  8. package/public/static/icon/Cancle Filled.svg +1 -0
  9. package/public/static/icon/Cancle.svg +1 -0
  10. package/public/static/icon/Cart Filled.svg +1 -0
  11. package/public/static/icon/Cart.svg +1 -0
  12. package/public/static/icon/Check.svg +1 -0
  13. package/public/static/icon/CheckBoxRound.svg +3 -0
  14. package/public/static/icon/Company.svg +1 -0
  15. package/public/static/icon/DragHandle.svg +1 -0
  16. package/public/static/icon/Eco.svg +1 -0
  17. package/public/static/icon/EditSquare.svg +1 -0
  18. package/public/static/icon/Error Filled.svg +1 -0
  19. package/public/static/icon/Error.svg +1 -0
  20. package/public/static/icon/File Upload.svg +1 -0
  21. package/public/static/icon/FilecheckInactive.svg +1 -0
  22. package/public/static/icon/Go Before.svg +1 -0
  23. package/public/static/icon/Go Next.svg +1 -0
  24. package/public/static/icon/Hamburger Menu.svg +1 -0
  25. package/public/static/icon/ImagecheckPlay.svg +4 -0
  26. package/public/static/icon/Kakao Talk.svg +1 -0
  27. package/public/static/icon/Minus.svg +1 -0
  28. package/public/static/icon/Naver Blog.svg +1 -0
  29. package/public/static/icon/Naver.svg +1 -0
  30. package/public/static/icon/OrderApprove.svg +1 -0
  31. package/public/static/icon/OrderInactive.svg +1 -0
  32. package/public/static/icon/OrderPlay.svg +1 -0
  33. package/public/static/icon/PayApprove.svg +1 -0
  34. package/public/static/icon/PayInactive.svg +1 -0
  35. package/public/static/icon/PrintInactive.svg +1 -0
  36. package/public/static/icon/PrintPlay.svg +1 -0
  37. package/public/static/icon/Search.svg +1 -0
  38. package/public/static/icon/ShippingDone.svg +1 -0
  39. package/public/static/icon/Support Agent.svg +1 -0
  40. package/public/static/icon/Task.svg +1 -0
  41. package/public/static/icon/User.svg +1 -0
  42. package/public/static/icon/Warning Filled.svg +1 -0
  43. package/public/static/icon/Warning.svg +1 -0
  44. package/public/static/icon/add_a_photo.svg +3 -0
  45. package/public/static/icon/attach_file.svg +1 -0
  46. package/public/static/icon/check_circle.svg +1 -0
  47. package/public/static/icon/close.svg +1 -0
  48. package/public/static/icon/download.svg +1 -0
  49. package/public/static/icon/expand all.svg +1 -0
  50. package/public/static/icon/expand_less.svg +1 -0
  51. package/public/static/icon/expand_more.svg +1 -0
  52. package/public/static/icon/factory.svg +1 -0
  53. package/public/static/icon/inventory.svg +1 -0
  54. package/public/static/icon/listAll.svg +1 -0
  55. package/public/static/icon/local_shipping.svg +1 -0
  56. package/public/static/icon/logout.svg +1 -0
  57. package/public/static/icon/menu.svg +1 -0
  58. package/public/static/icon/mode_edit.svg +4 -0
  59. package/public/static/icon/more.svg +1 -0
  60. package/public/static/icon/outward.svg +1 -0
  61. package/public/static/icon/personcard.svg +1 -0
  62. package/public/static/icon/product.svg +1 -0
  63. package/public/static/icon/settings.svg +1 -0
  64. package/public/static/icon/swap_vert.svg +3 -0
  65. package/public/static/icon/upload.svg +1 -0
  66. package/public/static/icon/warning_amber.svg +3 -0
  67. package/public/static/image/Thumbnail.png +0 -0
  68. package/public/vite.svg +1 -0
  69. package/public/weing_logo_final.png +0 -0
  70. package/scripts/icons/setIcon.cjs +151 -0
  71. package/scripts/icons/svg-template.cjs +18 -0
  72. package/scripts/icons/svgo-config.json +9 -0
  73. package/scripts/icons/svgr-config.json +16 -0
  74. package/src/@types/color.d.ts +51 -0
  75. package/src/@types/common.d.ts +6 -0
  76. package/src/@types/extends/react.extends.d.ts +7 -0
  77. package/src/@types/quill-image-resize-module-react/index.d.ts +1 -0
  78. package/src/App.css +0 -0
  79. package/src/App.tsx +84 -0
  80. package/src/assets/react.svg +1 -0
  81. package/src/components/Accordion/Accordion.context.tsx +62 -0
  82. package/src/components/Accordion/Accordion.module.scss +163 -0
  83. package/src/components/Accordion/Accordion.tsx +242 -0
  84. package/src/components/Accordion/README.md +408 -0
  85. package/src/components/Accordion/images/as_trigger.gif +0 -0
  86. package/src/components/Accordion/images/closed.png +0 -0
  87. package/src/components/Accordion/images/open.png +0 -0
  88. package/src/components/Accordion/images/thumbnail1.png +0 -0
  89. package/src/components/Accordion/images/thumbnail2.png +0 -0
  90. package/src/components/Accordion/images/thumbnail3.png +0 -0
  91. package/src/components/Avatar/Avatar.module.scss +47 -0
  92. package/src/components/Avatar/Avatar.tsx +230 -0
  93. package/src/components/Avatar/README.md +333 -0
  94. package/src/components/Badge/Badge.module.scss +49 -0
  95. package/src/components/Badge/Badge.tsx +87 -0
  96. package/src/components/Badge/README.md +223 -0
  97. package/src/components/BarCode/BarCode.module.scss +0 -0
  98. package/src/components/BarCode/BarCode.tsx +153 -0
  99. package/src/components/Breadcrumb/Breadcrumb.module.scss +36 -0
  100. package/src/components/Breadcrumb/Breadcrumb.tsx +114 -0
  101. package/src/components/Breadcrumb/README.md +339 -0
  102. package/src/components/Button/Button.colors.tsx +322 -0
  103. package/src/components/Button/Button.context.tsx +47 -0
  104. package/src/components/Button/Button.module.scss +156 -0
  105. package/src/components/Button/Button.tsx +190 -0
  106. package/src/components/Button/Button.type.ts +11 -0
  107. package/src/components/Calendar/Calendar.context.tsx +138 -0
  108. package/src/components/Calendar/Calendar.module.scss +116 -0
  109. package/src/components/Calendar/Calendar.tsx +440 -0
  110. package/src/components/Calendar/dayjs.util.ts +312 -0
  111. package/src/components/Cascader/Cacader.data.ts +1185 -0
  112. package/src/components/Cascader/Cascader.context.tsx +61 -0
  113. package/src/components/Cascader/Cascader.hook.ts +502 -0
  114. package/src/components/Cascader/Cascader.module.scss +143 -0
  115. package/src/components/Cascader/Cascader.tsx +281 -0
  116. package/src/components/Cascader/README.md +735 -0
  117. package/src/components/Chart/Chart.module.scss +0 -0
  118. package/src/components/Chart/Chart.tsx +230 -0
  119. package/src/components/Chart/README.md +85 -0
  120. package/src/components/Chart/images/barChart.png +0 -0
  121. package/src/components/Chips/Chips.colors.tsx +185 -0
  122. package/src/components/Chips/Chips.module.scss +49 -0
  123. package/src/components/Chips/Chips.tsx +78 -0
  124. package/src/components/Comment/Comment.module.scss +14 -0
  125. package/src/components/Comment/Comment.tsx +105 -0
  126. package/src/components/Dialog/Dialog.module.scss +0 -0
  127. package/src/components/Dialog/Dialog.tsx +12 -0
  128. package/src/components/Divider/Divider.module.scss +12 -0
  129. package/src/components/Divider/Divider.tsx +33 -0
  130. package/src/components/Editor/Editor.context.tsx +12 -0
  131. package/src/components/Editor/Editor.module.scss +40 -0
  132. package/src/components/Editor/Editor.tsx +174 -0
  133. package/src/components/Form/CheckBox/CheckBox.context.tsx +56 -0
  134. package/src/components/Form/CheckBox/CheckBox.module.scss +81 -0
  135. package/src/components/Form/CheckBox/CheckBox.tsx +196 -0
  136. package/src/components/Form/CheckBox/Checkbox.colors.tsx +42 -0
  137. package/src/components/Form/Dropdown/Dropdown.colors.tsx +125 -0
  138. package/src/components/Form/Dropdown/Dropdown.context.tsx +62 -0
  139. package/src/components/Form/Dropdown/Dropdown.module.scss +133 -0
  140. package/src/components/Form/Dropdown/Dropdown.tsx +404 -0
  141. package/src/components/Form/OTPInput/OTPInput.context.tsx +104 -0
  142. package/src/components/Form/OTPInput/OTPInput.module.scss +22 -0
  143. package/src/components/Form/OTPInput/OTPInput.tsx +67 -0
  144. package/src/components/Form/Radio/Radio.colors.tsx +42 -0
  145. package/src/components/Form/Radio/Radio.context.tsx +40 -0
  146. package/src/components/Form/Radio/Radio.module.scss +61 -0
  147. package/src/components/Form/Radio/Radio.tsx +174 -0
  148. package/src/components/Form/Switch/Switch.colors.tsx +42 -0
  149. package/src/components/Form/Switch/Switch.context.tsx +50 -0
  150. package/src/components/Form/Switch/Switch.module.scss +53 -0
  151. package/src/components/Form/Switch/Switch.tsx +151 -0
  152. package/src/components/Form/TextArea/TextArea.colors.tsx +76 -0
  153. package/src/components/Form/TextArea/TextArea.context.tsx +97 -0
  154. package/src/components/Form/TextArea/TextArea.module.scss +138 -0
  155. package/src/components/Form/TextArea/TextArea.tsx +246 -0
  156. package/src/components/Form/TextInput/TextInput.colors.tsx +76 -0
  157. package/src/components/Form/TextInput/TextInput.context.tsx +106 -0
  158. package/src/components/Form/TextInput/TextInput.module.scss +160 -0
  159. package/src/components/Form/TextInput/TextInput.tsx +225 -0
  160. package/src/components/GlobalStyle/GlobalStyle.tsx +20 -0
  161. package/src/components/HelperText/HelperText.module.scss +28 -0
  162. package/src/components/HelperText/HelperText.tsx +130 -0
  163. package/src/components/Icon/Icon.tsx +29 -0
  164. package/src/components/Icons/Add.tsx +20 -0
  165. package/src/components/Icons/AddAPhoto.tsx +21 -0
  166. package/src/components/Icons/ApprovalInactive.tsx +26 -0
  167. package/src/components/Icons/ArrowRight.tsx +24 -0
  168. package/src/components/Icons/AttachFile.tsx +20 -0
  169. package/src/components/Icons/Cancle.tsx +20 -0
  170. package/src/components/Icons/CancleFilled.tsx +20 -0
  171. package/src/components/Icons/Cart.tsx +20 -0
  172. package/src/components/Icons/CartFilled.tsx +20 -0
  173. package/src/components/Icons/Check.tsx +20 -0
  174. package/src/components/Icons/CheckBoxRound.tsx +21 -0
  175. package/src/components/Icons/CheckCircle.tsx +35 -0
  176. package/src/components/Icons/Close.tsx +20 -0
  177. package/src/components/Icons/Company.tsx +20 -0
  178. package/src/components/Icons/Download.tsx +20 -0
  179. package/src/components/Icons/DragHandle.tsx +20 -0
  180. package/src/components/Icons/Eco.tsx +20 -0
  181. package/src/components/Icons/EditSquare.tsx +20 -0
  182. package/src/components/Icons/Error.tsx +20 -0
  183. package/src/components/Icons/ErrorFilled.tsx +20 -0
  184. package/src/components/Icons/ExpandAll.tsx +20 -0
  185. package/src/components/Icons/ExpandLess.tsx +20 -0
  186. package/src/components/Icons/ExpandMore.tsx +20 -0
  187. package/src/components/Icons/Factory.tsx +20 -0
  188. package/src/components/Icons/FileUpload.tsx +20 -0
  189. package/src/components/Icons/FilecheckInactive.tsx +22 -0
  190. package/src/components/Icons/GoBefore.tsx +20 -0
  191. package/src/components/Icons/GoNext.tsx +20 -0
  192. package/src/components/Icons/HamburgerMenu.tsx +20 -0
  193. package/src/components/Icons/ImagecheckPlay.tsx +27 -0
  194. package/src/components/Icons/Inventory.tsx +20 -0
  195. package/src/components/Icons/KakaoTalk.tsx +22 -0
  196. package/src/components/Icons/ListAll.tsx +20 -0
  197. package/src/components/Icons/LocalShipping.tsx +20 -0
  198. package/src/components/Icons/Logout.tsx +20 -0
  199. package/src/components/Icons/Menu.tsx +20 -0
  200. package/src/components/Icons/Minus.tsx +18 -0
  201. package/src/components/Icons/ModeEdit.tsx +25 -0
  202. package/src/components/Icons/More.tsx +20 -0
  203. package/src/components/Icons/Naver.tsx +20 -0
  204. package/src/components/Icons/NaverBlog.tsx +22 -0
  205. package/src/components/Icons/OrderApprove.tsx +20 -0
  206. package/src/components/Icons/OrderInactive.tsx +20 -0
  207. package/src/components/Icons/OrderPlay.tsx +20 -0
  208. package/src/components/Icons/Outward.tsx +20 -0
  209. package/src/components/Icons/PayApprove.tsx +24 -0
  210. package/src/components/Icons/PayInactive.tsx +24 -0
  211. package/src/components/Icons/Personcard.tsx +20 -0
  212. package/src/components/Icons/PrintInactive.tsx +24 -0
  213. package/src/components/Icons/PrintPlay.tsx +24 -0
  214. package/src/components/Icons/Product.tsx +20 -0
  215. package/src/components/Icons/Search.tsx +20 -0
  216. package/src/components/Icons/Settings.tsx +20 -0
  217. package/src/components/Icons/ShippingDone.tsx +20 -0
  218. package/src/components/Icons/SupportAgent.tsx +20 -0
  219. package/src/components/Icons/SwapVert.tsx +21 -0
  220. package/src/components/Icons/Task.tsx +20 -0
  221. package/src/components/Icons/Upload.tsx +20 -0
  222. package/src/components/Icons/User.tsx +20 -0
  223. package/src/components/Icons/Warning.tsx +20 -0
  224. package/src/components/Icons/WarningAmber.tsx +21 -0
  225. package/src/components/Icons/WarningFilled.tsx +20 -0
  226. package/src/components/Icons/index.ts +62 -0
  227. package/src/components/LNB/LNB.context.tsx +32 -0
  228. package/src/components/LNB/LNB.module.scss +99 -0
  229. package/src/components/LNB/LNB.tsx +190 -0
  230. package/src/components/LNB/README.md +411 -0
  231. package/src/components/LNB/makeNavigation.ts +51 -0
  232. package/src/components/LNB/navigation.d.ts +15 -0
  233. package/src/components/LNB/navigation.json +211 -0
  234. package/src/components/Label/Label.colors.tsx +241 -0
  235. package/src/components/Label/Label.module.scss +31 -0
  236. package/src/components/Label/Label.tsx +54 -0
  237. package/src/components/LazyImage/LazyImage.module.scss +6 -0
  238. package/src/components/LazyImage/LazyImage.tsx +107 -0
  239. package/src/components/List/List.module.scss +81 -0
  240. package/src/components/List/List.tsx +91 -0
  241. package/src/components/List/README.md +87 -0
  242. package/src/components/MobilePicker/MobilePicker.context.tsx +22 -0
  243. package/src/components/MobilePicker/MobilePicker.module.scss +57 -0
  244. package/src/components/MobilePicker/MobilePicker.tsx +336 -0
  245. package/src/components/MobilePicker/README.md +159 -0
  246. package/src/components/Modal/Modal.tsx +77 -0
  247. package/src/components/Modal/README.md +130 -0
  248. package/src/components/Pagination/Pagination.colors.tsx +85 -0
  249. package/src/components/Pagination/Pagination.context.tsx +28 -0
  250. package/src/components/Pagination/Pagination.module.scss +60 -0
  251. package/src/components/Pagination/Pagination.tsx +234 -0
  252. package/src/components/Popup/Popup.context.tsx +15 -0
  253. package/src/components/Popup/Popup.module.scss +53 -0
  254. package/src/components/Popup/Popup.tsx +116 -0
  255. package/src/components/Popup/README.md +170 -0
  256. package/src/components/QRCode/QRCode.module.scss +2 -0
  257. package/src/components/QRCode/QRCode.tsx +61 -0
  258. package/src/components/ScrollCalendar/ScrollCalendar.context.tsx +26 -0
  259. package/src/components/ScrollCalendar/ScrollCalendar.module.scss +42 -0
  260. package/src/components/ScrollCalendar/ScrollCalendar.tsx +448 -0
  261. package/src/components/ScrollSpy/README.md +62 -0
  262. package/src/components/ScrollSpy/ScrollSpy.tsx +130 -0
  263. package/src/components/Sheet/README.md +92 -0
  264. package/src/components/Sheet/Sheet.context.tsx +23 -0
  265. package/src/components/Sheet/Sheet.module.scss +68 -0
  266. package/src/components/Sheet/Sheet.tsx +146 -0
  267. package/src/components/Slider/README.md +639 -0
  268. package/src/components/Slider/Slider.backup.tsx +477 -0
  269. package/src/components/Slider/Slider.context.tsx +67 -0
  270. package/src/components/Slider/Slider.module.scss +123 -0
  271. package/src/components/Slider/Slider.tsx +467 -0
  272. package/src/components/Stepper/README.md +320 -0
  273. package/src/components/Stepper/Stepper.context.tsx +49 -0
  274. package/src/components/Stepper/Stepper.module.scss +163 -0
  275. package/src/components/Stepper/Stepper.tsx +219 -0
  276. package/src/components/Tab/Tab.colors.tsx +54 -0
  277. package/src/components/Tab/Tab.context.tsx +64 -0
  278. package/src/components/Tab/Tab.module.scss +239 -0
  279. package/src/components/Tab/Tab.tsx +123 -0
  280. package/src/components/Tab/cx.ts +6 -0
  281. package/src/components/Table/README.md +162 -0
  282. package/src/components/Table/Table.context.tsx +23 -0
  283. package/src/components/Table/Table.hook.ts +51 -0
  284. package/src/components/Table/Table.module.scss +83 -0
  285. package/src/components/Table/Table.tsx +147 -0
  286. package/src/components/Thumbnail/README.md +302 -0
  287. package/src/components/Thumbnail/Thumbnail.context.tsx +42 -0
  288. package/src/components/Thumbnail/Thumbnail.module.scss +149 -0
  289. package/src/components/Thumbnail/Thumbnail.tsx +391 -0
  290. package/src/components/TimeInput/README.md +118 -0
  291. package/src/components/TimeInput/TimeInput.colors.tsx +76 -0
  292. package/src/components/TimeInput/TimeInput.context.tsx +58 -0
  293. package/src/components/TimeInput/TimeInput.module.scss +211 -0
  294. package/src/components/TimeInput/TimeInput.tsx +411 -0
  295. package/src/components/WeeklyCalendar/WeeklyCalendar.context.tsx +88 -0
  296. package/src/components/WeeklyCalendar/WeeklyCalendar.module.scss +225 -0
  297. package/src/components/WeeklyCalendar/WeeklyCalendar.tsx +772 -0
  298. package/src/components/WeeklyCalendar/dayjs.util.ts +336 -0
  299. package/src/components/WeeklyCalendar/weeklyCalendar.util.ts +583 -0
  300. package/src/components/WisywygEditor/Quill/Editor.tsx +148 -0
  301. package/src/constant/locale.constant.ts +6 -0
  302. package/src/globals.scss +80 -0
  303. package/src/hooks/useColorScheme.tsx +48 -0
  304. package/src/hooks/useElementRect.tsx +128 -0
  305. package/src/hooks/useIntersectionObserver.tsx +33 -0
  306. package/src/index.css +17 -0
  307. package/src/index.ts +187 -0
  308. package/src/main.tsx +10 -0
  309. package/src/styles/_fontSize.mixin.scss +57 -0
  310. package/src/styles/_fontWeight.mixin.scss +15 -0
  311. package/src/styles/_lineHeight.mixin.scss +57 -0
  312. package/src/styles/_scrollbar.mixin.scss +49 -0
  313. package/src/styles/baseColor.ts +297 -0
  314. package/src/styles/color.ts +272 -0
  315. package/src/styles/color2.ts +200 -0
  316. package/src/styles/scrollbar.README.md +72 -0
  317. package/src/styles/typography.scss +170 -0
  318. package/src/utils/aws.util.ts +180 -0
  319. package/src/utils/common.utill.ts +45 -0
  320. package/src/vite-env.d.ts +1 -0
  321. package/tsconfig.app.json +24 -0
  322. package/tsconfig.json +29 -0
  323. package/tsconfig.node.json +10 -0
  324. package/tsconfig.node.tsbuildinfo +1 -0
  325. package/tsconfig.tsbuildinfo +1 -0
  326. package/vite.config.d.ts +2 -0
  327. package/vite.config.js +47 -0
  328. package/vite.config.ts +50 -0
@@ -0,0 +1,391 @@
1
+ import React from "react";
2
+ import {
3
+ initialState,
4
+ THUMBNAIL_SIZES,
5
+ ThumbnailContext,
6
+ ThumbnailProviderProps,
7
+ ThumbnailSize,
8
+ } from "./Thumbnail.context";
9
+ import cn from "classnames/bind";
10
+ import styles from "./Thumbnail.module.scss";
11
+ import { Icon } from "../Icon/Icon";
12
+
13
+ const cx = cn.bind(styles);
14
+
15
+ type RemoveButtonProps = {
16
+ index?: number;
17
+ iconSize?: number;
18
+ } & React.JSX.IntrinsicElements["div"];
19
+ const RemoveButton = (props: RemoveButtonProps) => {
20
+ const { index, iconSize, ...rest } = props;
21
+ const { size, variant, onRemove } = React.use(ThumbnailContext);
22
+
23
+ const getPixelSize = React.useCallback((size?: ThumbnailSize) => {
24
+ if (!size) return 0;
25
+ return typeof size === "string" ? THUMBNAIL_SIZES[size] : size;
26
+ }, []);
27
+
28
+ const styles = React.useMemo(() => {
29
+ const pixelSize = getPixelSize(size);
30
+
31
+ if (variant === "circle") {
32
+ return {
33
+ iconSize:
34
+ iconSize ||
35
+ (() => {
36
+ if (typeof size !== "string") return 24;
37
+ if (pixelSize <= 72) return 16;
38
+ if (pixelSize <= 96) return 20;
39
+ return 24;
40
+ })(),
41
+ borderWidth: pixelSize <= 72 ? 2 : 3,
42
+ exactSize: (() => {
43
+ if (pixelSize <= 72) return 24;
44
+ if (pixelSize <= 96) return 32;
45
+ if (pixelSize <= 120) return 40;
46
+ return 48;
47
+ })(),
48
+ };
49
+ }
50
+
51
+ return {
52
+ iconSize:
53
+ iconSize ||
54
+ (() => {
55
+ if (pixelSize <= 72) return 16;
56
+ if (pixelSize <= 96) return 20;
57
+ return 24;
58
+ })(),
59
+ exactSize: (() => {
60
+ if (pixelSize <= 96) return 24;
61
+ if (pixelSize <= 160) return 40;
62
+ return 48;
63
+ })(),
64
+ };
65
+ }, [variant, size, iconSize, getPixelSize]);
66
+
67
+ return (
68
+ <div
69
+ {...rest}
70
+ className={cx("RemoveButton", rest.className, variant)}
71
+ style={{
72
+ "--borderWidth": `${styles.borderWidth}px`,
73
+ "--exactSize": `${styles.exactSize}px`,
74
+ ...rest.style,
75
+ }}
76
+ onClick={(e) => {
77
+ props.onClick?.(e);
78
+ onRemove?.(index);
79
+ }}
80
+ >
81
+ <Icon name="Close" size={styles.iconSize} color="white" />
82
+ </div>
83
+ );
84
+ };
85
+
86
+ type EditButtonProps = {
87
+ index?: number;
88
+ iconName?: React.ComponentProps<typeof Icon>["name"];
89
+ iconSize?: number;
90
+ } & React.JSX.IntrinsicElements["div"];
91
+
92
+ const EditButton = (props: EditButtonProps) => {
93
+ const { index, iconName = "ModeEdit", ...rest } = props;
94
+ const { size, variant, onEdit } = React.use(ThumbnailContext);
95
+
96
+ const id = React.useId();
97
+
98
+ const getPixelSize = React.useCallback((size?: ThumbnailSize) => {
99
+ if (!size) return 0;
100
+ return typeof size === "string" ? THUMBNAIL_SIZES[size] : size;
101
+ }, []);
102
+
103
+ const styles = React.useMemo(() => {
104
+ const pixelSize = getPixelSize(size);
105
+ return {
106
+ iconSize:
107
+ props.iconSize ||
108
+ (() => {
109
+ if (typeof size !== "string") return 24;
110
+ if (pixelSize <= 72) return 16;
111
+ if (pixelSize <= 96) return 20;
112
+ return 24;
113
+ })(),
114
+ borderWidth: pixelSize <= 72 ? 2 : 3,
115
+ translate: pixelSize <= 120 ? 8 : 12,
116
+ exactSize: (() => {
117
+ if (pixelSize <= 72) return 24;
118
+ if (pixelSize <= 96) return 32;
119
+ if (pixelSize <= 120) return 40;
120
+ return 48;
121
+ })(),
122
+ };
123
+ }, [size, props.iconSize, getPixelSize]);
124
+
125
+ return (
126
+ <label htmlFor={id}>
127
+ <div
128
+ {...rest}
129
+ className={cx("EditButton", rest.className, variant)}
130
+ style={{
131
+ "--borderWidth": `${styles.borderWidth}px`,
132
+ "--translate": `${styles.translate}px`,
133
+ "--exactSize": `${styles.exactSize}px`,
134
+ ...rest.style,
135
+ }}
136
+ >
137
+ <Icon name={iconName} size={styles.iconSize} />
138
+ </div>
139
+ <input
140
+ className={cx("FileInput")}
141
+ type="file"
142
+ id={id}
143
+ onChange={(e) => {
144
+ props.onChange?.(e);
145
+ onEdit?.(e, index);
146
+ }}
147
+ />
148
+ </label>
149
+ );
150
+ };
151
+
152
+ type EmptyImageProps = {
153
+ iconName?: React.ComponentProps<typeof Icon>["name"];
154
+ iconSize?: number;
155
+ } & React.JSX.IntrinsicElements["div"];
156
+ const EmptyImage = (props: EmptyImageProps) => {
157
+ const { iconName = "AddAPhoto", ...rest } = props;
158
+ const { size, aspectRatio, variant } = React.use(ThumbnailContext);
159
+
160
+ const iconSize = React.useMemo(() => {
161
+ if (typeof size === "string") {
162
+ const pixelSize = THUMBNAIL_SIZES[size];
163
+ if (pixelSize <= 56) return 16;
164
+ if (pixelSize <= 72) return 20;
165
+ return 24;
166
+ }
167
+ return 24;
168
+ }, [size]);
169
+
170
+ return (
171
+ <div
172
+ {...rest}
173
+ className={cx("EmptyImage", rest.className, variant)}
174
+ style={{
175
+ "--aspectRatio": aspectRatio,
176
+ ...(typeof size === "string"
177
+ ? { width: THUMBNAIL_SIZES[size] }
178
+ : {
179
+ width: size,
180
+ }),
181
+ ...rest.style,
182
+ }}
183
+ >
184
+ <Icon name={iconName} size={props.iconSize || iconSize} />
185
+ <span className={cx("Text")}>사진 추가</span>
186
+ </div>
187
+ );
188
+ };
189
+
190
+ type UploadProps = React.PropsWithChildren<
191
+ React.JSX.IntrinsicElements["input"]
192
+ >;
193
+ const Upload = (props: UploadProps) => {
194
+ const { children, id, ...rest } = props;
195
+ const { onUpload } = React.use(ThumbnailContext);
196
+
197
+ return (
198
+ <label htmlFor={id}>
199
+ <input
200
+ {...rest}
201
+ type="file"
202
+ id={id}
203
+ className={cx("UploadInput")}
204
+ onChange={rest.onChange ?? onUpload}
205
+ />
206
+ {children}
207
+ </label>
208
+ );
209
+ };
210
+
211
+ type ImageProps = React.PropsWithChildren<
212
+ { index?: number } & React.JSX.IntrinsicElements["img"]
213
+ >;
214
+ const ImageComponent = (props: ImageProps) => {
215
+ const { children, index, ...rest } = props;
216
+ const { size, aspectRatio, variant, objectFit, isReload, onEdit, onRemove } =
217
+ React.use(ThumbnailContext);
218
+ const [isLoading, setIsLoading] = React.useState(true);
219
+ const [hasError, setHasError] = React.useState(false);
220
+
221
+ const [src, setSrc] = React.useState(rest.src);
222
+
223
+ // Check if EditButton or RemoveButton exists in children
224
+ const hasEditButton = React.Children.toArray(children).some(
225
+ (child) => React.isValidElement(child) && child.type === EditButton
226
+ );
227
+ const hasRemoveButton = React.Children.toArray(children).some(
228
+ (child) => React.isValidElement(child) && child.type === RemoveButton
229
+ );
230
+
231
+ const iconSize = React.useMemo(() => {
232
+ if (typeof size === "string") {
233
+ const pixelSize = THUMBNAIL_SIZES[size];
234
+ if (pixelSize <= 96) return 24;
235
+ if (pixelSize <= 160) return 32;
236
+ return 40;
237
+ }
238
+ return 24;
239
+ }, [size]);
240
+
241
+ const errorImageRefCallback = React.useCallback(() => {
242
+ if (src && isReload)
243
+ return () => {
244
+ setIsLoading(true);
245
+ setHasError(false);
246
+ };
247
+ }, [src, isReload]);
248
+
249
+ React.useEffect(() => {
250
+ setSrc(rest.src);
251
+ }, [rest.src]);
252
+
253
+ if (!src || hasError) {
254
+ return (
255
+ <div className={cx("ImageWrapper")}>
256
+ <div
257
+ ref={errorImageRefCallback}
258
+ className={cx("ErrorImage", rest.className, variant)}
259
+ style={{
260
+ "--aspectRatio": aspectRatio,
261
+ ...(typeof size === "string"
262
+ ? { width: THUMBNAIL_SIZES[size] }
263
+ : {
264
+ width: size,
265
+ }),
266
+ ...rest.style,
267
+ }}
268
+ >
269
+ <Icon name="WarningAmber" size={iconSize} />
270
+ {children}
271
+ </div>
272
+ {hasEditButton || hasRemoveButton ? (
273
+ children
274
+ ) : (
275
+ <>
276
+ {onEdit && <EditButton index={index} />}
277
+ {onRemove && <RemoveButton index={index} />}
278
+ </>
279
+ )}
280
+ </div>
281
+ );
282
+ }
283
+
284
+ return (
285
+ <div className={cx("ImageWrapper")}>
286
+ <img
287
+ {...rest}
288
+ src={src}
289
+ className={cx("Image", rest.className, variant, objectFit, {
290
+ isLoading,
291
+ })}
292
+ style={{
293
+ "--aspectRatio": aspectRatio,
294
+ ...(typeof size === "string"
295
+ ? { width: THUMBNAIL_SIZES[size] }
296
+ : {
297
+ width: size,
298
+ }),
299
+ ...rest.style,
300
+ }}
301
+ onLoad={() => {
302
+ setIsLoading(false);
303
+ setHasError(false);
304
+ }}
305
+ onError={() => {
306
+ setIsLoading(false);
307
+ setHasError(true);
308
+ }}
309
+ />
310
+ {hasEditButton || hasRemoveButton ? (
311
+ children
312
+ ) : (
313
+ <>
314
+ {onEdit && <EditButton index={index} />}
315
+ {onRemove && <RemoveButton index={index} />}
316
+ </>
317
+ )}
318
+ </div>
319
+ );
320
+ };
321
+
322
+ type ThumbnailRootProps = ThumbnailProviderProps &
323
+ React.JSX.IntrinsicElements["div"];
324
+ const Root = (props: ThumbnailRootProps) => {
325
+ const {
326
+ children,
327
+ size,
328
+ aspectRatio,
329
+ variant,
330
+ objectFit,
331
+ isReload,
332
+ onEdit,
333
+ onRemove,
334
+ onUpload,
335
+ ...rest
336
+ } = props;
337
+
338
+ const contextValue = React.useMemo(
339
+ () => ({
340
+ ...initialState,
341
+ size,
342
+ aspectRatio,
343
+ variant,
344
+ objectFit,
345
+ isReload,
346
+ onEdit,
347
+ onRemove,
348
+ onUpload,
349
+ }),
350
+ [
351
+ size,
352
+ aspectRatio,
353
+ variant,
354
+ objectFit,
355
+ isReload,
356
+ onEdit,
357
+ onRemove,
358
+ onUpload,
359
+ ]
360
+ );
361
+ return (
362
+ <ThumbnailContext value={contextValue}>
363
+ <div {...rest} className={cx("Wrapper", rest.className)}>
364
+ {children}
365
+ </div>
366
+ </ThumbnailContext>
367
+ );
368
+ };
369
+
370
+ type IThumbnail = {
371
+ Root: typeof Root;
372
+ Image: typeof ImageComponent;
373
+ Upload: typeof Upload;
374
+ EmptyImage: typeof EmptyImage;
375
+ RemoveButton: typeof RemoveButton;
376
+ EditButton: typeof EditButton;
377
+ THUMBNAIL_SIZES: typeof THUMBNAIL_SIZES;
378
+ };
379
+ const Thumbnail: IThumbnail = {
380
+ Root,
381
+ Image: ImageComponent,
382
+ Upload,
383
+ EmptyImage,
384
+ RemoveButton,
385
+ EditButton,
386
+ THUMBNAIL_SIZES,
387
+ };
388
+
389
+ export type { ThumbnailRootProps };
390
+
391
+ export default Thumbnail;
@@ -0,0 +1,118 @@
1
+ # TimeInput UI
2
+
3
+ ## Contents List
4
+
5
+ 0. [Information](#Information)
6
+ 1. [Quick Start](#QuickStart)
7
+ 2. [Reference](#Reference)
8
+
9
+ ## Information
10
+
11
+ - TimeInput UI는 시간 범위를 입력받기 위한 컴포넌트입니다.
12
+ - 시작 시간과 종료 시간을 입력받을 수 있습니다.
13
+ - 각 입력 필드는 시간(0-23)과 분(0-59)으로 제한됩니다.
14
+ - 키보드 네비게이션을 지원합니다.
15
+ - 시간 범위 유효성 검사를 제공합니다.
16
+ - 다양한 스타일 변형(outlined/contained)을 지원합니다.
17
+
18
+ ## Quick Start
19
+
20
+ ```tsx
21
+ import { TimeInput } from "@weing-dev/ui-kit-primitive";
22
+
23
+ const Component = () => {
24
+ const handleTimeChange = (time: { start: string; end: string }) => {
25
+ console.log(time); // { start: "09:00", end: "18:00" }
26
+ };
27
+
28
+ return (
29
+ <TimeInput.Root onChange={handleTimeChange}>
30
+ <TimeInput.Label>시간</TimeInput.Label>
31
+ <TimeInput.InputWrapper>
32
+ <TimeInput.Input index={0} placeholder="00" />
33
+ <TimeInput.Colon />
34
+ <TimeInput.Input index={1} placeholder="00" />
35
+ <TimeInput.Separator />
36
+ <TimeInput.Input index={2} placeholder="00" />
37
+ <TimeInput.Colon />
38
+ <TimeInput.Input index={3} placeholder="00" />
39
+ </TimeInput.InputWrapper>
40
+ <TimeInput.HelperText />
41
+ </TimeInput.Root>
42
+ );
43
+ };
44
+ ```
45
+
46
+ ## Reference
47
+
48
+ ### Root
49
+
50
+ | Prop | Type | Required | Default Value |
51
+ | :-------- | :---------------------------------------------: | :------: | :-----------: |
52
+ | value | string | - | "" |
53
+ | type | "default" \| "range" | - | "range" |
54
+ | size | "small" \| "large" | - | "small" |
55
+ | disabled | boolean | - | false |
56
+ | readOnly | boolean | - | false |
57
+ | status | "error" \| "warning" \| "success" | - | - |
58
+ | variant | "outlined" \| "contained" | - | "outlined" |
59
+ | direction | "column" \| "row" | - | "column" |
60
+ | focused | boolean | - | false |
61
+ | onChange | (value: { start: string; end: string }) => void | - | - |
62
+
63
+ ### Input
64
+
65
+ | Prop | Type | Required | Default Value |
66
+ | :---------- | :----: | :------: | :-----------: |
67
+ | index | number | ✓ | - |
68
+ | placeholder | string | ✓ | - |
69
+
70
+ - `index`가 짝수(0,2)인 경우 시간(0-23) 입력
71
+ - `index`가 홀수(1,3)인 경우 분(0-59) 입력
72
+
73
+ ### Label
74
+
75
+ | Prop | Type | Required | Default Value |
76
+ | :----------- | :----: | :------: | :-----------: |
77
+ | focusedColor | string | - | - |
78
+ | className | string | - | - |
79
+
80
+ ### InputWrapper
81
+
82
+ 컴포넌트들을 감싸는 wrapper입니다. variant와 size prop에 따라 스타일이 변경됩니다.
83
+
84
+ ### Colon
85
+
86
+ 시간과 분을 구분하는 콜론(:) 컴포넌트입니다.
87
+
88
+ ### Separator
89
+
90
+ 시작 시간과 종료 시간을 구분하는 구분자(-) 컴포넌트입니다.
91
+
92
+ ### HelperText
93
+
94
+ 오류 메시지를 표시하는 컴포넌트입니다. status prop에 따라 스타일이 변경됩니다.
95
+
96
+ ## Features
97
+
98
+ ### 키보드 네비게이션
99
+
100
+ - `Tab` / `Shift+Tab`: 다음/이전 입력 필드로 이동
101
+ - `Backspace`: 입력값이 비어있을 때 이전 필드로 이동
102
+
103
+ ### 입력 제한
104
+
105
+ - 시간 필드(index 0,2): 0-23 범위로 제한
106
+ - 분 필드(index 1,3): 0-59 범위로 제한
107
+ - 최대값 초과 시 자동으로 최대값으로 설정
108
+
109
+ ### 유효성 검사
110
+
111
+ - 종료 시간이 시작 시간보다 작거나 같을 경우 오류 상태로 변경
112
+ - 오류 상태일 때 HelperText를 통해 오류 메시지 표시
113
+
114
+ ### 스타일 변형
115
+
116
+ - `outlined`: 테두리가 있는 스타일 (기본값)
117
+ - `contained`: 배경색이 있는 스타일
118
+ - 각 상태(focused, disabled, error 등)에 따른 스타일 자동 적용
@@ -0,0 +1,76 @@
1
+ import { colorThemes as colorSet } from "../../styles/color2";
2
+
3
+ const useTimeInputColors = () => {
4
+ const outlined = {
5
+ inactive: {
6
+ backgroundColor: "transparent",
7
+ color: `var(--textPrimary, ${colorSet.light.textPrimary})`,
8
+ placeholder: `var(--textTertiary, ${colorSet.light.textTertiary})`,
9
+ borderColor: `var(--componentTextfieldOutline, ${colorSet.light.componentTextfieldOutline})`,
10
+ },
11
+ focused: {
12
+ backgroundColor: "transparent",
13
+ color: `var(--textPrimary, ${colorSet.light.textPrimary})`,
14
+ placeholder: `var(--textTertiary, ${colorSet.light.textTertiary})`,
15
+ borderColor: `var(--actionAction, ${colorSet.light.actionAction})`,
16
+ },
17
+ readOnly: {
18
+ backgroundColor: "transparent",
19
+ color: `var(--textPrimary, ${colorSet.light.textPrimary})`,
20
+ placeholder: `var(--textTertiary, ${colorSet.light.textTertiary})`,
21
+ borderColor: `var(--componentTextfieldOutline, ${colorSet.light.componentTextfieldOutline})`,
22
+ },
23
+ disabled: {
24
+ backgroundColor: `var(--actionDisabledBackground, ${colorSet.light.actionDisabledBackground})`,
25
+ color: `var(--textDisabled, ${colorSet.light.textDisabled})`,
26
+ placeholder: `var(--textDisabled, ${colorSet.light.textDisabled})`,
27
+ borderColor: `var(--componentTextfieldOutline, ${colorSet.light.componentTextfieldOutline})`,
28
+ },
29
+ error: {
30
+ backgroundColor: "transparent",
31
+ color: `var(--textError, ${colorSet.light.textError})`,
32
+ placeholder: `var(--textError, ${colorSet.light.textError})`,
33
+ borderColor: `var(--componentTextfieldOutline, ${colorSet.light.componentTextfieldOutline})`,
34
+ },
35
+ };
36
+
37
+ const contained = {
38
+ inactive: {
39
+ backgroundColor: `var(--componentTextfieldFilled, ${colorSet.light.componentTextfieldFilled})`,
40
+ color: `var(--textPrimary, ${colorSet.light.textPrimary})`,
41
+ placeholder: `var(--textTertiary, ${colorSet.light.textTertiary})`,
42
+ borderColor: "transparent",
43
+ },
44
+ focused: {
45
+ backgroundColor: `var(--componentTextfieldFilled, ${colorSet.light.componentTextfieldFilled})`,
46
+ color: `var(--textPrimary, ${colorSet.light.textPrimary})`,
47
+ placeholder: `var(--textTertiary, ${colorSet.light.textTertiary})`,
48
+ borderColor: `var(--actionAction, ${colorSet.light.actionAction})`,
49
+ },
50
+ readOnly: {
51
+ backgroundColor: `var(--componentTextfieldFilled, ${colorSet.light.componentTextfieldFilled})`,
52
+ color: `var(--textPrimary, ${colorSet.light.textPrimary})`,
53
+ placeholder: `var(--textTertiary, ${colorSet.light.textTertiary})`,
54
+ borderColor: "transparent",
55
+ },
56
+ disabled: {
57
+ backgroundColor: `var(--actionDisabledBackground, ${colorSet.light.actionDisabledBackground})`,
58
+ color: `var(--textDisabled, ${colorSet.light.textDisabled})`,
59
+ placeholder: `var(--textDisabled, ${colorSet.light.textDisabled})`,
60
+ borderColor: "transparent",
61
+ },
62
+ error: {
63
+ backgroundColor: `var(--componentTextfieldFilled, ${colorSet.light.componentTextfieldFilled})`,
64
+ color: `var(--textError, ${colorSet.light.textError})`,
65
+ placeholder: `var(--textError, ${colorSet.light.textError})`,
66
+ borderColor: "transparent",
67
+ },
68
+ };
69
+
70
+ return {
71
+ outlined,
72
+ contained,
73
+ };
74
+ };
75
+
76
+ export default useTimeInputColors;
@@ -0,0 +1,58 @@
1
+ import React from "react";
2
+
3
+ export type TimeInputContextType = {
4
+ /** 입력값 */
5
+ value: string;
6
+ /** 크기 */
7
+ size?: "small" | "large";
8
+ /** 비활성화 여부 */
9
+ disabled?: boolean;
10
+ /** 읽기 전용 여부 */
11
+ readOnly?: boolean;
12
+ /** focused */
13
+ focused?: boolean;
14
+ /** 방향 */
15
+ direction?: "column" | "row";
16
+ /** flex basis */
17
+ flexBasis?: string;
18
+ /** 입력 요소 참조 */
19
+ inputRefs?: React.RefObject<HTMLInputElement[]>;
20
+ /** 상태 */
21
+ status?: "error" | "warning" | "success";
22
+ /** variant */
23
+ variant?: "outlined" | "contained";
24
+ /** 값 변경 핸들러 */
25
+ onChange?: (value: { start: string; end: string }) => void;
26
+ /** 입력값 변경 처리 */
27
+ handleChange?: (
28
+ e: React.ChangeEvent<HTMLInputElement>,
29
+ index: number
30
+ ) => void;
31
+ /** 포커스 아웃 처리 */
32
+ handleBlur?: (value: string, max: number, index: number) => void;
33
+ /** 키 입력 처리 */
34
+ handleKeyDown?: (
35
+ e: React.KeyboardEvent<HTMLInputElement>,
36
+ index: number
37
+ ) => void;
38
+ /** 포커스 처리 */
39
+ handleFocus?: () => void;
40
+ };
41
+
42
+ const initialState: TimeInputContextType = {
43
+ value: "",
44
+ variant: "outlined",
45
+ focused: false,
46
+ direction: "column",
47
+ };
48
+
49
+ export const TimeInputContext =
50
+ React.createContext<TimeInputContextType>(initialState);
51
+
52
+ export type TimeInputProviderProps = React.PropsWithChildren<{
53
+ value?: string[];
54
+ type?: "default" | "range";
55
+ disabled?: boolean;
56
+ readOnly?: boolean;
57
+ onChange?: (value: { start: string; end: string }) => void;
58
+ }>;