index.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. <template>
  2. <view>
  3. <van-popup
  4. v-if="poppable"
  5. :custom-class="'van-calendar__popup--' + position"
  6. close-icon-class="van-calendar__close-icon"
  7. :show="show"
  8. :round="round"
  9. :position="position"
  10. :closeable="showTitle || showSubtitle"
  11. :close-on-click-overlay="closeOnClickOverlay"
  12. @enter="onOpen"
  13. @close="onClose"
  14. @after-enter="onOpened"
  15. @after-leave="onClosed"
  16. >
  17. <!-- parse <template is="calendar" :data="title, subtitle, showTitle, showSubtitle, minDate, maxDate, type, color, showMark, formatter, rowHeight, currentDate, safeAreaInsetBottom, showConfirm, confirmDisabledText, confirmText, scrollIntoView, allowSameDay"/> -->
  18. <block name="calendar">
  19. <view class="van-calendar">
  20. <header :title="title" :showTitle="showTitle" :subtitle="subtitle" :showSubtitle="showSubtitle">
  21. <slot name="title" slot="title"></slot>
  22. </header>
  23. <scroll-view class="van-calendar__body" scroll-y :scroll-into-view="scrollIntoView">
  24. <month
  25. :id="'month' + index"
  26. class="month"
  27. :data-date="item"
  28. :date="item"
  29. :type="type"
  30. :color="color"
  31. :minDate="minDate"
  32. :maxDate="maxDate"
  33. :showMark="showMark"
  34. :formatter="formatter"
  35. :rowHeight="rowHeight"
  36. :currentDate="currentDate"
  37. :showSubtitle="showSubtitle"
  38. :allowSameDay="allowSameDay"
  39. :showMonthTitle="index !== 0 || !showSubtitle"
  40. @click="onClickDay($event, { date: item, tagId: 'month' + index })"
  41. v-for="(item, index) in computed.getMonths(minDate, maxDate)"
  42. :key="index"
  43. ></month>
  44. </scroll-view>
  45. <view :class="'van-calendar__footer ' + (safeAreaInsetBottom ? 'van-calendar__footer--safe-area-inset-bottom' : '')">
  46. <slot name="footer"></slot>
  47. </view>
  48. <view :class="'van-calendar__footer ' + (safeAreaInsetBottom ? 'van-calendar__footer--safe-area-inset-bottom' : '')">
  49. <van-button
  50. v-if="showConfirm"
  51. round
  52. block
  53. type="danger"
  54. :color="color"
  55. custom-class="van-calendar__confirm"
  56. :disabled="computed.getButtonDisabled(type, currentDate)"
  57. nativeType="text"
  58. @click="onConfirm"
  59. >
  60. {{ computed.getButtonDisabled(type, currentDate) ? confirmDisabledText : confirmText }}
  61. </van-button>
  62. </view>
  63. </view>
  64. </block>
  65. </van-popup>
  66. <!-- parse <template v-else is="calendar" :data="title, subtitle, showTitle, showSubtitle, minDate, maxDate, type, color, showMark, formatter, rowHeight, currentDate, safeAreaInsetBottom, showConfirm, confirmDisabledText, confirmText, scrollIntoView, allowSameDay"/> -->
  67. <block name="calendar" v-if="false" v-else>
  68. <view class="van-calendar">
  69. <header :title="title" :showTitle="showTitle" :subtitle="subtitle" :showSubtitle="showSubtitle">
  70. <slot name="title" slot="title"></slot>
  71. </header>
  72. <scroll-view class="van-calendar__body" scroll-y :scroll-into-view="scrollIntoView">
  73. <month
  74. :id="'month' + index"
  75. class="month"
  76. :data-date="item"
  77. :date="item"
  78. :type="type"
  79. :color="color"
  80. :minDate="minDate"
  81. :maxDate="maxDate"
  82. :showMark="showMark"
  83. :formatter="formatter"
  84. :rowHeight="rowHeight"
  85. :currentDate="currentDate"
  86. :showSubtitle="showSubtitle"
  87. :allowSameDay="allowSameDay"
  88. :showMonthTitle="index !== 0 || !showSubtitle"
  89. @click="onClickDay($event, { date: item, tagId: 'month' + index })"
  90. v-for="(item, index) in computed.getMonths(minDate, maxDate)"
  91. :key="index"
  92. ></month>
  93. </scroll-view>
  94. <view :class="'van-calendar__footer ' + (safeAreaInsetBottom ? 'van-calendar__footer--safe-area-inset-bottom' : '')">
  95. <slot name="footer"></slot>
  96. </view>
  97. <view :class="'van-calendar__footer ' + (safeAreaInsetBottom ? 'van-calendar__footer--safe-area-inset-bottom' : '')">
  98. <van-button
  99. v-if="showConfirm"
  100. round
  101. block
  102. type="danger"
  103. :color="color"
  104. custom-class="van-calendar__confirm"
  105. :disabled="computed.getButtonDisabled(type, currentDate)"
  106. nativeType="text"
  107. @click="onConfirm"
  108. >
  109. {{ computed.getButtonDisabled(type, currentDate) ? confirmDisabledText : confirmText }}
  110. </van-button>
  111. </view>
  112. </view>
  113. </block>
  114. <van-toast id="van-toast" />
  115. </view>
  116. </template>
  117. <script module="computed" lang="wxs" src="@/node_modules/@vant/weapp/dist/calendar/index.wxs"></script>
  118. <script>
  119. import header from './components/header/index';
  120. import month from './components/month/index';
  121. import { VantComponent } from '../common/component';
  122. import { ROW_HEIGHT, getNextDay, compareDay, copyDates, calcDateNum, formatMonthTitle, compareMonth, getMonths, getDayByOffset } from './utils';
  123. import Toast from '../toast/toast';
  124. export default {
  125. components: {
  126. header,
  127. month
  128. },
  129. data() {
  130. return {
  131. subtitle: '',
  132. currentDate: null,
  133. scrollIntoView: ''
  134. };
  135. },
  136. props: {
  137. title: {
  138. type: String,
  139. default: '日期选择'
  140. },
  141. color: String,
  142. show: {
  143. type: Boolean
  144. },
  145. formatter: null,
  146. confirmText: {
  147. type: String,
  148. default: '确定'
  149. },
  150. rangePrompt: String,
  151. defaultDate: {
  152. type: [Number, Array]
  153. },
  154. allowSameDay: Boolean,
  155. confirmDisabledText: String,
  156. type: {
  157. type: String,
  158. default: 'single'
  159. },
  160. minDate: {
  161. type: null,
  162. default: Date.now()
  163. },
  164. maxDate: {
  165. type: null,
  166. default: new Date(new Date().getFullYear(), new Date().getMonth() + 6, new Date().getDate()).getTime()
  167. },
  168. position: {
  169. type: String,
  170. default: 'bottom'
  171. },
  172. rowHeight: {
  173. type: [Number, String],
  174. default: ROW_HEIGHT
  175. },
  176. round: {
  177. type: Boolean,
  178. default: true
  179. },
  180. poppable: {
  181. type: Boolean,
  182. default: true
  183. },
  184. showMark: {
  185. type: Boolean,
  186. default: true
  187. },
  188. showTitle: {
  189. type: Boolean,
  190. default: true
  191. },
  192. showConfirm: {
  193. type: Boolean,
  194. default: true
  195. },
  196. showSubtitle: {
  197. type: Boolean,
  198. default: true
  199. },
  200. safeAreaInsetBottom: {
  201. type: Boolean,
  202. default: true
  203. },
  204. closeOnClickOverlay: {
  205. type: Boolean,
  206. default: true
  207. },
  208. maxRange: {
  209. type: [Number, String],
  210. default: null
  211. }
  212. },
  213. created() {
  214. this.setData({
  215. currentDate: this.getInitialDate()
  216. });
  217. },
  218. mounted() {
  219. if (this.show || !this.poppable) {
  220. this.initRect();
  221. this.scrollIntoViewFun();
  222. }
  223. },
  224. methods: {
  225. reset() {
  226. this.setData({
  227. currentDate: this.getInitialDate()
  228. });
  229. this.scrollIntoViewFun();
  230. },
  231. initRect() {
  232. if (this.contentObserver != null) {
  233. this.contentObserver.disconnect();
  234. }
  235. const contentObserver = this.createIntersectionObserver({
  236. thresholds: [0, 0.1, 0.9, 1],
  237. observeAll: true
  238. });
  239. this.contentObserver = contentObserver;
  240. contentObserver.relativeTo('.van-calendar__body');
  241. contentObserver.observe('.month', (res) => {
  242. if (res.boundingClientRect.top <= res.relativeRect.top) {
  243. // @ts-ignore
  244. this.setData({
  245. subtitle: formatMonthTitle(res.dataset.date)
  246. });
  247. }
  248. });
  249. },
  250. getInitialDate() {
  251. const { type, defaultDate, minDate } = this;
  252. if (type === 'range') {
  253. const [startDay, endDay] = defaultDate || [];
  254. return [startDay || minDate, endDay || getNextDay(new Date(minDate)).getTime()];
  255. }
  256. if (type === 'multiple') {
  257. return defaultDate || [minDate];
  258. }
  259. return defaultDate || minDate;
  260. },
  261. scrollIntoViewFun() {
  262. setTimeout(() => {
  263. const { currentDate, type, show, poppable, minDate, maxDate } = this;
  264. const targetDate = type === 'single' ? currentDate : currentDate[0];
  265. const displayed = show || !poppable;
  266. if (!targetDate || !displayed) {
  267. return;
  268. }
  269. const months = getMonths(minDate, maxDate);
  270. months.some((month, index) => {
  271. if (compareMonth(month, targetDate) === 0) {
  272. this.setData({
  273. scrollIntoView: `month${index}`
  274. });
  275. return true;
  276. }
  277. return false;
  278. });
  279. }, 100);
  280. },
  281. onOpen() {
  282. this.$emit('open');
  283. },
  284. onOpened() {
  285. this.$emit('opened');
  286. },
  287. onClose() {
  288. this.$emit('close');
  289. },
  290. onClosed() {
  291. this.$emit('closed');
  292. },
  293. onClickDay(event, _dataset) {
  294. /* ---处理dataset begin--- */
  295. this.handleDataset(event, _dataset);
  296. /* ---处理dataset end--- */
  297. const { date } = event.detail;
  298. const { type, currentDate, allowSameDay } = this;
  299. if (type === 'range') {
  300. const [startDay, endDay] = currentDate;
  301. if (startDay && !endDay) {
  302. const compareToStart = compareDay(date, startDay);
  303. if (compareToStart === 1) {
  304. this.select([startDay, date], true);
  305. } else if (compareToStart === -1) {
  306. this.select([date, null]);
  307. } else if (allowSameDay) {
  308. this.select([date, date]);
  309. }
  310. } else {
  311. this.select([date, null]);
  312. }
  313. } else if (type === 'multiple') {
  314. let selectedIndex;
  315. const selected = currentDate.some((dateItem, index) => {
  316. const equal = compareDay(dateItem, date) === 0;
  317. if (equal) {
  318. selectedIndex = index;
  319. }
  320. return equal;
  321. });
  322. if (selected) {
  323. const cancelDate = currentDate.splice(selectedIndex, 1);
  324. this.setData({
  325. currentDate
  326. });
  327. this.unselect(cancelDate);
  328. } else {
  329. this.select([...currentDate, date]);
  330. }
  331. } else {
  332. this.select(date, true);
  333. }
  334. },
  335. unselect(dateArray) {
  336. const date = dateArray[0];
  337. if (date) {
  338. this.$emit('unselect', copyDates(date));
  339. }
  340. },
  341. select(date, complete) {
  342. if (complete && this.type === 'range') {
  343. const valid = this.checkRange(date);
  344. if (!valid) {
  345. // auto selected to max range if showConfirm
  346. if (this.showConfirm) {
  347. this.emit([date[0], getDayByOffset(date[0], this.maxRange - 1)]);
  348. } else {
  349. this.emit(date);
  350. }
  351. return;
  352. }
  353. }
  354. this.emit(date);
  355. if (complete && !this.showConfirm) {
  356. this.onConfirm();
  357. }
  358. },
  359. emit(date) {
  360. const getTime = (date) => (date instanceof Date ? date.getTime() : date);
  361. this.setData({
  362. currentDate: Array.isArray(date) ? date.map(getTime) : getTime(date)
  363. });
  364. this.$emit('select', copyDates(date));
  365. },
  366. checkRange(date) {
  367. const { maxRange, rangePrompt } = this;
  368. if (maxRange && calcDateNum(date) > maxRange) {
  369. Toast({
  370. context: this,
  371. message: rangePrompt || `选择天数不能超过 ${maxRange} 天`
  372. });
  373. return false;
  374. }
  375. return true;
  376. },
  377. onConfirm() {
  378. if (this.type === 'range' && !this.checkRange(this.currentDate)) {
  379. return;
  380. }
  381. this.$nextTick(() => {
  382. this.$emit('confirm', copyDates(this.currentDate));
  383. });
  384. }
  385. },
  386. watch: {
  387. show: {
  388. handler: function (val) {
  389. if (val) {
  390. this.initRect();
  391. this.scrollIntoViewFun();
  392. }
  393. },
  394. immediate: true
  395. },
  396. defaultDate: {
  397. handler: function (val) {
  398. this.setData({
  399. currentDate: val
  400. });
  401. this.scrollIntoViewFun();
  402. },
  403. immediate: true
  404. },
  405. type: {
  406. handler: function () {
  407. this.setData({
  408. currentDate: this.getInitialDate()
  409. });
  410. this.scrollIntoViewFun();
  411. },
  412. immediate: true
  413. }
  414. }
  415. };
  416. </script>
  417. <style>
  418. @import './index.css';
  419. @import 'undefined';
  420. </style>