import * as moment from 'moment'; import omit from 'omit.js'; import VcTimePicker from '../vc-time-picker'; import LocaleReceiver from '../locale-provider/LocaleReceiver'; import defaultLocale from './locale/en_US'; import BaseMixin from '../_util/BaseMixin'; import PropTypes from '../_util/vue-types'; import warning from '../_util/warning'; import Icon from '../icon'; import interopDefault from '../_util/interopDefault'; import { initDefaultProps, hasProp, getOptionProps, getComponentFromProp, isValidElement, getListeners, } from '../_util/props-util'; import { cloneElement } from '../_util/vnode'; import { ConfigConsumerProps } from '../config-provider'; import Base from '../base'; export function generateShowHourMinuteSecond(format) { // Ref: http://momentjs.com/docs/#/parsing/string-format/ return { showHour: format.indexOf('H') > -1 || format.indexOf('h') > -1 || format.indexOf('k') > -1, showMinute: format.indexOf('m') > -1, showSecond: format.indexOf('s') > -1, }; } function isMoment(value) { if (Array.isArray(value)) { return ( value.length === 0 || value.findIndex(val => val === undefined || moment.isMoment(val)) !== -1 ); } else { return value === undefined || moment.isMoment(value); } } const MomentType = PropTypes.custom(isMoment); export const TimePickerProps = () => ({ size: PropTypes.oneOf(['large', 'default', 'small']), value: MomentType, defaultValue: MomentType, open: PropTypes.bool, format: PropTypes.string, disabled: PropTypes.bool, placeholder: PropTypes.string, prefixCls: PropTypes.string, hideDisabledOptions: PropTypes.bool, disabledHours: PropTypes.func, disabledMinutes: PropTypes.func, disabledSeconds: PropTypes.func, getPopupContainer: PropTypes.func, use12Hours: PropTypes.bool, focusOnOpen: PropTypes.bool, hourStep: PropTypes.number, minuteStep: PropTypes.number, secondStep: PropTypes.number, allowEmpty: PropTypes.bool, allowClear: PropTypes.bool, inputReadOnly: PropTypes.bool, clearText: PropTypes.string, defaultOpenValue: PropTypes.object, popupClassName: PropTypes.string, popupStyle: PropTypes.object, suffixIcon: PropTypes.any, align: PropTypes.object, placement: PropTypes.any, transitionName: PropTypes.string, autoFocus: PropTypes.bool, addon: PropTypes.any, }); const TimePicker = { name: 'ATimePicker', mixins: [BaseMixin], props: initDefaultProps(TimePickerProps(), { align: { offset: [0, -2], }, disabled: false, disabledHours: undefined, disabledMinutes: undefined, disabledSeconds: undefined, hideDisabledOptions: false, placement: 'bottomLeft', transitionName: 'slide-up', focusOnOpen: true, }), model: { prop: 'value', event: 'change', }, provide() { return { savePopupRef: this.savePopupRef, }; }, inject: { configProvider: { default: () => ConfigConsumerProps }, }, data() { const value = this.value || this.defaultValue; if (value && !interopDefault(moment).isMoment(value)) { throw new Error('The value/defaultValue of TimePicker must be a moment object, '); } warning( !hasProp(this, 'allowEmpty'), '`allowEmpty` in TimePicker is deprecated. Please use `allowClear` instead.', ); return { sValue: value, }; }, watch: { value(val) { this.setState({ sValue: val }); }, }, methods: { savePopupRef(ref) { this.popupRef = ref; }, handleChange(value) { if (!hasProp(this, 'value')) { this.setState({ sValue: value }); } const { format = 'HH:mm:ss' } = this; this.$emit('change', value, (value && value.format(format)) || ''); }, handleOpenClose({ open }) { this.$emit('openChange', open); this.$emit('update:open', open); }, focus() { this.$refs.timePicker.focus(); }, blur() { this.$refs.timePicker.blur(); }, getDefaultFormat() { const { format, use12Hours } = this; if (format) { return format; } else if (use12Hours) { return 'h:mm:ss a'; } return 'HH:mm:ss'; }, getAllowClear() { const { allowClear, allowEmpty } = this.$props; if (hasProp(this, 'allowClear')) { return allowClear; } return allowEmpty; }, renderInputIcon(prefixCls) { let suffixIcon = getComponentFromProp(this, 'suffixIcon'); suffixIcon = Array.isArray(suffixIcon) ? suffixIcon[0] : suffixIcon; const clockIcon = (suffixIcon && (isValidElement(suffixIcon) ? ( cloneElement(suffixIcon, { class: `${prefixCls}-clock-icon`, }) ) : ( {suffixIcon} ))) || ; return {clockIcon}; }, renderClearIcon(prefixCls) { const clearIcon = ; return clearIcon; }, renderTimePicker(locale) { let props = getOptionProps(this); props = omit(props, ['defaultValue', 'suffixIcon', 'allowEmpty', 'allowClear']); const { prefixCls: customizePrefixCls, getPopupContainer, placeholder, size } = props; const getPrefixCls = this.configProvider.getPrefixCls; const prefixCls = getPrefixCls('time-picker', customizePrefixCls); const format = this.getDefaultFormat(); const pickerClassName = { [`${prefixCls}-${size}`]: !!size, }; const tempAddon = getComponentFromProp(this, 'addon', {}, false); const pickerAddon = panel => { return tempAddon ? (
{typeof tempAddon === 'function' ? tempAddon(panel) : tempAddon}
) : null; }; const inputIcon = this.renderInputIcon(prefixCls); const clearIcon = this.renderClearIcon(prefixCls); const { getPopupContainer: getContextPopupContainer } = this.configProvider; const timeProps = { props: { ...generateShowHourMinuteSecond(format), ...props, allowEmpty: this.getAllowClear(), prefixCls, getPopupContainer: getPopupContainer || getContextPopupContainer, format, value: this.sValue, placeholder: placeholder === undefined ? locale.placeholder : placeholder, addon: pickerAddon, inputIcon, clearIcon, }, class: pickerClassName, ref: 'timePicker', on: { ...getListeners(this), change: this.handleChange, open: this.handleOpenClose, close: this.handleOpenClose, }, }; return ; }, }, render() { return ( ); }, }; /* istanbul ignore next */ TimePicker.install = function(Vue) { Vue.use(Base); Vue.component(TimePicker.name, TimePicker); }; export default TimePicker;