useRangePickerValue.js 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196
  1. "use strict";
  2. var _interopRequireWildcard = require("@babel/runtime/helpers/interopRequireWildcard").default;
  3. var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault").default;
  4. Object.defineProperty(exports, "__esModule", {
  5. value: true
  6. });
  7. exports.default = useRangePickerValue;
  8. exports.offsetPanelDate = offsetPanelDate;
  9. var _slicedToArray2 = _interopRequireDefault(require("@babel/runtime/helpers/slicedToArray"));
  10. var _rcUtil = require("rc-util");
  11. var _useLayoutEffect = _interopRequireDefault(require("rc-util/lib/hooks/useLayoutEffect"));
  12. var React = _interopRequireWildcard(require("react"));
  13. var _dateUtil = require("../../utils/dateUtil");
  14. function offsetPanelDate(generateConfig, picker, date, offset) {
  15. switch (picker) {
  16. case 'date':
  17. case 'week':
  18. return generateConfig.addMonth(date, offset);
  19. case 'month':
  20. case 'quarter':
  21. return generateConfig.addYear(date, offset);
  22. case 'year':
  23. return generateConfig.addYear(date, offset * 10);
  24. case 'decade':
  25. return generateConfig.addYear(date, offset * 100);
  26. default:
  27. return date;
  28. }
  29. }
  30. var EMPTY_LIST = [];
  31. function useRangePickerValue(generateConfig, locale, calendarValue, modes, open, activeIndex, pickerMode, multiplePanel) {
  32. var defaultPickerValue = arguments.length > 8 && arguments[8] !== undefined ? arguments[8] : EMPTY_LIST;
  33. var pickerValue = arguments.length > 9 && arguments[9] !== undefined ? arguments[9] : EMPTY_LIST;
  34. var timeDefaultValue = arguments.length > 10 && arguments[10] !== undefined ? arguments[10] : EMPTY_LIST;
  35. var onPickerValueChange = arguments.length > 11 ? arguments[11] : undefined;
  36. var minDate = arguments.length > 12 ? arguments[12] : undefined;
  37. var maxDate = arguments.length > 13 ? arguments[13] : undefined;
  38. var isTimePicker = pickerMode === 'time';
  39. // ======================== Active ========================
  40. // `activeIndex` must be valid to avoid getting empty `pickerValue`
  41. var mergedActiveIndex = activeIndex || 0;
  42. // ===================== Picker Value =====================
  43. var getDefaultPickerValue = function getDefaultPickerValue(index) {
  44. var now = generateConfig.getNow();
  45. if (isTimePicker) {
  46. now = (0, _dateUtil.fillTime)(generateConfig, now);
  47. }
  48. return defaultPickerValue[index] || calendarValue[index] || now;
  49. };
  50. // Align `pickerValue` with `showTime.defaultValue`
  51. var _pickerValue = (0, _slicedToArray2.default)(pickerValue, 2),
  52. startPickerValue = _pickerValue[0],
  53. endPickerValue = _pickerValue[1];
  54. // PickerValue state
  55. var _useMergedState = (0, _rcUtil.useMergedState)(function () {
  56. return getDefaultPickerValue(0);
  57. }, {
  58. value: startPickerValue
  59. }),
  60. _useMergedState2 = (0, _slicedToArray2.default)(_useMergedState, 2),
  61. mergedStartPickerValue = _useMergedState2[0],
  62. setStartPickerValue = _useMergedState2[1];
  63. var _useMergedState3 = (0, _rcUtil.useMergedState)(function () {
  64. return getDefaultPickerValue(1);
  65. }, {
  66. value: endPickerValue
  67. }),
  68. _useMergedState4 = (0, _slicedToArray2.default)(_useMergedState3, 2),
  69. mergedEndPickerValue = _useMergedState4[0],
  70. setEndPickerValue = _useMergedState4[1];
  71. // Current PickerValue
  72. var currentPickerValue = React.useMemo(function () {
  73. var current = [mergedStartPickerValue, mergedEndPickerValue][mergedActiveIndex];
  74. // Merge the `showTime.defaultValue` into `pickerValue`
  75. return isTimePicker ? current : (0, _dateUtil.fillTime)(generateConfig, current, timeDefaultValue[mergedActiveIndex]);
  76. }, [isTimePicker, mergedStartPickerValue, mergedEndPickerValue, mergedActiveIndex, generateConfig, timeDefaultValue]);
  77. var setCurrentPickerValue = function setCurrentPickerValue(nextPickerValue) {
  78. var source = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 'panel';
  79. var updater = [setStartPickerValue, setEndPickerValue][mergedActiveIndex];
  80. updater(nextPickerValue);
  81. var clone = [mergedStartPickerValue, mergedEndPickerValue];
  82. clone[mergedActiveIndex] = nextPickerValue;
  83. if (onPickerValueChange && (!(0, _dateUtil.isSame)(generateConfig, locale, mergedStartPickerValue, clone[0], pickerMode) || !(0, _dateUtil.isSame)(generateConfig, locale, mergedEndPickerValue, clone[1], pickerMode))) {
  84. onPickerValueChange(clone, {
  85. source: source,
  86. range: mergedActiveIndex === 1 ? 'end' : 'start',
  87. mode: modes
  88. });
  89. }
  90. };
  91. // ======================== Effect ========================
  92. /**
  93. * EndDate pickerValue is little different. It should be:
  94. * - If date picker (without time), endDate is not same year & month as startDate
  95. * - pickerValue minus one month
  96. * - Else pass directly
  97. */
  98. var getEndDatePickerValue = function getEndDatePickerValue(startDate, endDate) {
  99. if (multiplePanel) {
  100. // Basic offset
  101. var SAME_CHECKER = {
  102. date: 'month',
  103. week: 'month',
  104. month: 'year',
  105. quarter: 'year'
  106. };
  107. var mode = SAME_CHECKER[pickerMode];
  108. if (mode && !(0, _dateUtil.isSame)(generateConfig, locale, startDate, endDate, mode)) {
  109. return offsetPanelDate(generateConfig, pickerMode, endDate, -1);
  110. }
  111. // Year offset
  112. if (pickerMode === 'year' && startDate) {
  113. var srcYear = Math.floor(generateConfig.getYear(startDate) / 10);
  114. var tgtYear = Math.floor(generateConfig.getYear(endDate) / 10);
  115. if (srcYear !== tgtYear) {
  116. return offsetPanelDate(generateConfig, pickerMode, endDate, -1);
  117. }
  118. }
  119. }
  120. return endDate;
  121. };
  122. // >>> When switch field, reset the picker value as prev field picker value
  123. var prevActiveIndexRef = React.useRef(null);
  124. (0, _useLayoutEffect.default)(function () {
  125. if (open) {
  126. if (!defaultPickerValue[mergedActiveIndex]) {
  127. var nextPickerValue = isTimePicker ? null : generateConfig.getNow();
  128. /**
  129. * 1. If has prevActiveIndex, use it to avoid panel jump
  130. * 2. If current field has value
  131. * - If `activeIndex` is 1 and `calendarValue[0]` is not same panel as `calendarValue[1]`,
  132. * offset `calendarValue[1]` and set it
  133. * - Else use `calendarValue[activeIndex]`
  134. * 3. If current field has no value but another field has value, use another field value
  135. * 4. Else use now (not any `calendarValue` can ref)
  136. */
  137. if (prevActiveIndexRef.current !== null && prevActiveIndexRef.current !== mergedActiveIndex) {
  138. // If from another field, not jump picker value
  139. nextPickerValue = [mergedStartPickerValue, mergedEndPickerValue][mergedActiveIndex ^ 1];
  140. } else if (calendarValue[mergedActiveIndex]) {
  141. // Current field has value
  142. nextPickerValue = mergedActiveIndex === 0 ? calendarValue[0] : getEndDatePickerValue(calendarValue[0], calendarValue[1]);
  143. } else if (calendarValue[mergedActiveIndex ^ 1]) {
  144. // Current field has no value but another field has value
  145. nextPickerValue = calendarValue[mergedActiveIndex ^ 1];
  146. }
  147. // Only sync when has value, this will sync in the `min-max` logic
  148. if (nextPickerValue) {
  149. // nextPickerValue < minDate
  150. if (minDate && generateConfig.isAfter(minDate, nextPickerValue)) {
  151. nextPickerValue = minDate;
  152. }
  153. // maxDate < nextPickerValue
  154. var offsetPickerValue = multiplePanel ? offsetPanelDate(generateConfig, pickerMode, nextPickerValue, 1) : nextPickerValue;
  155. if (maxDate && generateConfig.isAfter(offsetPickerValue, maxDate)) {
  156. nextPickerValue = multiplePanel ? offsetPanelDate(generateConfig, pickerMode, maxDate, -1) : maxDate;
  157. }
  158. setCurrentPickerValue(nextPickerValue, 'reset');
  159. }
  160. }
  161. }
  162. }, [open, mergedActiveIndex, calendarValue[mergedActiveIndex]]);
  163. // >>> Reset prevActiveIndex when panel closed
  164. React.useEffect(function () {
  165. if (open) {
  166. prevActiveIndexRef.current = mergedActiveIndex;
  167. } else {
  168. prevActiveIndexRef.current = null;
  169. }
  170. }, [open, mergedActiveIndex]);
  171. // >>> defaultPickerValue: Resync to `defaultPickerValue` for each panel focused
  172. (0, _useLayoutEffect.default)(function () {
  173. if (open && defaultPickerValue) {
  174. if (defaultPickerValue[mergedActiveIndex]) {
  175. setCurrentPickerValue(defaultPickerValue[mergedActiveIndex], 'reset');
  176. }
  177. }
  178. }, [open, mergedActiveIndex]);
  179. return [currentPickerValue, setCurrentPickerValue];
  180. }