uni-datetime-picker.vue 25 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019
  1. <template>
  2. <view class="uni-date">
  3. <view class="uni-date-editor" @click="show">
  4. <slot>
  5. <view class="uni-date-editor--x" :class="{'uni-date-editor--x__disabled': disabled,
  6. 'uni-date-x--border': border}">
  7. <view v-if="!isRange" class="uni-date-x uni-date-single">
  8. <uni-icons type="calendar" color="#c0c4cc" size="22"></uni-icons>
  9. <!-- <input class="uni-date__x-input" type="text" v-model="singleVal"
  10. :placeholder="singlePlaceholderText" readonly /> -->
  11. <view class="uni-date__x-input">{{singleVal}}</view>
  12. </view>
  13. <view v-else class="uni-date-x uni-date-range">
  14. <uni-icons type="calendar" color="#c0c4cc" size="22"></uni-icons>
  15. <input class="uni-date__x-input t-c" type="text" v-model="range.startDate"
  16. :placeholder="startPlaceholderText" readonly />
  17. <slot>
  18. <view class="">{{rangeSeparator}}</view>
  19. </slot>
  20. <input class="uni-date__x-input t-c" type="text" v-model="range.endDate"
  21. :placeholder="endPlaceholderText" readonly />
  22. </view>
  23. <view v-if="showClearIcon" class="uni-date__icon-clear" @click.stop="clear">
  24. <uni-icons type="clear" color="#c0c4cc" size="24"></uni-icons>
  25. </view>
  26. </view>
  27. </slot>
  28. </view>
  29. <view v-show="popup" class="uni-date-mask" @click="close"></view>
  30. <view v-if="!isPhone" ref="datePicker" v-show="popup" class="uni-date-picker__container">
  31. <view v-if="!isRange" class="uni-date-single--x" :style="popover">
  32. <view class="uni-popper__arrow"></view>
  33. <view v-if="hasTime" class="uni-date-changed popup-x-header">
  34. <input class="uni-date__input t-c" type="text" v-model="tempSingleDate"
  35. :placeholder="selectDateText" />
  36. <time-picker type="time" v-model="time" :border="false" :disabled="!tempSingleDate"
  37. :start="reactStartTime" :end="reactEndTime" :hideSecond="hideSecond" style="width: 100%;">
  38. <input class="uni-date__input t-c" type="text" v-model="time" :placeholder="selectTimeText"
  39. :disabled="!tempSingleDate" />
  40. </time-picker>
  41. </view>
  42. <calendar ref="pcSingle" :showMonth="false" :start-date="caleRange.startDate"
  43. :end-date="caleRange.endDate" :date="defSingleDate" @change="singleChange"
  44. style="padding: 0 8px;" />
  45. <view v-if="hasTime" class="popup-x-footer">
  46. <!-- <text class="">此刻</text> -->
  47. <text class="confirm" @click="confirmSingleChange">{{okText}}</text>
  48. </view>
  49. <view class="uni-date-popper__arrow"></view>
  50. </view>
  51. <view v-else class="uni-date-range--x" :style="popover">
  52. <view class="uni-popper__arrow"></view>
  53. <view v-if="hasTime" class="popup-x-header uni-date-changed">
  54. <view class="popup-x-header--datetime">
  55. <input class="uni-date__input uni-date-range__input" type="text" v-model="tempRange.startDate"
  56. :placeholder="startDateText" />
  57. <time-picker type="time" v-model="tempRange.startTime" :start="reactStartTime" :border="false"
  58. :disabled="!tempRange.startDate" :hideSecond="hideSecond">
  59. <input class="uni-date__input uni-date-range__input" type="text"
  60. v-model="tempRange.startTime" :placeholder="startTimeText"
  61. :disabled="!tempRange.startDate" />
  62. </time-picker>
  63. </view>
  64. <uni-icons type="arrowthinright" color="#999" style="line-height: 40px;"></uni-icons>
  65. <view class="popup-x-header--datetime">
  66. <input class="uni-date__input uni-date-range__input" type="text" v-model="tempRange.endDate"
  67. :placeholder="endDateText" />
  68. <time-picker type="time" v-model="tempRange.endTime" :end="reactEndTime" :border="false"
  69. :disabled="!tempRange.endDate" :hideSecond="hideSecond">
  70. <input class="uni-date__input uni-date-range__input" type="text" v-model="tempRange.endTime"
  71. :placeholder="endTimeText" :disabled="!tempRange.endDate" />
  72. </time-picker>
  73. </view>
  74. </view>
  75. <view class="popup-x-body">
  76. <calendar ref="left" :showMonth="false" :start-date="caleRange.startDate"
  77. :end-date="caleRange.endDate" :range="true" @change="leftChange" :pleStatus="endMultipleStatus"
  78. @firstEnterCale="updateRightCale" @monthSwitch="leftMonthSwitch" style="padding: 0 8px;" />
  79. <calendar ref="right" :showMonth="false" :start-date="caleRange.startDate"
  80. :end-date="caleRange.endDate" :range="true" @change="rightChange"
  81. :pleStatus="startMultipleStatus" @firstEnterCale="updateLeftCale"
  82. @monthSwitch="rightMonthSwitch" style="padding: 0 8px;border-left: 1px solid #F1F1F1;" />
  83. </view>
  84. <view v-if="hasTime" class="popup-x-footer">
  85. <text class="" @click="clear">{{clearText}}</text>
  86. <text class="confirm" @click="confirmRangeChange">{{okText}}</text>
  87. </view>
  88. </view>
  89. </view>
  90. <calendar v-show="isPhone" ref="mobile" :clearDate="false" :date="defSingleDate" :defTime="reactMobDefTime"
  91. :start-date="caleRange.startDate" :end-date="caleRange.endDate" :selectableTimes="mobSelectableTime"
  92. :pleStatus="endMultipleStatus" :showMonth="false" :range="isRange" :typeHasTime="hasTime" :insert="false"
  93. :hideSecond="hideSecond" @confirm="mobileChange" @maskClose="close" />
  94. </view>
  95. </template>
  96. <script>
  97. /**
  98. * DatetimePicker 时间选择器
  99. * @description 同时支持 PC 和移动端使用日历选择日期和日期范围
  100. * @tutorial https://ext.dcloud.net.cn/plugin?id=3962
  101. * @property {String} type 选择器类型
  102. * @property {String|Number|Array|Date} value 绑定值
  103. * @property {String} placeholder 单选择时的占位内容
  104. * @property {String} start 起始时间
  105. * @property {String} end 终止时间
  106. * @property {String} start-placeholder 范围选择时开始日期的占位内容
  107. * @property {String} end-placeholder 范围选择时结束日期的占位内容
  108. * @property {String} range-separator 选择范围时的分隔符
  109. * @property {Boolean} border = [true|false] 是否有边框
  110. * @property {Boolean} disabled = [true|false] 是否禁用
  111. * @property {Boolean} clearIcon = [true|false] 是否显示清除按钮(仅PC端适用)
  112. * @event {Function} change 确定日期时触发的事件
  113. * @event {Function} show 打开弹出层
  114. * @event {Function} close 关闭弹出层
  115. * @event {Function} clear 清除上次选中的状态和值
  116. **/
  117. import calendar from './calendar.vue'
  118. import timePicker from './time-picker.vue'
  119. import {
  120. initVueI18n
  121. } from '@dcloudio/uni-i18n'
  122. import messages from './i18n/index.js'
  123. const {
  124. t
  125. } = initVueI18n(messages)
  126. export default {
  127. name: 'UniDatetimePicker',
  128. options: {
  129. virtualHost: true
  130. },
  131. components: {
  132. calendar,
  133. timePicker
  134. },
  135. inject: {
  136. form: {
  137. from: 'uniForm',
  138. default: null
  139. },
  140. formItem: {
  141. from: 'uniFormItem',
  142. default: null
  143. },
  144. },
  145. data() {
  146. return {
  147. isRange: false,
  148. hasTime: false,
  149. mobileRange: false,
  150. // 单选
  151. singleVal: '',
  152. tempSingleDate: '',
  153. defSingleDate: '',
  154. time: '',
  155. // 范围选
  156. caleRange: {
  157. startDate: '',
  158. startTime: '',
  159. endDate: '',
  160. endTime: ''
  161. },
  162. range: {
  163. startDate: '',
  164. // startTime: '',
  165. endDate: '',
  166. // endTime: ''
  167. },
  168. tempRange: {
  169. startDate: '',
  170. startTime: '',
  171. endDate: '',
  172. endTime: ''
  173. },
  174. // 左右日历同步数据
  175. startMultipleStatus: {
  176. before: '',
  177. after: '',
  178. data: [],
  179. fulldate: ''
  180. },
  181. endMultipleStatus: {
  182. before: '',
  183. after: '',
  184. data: [],
  185. fulldate: ''
  186. },
  187. visible: false,
  188. popup: false,
  189. popover: null,
  190. isEmitValue: false,
  191. isPhone: false,
  192. isFirstShow: true,
  193. }
  194. },
  195. props: {
  196. type: {
  197. type: String,
  198. default: 'datetime'
  199. },
  200. value: {
  201. type: [String, Number, Array, Date],
  202. default: ''
  203. },
  204. modelValue: {
  205. type: [String, Number, Array, Date],
  206. default: ''
  207. },
  208. start: {
  209. type: [Number, String],
  210. default: ''
  211. },
  212. end: {
  213. type: [Number, String],
  214. default: ''
  215. },
  216. returnType: {
  217. type: String,
  218. default: 'string'
  219. },
  220. placeholder: {
  221. type: String,
  222. default: ''
  223. },
  224. startPlaceholder: {
  225. type: String,
  226. default: ''
  227. },
  228. endPlaceholder: {
  229. type: String,
  230. default: ''
  231. },
  232. rangeSeparator: {
  233. type: String,
  234. default: '-'
  235. },
  236. border: {
  237. type: [Boolean],
  238. default: true
  239. },
  240. disabled: {
  241. type: [Boolean],
  242. default: false
  243. },
  244. clearIcon: {
  245. type: [Boolean],
  246. default: true
  247. },
  248. hideSecond: {
  249. type: [Boolean],
  250. default: false
  251. }
  252. },
  253. watch: {
  254. type: {
  255. immediate: true,
  256. handler(newVal, oldVal) {
  257. if (newVal.indexOf('time') !== -1) {
  258. this.hasTime = true
  259. } else {
  260. this.hasTime = false
  261. }
  262. if (newVal.indexOf('range') !== -1) {
  263. this.isRange = true
  264. } else {
  265. this.isRange = false
  266. }
  267. }
  268. },
  269. // #ifndef VUE3
  270. value: {
  271. immediate: true,
  272. handler(newVal, oldVal) {
  273. if (this.isEmitValue) {
  274. this.isEmitValue = false
  275. return
  276. }
  277. this.initPicker(newVal)
  278. }
  279. },
  280. // #endif
  281. // #ifdef VUE3
  282. modelValue: {
  283. immediate: true,
  284. handler(newVal, oldVal) {
  285. if (this.isEmitValue) {
  286. this.isEmitValue = false
  287. return
  288. }
  289. this.initPicker(newVal)
  290. }
  291. },
  292. // #endif
  293. start: {
  294. immediate: true,
  295. handler(newVal, oldVal) {
  296. if (!newVal) return
  297. const {
  298. defDate,
  299. defTime
  300. } = this.parseDate(newVal)
  301. this.caleRange.startDate = defDate
  302. if (this.hasTime) {
  303. this.caleRange.startTime = defTime
  304. }
  305. }
  306. },
  307. end: {
  308. immediate: true,
  309. handler(newVal, oldVal) {
  310. if (!newVal) return
  311. const {
  312. defDate,
  313. defTime
  314. } = this.parseDate(newVal)
  315. this.caleRange.endDate = defDate
  316. if (this.hasTime) {
  317. this.caleRange.endTime = defTime
  318. }
  319. }
  320. },
  321. },
  322. computed: {
  323. reactStartTime() {
  324. const activeDate = this.isRange ? this.tempRange.startDate : this.tempSingleDate
  325. const res = activeDate === this.caleRange.startDate ? this.caleRange.startTime : ''
  326. return res
  327. },
  328. reactEndTime() {
  329. const activeDate = this.isRange ? this.tempRange.endDate : this.tempSingleDate
  330. const res = activeDate === this.caleRange.endDate ? this.caleRange.endTime : ''
  331. return res
  332. },
  333. reactMobDefTime() {
  334. const times = {
  335. start: this.tempRange.startTime,
  336. end: this.tempRange.endTime
  337. }
  338. return this.isRange ? times : this.time
  339. },
  340. mobSelectableTime() {
  341. return {
  342. start: this.caleRange.startTime,
  343. end: this.caleRange.endTime
  344. }
  345. },
  346. datePopupWidth() {
  347. // todo
  348. return this.isRange ? 653 : 301
  349. },
  350. /**
  351. * for i18n
  352. */
  353. singlePlaceholderText() {
  354. return this.placeholder || (this.type === 'date' ? this.selectDateText : t(
  355. "uni-datetime-picker.selectDateTime"))
  356. },
  357. startPlaceholderText() {
  358. return this.startPlaceholder || this.startDateText
  359. },
  360. endPlaceholderText() {
  361. return this.endPlaceholder || this.endDateText
  362. },
  363. selectDateText() {
  364. return t("uni-datetime-picker.selectDate")
  365. },
  366. selectTimeText() {
  367. return t("uni-datetime-picker.selectTime")
  368. },
  369. startDateText() {
  370. return this.startPlaceholder || t("uni-datetime-picker.startDate")
  371. },
  372. startTimeText() {
  373. return t("uni-datetime-picker.startTime")
  374. },
  375. endDateText() {
  376. return this.endPlaceholder || t("uni-datetime-picker.endDate")
  377. },
  378. endTimeText() {
  379. return t("uni-datetime-picker.endTime")
  380. },
  381. okText() {
  382. return t("uni-datetime-picker.ok")
  383. },
  384. clearText() {
  385. return t("uni-datetime-picker.clear")
  386. },
  387. showClearIcon() {
  388. const {
  389. clearIcon,
  390. disabled,
  391. singleVal,
  392. range
  393. } = this
  394. const bool = clearIcon && !disabled && (singleVal || (range.startDate && range.endDate))
  395. return bool
  396. }
  397. },
  398. created() {
  399. // if (this.form && this.formItem) {
  400. // this.$watch('formItem.errMsg', (newVal) => {
  401. // this.localMsg = newVal
  402. // })
  403. // }
  404. },
  405. mounted() {
  406. this.platform()
  407. },
  408. methods: {
  409. initPicker(newVal) {
  410. if (!newVal || Array.isArray(newVal) && !newVal.length) {
  411. this.$nextTick(() => {
  412. this.clear(false)
  413. })
  414. return
  415. }
  416. if (!Array.isArray(newVal) && !this.isRange) {
  417. const {
  418. defDate,
  419. defTime
  420. } = this.parseDate(newVal)
  421. this.singleVal = defDate
  422. this.tempSingleDate = defDate
  423. this.defSingleDate = defDate
  424. if (this.hasTime) {
  425. this.singleVal = defDate + ' ' + defTime
  426. this.time = defTime
  427. }
  428. } else {
  429. const [before, after] = newVal
  430. if (!before && !after) return
  431. const defBefore = this.parseDate(before)
  432. const defAfter = this.parseDate(after)
  433. const startDate = defBefore.defDate
  434. const endDate = defAfter.defDate
  435. this.range.startDate = this.tempRange.startDate = startDate
  436. this.range.endDate = this.tempRange.endDate = endDate
  437. if (this.hasTime) {
  438. this.range.startDate = defBefore.defDate + ' ' + defBefore.defTime
  439. this.range.endDate = defAfter.defDate + ' ' + defAfter.defTime
  440. this.tempRange.startTime = defBefore.defTime
  441. this.tempRange.endTime = defAfter.defTime
  442. }
  443. const defaultRange = {
  444. before: defBefore.defDate,
  445. after: defAfter.defDate
  446. }
  447. this.startMultipleStatus = Object.assign({}, this.startMultipleStatus, defaultRange, {
  448. which: 'right'
  449. })
  450. this.endMultipleStatus = Object.assign({}, this.endMultipleStatus, defaultRange, {
  451. which: 'left'
  452. })
  453. }
  454. },
  455. updateLeftCale(e) {
  456. const left = this.$refs.left
  457. // 设置范围选
  458. left.cale.setHoverMultiple(e.after)
  459. left.setDate(this.$refs.left.nowDate.fullDate)
  460. },
  461. updateRightCale(e) {
  462. const right = this.$refs.right
  463. // 设置范围选
  464. right.cale.setHoverMultiple(e.after)
  465. right.setDate(this.$refs.right.nowDate.fullDate)
  466. },
  467. platform() {
  468. const systemInfo = uni.getSystemInfoSync()
  469. this.isPhone = systemInfo.windowWidth <= 500
  470. this.windowWidth = systemInfo.windowWidth
  471. },
  472. show(event) {
  473. if (this.disabled) {
  474. return
  475. }
  476. this.platform()
  477. if (this.isPhone) {
  478. this.$refs.mobile.open()
  479. return
  480. }
  481. this.popover = {
  482. top: '10px'
  483. }
  484. const dateEditor = uni.createSelectorQuery().in(this).select(".uni-date-editor")
  485. dateEditor.boundingClientRect(rect => {
  486. if (this.windowWidth - rect.left < this.datePopupWidth) {
  487. this.popover.right = 0
  488. }
  489. }).exec()
  490. setTimeout(() => {
  491. this.popup = !this.popup
  492. if (!this.isPhone && this.isRange && this.isFirstShow) {
  493. this.isFirstShow = false
  494. const {
  495. startDate,
  496. endDate
  497. } = this.range
  498. if (startDate && endDate) {
  499. if (this.diffDate(startDate, endDate) < 30) {
  500. this.$refs.right.next()
  501. }
  502. } else {
  503. this.$refs.right.next()
  504. this.$refs.right.cale.lastHover = false
  505. }
  506. }
  507. }, 50)
  508. },
  509. close() {
  510. setTimeout(() => {
  511. this.popup = false
  512. this.$emit('maskClick', this.value)
  513. this.$refs.mobile.close()
  514. }, 20)
  515. },
  516. setEmit(value) {
  517. if (this.returnType === "timestamp" || this.returnType === "date") {
  518. if (!Array.isArray(value)) {
  519. if (!this.hasTime) {
  520. value = value + ' ' + '00:00:00'
  521. }
  522. value = this.createTimestamp(value)
  523. if (this.returnType === "date") {
  524. value = new Date(value)
  525. }
  526. } else {
  527. if (!this.hasTime) {
  528. value[0] = value[0] + ' ' + '00:00:00'
  529. value[1] = value[1] + ' ' + '00:00:00'
  530. }
  531. value[0] = this.createTimestamp(value[0])
  532. value[1] = this.createTimestamp(value[1])
  533. if (this.returnType === "date") {
  534. value[0] = new Date(value[0])
  535. value[1] = new Date(value[1])
  536. }
  537. }
  538. }
  539. this.$emit('change', value)
  540. this.$emit('input', value)
  541. this.$emit('update:modelValue', value)
  542. this.isEmitValue = true
  543. },
  544. createTimestamp(date) {
  545. date = this.fixIosDateFormat(date)
  546. return Date.parse(new Date(date))
  547. },
  548. singleChange(e) {
  549. this.tempSingleDate = e.fulldate
  550. if (this.hasTime) return
  551. this.confirmSingleChange()
  552. },
  553. confirmSingleChange() {
  554. if (!this.tempSingleDate) {
  555. this.popup = false
  556. return
  557. }
  558. if (this.hasTime) {
  559. this.singleVal = this.tempSingleDate + ' ' + (this.time ? this.time : '00:00:00')
  560. } else {
  561. this.singleVal = this.tempSingleDate
  562. }
  563. this.setEmit(this.singleVal)
  564. this.popup = false
  565. },
  566. leftChange(e) {
  567. const {
  568. before,
  569. after
  570. } = e.range
  571. this.rangeChange(before, after)
  572. const obj = {
  573. before: e.range.before,
  574. after: e.range.after,
  575. data: e.range.data,
  576. fulldate: e.fulldate
  577. }
  578. this.startMultipleStatus = Object.assign({}, this.startMultipleStatus, obj)
  579. },
  580. rightChange(e) {
  581. const {
  582. before,
  583. after
  584. } = e.range
  585. this.rangeChange(before, after)
  586. const obj = {
  587. before: e.range.before,
  588. after: e.range.after,
  589. data: e.range.data,
  590. fulldate: e.fulldate
  591. }
  592. this.endMultipleStatus = Object.assign({}, this.endMultipleStatus, obj)
  593. },
  594. mobileChange(e) {
  595. if (this.isRange) {
  596. const {
  597. before,
  598. after
  599. } = e.range
  600. this.handleStartAndEnd(before, after, true)
  601. if (this.hasTime) {
  602. const {
  603. startTime,
  604. endTime
  605. } = e.timeRange
  606. this.tempRange.startTime = startTime
  607. this.tempRange.endTime = endTime
  608. }
  609. this.confirmRangeChange()
  610. } else {
  611. if (this.hasTime) {
  612. this.singleVal = e.fulldate + ' ' + e.time
  613. } else {
  614. this.singleVal = e.fulldate
  615. }
  616. this.setEmit(this.singleVal)
  617. }
  618. this.$refs.mobile.close()
  619. },
  620. rangeChange(before, after) {
  621. if (!(before && after)) return
  622. this.handleStartAndEnd(before, after, true)
  623. if (this.hasTime) return
  624. this.confirmRangeChange()
  625. },
  626. confirmRangeChange() {
  627. if (!this.tempRange.startDate && !this.tempRange.endDate) {
  628. this.popup = false
  629. return
  630. }
  631. let start, end
  632. if (!this.hasTime) {
  633. start = this.range.startDate = this.tempRange.startDate
  634. end = this.range.endDate = this.tempRange.endDate
  635. } else {
  636. start = this.range.startDate = this.tempRange.startDate + ' ' +
  637. (this.tempRange.startTime ? this.tempRange.startTime : '00:00:00')
  638. end = this.range.endDate = this.tempRange.endDate + ' ' +
  639. (this.tempRange.endTime ? this.tempRange.endTime : '00:00:00')
  640. }
  641. const displayRange = [start, end]
  642. this.setEmit(displayRange)
  643. this.popup = false
  644. },
  645. handleStartAndEnd(before, after, temp = false) {
  646. if (!(before && after)) return
  647. const type = temp ? 'tempRange' : 'range'
  648. if (this.dateCompare(before, after)) {
  649. this[type].startDate = before
  650. this[type].endDate = after
  651. } else {
  652. this[type].startDate = after
  653. this[type].endDate = before
  654. }
  655. },
  656. /**
  657. * 比较时间大小
  658. */
  659. dateCompare(startDate, endDate) {
  660. // 计算截止时间
  661. startDate = new Date(startDate.replace('-', '/').replace('-', '/'))
  662. // 计算详细项的截止时间
  663. endDate = new Date(endDate.replace('-', '/').replace('-', '/'))
  664. if (startDate <= endDate) {
  665. return true
  666. } else {
  667. return false
  668. }
  669. },
  670. /**
  671. * 比较时间差
  672. */
  673. diffDate(startDate, endDate) {
  674. // 计算截止时间
  675. startDate = new Date(startDate.replace('-', '/').replace('-', '/'))
  676. // 计算详细项的截止时间
  677. endDate = new Date(endDate.replace('-', '/').replace('-', '/'))
  678. const diff = (endDate - startDate) / (24 * 60 * 60 * 1000)
  679. return Math.abs(diff)
  680. },
  681. clear(needEmit = true) {
  682. if (!this.isRange) {
  683. this.singleVal = ''
  684. this.tempSingleDate = ''
  685. this.time = ''
  686. if (this.isPhone) {
  687. this.$refs.mobile && this.$refs.mobile.clearCalender()
  688. } else {
  689. this.$refs.pcSingle && this.$refs.pcSingle.clearCalender()
  690. }
  691. if (needEmit) {
  692. // 校验规则
  693. // if(this.form && this.formItem){
  694. // const {
  695. // validateTrigger
  696. // } = this.form
  697. // if (validateTrigger === 'blur') {
  698. // this.formItem.onFieldChange()
  699. // }
  700. // }
  701. this.$emit('change', '')
  702. this.$emit('input', '')
  703. this.$emit('update:modelValue', '')
  704. }
  705. } else {
  706. this.range.startDate = ''
  707. this.range.endDate = ''
  708. this.tempRange.startDate = ''
  709. this.tempRange.startTime = ''
  710. this.tempRange.endDate = ''
  711. this.tempRange.endTime = ''
  712. if (this.isPhone) {
  713. this.$refs.mobile && this.$refs.mobile.clearCalender()
  714. } else {
  715. this.$refs.left && this.$refs.left.clearCalender()
  716. this.$refs.right && this.$refs.right.clearCalender()
  717. this.$refs.right && this.$refs.right.next()
  718. }
  719. if (needEmit) {
  720. this.$emit('change', [])
  721. this.$emit('input', [])
  722. this.$emit('update:modelValue', [])
  723. }
  724. }
  725. },
  726. parseDate(date) {
  727. date = this.fixIosDateFormat(date)
  728. const defVal = new Date(date)
  729. const year = defVal.getFullYear()
  730. const month = defVal.getMonth() + 1
  731. const day = defVal.getDate()
  732. const hour = defVal.getHours()
  733. const minute = defVal.getMinutes()
  734. const second = defVal.getSeconds()
  735. const defDate = year + '-' + this.lessTen(month) + '-' + this.lessTen(day)
  736. const defTime = this.lessTen(hour) + ':' + this.lessTen(minute) + (this.hideSecond ? '' : (':' + this
  737. .lessTen(second)))
  738. return {
  739. defDate,
  740. defTime
  741. }
  742. },
  743. lessTen(item) {
  744. return item < 10 ? '0' + item : item
  745. },
  746. //兼容 iOS、safari 日期格式
  747. fixIosDateFormat(value) {
  748. if (typeof value === 'string') {
  749. value = value.replace(/-/g, '/')
  750. }
  751. return value
  752. },
  753. leftMonthSwitch(e) {
  754. // console.log('leftMonthSwitch 返回:', e)
  755. },
  756. rightMonthSwitch(e) {
  757. // console.log('rightMonthSwitch 返回:', e)
  758. }
  759. }
  760. }
  761. </script>
  762. <style lang="scss">
  763. $uni-primary: #007aff !default;
  764. .uni-date {
  765. /* #ifndef APP-NVUE */
  766. width: 100%;
  767. /* #endif */
  768. flex: 1;
  769. }
  770. .uni-date-x {
  771. display: flex;
  772. flex-direction: row;
  773. align-items: center;
  774. justify-content: center;
  775. padding: 0 10px;
  776. border-radius: 4px;
  777. background-color: #fff;
  778. color: #666;
  779. font-size: 14px;
  780. flex: 1;
  781. }
  782. .uni-date-x--border {
  783. box-sizing: border-box;
  784. border-radius: 4px;
  785. border: 1px solid #e5e5e5;
  786. }
  787. .uni-date-editor--x {
  788. display: flex;
  789. align-items: center;
  790. position: relative;
  791. }
  792. .uni-date-editor--x .uni-date__icon-clear {
  793. padding: 0 5px;
  794. display: flex;
  795. align-items: center;
  796. /* #ifdef H5 */
  797. cursor: pointer;
  798. /* #endif */
  799. }
  800. .uni-date__x-input {
  801. padding: 0 8px;
  802. /* #ifndef APP-NVUE */
  803. width: auto;
  804. /* #endif */
  805. position: relative;
  806. overflow: hidden;
  807. flex: 1;
  808. line-height: 1;
  809. font-size: 12px;
  810. height: 35px;
  811. display: flex;
  812. align-items: center;
  813. }
  814. .t-c {
  815. text-align: center;
  816. }
  817. .uni-date__input {
  818. height: 40px;
  819. width: 100%;
  820. line-height: 40px;
  821. font-size: 14px;
  822. }
  823. .uni-date-range__input {
  824. text-align: center;
  825. max-width: 142px;
  826. }
  827. .uni-date-picker__container {
  828. position: relative;
  829. /* position: fixed;
  830. left: 0;
  831. right: 0;
  832. top: 0;
  833. bottom: 0;
  834. box-sizing: border-box;
  835. z-index: 996;
  836. font-size: 14px; */
  837. }
  838. .uni-date-mask {
  839. position: fixed;
  840. bottom: 0px;
  841. top: 0px;
  842. left: 0px;
  843. right: 0px;
  844. background-color: rgba(0, 0, 0, 0);
  845. transition-duration: 0.3s;
  846. z-index: 996;
  847. }
  848. .uni-date-single--x {
  849. /* padding: 0 8px; */
  850. background-color: #fff;
  851. position: absolute;
  852. top: 0;
  853. z-index: 999;
  854. border: 1px solid #EBEEF5;
  855. box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
  856. border-radius: 4px;
  857. }
  858. .uni-date-range--x {
  859. /* padding: 0 8px; */
  860. background-color: #fff;
  861. position: absolute;
  862. top: 0;
  863. z-index: 999;
  864. border: 1px solid #EBEEF5;
  865. box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
  866. border-radius: 4px;
  867. }
  868. .uni-date-editor--x__disabled {
  869. opacity: 0.4;
  870. cursor: default;
  871. }
  872. .uni-date-editor--logo {
  873. width: 16px;
  874. height: 16px;
  875. vertical-align: middle;
  876. }
  877. /* 添加时间 */
  878. .popup-x-header {
  879. /* #ifndef APP-NVUE */
  880. display: flex;
  881. /* #endif */
  882. flex-direction: row;
  883. /* justify-content: space-between; */
  884. }
  885. .popup-x-header--datetime {
  886. /* #ifndef APP-NVUE */
  887. display: flex;
  888. /* #endif */
  889. flex-direction: row;
  890. flex: 1;
  891. }
  892. .popup-x-body {
  893. display: flex;
  894. }
  895. .popup-x-footer {
  896. padding: 0 15px;
  897. border-top-color: #F1F1F1;
  898. border-top-style: solid;
  899. border-top-width: 1px;
  900. /* background-color: #fff; */
  901. line-height: 40px;
  902. text-align: right;
  903. color: #666;
  904. }
  905. .popup-x-footer text:hover {
  906. color: $uni-primary;
  907. cursor: pointer;
  908. opacity: 0.8;
  909. }
  910. .popup-x-footer .confirm {
  911. margin-left: 20px;
  912. color: $uni-primary;
  913. }
  914. .uni-date-changed {
  915. /* background-color: #fff; */
  916. text-align: center;
  917. color: #333;
  918. border-bottom-color: #F1F1F1;
  919. border-bottom-style: solid;
  920. border-bottom-width: 1px;
  921. /* padding: 0 50px; */
  922. }
  923. .uni-date-changed--time text {
  924. /* padding: 0 20px; */
  925. height: 50px;
  926. line-height: 50px;
  927. }
  928. .uni-date-changed .uni-date-changed--time {
  929. /* display: flex; */
  930. flex: 1;
  931. }
  932. .uni-date-changed--time-date {
  933. color: #333;
  934. opacity: 0.6;
  935. }
  936. .mr-50 {
  937. margin-right: 50px;
  938. }
  939. /* picker 弹出层通用的指示小三角, todo:扩展至上下左右方向定位 */
  940. .uni-popper__arrow,
  941. .uni-popper__arrow::after {
  942. position: absolute;
  943. display: block;
  944. width: 0;
  945. height: 0;
  946. border-color: transparent;
  947. border-style: solid;
  948. border-width: 6px;
  949. }
  950. .uni-popper__arrow {
  951. filter: drop-shadow(0 2px 12px rgba(0, 0, 0, 0.03));
  952. top: -6px;
  953. left: 10%;
  954. margin-right: 3px;
  955. border-top-width: 0;
  956. border-bottom-color: #EBEEF5;
  957. }
  958. .uni-popper__arrow::after {
  959. content: " ";
  960. top: 1px;
  961. margin-left: -6px;
  962. border-top-width: 0;
  963. border-bottom-color: #fff;
  964. }
  965. </style>