iov-design 2.15.60 → 2.15.62

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 (242) hide show
  1. package/README.md +148 -148
  2. package/lib/alert.js +24 -105
  3. package/lib/aside.js +24 -34
  4. package/lib/autocomplete.js +36 -240
  5. package/lib/avatar.js +21 -20
  6. package/lib/backtop.js +27 -52
  7. package/lib/badge.js +30 -58
  8. package/lib/breadcrumb-item.js +24 -53
  9. package/lib/breadcrumb.js +24 -37
  10. package/lib/button-group.js +24 -29
  11. package/lib/button.js +38 -60
  12. package/lib/calendar.js +29 -146
  13. package/lib/card.js +24 -51
  14. package/lib/carousel-item.js +30 -68
  15. package/lib/carousel.js +27 -159
  16. package/lib/cascader-panel.js +41 -64
  17. package/lib/cascader.js +37 -470
  18. package/lib/checkbox-button.js +28 -164
  19. package/lib/checkbox-group.js +24 -37
  20. package/lib/checkbox.js +35 -189
  21. package/lib/col.js +1 -0
  22. package/lib/collapse-item.js +27 -125
  23. package/lib/collapse.js +24 -37
  24. package/lib/color-picker.js +104 -424
  25. package/lib/container.js +24 -34
  26. package/lib/date-picker.js +126 -1981
  27. package/lib/descriptions-item.js +1 -0
  28. package/lib/descriptions.js +1 -0
  29. package/lib/dialog.js +24 -137
  30. package/lib/divider.js +29 -64
  31. package/lib/drawer.js +24 -139
  32. package/lib/dropdown-item.js +27 -45
  33. package/lib/dropdown-menu.js +24 -51
  34. package/lib/dropdown.js +21 -20
  35. package/lib/empty.js +24 -61
  36. package/lib/footer.js +24 -34
  37. package/lib/form-item.js +42 -128
  38. package/lib/form.js +27 -40
  39. package/lib/header.js +24 -34
  40. package/lib/icon.js +24 -29
  41. package/lib/image.js +36 -220
  42. package/lib/index.js +1 -1
  43. package/lib/infinite-scroll.js +1 -0
  44. package/lib/input-number.js +30 -163
  45. package/lib/input.js +42 -274
  46. package/lib/iov-design.common.js +1635 -11132
  47. package/lib/link.js +31 -56
  48. package/lib/loading.js +24 -77
  49. package/lib/locale/format.js +10 -10
  50. package/lib/main.js +24 -29
  51. package/lib/menu-item-group.js +24 -40
  52. package/lib/menu-item.js +27 -83
  53. package/lib/menu.js +21 -20
  54. package/lib/message-box.js +24 -321
  55. package/lib/message.js +30 -82
  56. package/lib/mixins/migrating.js +20 -20
  57. package/lib/notification.js +24 -109
  58. package/lib/option-group.js +24 -50
  59. package/lib/option.js +27 -76
  60. package/lib/page-header.js +24 -56
  61. package/lib/pagination.js +24 -97
  62. package/lib/popconfirm.js +24 -97
  63. package/lib/popover.js +24 -89
  64. package/lib/progress.js +30 -133
  65. package/lib/radio-button.js +29 -111
  66. package/lib/radio-group.js +24 -39
  67. package/lib/radio.js +36 -152
  68. package/lib/rate.js +24 -94
  69. package/lib/result.js +72 -187
  70. package/lib/row.js +1 -0
  71. package/lib/scrollbar.js +1 -0
  72. package/lib/select.js +48 -637
  73. package/lib/skeleton-item.js +36 -68
  74. package/lib/skeleton.js +28 -70
  75. package/lib/slider.js +36 -267
  76. package/lib/spinner.js +24 -51
  77. package/lib/statistic.js +24 -76
  78. package/lib/step.js +29 -118
  79. package/lib/steps.js +27 -40
  80. package/lib/submenu.js +21 -20
  81. package/lib/switch.js +24 -130
  82. package/lib/tab-pane.js +24 -52
  83. package/lib/table-column.js +1 -0
  84. package/lib/table.js +93 -633
  85. package/lib/tabs.js +34 -49
  86. package/lib/tag.js +21 -20
  87. package/lib/theme-chalk/autocomplete.css +1 -1
  88. package/lib/theme-chalk/base.css +1 -1
  89. package/lib/theme-chalk/cascader.css +1 -1
  90. package/lib/theme-chalk/date-picker.css +1 -1
  91. package/lib/theme-chalk/index.css +1 -1
  92. package/lib/theme-chalk/input-number.css +1 -1
  93. package/lib/theme-chalk/input.css +1 -1
  94. package/lib/theme-chalk/iovfont.css +1 -1
  95. package/lib/theme-chalk/link.css +1 -1
  96. package/lib/theme-chalk/message-box.css +1 -1
  97. package/lib/theme-chalk/pagination.css +1 -1
  98. package/lib/theme-chalk/select.css +1 -1
  99. package/lib/theme-chalk/slider.css +1 -1
  100. package/lib/theme-chalk/time-picker.css +1 -1
  101. package/lib/theme-chalk/transfer.css +1 -1
  102. package/lib/time-picker.js +58 -820
  103. package/lib/time-select.js +42 -303
  104. package/lib/timeline-item.js +29 -75
  105. package/lib/timeline.js +21 -20
  106. package/lib/tooltip.js +1 -0
  107. package/lib/transfer.js +36 -299
  108. package/lib/tree.js +54 -243
  109. package/lib/upload.js +86 -654
  110. package/lib/utils/clickoutside.js +7 -7
  111. package/lib/utils/vue-popper.js +7 -7
  112. package/package.json +154 -154
  113. package/packages/autocomplete/src/autocomplete-suggestions.vue +76 -76
  114. package/packages/autocomplete/src/autocomplete.vue +285 -285
  115. package/packages/button/src/button.vue +90 -90
  116. package/packages/calendar/src/date-table.vue +200 -200
  117. package/packages/calendar/src/main.vue +280 -280
  118. package/packages/carousel/src/item.vue +138 -138
  119. package/packages/carousel/src/main.vue +315 -315
  120. package/packages/cascader/src/cascader.vue +776 -776
  121. package/packages/cascader-panel/src/cascader-menu.vue +138 -138
  122. package/packages/cascader-panel/src/cascader-node.vue +246 -246
  123. package/packages/cascader-panel/src/cascader-panel.vue +391 -391
  124. package/packages/cascader-panel/src/node.js +166 -166
  125. package/packages/cascader-panel/src/store.js +58 -58
  126. package/packages/checkbox/src/checkbox-button.vue +199 -199
  127. package/packages/checkbox/src/checkbox-group.vue +49 -49
  128. package/packages/checkbox/src/checkbox.vue +225 -225
  129. package/packages/collapse/src/collapse-item.vue +114 -114
  130. package/packages/color-picker/src/components/picker-dropdown.vue +121 -121
  131. package/packages/color-picker/src/main.vue +188 -188
  132. package/packages/date-picker/src/basic/date-table.vue +441 -441
  133. package/packages/date-picker/src/basic/month-table.vue +269 -269
  134. package/packages/date-picker/src/basic/time-spinner.vue +304 -304
  135. package/packages/date-picker/src/basic/year-table.vue +111 -111
  136. package/packages/date-picker/src/panel/date-range.vue +680 -680
  137. package/packages/date-picker/src/panel/date.vue +609 -609
  138. package/packages/date-picker/src/panel/month-range.vue +289 -289
  139. package/packages/date-picker/src/panel/time-range.vue +248 -248
  140. package/packages/date-picker/src/panel/time-select.vue +178 -178
  141. package/packages/date-picker/src/panel/time.vue +186 -186
  142. package/packages/date-picker/src/picker.vue +967 -967
  143. package/packages/descriptions/src/index.js +180 -180
  144. package/packages/dialog/src/component.vue +262 -262
  145. package/packages/drawer/src/main.vue +205 -205
  146. package/packages/dropdown/src/dropdown-item.vue +37 -37
  147. package/packages/dropdown/src/dropdown-menu.vue +63 -63
  148. package/packages/dropdown/src/dropdown.vue +293 -293
  149. package/packages/empty/src/index.vue +70 -70
  150. package/packages/form/src/form-item.vue +324 -324
  151. package/packages/form/src/form.vue +182 -182
  152. package/packages/image/src/image-viewer.vue +330 -330
  153. package/packages/image/src/main.vue +249 -249
  154. package/packages/infinite-scroll/src/main.js +150 -150
  155. package/packages/input/src/input.vue +477 -477
  156. package/packages/input-number/src/input-number.vue +283 -283
  157. package/packages/loading/src/directive.js +133 -133
  158. package/packages/loading/src/index.js +106 -106
  159. package/packages/menu/src/menu-item.vue +112 -112
  160. package/packages/menu/src/menu.vue +325 -325
  161. package/packages/menu/src/submenu.vue +349 -349
  162. package/packages/message/src/main.js +91 -91
  163. package/packages/message-box/src/main.js +216 -216
  164. package/packages/message-box/src/main.vue +333 -333
  165. package/packages/notification/src/main.js +94 -94
  166. package/packages/page-header/src/main.vue +30 -30
  167. package/packages/pagination/src/pagination.js +390 -390
  168. package/packages/popconfirm/src/main.vue +104 -104
  169. package/packages/popover/src/main.vue +239 -239
  170. package/packages/radio/src/radio-button.vue +115 -115
  171. package/packages/radio/src/radio-group.vue +115 -115
  172. package/packages/radio/src/radio.vue +148 -148
  173. package/packages/rate/src/main.vue +348 -348
  174. package/packages/scrollbar/src/bar.js +92 -92
  175. package/packages/scrollbar/src/main.js +130 -130
  176. package/packages/select/src/option-group.vue +60 -60
  177. package/packages/select/src/option.vue +171 -171
  178. package/packages/select/src/select-dropdown.vue +74 -74
  179. package/packages/select/src/select.vue +979 -979
  180. package/packages/slider/src/button.vue +238 -238
  181. package/packages/slider/src/main.vue +427 -427
  182. package/packages/statistic/src/main.vue +204 -204
  183. package/packages/steps/src/steps.vue +68 -68
  184. package/packages/switch/src/component.vue +182 -182
  185. package/packages/table/src/config.js +153 -153
  186. package/packages/table/src/filter-panel.vue +194 -194
  187. package/packages/table/src/store/current.js +76 -76
  188. package/packages/table/src/store/helper.js +41 -41
  189. package/packages/table/src/store/index.js +147 -147
  190. package/packages/table/src/store/watcher.js +502 -502
  191. package/packages/table/src/table-body.js +469 -469
  192. package/packages/table/src/table-column.js +328 -328
  193. package/packages/table/src/table-header.js +571 -571
  194. package/packages/table/src/table-layout.js +249 -249
  195. package/packages/table/src/table-row.js +101 -101
  196. package/packages/table/src/table.vue +740 -740
  197. package/packages/table/src/util.js +273 -273
  198. package/packages/tabs/src/tab-bar.vue +57 -57
  199. package/packages/tabs/src/tab-nav.vue +294 -294
  200. package/packages/tabs/src/tabs.vue +201 -201
  201. package/packages/theme-chalk/src/button.scss +416 -416
  202. package/packages/theme-chalk/src/cascader.scss +252 -252
  203. package/packages/theme-chalk/src/checkbox.scss +419 -419
  204. package/packages/theme-chalk/src/common/var.scss +795 -795
  205. package/packages/theme-chalk/src/form.scss +201 -201
  206. package/packages/theme-chalk/src/input.scss +591 -591
  207. package/packages/theme-chalk/src/iovfont.scss +818 -803
  208. package/packages/theme-chalk/src/link.scss +8 -8
  209. package/packages/theme-chalk/src/message-box.scss +207 -207
  210. package/packages/theme-chalk/src/mixins/_button.scss +136 -136
  211. package/packages/theme-chalk/src/mixins/mixins.scss +190 -190
  212. package/packages/theme-chalk/src/radio-button.scss +115 -115
  213. package/packages/theme-chalk/src/select.scss +270 -270
  214. package/packages/theme-chalk/src/tabs.scss +755 -755
  215. package/packages/theme-chalk/src/upload.scss +568 -568
  216. package/packages/tooltip/src/main.js +242 -242
  217. package/packages/transfer/src/main.vue +231 -231
  218. package/packages/transfer/src/transfer-panel.vue +251 -251
  219. package/packages/tree/src/model/node.js +484 -484
  220. package/packages/tree/src/tree-node.vue +279 -279
  221. package/packages/tree/src/tree.vue +496 -496
  222. package/packages/upload/src/index.vue +420 -420
  223. package/packages/upload/src/upload-file.vue +98 -98
  224. package/packages/upload/src/upload-list.vue +115 -115
  225. package/packages/upload/src/upload-picture.vue +98 -98
  226. package/packages/upload/src/upload.vue +231 -231
  227. package/packages/upload/src/utils.js +31 -31
  228. package/src/directives/repeat-click.js +26 -26
  229. package/src/index.js +259 -259
  230. package/src/locale/format.js +46 -46
  231. package/src/locale/index.js +48 -48
  232. package/src/locale/lang/lo-LA.js +126 -126
  233. package/src/mixins/locale.js +9 -9
  234. package/src/mixins/migrating.js +54 -54
  235. package/src/transitions/collapse-transition.js +77 -77
  236. package/src/utils/clickoutside.js +76 -76
  237. package/src/utils/date-util.js +282 -282
  238. package/src/utils/popup/index.js +218 -218
  239. package/src/utils/popup/popup-manager.js +194 -194
  240. package/src/utils/util.js +245 -245
  241. package/src/utils/vdom.js +5 -5
  242. package/src/utils/vue-popper.js +198 -198
