@seakoi/native-ui 1.1.3 → 1.2.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 (224) hide show
  1. package/CHANGELOG.md +17 -0
  2. package/dist/commonjs/components/base/carousel/carousel-indicator.js +56 -0
  3. package/dist/commonjs/components/base/carousel/carousel-slides.js +140 -0
  4. package/dist/commonjs/components/base/carousel/carousel.js +114 -122
  5. package/dist/commonjs/components/base/carousel/hooks/index.js +0 -14
  6. package/dist/commonjs/components/base/carousel/hooks/use-carousel-index.js +16 -13
  7. package/dist/commonjs/components/base/carousel/hooks/use-carousel-lifecycle.js +6 -2
  8. package/dist/commonjs/components/base/carousel/hooks/use-carousel-pan-responder.js +40 -12
  9. package/dist/commonjs/components/base/carousel/hooks/use-carousel-position.js +6 -2
  10. package/dist/commonjs/components/base/carousel/index.js +1 -15
  11. package/dist/commonjs/components/base/carousel/style/index.js +12 -0
  12. package/dist/commonjs/components/base/date-picker/date-picker.js +56 -44
  13. package/dist/commonjs/components/base/date-picker/date-range-picker.js +142 -50
  14. package/dist/commonjs/components/base/date-picker/style/index.js +15 -0
  15. package/dist/commonjs/components/base/date-picker-view/date-picker-view.js +19 -53
  16. package/dist/commonjs/components/base/date-picker-view/index.js +0 -22
  17. package/dist/commonjs/components/base/index.js +30 -8
  18. package/dist/commonjs/components/base/picker/index.js +26 -4
  19. package/dist/commonjs/components/base/picker/picker-content.js +60 -0
  20. package/dist/commonjs/components/base/picker/picker-context.js +9 -0
  21. package/dist/commonjs/components/base/picker/picker-field.js +37 -0
  22. package/dist/commonjs/components/base/picker/picker.js +22 -96
  23. package/dist/commonjs/components/base/picker/style/index.js +1 -3
  24. package/dist/commonjs/components/base/picker-backup/base-picker-container.js +50 -0
  25. package/dist/commonjs/components/base/picker-backup/index.js +27 -0
  26. package/dist/commonjs/components/base/picker-backup/picker-backup.js +75 -0
  27. package/dist/commonjs/components/base/picker-backup/picker-copy.js +106 -0
  28. package/dist/commonjs/components/base/{picker → picker-backup}/picker-trigger.js +5 -5
  29. package/dist/commonjs/components/base/picker-backup/style/index.js +19 -0
  30. package/dist/commonjs/components/base/picker-backup/utils.js +53 -0
  31. package/dist/commonjs/components/base/picker-view/picker-view-column.js +15 -0
  32. package/dist/commonjs/components/base/picker-view/picker-view.js +4 -4
  33. package/dist/commonjs/components/base/tag/index.js +20 -0
  34. package/dist/commonjs/components/base/tag/style/index.js +89 -0
  35. package/dist/commonjs/components/base/tag/tag-context.js +12 -0
  36. package/dist/commonjs/components/base/tag/tag-group.js +35 -0
  37. package/dist/commonjs/components/base/tag/tag.js +47 -0
  38. package/dist/commonjs/components/base/tag/types.js +5 -0
  39. package/dist/module/components/base/carousel/carousel-indicator.js +51 -0
  40. package/dist/module/components/base/carousel/carousel-slides.js +135 -0
  41. package/dist/module/components/base/carousel/carousel.js +116 -124
  42. package/dist/module/components/base/carousel/hooks/index.js +0 -2
  43. package/dist/module/components/base/carousel/hooks/use-carousel-index.js +15 -11
  44. package/dist/module/components/base/carousel/hooks/use-carousel-lifecycle.js +6 -2
  45. package/dist/module/components/base/carousel/hooks/use-carousel-pan-responder.js +40 -11
  46. package/dist/module/components/base/carousel/hooks/use-carousel-position.js +5 -1
  47. package/dist/module/components/base/carousel/index.js +1 -5
  48. package/dist/module/components/base/carousel/style/index.js +12 -0
  49. package/dist/module/components/base/date-picker/date-picker.js +60 -48
  50. package/dist/module/components/base/date-picker/date-range-picker.js +146 -54
  51. package/dist/module/components/base/date-picker/style/index.js +11 -0
  52. package/dist/module/components/base/date-picker-view/date-picker-view.js +23 -57
  53. package/dist/module/components/base/date-picker-view/index.js +1 -3
  54. package/dist/module/components/base/index.js +2 -0
  55. package/dist/module/components/base/picker/index.js +9 -1
  56. package/dist/module/components/base/picker/picker-content.js +54 -0
  57. package/dist/module/components/base/picker/picker-context.js +4 -0
  58. package/dist/module/components/base/picker/picker-field.js +32 -0
  59. package/dist/module/components/base/picker/picker.js +25 -99
  60. package/dist/module/components/base/picker/style/index.js +1 -3
  61. package/dist/module/components/base/picker-backup/base-picker-container.js +44 -0
  62. package/dist/module/components/base/picker-backup/index.js +4 -0
  63. package/dist/module/components/base/picker-backup/picker-backup.js +69 -0
  64. package/dist/module/components/base/picker-backup/picker-copy.js +101 -0
  65. package/dist/module/components/base/{picker → picker-backup}/picker-trigger.js +2 -2
  66. package/dist/module/components/base/picker-backup/style/index.js +15 -0
  67. package/dist/module/components/base/picker-backup/utils.js +48 -0
  68. package/dist/module/components/base/picker-view/picker-view-column.js +15 -0
  69. package/dist/module/components/base/picker-view/picker-view.js +4 -4
  70. package/dist/module/components/base/tag/index.js +5 -0
  71. package/dist/module/components/base/tag/style/index.js +85 -0
  72. package/dist/module/components/base/tag/tag-context.js +8 -0
  73. package/dist/module/components/base/tag/tag-group.js +29 -0
  74. package/dist/module/components/base/tag/tag.js +41 -0
  75. package/dist/module/components/base/tag/types.js +3 -0
  76. package/dist/typescript/components/base/carousel/carousel-indicator.d.ts +42 -0
  77. package/dist/typescript/components/base/carousel/carousel-indicator.d.ts.map +1 -0
  78. package/dist/typescript/components/base/carousel/carousel-slides.d.ts +49 -0
  79. package/dist/typescript/components/base/carousel/carousel-slides.d.ts.map +1 -0
  80. package/dist/typescript/components/base/carousel/carousel.d.ts +16 -11
  81. package/dist/typescript/components/base/carousel/carousel.d.ts.map +1 -1
  82. package/dist/typescript/components/base/carousel/hooks/index.d.ts +0 -2
  83. package/dist/typescript/components/base/carousel/hooks/index.d.ts.map +1 -1
  84. package/dist/typescript/components/base/carousel/hooks/use-carousel-index.d.ts.map +1 -1
  85. package/dist/typescript/components/base/carousel/hooks/use-carousel-lifecycle.d.ts.map +1 -1
  86. package/dist/typescript/components/base/carousel/hooks/use-carousel-pan-responder.d.ts.map +1 -1
  87. package/dist/typescript/components/base/carousel/hooks/use-carousel-position.d.ts.map +1 -1
  88. package/dist/typescript/components/base/carousel/index.d.ts +1 -4
  89. package/dist/typescript/components/base/carousel/index.d.ts.map +1 -1
  90. package/dist/typescript/components/base/carousel/style/index.d.ts +12 -0
  91. package/dist/typescript/components/base/carousel/style/index.d.ts.map +1 -1
  92. package/dist/typescript/components/base/carousel/types.d.ts +8 -17
  93. package/dist/typescript/components/base/carousel/types.d.ts.map +1 -1
  94. package/dist/typescript/components/base/date-picker/date-picker.d.ts +4 -2
  95. package/dist/typescript/components/base/date-picker/date-picker.d.ts.map +1 -1
  96. package/dist/typescript/components/base/date-picker/date-range-picker.d.ts +12 -3
  97. package/dist/typescript/components/base/date-picker/date-range-picker.d.ts.map +1 -1
  98. package/dist/typescript/components/base/date-picker/style/index.d.ts +9 -0
  99. package/dist/typescript/components/base/date-picker/style/index.d.ts.map +1 -0
  100. package/dist/typescript/components/base/date-picker-view/date-picker-view.d.ts +1 -6
  101. package/dist/typescript/components/base/date-picker-view/date-picker-view.d.ts.map +1 -1
  102. package/dist/typescript/components/base/date-picker-view/index.d.ts +0 -2
  103. package/dist/typescript/components/base/date-picker-view/index.d.ts.map +1 -1
  104. package/dist/typescript/components/base/date-picker-view/types.d.ts +1 -1
  105. package/dist/typescript/components/base/index.d.ts +2 -0
  106. package/dist/typescript/components/base/index.d.ts.map +1 -1
  107. package/dist/typescript/components/base/picker/index.d.ts +7 -1
  108. package/dist/typescript/components/base/picker/index.d.ts.map +1 -1
  109. package/dist/typescript/components/base/picker/picker-content.d.ts +15 -0
  110. package/dist/typescript/components/base/picker/picker-content.d.ts.map +1 -0
  111. package/dist/typescript/components/base/picker/picker-context.d.ts +18 -0
  112. package/dist/typescript/components/base/picker/picker-context.d.ts.map +1 -0
  113. package/dist/typescript/components/base/picker/picker-field.d.ts +10 -0
  114. package/dist/typescript/components/base/picker/picker-field.d.ts.map +1 -0
  115. package/dist/typescript/components/base/picker/picker.d.ts +13 -11
  116. package/dist/typescript/components/base/picker/picker.d.ts.map +1 -1
  117. package/dist/typescript/components/base/picker/style/index.d.ts +0 -2
  118. package/dist/typescript/components/base/picker/style/index.d.ts.map +1 -1
  119. package/dist/typescript/components/base/picker-backup/base-picker-container.d.ts +15 -0
  120. package/dist/typescript/components/base/picker-backup/base-picker-container.d.ts.map +1 -0
  121. package/dist/typescript/components/base/picker-backup/index.d.ts +3 -0
  122. package/dist/typescript/components/base/picker-backup/index.d.ts.map +1 -0
  123. package/dist/typescript/components/base/picker-backup/picker-backup.d.ts +26 -0
  124. package/dist/typescript/components/base/picker-backup/picker-backup.d.ts.map +1 -0
  125. package/dist/typescript/components/base/picker-backup/picker-copy.d.ts +13 -0
  126. package/dist/typescript/components/base/picker-backup/picker-copy.d.ts.map +1 -0
  127. package/dist/typescript/components/base/picker-backup/picker-trigger.d.ts.map +1 -0
  128. package/dist/typescript/components/base/picker-backup/style/index.d.ts +13 -0
  129. package/dist/typescript/components/base/picker-backup/style/index.d.ts.map +1 -0
  130. package/dist/typescript/components/base/picker-backup/utils.d.ts +8 -0
  131. package/dist/typescript/components/base/picker-backup/utils.d.ts.map +1 -0
  132. package/dist/typescript/components/base/picker-view/picker-view-column.d.ts.map +1 -1
  133. package/dist/typescript/components/base/picker-view/utils/picker.d.ts +3 -3
  134. package/dist/typescript/components/base/picker-view/utils/picker.d.ts.map +1 -1
  135. package/dist/typescript/components/base/tag/index.d.ts +5 -0
  136. package/dist/typescript/components/base/tag/index.d.ts.map +1 -0
  137. package/dist/typescript/components/base/tag/style/index.d.ts +61 -0
  138. package/dist/typescript/components/base/tag/style/index.d.ts.map +1 -0
  139. package/dist/typescript/components/base/tag/tag-context.d.ts +3 -0
  140. package/dist/typescript/components/base/tag/tag-context.d.ts.map +1 -0
  141. package/dist/typescript/components/base/tag/tag-group.d.ts +4 -0
  142. package/dist/typescript/components/base/tag/tag-group.d.ts.map +1 -0
  143. package/dist/typescript/components/base/tag/tag.d.ts +4 -0
  144. package/dist/typescript/components/base/tag/tag.d.ts.map +1 -0
  145. package/dist/typescript/components/base/tag/types.d.ts +48 -0
  146. package/dist/typescript/components/base/tag/types.d.ts.map +1 -0
  147. package/package.json +1 -1
  148. package/src/components/base/carousel/carousel-indicator.tsx +80 -0
  149. package/src/components/base/carousel/carousel-slides.tsx +177 -0
  150. package/src/components/base/carousel/carousel.tsx +108 -118
  151. package/src/components/base/carousel/hooks/index.ts +0 -2
  152. package/src/components/base/carousel/hooks/use-carousel-index.ts +13 -9
  153. package/src/components/base/carousel/hooks/use-carousel-lifecycle.ts +4 -3
  154. package/src/components/base/carousel/hooks/use-carousel-pan-responder.ts +40 -16
  155. package/src/components/base/carousel/hooks/use-carousel-position.ts +4 -1
  156. package/src/components/base/carousel/index.ts +1 -3
  157. package/src/components/base/carousel/style/index.ts +12 -0
  158. package/src/components/base/carousel/types.ts +8 -21
  159. package/src/components/base/date-picker/date-picker.tsx +64 -61
  160. package/src/components/base/date-picker/date-range-picker.tsx +178 -70
  161. package/src/components/base/date-picker/style/index.ts +10 -0
  162. package/src/components/base/date-picker-view/date-picker-view.tsx +21 -68
  163. package/src/components/base/date-picker-view/index.ts +0 -2
  164. package/src/components/base/date-picker-view/types.ts +1 -1
  165. package/src/components/base/index.ts +2 -0
  166. package/src/components/base/picker/index.ts +11 -1
  167. package/src/components/base/picker/picker-content.tsx +75 -0
  168. package/src/components/base/picker/picker-context.ts +19 -0
  169. package/src/components/base/picker/picker-field.tsx +50 -0
  170. package/src/components/base/picker/picker.tsx +38 -114
  171. package/src/components/base/picker/style/index.ts +0 -2
  172. package/src/components/base/picker-backup/base-picker-container.tsx +55 -0
  173. package/src/components/base/picker-backup/index.ts +2 -0
  174. package/src/components/base/picker-backup/picker-backup.tsx +110 -0
  175. package/src/components/base/picker-backup/picker-copy.tsx +125 -0
  176. package/src/components/base/{picker → picker-backup}/picker-trigger.tsx +2 -2
  177. package/src/components/base/picker-backup/style/index.ts +14 -0
  178. package/src/components/base/picker-backup/utils.ts +62 -0
  179. package/src/components/base/picker-view/picker-view-column.tsx +20 -0
  180. package/src/components/base/picker-view/picker-view.tsx +4 -4
  181. package/src/components/base/picker-view/utils/picker.ts +3 -5
  182. package/src/components/base/tag/index.ts +5 -0
  183. package/src/components/base/tag/style/index.tsx +84 -0
  184. package/src/components/base/tag/tag-context.ts +9 -0
  185. package/src/components/base/tag/tag-group.tsx +31 -0
  186. package/src/components/base/tag/tag.tsx +50 -0
  187. package/src/components/base/tag/types.ts +71 -0
  188. package/dist/commonjs/components/base/carousel/carousel-item.js +0 -45
  189. package/dist/commonjs/components/base/carousel/constants.js +0 -25
  190. package/dist/commonjs/components/base/carousel/hooks/use-carousel-indicator.js +0 -63
  191. package/dist/commonjs/components/base/carousel/hooks/use-carousel-slides.js +0 -95
  192. package/dist/commonjs/components/base/carousel/utils.js +0 -63
  193. package/dist/commonjs/components/base/date-picker-view/date-range-picker-view.js +0 -145
  194. package/dist/commonjs/components/base/date-picker-view/date-time-picker.js +0 -39
  195. package/dist/module/components/base/carousel/carousel-item.js +0 -40
  196. package/dist/module/components/base/carousel/constants.js +0 -21
  197. package/dist/module/components/base/carousel/hooks/use-carousel-indicator.js +0 -58
  198. package/dist/module/components/base/carousel/hooks/use-carousel-slides.js +0 -90
  199. package/dist/module/components/base/carousel/utils.js +0 -55
  200. package/dist/module/components/base/date-picker-view/date-range-picker-view.js +0 -138
  201. package/dist/module/components/base/date-picker-view/date-time-picker.js +0 -34
  202. package/dist/typescript/components/base/carousel/carousel-item.d.ts +0 -26
  203. package/dist/typescript/components/base/carousel/carousel-item.d.ts.map +0 -1
  204. package/dist/typescript/components/base/carousel/constants.d.ts +0 -17
  205. package/dist/typescript/components/base/carousel/constants.d.ts.map +0 -1
  206. package/dist/typescript/components/base/carousel/hooks/use-carousel-indicator.d.ts +0 -37
  207. package/dist/typescript/components/base/carousel/hooks/use-carousel-indicator.d.ts.map +0 -1
  208. package/dist/typescript/components/base/carousel/hooks/use-carousel-slides.d.ts +0 -51
  209. package/dist/typescript/components/base/carousel/hooks/use-carousel-slides.d.ts.map +0 -1
  210. package/dist/typescript/components/base/carousel/utils.d.ts +0 -25
  211. package/dist/typescript/components/base/carousel/utils.d.ts.map +0 -1
  212. package/dist/typescript/components/base/date-picker-view/date-range-picker-view.d.ts +0 -26
  213. package/dist/typescript/components/base/date-picker-view/date-range-picker-view.d.ts.map +0 -1
  214. package/dist/typescript/components/base/date-picker-view/date-time-picker.d.ts +0 -3
  215. package/dist/typescript/components/base/date-picker-view/date-time-picker.d.ts.map +0 -1
  216. package/dist/typescript/components/base/picker/picker-trigger.d.ts.map +0 -1
  217. package/src/components/base/carousel/carousel-item.tsx +0 -35
  218. package/src/components/base/carousel/constants.ts +0 -19
  219. package/src/components/base/carousel/hooks/use-carousel-indicator.tsx +0 -84
  220. package/src/components/base/carousel/hooks/use-carousel-slides.tsx +0 -131
  221. package/src/components/base/carousel/utils.ts +0 -55
  222. package/src/components/base/date-picker-view/date-range-picker-view.tsx +0 -191
  223. package/src/components/base/date-picker-view/date-time-picker.tsx +0 -34
  224. /package/dist/typescript/components/base/{picker → picker-backup}/picker-trigger.d.ts +0 -0