@@ -1,609 +1,609 @@
1
- <template>
2
- <transition name="el-zoom-in-top" @after-enter="handleEnter" @after-leave="handleLeave">
3
- <div
4
- v-show="visible"
5
- class="el-picker-panel el-date-picker el-popper"
6
- :class="[{
7
- 'has-sidebar': $slots.sidebar || shortcuts,
8
- 'has-time': showTime
9
- }, popperClass]">
10
- <div class="el-picker-panel__body-wrapper">
11
- <slot name="sidebar" class="el-picker-panel__sidebar"></slot>
12
- <div class="el-picker-panel__sidebar" v-if="shortcuts">
13
- <button
14
- type="button"
15
- class="el-picker-panel__shortcut"
16
- v-for="(shortcut, key) in shortcuts"
17
- :key="key"
18
- @click="handleShortcutClick(shortcut)">{{ shortcut.text }}</button>
19
- </div>
20
- <div class="el-picker-panel__body">
21
- <div class="el-date-picker__time-header" v-if="showTime">
22
- <span class="el-date-picker__editor-wrap">
23
- <el-input
24
- :placeholder="t('el.datepicker.selectDate')"
25
- :value="visibleDate"
26
- size="small"
27
- @input="val => userInputDate = val"
28
- @change="handleVisibleDateChange" />
29
- </span>
30
- <span class="el-date-picker__editor-wrap" v-clickoutside="handleTimePickClose">
31
- <el-input
32
- ref="input"
33
- @focus="timePickerVisible = true"
34
- :placeholder="t('el.datepicker.selectTime')"
35
- :value="visibleTime"
36
- size="small"
37
- @input="val => userInputTime = val"
38
- @change="handleVisibleTimeChange" />
39
- <time-picker
40
- ref="timepicker"
41
- :time-arrow-control="arrowControl"
42
- @pick="handleTimePick"
43
- :visible="timePickerVisible"
44
- @mounted="proxyTimePickerDataProperties">
45
- </time-picker>
46
- </span>
47
- </div>
48
- <div
49
- class="el-date-picker__header"
50
- :class="{ 'el-date-picker__header--bordered': currentView === 'year' || currentView === 'month' }"
51
- v-show="currentView !== 'time'">
52
- <button
53
- type="button"
54
- @click="prevYear"
55
- :aria-label="t(`el.datepicker.prevYear`)"
56
- class="el-picker-panel__icon-btn el-date-picker__prev-btn el-icon-d-arrow-left">
57
- </button>
58
- <button
59
- type="button"
60
- @click="prevMonth"
61
- v-show="currentView === 'date'"
62
- :aria-label="t(`el.datepicker.prevMonth`)"
63
- class="el-picker-panel__icon-btn el-date-picker__prev-btn el-icon-arrow-left">
64
- </button>
65
- <span
66
- @click="showYearPicker"
67
- role="button"
68
- class="el-date-picker__header-label">{{ yearLabel }}</span>
69
- <span
70
- @click="showMonthPicker"
71
- v-show="currentView === 'date'"
72
- role="button"
73
- class="el-date-picker__header-label"
74
- :class="{ active: currentView === 'month' }">{{t(`el.datepicker.month${ month + 1 }`)}}</span>
75
- <button
76
- type="button"
77
- @click="nextYear"
78
- :aria-label="t(`el.datepicker.nextYear`)"
79
- class="el-picker-panel__icon-btn el-date-picker__next-btn el-icon-d-arrow-right">
80
- </button>
81
- <button
82
- type="button"
83
- @click="nextMonth"
84
- v-show="currentView === 'date'"
85
- :aria-label="t(`el.datepicker.nextMonth`)"
86
- class="el-picker-panel__icon-btn el-date-picker__next-btn el-icon-arrow-right">
87
- </button>
88
- </div>
89
-
90
- <div class="el-picker-panel__content">
91
- <date-table
92
- v-show="currentView === 'date'"
93
- @pick="handleDatePick"
94
- :selection-mode="selectionMode"
95
- :first-day-of-week="firstDayOfWeek"
96
- :value="value"
97
- :default-value="defaultValue ? new Date(defaultValue) : null"
98
- :date="date"
99
- :cell-class-name="cellClassName"
100
- :disabled-date="disabledDate">
101
- </date-table>
102
- <year-table
103
- v-show="currentView === 'year'"
104
- @pick="handleYearPick"
105
- :selection-mode="selectionMode"
106
- :value="value"
107
- :default-value="defaultValue ? new Date(defaultValue) : null"
108
- :date="date"
109
- :disabled-date="disabledDate">
110
- </year-table>
111
- <month-table
112
- v-show="currentView === 'month'"
113
- @pick="handleMonthPick"
114
- :selection-mode="selectionMode"
115
- :value="value"
116
- :default-value="defaultValue ? new Date(defaultValue) : null"
117
- :date="date"
118
- :disabled-date="disabledDate">
119
- </month-table>
120
- </div>
121
- </div>
122
- </div>
123
-
124
- <div
125
- class="el-picker-panel__footer"
126
- v-show="footerVisible && (currentView === 'date' || currentView === 'month' || currentView === 'year')">
127
- <el-button
128
- size="mini"
129
- type="text"
130
- class="el-picker-panel__link-btn"
131
- @click="changeToNow"
132
- v-show="selectionMode !== 'dates' && selectionMode !== 'months' && selectionMode !== 'years'">
133
- {{ t('el.datepicker.now') }}
134
- </el-button>
135
- <el-button
136
- plain
137
- size="mini"
138
- class="el-picker-panel__link-btn"
139
- @click="confirm">
140
- {{ t('el.datepicker.confirm') }}
141
- </el-button>
142
- </div>
143
- </div>
144
- </transition>
145
- </template>
146
-
147
- <script type="text/babel">
148
- import {
149
- formatDate,
150
- parseDate,
151
- getWeekNumber,
152
- isDate,
153
- modifyDate,
154
- modifyTime,
155
- modifyWithTimeString,
156
- clearMilliseconds,
157
- clearTime,
158
- prevYear,
159
- nextYear,
160
- prevMonth,
161
- nextMonth,
162
- changeYearMonthAndClampDate,
163
- extractDateFormat,
164
- extractTimeFormat,
165
- timeWithinRange
166
- } from 'iov-design/src/utils/date-util';
167
- import Clickoutside from 'iov-design/src/utils/clickoutside';
168
- import Locale from 'iov-design/src/mixins/locale';
169
- import ElInput from 'iov-design/packages/input';
170
- import ElButton from 'iov-design/packages/button';
171
- import TimePicker from './time';
172
- import YearTable from '../basic/year-table';
173
- import MonthTable from '../basic/month-table';
174
- import DateTable from '../basic/date-table';
175
-
176
- export default {
177
- mixins: [Locale],
178
-
179
- directives: { Clickoutside },
180
-
181
- watch: {
182
- showTime(val) {
183
- /* istanbul ignore if */
184
- if (!val) return;
185
- this.$nextTick(_ => {
186
- const inputElm = this.$refs.input.$el;
187
- if (inputElm) {
188
- this.pickerWidth = inputElm.getBoundingClientRect().width + 10;
189
- }
190
- });
191
- },
192
-
193
- value(val) {
194
- if (this.selectionMode === 'dates' && this.value) return;
195
- if (this.selectionMode === 'months' && this.value) return;
196
- if (this.selectionMode === 'years' && this.value) return;
197
- if (isDate(val)) {
198
- this.date = new Date(val);
199
- } else {
200
- this.date = this.getDefaultValue();
201
- }
202
- },
203
-
204
- defaultValue(val) {
205
- if (!isDate(this.value)) {
206
- this.date = val ? new Date(val) : new Date();
207
- }
208
- },
209
-
210
- timePickerVisible(val) {
211
- if (val) this.$nextTick(() => this.$refs.timepicker.adjustSpinners());
212
- },
213
-
214
- selectionMode(newVal) {
215
- if (newVal === 'month') {
216
- /* istanbul ignore next */
217
- if (this.currentView !== 'year' || this.currentView !== 'month') {
218
- this.currentView = 'month';
219
- }
220
- } else if (newVal === 'dates') {
221
- this.currentView = 'date';
222
- } else if (newVal === 'years') {
223
- this.currentView = 'year';
224
- } else if (newVal === 'months') {
225
- this.currentView = 'month';
226
- }
227
- }
228
- },
229
-
230
- methods: {
231
- proxyTimePickerDataProperties() {
232
- const format = timeFormat => {this.$refs.timepicker.format = timeFormat;};
233
- const value = value => {this.$refs.timepicker.value = value;};
234
- const date = date => {this.$refs.timepicker.date = date;};
235
- const selectableRange = selectableRange => {this.$refs.timepicker.selectableRange = selectableRange;};
236
-
237
- this.$watch('value', value);
238
- this.$watch('date', date);
239
- this.$watch('selectableRange', selectableRange);
240
-
241
- format(this.timeFormat);
242
- value(this.value);
243
- date(this.date);
244
- selectableRange(this.selectableRange);
245
- },
246
-
247
- handleClear() {
248
- this.date = this.getDefaultValue();
249
- this.$emit('pick', null);
250
- },
251
-
252
- emit(value, ...args) {
253
- if (!value) {
254
- this.$emit('pick', value, ...args);
255
- } else if (Array.isArray(value)) {
256
- const dates = value.map(date => this.showTime ? clearMilliseconds(date) : clearTime(date));
257
- this.$emit('pick', dates, ...args);
258
- } else {
259
- this.$emit('pick', this.showTime ? clearMilliseconds(value) : clearTime(value), ...args);
260
- }
261
- this.userInputDate = null;
262
- this.userInputTime = null;
263
- },
264
-
265
- // resetDate() {
266
- // this.date = new Date(this.date);
267
- // },
268
-
269
- showMonthPicker() {
270
- this.currentView = 'month';
271
- },
272
-
273
- showYearPicker() {
274
- this.currentView = 'year';
275
- },
276
-
277
- // XXX: 没用到
278
- // handleLabelClick() {
279
- // if (this.currentView === 'date') {
280
- // this.showMonthPicker();
281
- // } else if (this.currentView === 'month') {
282
- // this.showYearPicker();
283
- // }
284
- // },
285
-
286
- prevMonth() {
287
- this.date = prevMonth(this.date);
288
- },
289
-
290
- nextMonth() {
291
- this.date = nextMonth(this.date);
292
- },
293
-
294
- prevYear() {
295
- if (this.currentView === 'year') {
296
- this.date = prevYear(this.date, 10);
297
- } else {
298
- this.date = prevYear(this.date);
299
- }
300
- },
301
-
302
- nextYear() {
303
- if (this.currentView === 'year') {
304
- this.date = nextYear(this.date, 10);
305
- } else {
306
- this.date = nextYear(this.date);
307
- }
308
- },
309
-
310
- handleShortcutClick(shortcut) {
311
- if (shortcut.onClick) {
312
- shortcut.onClick(this);
313
- }
314
- },
315
-
316
- handleTimePick(value, visible, first) {
317
- if (isDate(value)) {
318
- const newDate = this.value
319
- ? modifyTime(this.value, value.getHours(), value.getMinutes(), value.getSeconds())
320
- : modifyWithTimeString(this.getDefaultValue(), this.defaultTime);
321
- this.date = newDate;
322
- this.emit(this.date, true);
323
- } else {
324
- this.emit(value, true);
325
- }
326
- if (!first) {
327
- this.timePickerVisible = visible;
328
- }
329
- },
330
-
331
- handleTimePickClose() {
332
- this.timePickerVisible = false;
333
- },
334
-
335
- handleMonthPick(month) {
336
- if (this.selectionMode === 'month') {
337
- this.date = modifyDate(this.date, this.year, month, 1);
338
- this.emit(this.date);
339
- } else if (this.selectionMode === 'months') {
340
- this.emit(month, true);
341
- } else {
342
- this.date = changeYearMonthAndClampDate(this.date, this.year, month);
343
- // TODO: should emit intermediate value ??
344
- // this.emit(this.date);
345
- this.currentView = 'date';
346
- }
347
- },
348
-
349
- handleDatePick(value) {
350
- if (this.selectionMode === 'day') {
351
- let newDate = this.value
352
- ? modifyDate(this.value, value.getFullYear(), value.getMonth(), value.getDate())
353
- : modifyWithTimeString(value, this.defaultTime);
354
- // change default time while out of selectableRange
355
- if (!this.checkDateWithinRange(newDate)) {
356
- newDate = modifyDate(this.selectableRange[0][0], value.getFullYear(), value.getMonth(), value.getDate());
357
- }
358
- this.date = newDate;
359
- this.emit(this.date, this.showTime);
360
- } else if (this.selectionMode === 'week') {
361
- this.emit(value.date);
362
- } else if (this.selectionMode === 'dates') {
363
- this.emit(value, true); // set false to keep panel open
364
- }
365
- },
366
-
367
- handleYearPick(year) {
368
- if (this.selectionMode === 'year') {
369
- this.date = modifyDate(this.date, year, 0, 1);
370
- this.emit(this.date);
371
- } else if (this.selectionMode === 'years') {
372
- this.emit(year, true);
373
- } else {
374
- this.date = changeYearMonthAndClampDate(this.date, year, this.month);
375
- // TODO: should emit intermediate value ??
376
- // this.emit(this.date, true);
377
- this.currentView = 'month';
378
- }
379
- },
380
-
381
- changeToNow() {
382
- // NOTE: not a permanent solution
383
- // consider disable "now" button in the future
384
- if ((!this.disabledDate || !this.disabledDate(new Date())) && this.checkDateWithinRange(new Date())) {
385
- this.date = new Date();
386
- this.emit(this.date);
387
- }
388
- },
389
-
390
- confirm() {
391
- if (this.selectionMode === 'dates' || this.selectionMode === 'months' || this.selectionMode === 'years') {
392
- this.emit(this.value);
393
- } else {
394
- // value were emitted in handle{Date,Time}Pick, nothing to update here
395
- // deal with the scenario where: user opens the picker, then confirm without doing anything
396
- const value = this.value
397
- ? this.value
398
- : modifyWithTimeString(this.getDefaultValue(), this.defaultTime);
399
- this.date = new Date(value); // refresh date
400
- this.emit(value);
401
- }
402
- },
403
-
404
- resetView() {
405
- if (this.selectionMode === 'month' || this.selectionMode === 'months') {
406
- this.currentView = 'month';
407
- } else if (this.selectionMode === 'year' || this.selectionMode === 'years') {
408
- this.currentView = 'year';
409
- } else {
410
- this.currentView = 'date';
411
- }
412
- },
413
-
414
- handleEnter() {
415
- document.body.addEventListener('keydown', this.handleKeydown);
416
- },
417
-
418
- handleLeave() {
419
- this.$emit('dodestroy');
420
- document.body.removeEventListener('keydown', this.handleKeydown);
421
- },
422
-
423
- handleKeydown(event) {
424
- const keyCode = event.keyCode;
425
- const list = [38, 40, 37, 39];
426
- if (this.visible && !this.timePickerVisible) {
427
- if (list.indexOf(keyCode) !== -1) {
428
- this.handleKeyControl(keyCode);
429
- event.stopPropagation();
430
- event.preventDefault();
431
- }
432
- if (keyCode === 13 && this.userInputDate === null && this.userInputTime === null) { // Enter
433
- this.emit(this.date, false);
434
- }
435
- }
436
- },
437
-
438
- handleKeyControl(keyCode) {
439
- const mapping = {
440
- 'year': {
441
- 38: -4, 40: 4, 37: -1, 39: 1, offset: (date, step) => date.setFullYear(date.getFullYear() + step)
442
- },
443
- 'month': {
444
- 38: -4, 40: 4, 37: -1, 39: 1, offset: (date, step) => date.setMonth(date.getMonth() + step)
445
- },
446
- 'week': {
447
- 38: -1, 40: 1, 37: -1, 39: 1, offset: (date, step) => date.setDate(date.getDate() + step * 7)
448
- },
449
- 'day': {
450
- 38: -7, 40: 7, 37: -1, 39: 1, offset: (date, step) => date.setDate(date.getDate() + step)
451
- }
452
- };
453
- const mode = this.selectionMode;
454
- const year = 3.1536e10;
455
- const now = this.date.getTime();
456
- const newDate = new Date(this.date.getTime());
457
- while (Math.abs(now - newDate.getTime()) <= year) {
458
- const map = mapping[mode];
459
- map.offset(newDate, map[keyCode]);
460
- if (typeof this.disabledDate === 'function' && this.disabledDate(newDate)) {
461
- continue;
462
- }
463
- this.date = newDate;
464
- this.$emit('pick', newDate, true);
465
- break;
466
- }
467
- },
468
-
469
- handleVisibleTimeChange(value) {
470
- const time = parseDate(value, this.timeFormat);
471
- if (time && this.checkDateWithinRange(time)) {
472
- this.date = modifyDate(time, this.year, this.month, this.monthDate);
473
- this.userInputTime = null;
474
- this.$refs.timepicker.value = this.date;
475
- this.timePickerVisible = false;
476
- this.emit(this.date, true);
477
- }
478
- },
479
-
480
- handleVisibleDateChange(value) {
481
- const date = parseDate(value, this.dateFormat);
482
- if (date) {
483
- if (typeof this.disabledDate === 'function' && this.disabledDate(date)) {
484
- return;
485
- }
486
- this.date = modifyTime(date, this.date.getHours(), this.date.getMinutes(), this.date.getSeconds());
487
- this.userInputDate = null;
488
- this.resetView();
489
- this.emit(this.date, true);
490
- }
491
- },
492
-
493
- isValidValue(value) {
494
- return value && !isNaN(value) && (
495
- typeof this.disabledDate === 'function'
496
- ? !this.disabledDate(value)
497
- : true
498
- ) && this.checkDateWithinRange(value);
499
- },
500
-
501
- getDefaultValue() {
502
- // if default-value is set, return it
503
- // otherwise, return now (the moment this method gets called)
504
- return this.defaultValue ? new Date(this.defaultValue) : new Date();
505
- },
506
-
507
- checkDateWithinRange(date) {
508
- return this.selectableRange.length > 0
509
- ? timeWithinRange(date, this.selectableRange, this.format || 'HH:mm:ss')
510
- : true;
511
- }
512
- },
513
-
514
- components: {
515
- TimePicker, YearTable, MonthTable, DateTable, ElInput, ElButton
516
- },
517
-
518
- data() {
519
- return {
520
- popperClass: '',
521
- date: new Date(),
522
- value: '',
523
- defaultValue: null, // use getDefaultValue() for time computation
524
- defaultTime: null,
525
- showTime: false,
526
- selectionMode: 'day',
527
- shortcuts: '',
528
- visible: false,
529
- currentView: 'date',
530
- disabledDate: '',
531
- cellClassName: '',
532
- selectableRange: [],
533
- firstDayOfWeek: 7,
534
- showWeekNumber: false,
535
- timePickerVisible: false,
536
- format: '',
537
- arrowControl: false,
538
- userInputDate: null,
539
- userInputTime: null
540
- };
541
- },
542
-
543
- computed: {
544
- year() {
545
- return this.date.getFullYear();
546
- },
547
-
548
- month() {
549
- return this.date.getMonth();
550
- },
551
-
552
- week() {
553
- return getWeekNumber(this.date);
554
- },
555
-
556
- monthDate() {
557
- return this.date.getDate();
558
- },
559
-
560
- footerVisible() {
561
- return this.showTime || this.selectionMode === 'dates' || this.selectionMode === 'months' || this.selectionMode === 'years';
562
- },
563
-
564
- visibleTime() {
565
- if (this.userInputTime !== null) {
566
- return this.userInputTime;
567
- } else {
568
- return formatDate(this.value || this.defaultValue, this.timeFormat);
569
- }
570
- },
571
-
572
- visibleDate() {
573
- if (this.userInputDate !== null) {
574
- return this.userInputDate;
575
- } else {
576
- return formatDate(this.value || this.defaultValue, this.dateFormat);
577
- }
578
- },
579
-
580
- yearLabel() {
581
- const yearTranslation = this.t('el.datepicker.year');
582
- if (this.currentView === 'year') {
583
- const startYear = Math.floor(this.year / 10) * 10;
584
- if (yearTranslation) {
585
- return startYear + ' ' + yearTranslation + ' - ' + (startYear + 9) + ' ' + yearTranslation;
586
- }
587
- return startYear + ' - ' + (startYear + 9);
588
- }
589
- return this.year + ' ' + yearTranslation;
590
- },
591
-
592
- timeFormat() {
593
- if (this.format) {
594
- return extractTimeFormat(this.format);
595
- } else {
596
- return 'HH:mm:ss';
597
- }
598
- },
599
-
600
- dateFormat() {
601
- if (this.format) {
602
- return extractDateFormat(this.format);
603
- } else {
604
- return 'yyyy-MM-dd';
605
- }
606
- }
607
- }
608
- };
609
- </script>
1
+ <template>
2
+ <transition name="el-zoom-in-top" @after-enter="handleEnter" @after-leave="handleLeave">
3
+ <div
4
+ v-show="visible"
5
+ class="el-picker-panel el-date-picker el-popper"
6
+ :class="[{
7
+ 'has-sidebar': $slots.sidebar || shortcuts,
8
+ 'has-time': showTime
9
+ }, popperClass]">
10
+ <div class="el-picker-panel__body-wrapper">
11
+ <slot name="sidebar" class="el-picker-panel__sidebar"></slot>
12
+ <div class="el-picker-panel__sidebar" v-if="shortcuts">
13
+ <button
14
+ type="button"
15
+ class="el-picker-panel__shortcut"
16
+ v-for="(shortcut, key) in shortcuts"
17
+ :key="key"
18
+ @click="handleShortcutClick(shortcut)">{{ shortcut.text }}</button>
19
+ </div>
20
+ <div class="el-picker-panel__body">
21
+ <div class="el-date-picker__time-header" v-if="showTime">
22
+ <span class="el-date-picker__editor-wrap">
23
+ <el-input
24
+ :placeholder="t('el.datepicker.selectDate')"
25
+ :value="visibleDate"
26
+ size="small"
27
+ @input="val => userInputDate = val"
28
+ @change="handleVisibleDateChange" />
29
+ </span>
30
+ <span class="el-date-picker__editor-wrap" v-clickoutside="handleTimePickClose">
31
+ <el-input
32
+ ref="input"
33
+ @focus="timePickerVisible = true"
34
+ :placeholder="t('el.datepicker.selectTime')"
35
+ :value="visibleTime"
36
+ size="small"
37
+ @input="val => userInputTime = val"
38
+ @change="handleVisibleTimeChange" />
39
+ <time-picker
40
+ ref="timepicker"
41
+ :time-arrow-control="arrowControl"
42
+ @pick="handleTimePick"
43
+ :visible="timePickerVisible"
44
+ @mounted="proxyTimePickerDataProperties">
45
+ </time-picker>
46
+ </span>
47
+ </div>
48
+ <div
49
+ class="el-date-picker__header"
50
+ :class="{ 'el-date-picker__header--bordered': currentView === 'year' || currentView === 'month' }"
51
+ v-show="currentView !== 'time'">
52
+ <button
53
+ type="button"
54
+ @click="prevYear"
55
+ :aria-label="t(`el.datepicker.prevYear`)"
56
+ class="el-picker-panel__icon-btn el-date-picker__prev-btn el-icon-d-arrow-left">
57
+ </button>
58
+ <button
59
+ type="button"
60
+ @click="prevMonth"
61
+ v-show="currentView === 'date'"
62
+ :aria-label="t(`el.datepicker.prevMonth`)"
63
+ class="el-picker-panel__icon-btn el-date-picker__prev-btn el-icon-arrow-left">
64
+ </button>
65
+ <span
66
+ @click="showYearPicker"
67
+ role="button"
68
+ class="el-date-picker__header-label">{{ yearLabel }}</span>
69
+ <span
70
+ @click="showMonthPicker"
71
+ v-show="currentView === 'date'"
72
+ role="button"
73
+ class="el-date-picker__header-label"
74
+ :class="{ active: currentView === 'month' }">{{t(`el.datepicker.month${ month + 1 }`)}}</span>
75
+ <button
76
+ type="button"
77
+ @click="nextYear"
78
+ :aria-label="t(`el.datepicker.nextYear`)"
79
+ class="el-picker-panel__icon-btn el-date-picker__next-btn el-icon-d-arrow-right">
80
+ </button>
81
+ <button
82
+ type="button"
83
+ @click="nextMonth"
84
+ v-show="currentView === 'date'"
85
+ :aria-label="t(`el.datepicker.nextMonth`)"
86
+ class="el-picker-panel__icon-btn el-date-picker__next-btn el-icon-arrow-right">
87
+ </button>
88
+ </div>
89
+
90
+ <div class="el-picker-panel__content">
91
+ <date-table
92
+ v-show="currentView === 'date'"
93
+ @pick="handleDatePick"
94
+ :selection-mode="selectionMode"
95
+ :first-day-of-week="firstDayOfWeek"
96
+ :value="value"
97
+ :default-value="defaultValue ? new Date(defaultValue) : null"
98
+ :date="date"
99
+ :cell-class-name="cellClassName"
100
+ :disabled-date="disabledDate">
101
+ </date-table>
102
+ <year-table
103
+ v-show="currentView === 'year'"
104
+ @pick="handleYearPick"
105
+ :selection-mode="selectionMode"
106
+ :value="value"
107
+ :default-value="defaultValue ? new Date(defaultValue) : null"
108
+ :date="date"
109
+ :disabled-date="disabledDate">
110
+ </year-table>
111
+ <month-table
112
+ v-show="currentView === 'month'"
113
+ @pick="handleMonthPick"
114
+ :selection-mode="selectionMode"
115
+ :value="value"
116
+ :default-value="defaultValue ? new Date(defaultValue) : null"
117
+ :date="date"
118
+ :disabled-date="disabledDate">
119
+ </month-table>
120
+ </div>
121
+ </div>
122
+ </div>
123
+
124
+ <div
125
+ class="el-picker-panel__footer"
126
+ v-show="footerVisible && (currentView === 'date' || currentView === 'month' || currentView === 'year')">
127
+ <el-button
128
+ size="mini"
129
+ type="text"
130
+ class="el-picker-panel__link-btn"
131
+ @click="changeToNow"
132
+ v-show="selectionMode !== 'dates' && selectionMode !== 'months' && selectionMode !== 'years'">
133
+ {{ t('el.datepicker.now') }}
134
+ </el-button>
135
+ <el-button
136
+ plain
137
+ size="mini"
138
+ class="el-picker-panel__link-btn"
139
+ @click="confirm">
140
+ {{ t('el.datepicker.confirm') }}
141
+ </el-button>
142
+ </div>
143
+ </div>
144
+ </transition>
145
+ </template>
146
+
147
+ <script type="text/babel">
148
+ import {
149
+ formatDate,
150
+ parseDate,
151
+ getWeekNumber,
152
+ isDate,
153
+ modifyDate,
154
+ modifyTime,
155
+ modifyWithTimeString,
156
+ clearMilliseconds,
157
+ clearTime,
158
+ prevYear,
159
+ nextYear,
160
+ prevMonth,
161
+ nextMonth,
162
+ changeYearMonthAndClampDate,
163
+ extractDateFormat,
164
+ extractTimeFormat,
165
+ timeWithinRange
166
+ } from 'iov-design/src/utils/date-util';
167
+ import Clickoutside from 'iov-design/src/utils/clickoutside';
168
+ import Locale from 'iov-design/src/mixins/locale';
169
+ import ElInput from 'iov-design/packages/input';
170
+ import ElButton from 'iov-design/packages/button';
171
+ import TimePicker from './time';
172
+ import YearTable from '../basic/year-table';
173
+ import MonthTable from '../basic/month-table';
174
+ import DateTable from '../basic/date-table';
175
+
176
+ export default {
177
+ mixins: [Locale],
178
+
179
+ directives: { Clickoutside },
180
+
181
+ watch: {
182
+ showTime(val) {
183
+ /* istanbul ignore if */
184
+ if (!val) return;
185
+ this.$nextTick(_ => {
186
+ const inputElm = this.$refs.input.$el;
187
+ if (inputElm) {
188
+ this.pickerWidth = inputElm.getBoundingClientRect().width + 10;
189
+ }
190
+ });
191
+ },
192
+
193
+ value(val) {
194
+ if (this.selectionMode === 'dates' && this.value) return;
195
+ if (this.selectionMode === 'months' && this.value) return;
196
+ if (this.selectionMode === 'years' && this.value) return;
197
+ if (isDate(val)) {
198
+ this.date = new Date(val);
199
+ } else {
200
+ this.date = this.getDefaultValue();
201
+ }
202
+ },
203
+
204
+ defaultValue(val) {
205
+ if (!isDate(this.value)) {
206
+ this.date = val ? new Date(val) : new Date();
207
+ }
208
+ },
209
+
210
+ timePickerVisible(val) {
211
+ if (val) this.$nextTick(() => this.$refs.timepicker.adjustSpinners());
212
+ },
213
+
214
+ selectionMode(newVal) {
215
+ if (newVal === 'month') {
216
+ /* istanbul ignore next */
217
+ if (this.currentView !== 'year' || this.currentView !== 'month') {
218
+ this.currentView = 'month';
219
+ }
220
+ } else if (newVal === 'dates') {
221
+ this.currentView = 'date';
222
+ } else if (newVal === 'years') {
223
+ this.currentView = 'year';
224
+ } else if (newVal === 'months') {
225
+ this.currentView = 'month';
226
+ }
227
+ }
228
+ },
229
+
230
+ methods: {
231
+ proxyTimePickerDataProperties() {
232
+ const format = timeFormat => {this.$refs.timepicker.format = timeFormat;};
233
+ const value = value => {this.$refs.timepicker.value = value;};
234
+ const date = date => {this.$refs.timepicker.date = date;};
235
+ const selectableRange = selectableRange => {this.$refs.timepicker.selectableRange = selectableRange;};
236
+
237
+ this.$watch('value', value);
238
+ this.$watch('date', date);
239
+ this.$watch('selectableRange', selectableRange);
240
+
241
+ format(this.timeFormat);
242
+ value(this.value);
243
+ date(this.date);
244
+ selectableRange(this.selectableRange);
245
+ },
246
+
247
+ handleClear() {
248
+ this.date = this.getDefaultValue();
249
+ this.$emit('pick', null);
250
+ },
251
+
252
+ emit(value, ...args) {
253
+ if (!value) {
254
+ this.$emit('pick', value, ...args);
255
+ } else if (Array.isArray(value)) {
256
+ const dates = value.map(date => this.showTime ? clearMilliseconds(date) : clearTime(date));
257
+ this.$emit('pick', dates, ...args);
258
+ } else {
259
+ this.$emit('pick', this.showTime ? clearMilliseconds(value) : clearTime(value), ...args);
260
+ }
261
+ this.userInputDate = null;
262
+ this.userInputTime = null;
263
+ },
264
+
265
+ // resetDate() {
266
+ // this.date = new Date(this.date);
267
+ // },
268
+
269
+ showMonthPicker() {
270
+ this.currentView = 'month';
271
+ },
272
+
273
+ showYearPicker() {
274
+ this.currentView = 'year';
275
+ },
276
+
277
+ // XXX: 没用到
278
+ // handleLabelClick() {
279
+ // if (this.currentView === 'date') {
280
+ // this.showMonthPicker();
281
+ // } else if (this.currentView === 'month') {
282
+ // this.showYearPicker();
283
+ // }
284
+ // },
285
+
286
+ prevMonth() {
287
+ this.date = prevMonth(this.date);
288
+ },
289
+
290
+ nextMonth() {
291
+ this.date = nextMonth(this.date);
292
+ },
293
+
294
+ prevYear() {
295
+ if (this.currentView === 'year') {
296
+ this.date = prevYear(this.date, 10);
297
+ } else {
298
+ this.date = prevYear(this.date);
299
+ }
300
+ },
301
+
302
+ nextYear() {
303
+ if (this.currentView === 'year') {
304
+ this.date = nextYear(this.date, 10);
305
+ } else {
306
+ this.date = nextYear(this.date);
307
+ }
308
+ },
309
+
310
+ handleShortcutClick(shortcut) {
311
+ if (shortcut.onClick) {
312
+ shortcut.onClick(this);
313
+ }
314
+ },
315
+
316
+ handleTimePick(value, visible, first) {
317
+ if (isDate(value)) {
318
+ const newDate = this.value
319
+ ? modifyTime(this.value, value.getHours(), value.getMinutes(), value.getSeconds())
320
+ : modifyWithTimeString(this.getDefaultValue(), this.defaultTime);
321
+ this.date = newDate;
322
+ this.emit(this.date, true);
323
+ } else {
324
+ this.emit(value, true);
325
+ }
326
+ if (!first) {
327
+ this.timePickerVisible = visible;
328
+ }
329
+ },
330
+
331
+ handleTimePickClose() {
332
+ this.timePickerVisible = false;
333
+ },
334
+
335
+ handleMonthPick(month) {
336
+ if (this.selectionMode === 'month') {
337
+ this.date = modifyDate(this.date, this.year, month, 1);
338
+ this.emit(this.date);
339
+ } else if (this.selectionMode === 'months') {
340
+ this.emit(month, true);
341
+ } else {
342
+ this.date = changeYearMonthAndClampDate(this.date, this.year, month);
343
+ // TODO: should emit intermediate value ??
344
+ // this.emit(this.date);
345
+ this.currentView = 'date';
346
+ }
347
+ },
348
+
349
+ handleDatePick(value) {
350
+ if (this.selectionMode === 'day') {
351
+ let newDate = this.value
352
+ ? modifyDate(this.value, value.getFullYear(), value.getMonth(), value.getDate())
353
+ : modifyWithTimeString(value, this.defaultTime);
354
+ // change default time while out of selectableRange
355
+ if (!this.checkDateWithinRange(newDate)) {
356
+ newDate = modifyDate(this.selectableRange[0][0], value.getFullYear(), value.getMonth(), value.getDate());
357
+ }
358
+ this.date = newDate;
359
+ this.emit(this.date, this.showTime);
360
+ } else if (this.selectionMode === 'week') {
361
+ this.emit(value.date);
362
+ } else if (this.selectionMode === 'dates') {
363
+ this.emit(value, true); // set false to keep panel open
364
+ }
365
+ },
366
+
367
+ handleYearPick(year) {
368
+ if (this.selectionMode === 'year') {
369
+ this.date = modifyDate(this.date, year, 0, 1);
370
+ this.emit(this.date);
371
+ } else if (this.selectionMode === 'years') {
372
+ this.emit(year, true);
373
+ } else {
374
+ this.date = changeYearMonthAndClampDate(this.date, year, this.month);
375
+ // TODO: should emit intermediate value ??
376
+ // this.emit(this.date, true);
377
+ this.currentView = 'month';
378
+ }
379
+ },
380
+
381
+ changeToNow() {
382
+ // NOTE: not a permanent solution
383
+ // consider disable "now" button in the future
384
+ if ((!this.disabledDate || !this.disabledDate(new Date())) && this.checkDateWithinRange(new Date())) {
385
+ this.date = new Date();
386
+ this.emit(this.date);
387
+ }
388
+ },
389
+
390
+ confirm() {
391
+ if (this.selectionMode === 'dates' || this.selectionMode === 'months' || this.selectionMode === 'years') {
392
+ this.emit(this.value);
393
+ } else {
394
+ // value were emitted in handle{Date,Time}Pick, nothing to update here
395
+ // deal with the scenario where: user opens the picker, then confirm without doing anything
396
+ const value = this.value
397
+ ? this.value
398
+ : modifyWithTimeString(this.getDefaultValue(), this.defaultTime);
399
+ this.date = new Date(value); // refresh date
400
+ this.emit(value);
401
+ }
402
+ },
403
+
404
+ resetView() {
405
+ if (this.selectionMode === 'month' || this.selectionMode === 'months') {
406
+ this.currentView = 'month';
407
+ } else if (this.selectionMode === 'year' || this.selectionMode === 'years') {
408
+ this.currentView = 'year';
409
+ } else {
410
+ this.currentView = 'date';
411
+ }
412
+ },
413
+
414
+ handleEnter() {
415
+ document.body.addEventListener('keydown', this.handleKeydown);
416
+ },
417
+
418
+ handleLeave() {
419
+ this.$emit('dodestroy');
420
+ document.body.removeEventListener('keydown', this.handleKeydown);
421
+ },
422
+
423
+ handleKeydown(event) {
424
+ const keyCode = event.keyCode;
425
+ const list = [38, 40, 37, 39];
426
+ if (this.visible && !this.timePickerVisible) {
427
+ if (list.indexOf(keyCode) !== -1) {
428
+ this.handleKeyControl(keyCode);
429
+ event.stopPropagation();
430
+ event.preventDefault();
431
+ }
432
+ if (keyCode === 13 && this.userInputDate === null && this.userInputTime === null) { // Enter
433
+ this.emit(this.date, false);
434
+ }
435
+ }
436
+ },
437
+
438
+ handleKeyControl(keyCode) {
439
+ const mapping = {
440
+ 'year': {
441
+ 38: -4, 40: 4, 37: -1, 39: 1, offset: (date, step) => date.setFullYear(date.getFullYear() + step)
442
+ },
443
+ 'month': {
444
+ 38: -4, 40: 4, 37: -1, 39: 1, offset: (date, step) => date.setMonth(date.getMonth() + step)
445
+ },
446
+ 'week': {
447
+ 38: -1, 40: 1, 37: -1, 39: 1, offset: (date, step) => date.setDate(date.getDate() + step * 7)
448
+ },
449
+ 'day': {
450
+ 38: -7, 40: 7, 37: -1, 39: 1, offset: (date, step) => date.setDate(date.getDate() + step)
451
+ }
452
+ };
453
+ const mode = this.selectionMode;
454
+ const year = 3.1536e10;
455
+ const now = this.date.getTime();
456
+ const newDate = new Date(this.date.getTime());
457
+ while (Math.abs(now - newDate.getTime()) <= year) {
458
+ const map = mapping[mode];
459
+ map.offset(newDate, map[keyCode]);
460
+ if (typeof this.disabledDate === 'function' && this.disabledDate(newDate)) {
461
+ continue;
462
+ }
463
+ this.date = newDate;
464
+ this.$emit('pick', newDate, true);
465
+ break;
466
+ }
467
+ },
468
+
469
+ handleVisibleTimeChange(value) {
470
+ const time = parseDate(value, this.timeFormat);
471
+ if (time && this.checkDateWithinRange(time)) {
472
+ this.date = modifyDate(time, this.year, this.month, this.monthDate);
473
+ this.userInputTime = null;
474
+ this.$refs.timepicker.value = this.date;
475
+ this.timePickerVisible = false;
476
+ this.emit(this.date, true);
477
+ }
478
+ },
479
+
480
+ handleVisibleDateChange(value) {
481
+ const date = parseDate(value, this.dateFormat);
482
+ if (date) {
483
+ if (typeof this.disabledDate === 'function' && this.disabledDate(date)) {
484
+ return;
485
+ }
486
+ this.date = modifyTime(date, this.date.getHours(), this.date.getMinutes(), this.date.getSeconds());
487
+ this.userInputDate = null;
488
+ this.resetView();
489
+ this.emit(this.date, true);
490
+ }
491
+ },
492
+
493
+ isValidValue(value) {
494
+ return value && !isNaN(value) && (
495
+ typeof this.disabledDate === 'function'
496
+ ? !this.disabledDate(value)
497
+ : true
498
+ ) && this.checkDateWithinRange(value);
499
+ },
500
+
501
+ getDefaultValue() {
502
+ // if default-value is set, return it
503
+ // otherwise, return now (the moment this method gets called)
504
+ return this.defaultValue ? new Date(this.defaultValue) : new Date();
505
+ },
506
+
507
+ checkDateWithinRange(date) {
508
+ return this.selectableRange.length > 0
509
+ ? timeWithinRange(date, this.selectableRange, this.format || 'HH:mm:ss')
510
+ : true;
511
+ }
512
+ },
513
+
514
+ components: {
515
+ TimePicker, YearTable, MonthTable, DateTable, ElInput, ElButton
516
+ },
517
+
518
+ data() {
519
+ return {
520
+ popperClass: '',
521
+ date: new Date(),
522
+ value: '',
523
+ defaultValue: null, // use getDefaultValue() for time computation
524
+ defaultTime: null,
525
+ showTime: false,
526
+ selectionMode: 'day',
527
+ shortcuts: '',
528
+ visible: false,
529
+ currentView: 'date',
530
+ disabledDate: '',
531
+ cellClassName: '',
532
+ selectableRange: [],
533
+ firstDayOfWeek: 7,
534
+ showWeekNumber: false,
535
+ timePickerVisible: false,
536
+ format: '',
537
+ arrowControl: false,
538
+ userInputDate: null,
539
+ userInputTime: null
540
+ };
541
+ },
542
+
543
+ computed: {
544
+ year() {
545
+ return this.date.getFullYear();
546
+ },
547
+
548
+ month() {
549
+ return this.date.getMonth();
550
+ },
551
+
552
+ week() {
553
+ return getWeekNumber(this.date);
554
+ },
555
+
556
+ monthDate() {
557
+ return this.date.getDate();
558
+ },
559
+
560
+ footerVisible() {
561
+ return this.showTime || this.selectionMode === 'dates' || this.selectionMode === 'months' || this.selectionMode === 'years';
562
+ },
563
+
564
+ visibleTime() {
565
+ if (this.userInputTime !== null) {
566
+ return this.userInputTime;
567
+ } else {
568
+ return formatDate(this.value || this.defaultValue, this.timeFormat);
569
+ }
570
+ },
571
+
572
+ visibleDate() {
573
+ if (this.userInputDate !== null) {
574
+ return this.userInputDate;
575
+ } else {
576
+ return formatDate(this.value || this.defaultValue, this.dateFormat);
577
+ }
578
+ },
579
+
580
+ yearLabel() {
581
+ const yearTranslation = this.t('el.datepicker.year');
582
+ if (this.currentView === 'year') {
583
+ const startYear = Math.floor(this.year / 10) * 10;
584
+ if (yearTranslation) {
585
+ return startYear + ' ' + yearTranslation + ' - ' + (startYear + 9) + ' ' + yearTranslation;
586
+ }
587
+ return startYear + ' - ' + (startYear + 9);
588
+ }
589
+ return this.year + ' ' + yearTranslation;
590
+ },
591
+
592
+ timeFormat() {
593
+ if (this.format) {
594
+ return extractTimeFormat(this.format);
595
+ } else {
596
+ return 'HH:mm:ss';
597
+ }
598
+ },
599
+
600
+ dateFormat() {
601
+ if (this.format) {
602
+ return extractDateFormat(this.format);
603
+ } else {
604
+ return 'yyyy-MM-dd';
605
+ }
606
+ }
607
+ }
608
+ };
609
+ </script>