@@ -1,95 +1,98 @@
1
- import { useControllableValue, useMemoizedFn } from 'ahooks';
1
+ import { useMemoizedFn } from 'ahooks';
2
2
  import dayjs from 'dayjs';
3
- import React, { useMemo, useState } from 'react';
3
+ import React, { useCallback, useMemo, useRef, useState } from 'react';
4
4
 
5
5
  import {
6
+ Button,
6
7
  DatePickerView,
7
8
  type DatePickerViewProps,
8
- Modal,
9
- PickerTrigger,
10
- type PickerTriggerControlledProps,
11
- Portal,
9
+ Flex,
10
+ Picker,
11
+ type PickerMergeProps,
12
12
  } from '#components/base';
13
13
 
14
+ import { useDatePickerStyles } from './style';
15
+
14
16
  export interface DatePickerProps
15
- extends PickerTriggerControlledProps<Date>,
16
- Omit<DatePickerViewProps, 'onClose' | 'onChange' | 'value'> {
17
+ extends PickerMergeProps<Date>,
18
+ Omit<DatePickerViewProps, 'onClose' | 'onChange' | 'value' | 'defaultValue'> {
17
19
  /**
18
20
  * 自定义格式化函数。
19
21
  * 配置参考 https://day.js.org/docs/zh-CN/display/format
20
22
  */
21
23
  formatter?: string | ((date: Date) => string);
24
+ /** 确定按钮文案 */
25
+ confirmText?: string;
22
26
  }
23
27
 
24
28
  export const DatePicker: React.FC<DatePickerProps> = ({
25
29
  placeholder = '请选择日期',
26
- disabled = false,
27
30
  formatter = 'YYYY/MM/DD',
28
- style,
29
- children,
30
31
  mode = 'Y-D',
31
- title,
32
- confirmText,
32
+ headerTitle,
33
+ confirmText = '确定',
33
34
  ...restProps
34
35
  }) => {
35
- const [value, setValue] = useControllableValue<Date | undefined>(restProps, {
36
- defaultValue: restProps.defaultValue,
37
- });
36
+ const [draftValue, setDraftValue] = useState<Date>();
37
+
38
+ const styles = useDatePickerStyles();
38
39
 
39
- const [visible, setVisible] = useState(false);
40
+ const committedValueRef = useRef<Date>(new Date());
41
+
42
+ const handleOpen = useCallback(() => {
43
+ setDraftValue(committedValueRef.current);
44
+ }, []);
40
45
 
41
46
  /** 格式化显示文案 */
42
- const displayText = useMemo(() => {
43
- if (!value) return '';
47
+ const _renderValue = useMemoizedFn(value => {
48
+ if (!value) return;
44
49
  if (typeof formatter === 'string') {
45
50
  return dayjs(value).format(formatter);
46
51
  }
47
52
  return formatter(value);
48
- }, [value, mode, formatter]);
49
-
50
- const handleOpen = useMemoizedFn(() => {
51
- setVisible(true);
52
53
  });
53
54
 
54
- const handleChange = useMemoizedFn((date: Date) => {
55
- setValue(date);
56
- setVisible(false);
57
- });
58
-
59
- const handleClose = useMemoizedFn(() => {
60
- setVisible(false);
61
- });
55
+ const _headerTitle = useMemo(() => {
56
+ if (headerTitle) {
57
+ return headerTitle;
58
+ }
59
+ switch (mode) {
60
+ case 'Y':
61
+ return '请选择年';
62
+ case 'Y-M':
63
+ return '请选择年月';
64
+ case 'Y-D':
65
+ return '请选择年月日';
66
+ default:
67
+ return '请选择日期';
68
+ }
69
+ }, [mode, headerTitle]);
62
70
 
63
71
  return (
64
- <>
65
- <PickerTrigger
66
- displayText={displayText}
67
- placeholder={placeholder}
68
- disabled={disabled}
69
- style={style}
70
- onPress={handleOpen}
71
- >
72
- {children}
73
- </PickerTrigger>
74
-
75
- <Portal>
76
- <Modal
77
- visible={visible}
78
- position="bottom"
79
- safeAreaInsetBottom
80
- contentHeight={'50%'}
81
- >
82
- <DatePickerView
83
- {...restProps}
84
- mode={mode}
85
- title={title}
86
- confirmText={confirmText}
87
- value={value}
88
- onChange={handleChange}
89
- onClose={handleClose}
90
- />
91
- </Modal>
92
- </Portal>
93
- </>
72
+ <Picker {...restProps}>
73
+ <Picker.Field placeholder={placeholder} renderValue={_renderValue} />
74
+ <Picker.Content headerTitle={_headerTitle} onOpen={handleOpen}>
75
+ {({ onChange, value: contextValue }) => {
76
+ // 同步 ref:始终保持最新的已提交值
77
+ committedValueRef.current = contextValue ?? new Date();
78
+ return (
79
+ <Flex vertical align="stretch" style={styles.container}>
80
+ <DatePickerView
81
+ mode={mode}
82
+ value={draftValue}
83
+ onChange={setDraftValue}
84
+ />
85
+ <Button
86
+ text={confirmText}
87
+ size="large"
88
+ onPress={() => {
89
+ onChange?.(draftValue);
90
+ }}
91
+ />
92
+ </Flex>
93
+ );
94
+ }}
95
+ </Picker.Content>
96
+ </Picker>
94
97
  );
95
98
  };
@@ -1,62 +1,87 @@
1
- import { useControllableValue } from 'ahooks';
1
+ import { useMemoizedFn } from 'ahooks';
2
2
  import dayjs from 'dayjs';
3
- import React, { useMemo, useState } from 'react';
3
+ import React, { useCallback, useMemo, useRef, useState } from 'react';
4
4
 
5
5
  import {
6
- DateRangePickerView,
7
- type DateRangePickerViewProps,
8
- Modal,
9
- PickerTrigger,
10
- type PickerTriggerControlledProps,
11
- Portal,
6
+ Button,
7
+ DatePickerView,
8
+ Flex,
9
+ Picker,
10
+ type PickerMergeProps,
11
+ Text,
12
12
  } from '#components/base';
13
13
 
14
+ import { useDateRangePickerStyles } from '../date-picker-view/style';
15
+ import type { DatePickerViewProps } from '../date-picker-view/types';
16
+ import { sortRange } from '../date-picker-view/utils';
17
+
14
18
  export interface DateRangePickerProps
15
- extends PickerTriggerControlledProps<[Date, Date]>,
16
- Omit<DateRangePickerViewProps, 'onClose' | 'onChange' | 'value' | 'formatter'> {
19
+ extends PickerMergeProps<[Date, Date]>,
20
+ Omit<DatePickerViewProps, 'onClose' | 'onChange' | 'value' | 'defaultValue'> {
17
21
  /**
18
22
  * 自定义格式化函数。
19
23
  * 配置参考 https://day.js.org/docs/zh-CN/display/format
20
24
  */
21
- formatter?: string | ((date: [Date, Date] | Date) => string);
25
+ formatter?: string | ((date: [Date, Date]) => string);
22
26
  /** 范围分隔符 */
23
27
  separator?: string;
28
+ /** start 开始时间的占位文案 */
29
+ startText?: string;
30
+ /** end 结束时间的占位文案 */
31
+ endText?: string;
32
+ /** 确定按钮文案 */
33
+ confirmText?: string;
34
+ /** 重置按钮文案 */
35
+ resetText?: string;
24
36
  }
25
37
 
38
+ type Selected = 'start' | 'end';
39
+
26
40
  export const DateRangePicker: React.FC<DateRangePickerProps> = ({
27
41
  placeholder = '请选择日期范围',
28
- disabled = false,
29
42
  formatter = 'YYYY/MM/DD',
30
- style,
31
43
  separator = ' ~ ',
32
- children,
33
- title,
34
- confirmText,
35
- resetText,
36
- startText,
37
- endText,
44
+ headerTitle,
45
+ startText = '请选择开始时间',
46
+ endText = '请选择结束时间',
47
+ confirmText = '确定',
48
+ resetText = '重置',
49
+ min,
50
+ max,
38
51
  ...restProps
39
52
  }) => {
40
- const [value, setValue] = useControllableValue<[Date, Date] | undefined>(
41
- restProps,
42
- {
43
- defaultValue: restProps.defaultValue,
44
- },
45
- );
53
+ const styles = useDateRangePickerStyles();
54
+
55
+ /** 当前编辑的是 start 还是 end */
56
+ const [selected, setSelected] = useState<Selected>('start');
46
57
 
47
- const [visible, setVisible] = useState(false);
58
+ /**
59
+ * 草稿值:用户在弹窗内选择时的临时值
60
+ * - 弹窗打开时重置为已提交的值(通过 onOpen)
61
+ */
62
+ const [draft, setDraft] = useState<[Date?, Date?]>([]);
48
63
 
49
- const openModal = () => {
50
- setVisible(true);
51
- };
64
+ /**
65
+ * 追踪 Picker context 中的已提交值
66
+ * 非受控模式下 restProps.value 始终是 undefined,真正的值在 Picker context 内
67
+ */
68
+ const committedValueRef = useRef<[Date, Date] | undefined>(undefined);
52
69
 
53
- const closeModal = () => {
54
- setVisible(false);
55
- };
70
+ /**
71
+ * 弹窗打开时,将草稿值重置为已提交的值
72
+ */
73
+ const handleOpen = useCallback(() => {
74
+ if (committedValueRef.current) {
75
+ setDraft(sortRange(committedValueRef.current));
76
+ } else {
77
+ setDraft([new Date(), undefined]);
78
+ }
79
+ setSelected('start');
80
+ }, []);
56
81
 
57
82
  /** 格式化显示文案 */
58
- const displayText = useMemo(() => {
59
- if (!value) return '';
83
+ const _renderValue = useMemoizedFn(value => {
84
+ if (!value) return;
60
85
  if (typeof formatter === 'string') {
61
86
  return (
62
87
  dayjs(value[0]).format(formatter) +
@@ -65,44 +90,127 @@ export const DateRangePicker: React.FC<DateRangePickerProps> = ({
65
90
  );
66
91
  }
67
92
  return formatter(value);
68
- }, [value, formatter, separator]);
93
+ });
94
+
95
+ /** Picker 当前值 */
96
+ const pickerValue = (selected === 'start' ? draft[0] : draft[1]) ?? new Date();
97
+
98
+ /** Picker 可选范围 */
99
+ const minDate = useMemo(() => {
100
+ return selected === 'end' ? (draft[0] ?? min) : min;
101
+ }, [selected, draft, min]);
102
+
103
+ const maxDate = useMemo(() => {
104
+ return selected === 'start' ? (draft[1] ?? max) : max;
105
+ }, [selected, draft, max]);
106
+
107
+ /** Picker 改 draft */
108
+ const onChangePicker = useCallback(
109
+ (date: Date) => {
110
+ setDraft(prev => {
111
+ const index = selected === 'start' ? 0 : 1;
112
+ if (prev[index]?.getTime() === date.getTime()) {
113
+ return prev;
114
+ }
115
+ const next: [Date?, Date?] = [prev[0], prev[1]];
116
+ next[index] = date;
117
+ return next;
118
+ });
119
+ },
120
+ [selected],
121
+ );
122
+
123
+ const changeSelected = useCallback((nextSelected: Selected) => {
124
+ setSelected(nextSelected);
125
+ setDraft(prev => {
126
+ const next: [Date?, Date?] = [prev[0], prev[1]];
127
+ const index = nextSelected === 'start' ? 0 : 1;
128
+ next[index] ??= new Date();
129
+ return next;
130
+ });
131
+ }, []);
132
+
133
+ /** 重置 */
134
+ const onReset = useCallback(() => {
135
+ setDraft([new Date(), undefined]);
136
+ setSelected('start');
137
+ }, []);
138
+
139
+ /** 是否可以确定 */
140
+ const canConfirm = Boolean(draft[0] && draft[1]);
141
+
142
+ const _startText =
143
+ (draft[0] &&
144
+ dayjs(draft[0]).format(
145
+ typeof formatter === 'string' ? formatter : 'YYYY/MM/DD',
146
+ )) ||
147
+ startText;
148
+ const _endText =
149
+ (draft[1] &&
150
+ dayjs(draft[1]).format(
151
+ typeof formatter === 'string' ? formatter : 'YYYY/MM/DD',
152
+ )) ||
153
+ endText;
69
154
 
70
155
  return (
71
- <>
72
- <PickerTrigger
73
- displayText={displayText}
74
- placeholder={placeholder}
75
- disabled={disabled}
76
- style={style}
77
- onPress={openModal}
156
+ <Picker {...restProps}>
157
+ <Picker.Field placeholder={placeholder} renderValue={_renderValue} />
158
+ <Picker.Content
159
+ headerTitle={headerTitle}
160
+ contentHeight={490}
161
+ onOpen={handleOpen}
78
162
  >
79
- {children}
80
- </PickerTrigger>
81
-
82
- <Portal>
83
- <Modal
84
- visible={visible}
85
- position="bottom"
86
- safeAreaInsetBottom
87
- contentHeight={490}
88
- >
89
- <DateRangePickerView
90
- {...restProps}
91
- title={title}
92
- confirmText={confirmText}
93
- resetText={resetText}
94
- startText={startText}
95
- endText={endText}
96
- value={value}
97
- onChange={range => {
98
- setValue(range);
99
- closeModal();
100
- }}
101
- onClose={closeModal}
102
- formatter={formatter}
103
- />
104
- </Modal>
105
- </Portal>
106
- </>
163
+ {({ onChange, value: contextValue }) => {
164
+ // 同步 ref:始终保持最新的已提交值
165
+ committedValueRef.current = contextValue;
166
+ return (
167
+ <Flex vertical align="stretch" style={styles.container}>
168
+ <Flex>
169
+ <Button
170
+ variant="outline"
171
+ theme={selected === 'start' ? 'primary' : 'default'}
172
+ block
173
+ text={_startText}
174
+ onPress={() => changeSelected('start')}
175
+ />
176
+ <Text>—</Text>
177
+ <Button
178
+ variant="outline"
179
+ theme={selected === 'end' ? 'primary' : 'default'}
180
+ block
181
+ text={_endText}
182
+ onPress={() => changeSelected('end')}
183
+ />
184
+ </Flex>
185
+
186
+ <DatePickerView
187
+ value={pickerValue}
188
+ min={minDate}
189
+ max={maxDate}
190
+ onChange={onChangePicker}
191
+ />
192
+
193
+ <Button.Group block>
194
+ <Button
195
+ size="large"
196
+ text={resetText}
197
+ theme="secondary"
198
+ onPress={onReset}
199
+ />
200
+ <Button
201
+ size="large"
202
+ text={confirmText}
203
+ onPress={() => {
204
+ if (!draft[0] || !draft[1]) return;
205
+ onChange?.(sortRange([draft[0], draft[1]]));
206
+ }}
207
+ disabled={!canConfirm}
208
+ />
209
+ </Button.Group>
210
+ </Flex>
211
+ );
212
+ }}
213
+ </Picker.Content>
214
+ </Picker>
107
215
  );
108
216
  };
@@ -0,0 +1,10 @@
1
+ import { createThemedStyles } from '#native-provider';
2
+
3
+ export const useDatePickerStyles = createThemedStyles(theme => ({
4
+ container: {
5
+ backgroundColor: theme.palette.white,
6
+ padding: 16,
7
+ rowGap: 16,
8
+ flex: 1,
9
+ },
10
+ }));
@@ -1,81 +1,34 @@
1
- import { useMemoizedFn } from 'ahooks';
2
- import { useMemo, useRef, useState } from 'react';
1
+ import { useControllableValue } from 'ahooks';
2
+ import { useMemo } from 'react';
3
3
 
4
- import {
5
- Button,
6
- DateTimePicker,
7
- Flex,
8
- ModalHeader,
9
- type ModalHeaderBaseProps,
10
- } from '#components/base';
4
+ import { PickerView } from '#components/base';
11
5
 
12
- import { useDatePickerStyles } from './style';
13
- import type { DateTimePickerProps } from './types';
14
-
15
- export interface DatePickerViewProps
16
- extends DateTimePickerProps,
17
- ModalHeaderBaseProps {
18
- /** 确定按钮文案 */
19
- confirmText?: string;
20
- }
6
+ import { useDateTimePicker } from './hooks/use-date-time-picker';
7
+ import type { DatePickerViewProps } from './types';
21
8
 
22
9
  export const DatePickerView: React.FC<DatePickerViewProps> = ({
23
10
  mode = 'Y-D',
24
- onChange,
25
- onClose,
26
- title,
27
- confirmText = '确定',
11
+ min,
12
+ max,
28
13
  ...restProps
29
14
  }) => {
30
- const styles = useDatePickerStyles();
31
-
32
- // 内部状态用于管理滚动时的临时值
33
- const [localValue, setLocalValue] = useState<Date>(
34
- restProps.value ?? restProps.defaultValue ?? new Date(),
35
- );
36
-
37
- const lastValueRef = useRef(restProps.value);
38
- if (restProps.value !== undefined && restProps.value !== lastValueRef.current) {
39
- lastValueRef.current = restProps.value;
40
- setLocalValue(restProps.value);
41
- }
15
+ // 确保有一个稳定的默认日期值
16
+ const defaultDate = useMemo(() => new Date(), []);
42
17
 
43
- const _title = useMemo(() => {
44
- if (title) {
45
- return title;
46
- }
47
- switch (mode) {
48
- case 'Y':
49
- return '请选择年';
50
- case 'Y-M':
51
- return '请选择年月';
52
- case 'Y-D':
53
- return '请选择年月日';
54
- default:
55
- return '请选择日期';
56
- }
57
- }, [mode, title]);
58
-
59
- // 滚动时只更新内部状态,不触发外部 onChange
60
- const onChangePicker = useMemoizedFn((value: Date) => {
61
- setLocalValue(value);
18
+ const [value, onChange] = useControllableValue(restProps, {
19
+ defaultValue: defaultDate,
62
20
  });
63
21
 
64
- // 只在点击确定按钮时才调用外部 onChange
65
- const onConfirm = useMemoizedFn(() => {
66
- onChange?.(localValue);
22
+ // 确保 value 始终有值,避免 undefined 传递给 hook
23
+ const safeValue = value ?? defaultDate;
24
+
25
+ const [values, columns, onChangePicker] = useDateTimePicker({
26
+ mode,
27
+ value: safeValue,
28
+ onChange,
29
+ min,
30
+ max,
67
31
  });
68
32
 
69
- return (
70
- <Flex vertical align="stretch" style={styles.container}>
71
- <ModalHeader title={_title} onClose={onClose} />
72
- <DateTimePicker
73
- {...restProps}
74
- mode={mode}
75
- value={localValue}
76
- onChange={onChangePicker}
77
- />
78
- <Button text={confirmText} size="large" onPress={onConfirm} />
79
- </Flex>
80
- );
33
+ return <PickerView columns={columns} value={values} onChange={onChangePicker} />;
81
34
  };
@@ -1,4 +1,2 @@
1
1
  export * from './date-picker-view';
2
- export * from './date-range-picker-view';
3
- export * from './date-time-picker';
4
2
  export type * from './types';
@@ -19,7 +19,7 @@ export type DatePickerColumnType = 'Y' | 'M' | 'D';
19
19
 
20
20
  export type RenderLabel = (t: DatePickerColumnType, n: number) => string;
21
21
 
22
- export interface DateTimePickerProps {
22
+ export interface DatePickerViewProps {
23
23
  /**
24
24
  * 选中项
25
25
  */
@@ -23,6 +23,7 @@ export * from './loading';
23
23
  export * from './modal';
24
24
  export * from './nav-bar';
25
25
  export * from './picker';
26
+ export * from './picker-backup';
26
27
  export * from './picker-view';
27
28
  export * from './portal';
28
29
  export * from './radio';
@@ -30,4 +31,5 @@ export * from './result';
30
31
  export * from './surface';
31
32
  export * from './switch';
32
33
  export * from './tabs';
34
+ export * from './tag';
33
35
  export * from './text';
@@ -1,2 +1,12 @@
1
+ import { Picker as _Picker } from './picker';
2
+ import { PickerContent } from './picker-content';
3
+ import { PickerField } from './picker-field';
4
+
1
5
  export * from './picker';
2
- export * from './picker-trigger';
6
+ export * from './picker-content';
7
+ export * from './picker-field';
8
+
9
+ export const Picker = Object.assign(_Picker, {
10
+ Field: PickerField,
11
+ Content: PickerContent,
12
+ });
@@ -0,0 +1,75 @@
1
+ import React, { useContext, useEffect } from 'react';
2
+ import type { StyleProp, ViewStyle } from 'react-native';
3
+
4
+ import {
5
+ Flex,
6
+ Modal,
7
+ ModalHeader,
8
+ type ModalProps,
9
+ Portal,
10
+ } from '#components/base';
11
+ import type { Any } from '#shared/utils';
12
+
13
+ import { PickerContext } from './picker-context';
14
+
15
+ export interface PickerContentProps
16
+ extends Omit<ModalProps, 'visible' | 'children'> {
17
+ hiddenHeaderCloseIcon?: boolean;
18
+ headerTitle?: React.ReactNode;
19
+ headerStyle?: StyleProp<ViewStyle>;
20
+ children?:
21
+ | React.ReactElement
22
+ | ((props: {
23
+ value: Any;
24
+ onChange?: (value: Any) => void;
25
+ }) => React.ReactElement);
26
+ }
27
+
28
+ export const PickerContent: React.FC<PickerContentProps> = ({
29
+ children,
30
+ headerTitle,
31
+ headerStyle,
32
+ hiddenHeaderCloseIcon,
33
+ ...modalProps
34
+ }) => {
35
+ const context = useContext(PickerContext);
36
+
37
+ let childNode: React.ReactElement<Any> | null = null;
38
+
39
+ if (React.isValidElement(children)) {
40
+ childNode = React.cloneElement<Any>(children, {
41
+ value: context?.value,
42
+ onChange: context?.onChange,
43
+ });
44
+ } else if (typeof children === 'function') {
45
+ childNode = children({
46
+ value: context?.value,
47
+ onChange: context?.onChange,
48
+ });
49
+ }
50
+ useEffect(() => {
51
+ context?.setDisabled?.(!!childNode?.props?.disabled);
52
+ }, [childNode?.props?.disabled, context?.setDisabled]);
53
+
54
+ return (
55
+ <Portal>
56
+ <Modal
57
+ safeAreaInsetBottom
58
+ position="bottom"
59
+ contentHeight={'50%'}
60
+ {...modalProps}
61
+ visible={!!context?.visible}
62
+ >
63
+ <Flex vertical align="stretch" gap={16} style={{ flex: 1, padding: 16 }}>
64
+ <ModalHeader
65
+ title={headerTitle}
66
+ style={headerStyle}
67
+ onClose={context?.toggleVisible}
68
+ showClose={!hiddenHeaderCloseIcon}
69
+ />
70
+ {childNode}
71
+ </Flex>
72
+ </Modal>
73
+ </Portal>
74
+ );
75
+ };