zoukankan      html  css  js  c++  java
  • sencha touch datepicker/datepickerfield(时间选择控件)扩展(废弃 仅参考)

    参考资料:
    https://market.sencha.com/extensions/datetimepicker

    上面的扩展在2.2有些问题,参考源码重新写了一个

    TimePicker:

      1 Ext.define('ux.TimePicker', {
      2     extend: 'Ext.picker.Picker',
      3     xtype: 'timePicker',
      4     alternateClassName: 'timePicker',
      5     requires: ['Ext.DateExtras'],
      6 
      7     /**
      8     * @event change
      9     * Fired when the value of this picker has changed and the done button is pressed.
     10     * @param {Ext.picker.Date} this This Picker
     11     * @param {Date} value The date value
     12     */
     13 
     14     config: {
     15         /**
     16         * @cfg {Number} yearFrom
     17         * 开始年份,如果他比yearTo大,则选择顺序颠倒
     18         * @accessor
     19         */
     20         yearFrom: 1980,
     21 
     22         /**
     23         * @cfg {Number} [yearTo=new Date().getFullYear()]
     24         * 结束年份
     25         * @accessor
     26         */
     27         yearTo: new Date().getFullYear(),
     28 
     29         /**
     30         * @cfg {String} monthText
     31         * 月显示值
     32         * @accessor
     33         */
     34         monthText: '月',
     35 
     36         /**
     37         * @cfg {String} dayText
     38         * 日显示值
     39         * @accessor
     40         */
     41         dayText: '日',
     42 
     43         /**
     44         * @cfg {String} yearText
     45         * 年显示值
     46         * @accessor
     47         */
     48         yearText: '年',
     49 
     50         /**
     51         * @cfg {String} hourText
     52         * 小时显示值
     53         */
     54         hourText: '时',
     55 
     56         /**
     57         * @cfg {String} minuteText
     58         * 分显示值
     59         */
     60         minuteText: '分',
     61 
     62         /**
     63         * @cfg {String} ampmText
     64         * 上午/下午显示值
     65         */
     66         ampmText: '上午/下午',
     67         /**
     68         * @cfg {Array} slotOrder
     69         * 默认的选项列表
     70         * @accessor
     71         */
     72         slotOrder: ['year', 'month', 'day', 'hour', 'minute'], //
     73 
     74         /**
     75         * @cfg {Int} 
     76         *分钟间隔
     77         * @accessor
     78         */
     79         minuteInterval: 1,
     80 
     81         /**
     82         * @cfg {Boolean} ampm
     83         *是否使用12小时制
     84         * @accessor
     85         */
     86         ampm: false,
     87         useTitles: true
     88     },
     89 
     90     platformConfig: [{
     91         theme: ['Windows'],
     92         doneButton: {
     93             iconCls: 'check2',
     94             ui: 'round',
     95             text: ''
     96         }
     97     }],
     98 
     99     initialize: function () {
    100         this.callParent();
    101 
    102         this.on({
    103             scope: this,
    104             delegate: '> slot',
    105             slotpick: this.onSlotPick
    106         });
    107 
    108         this.on({
    109             scope: this,
    110             show: this.onSlotPick
    111         });
    112     },
    113 
    114     setValue: function (value, animated) {
    115         if (Ext.isDate(value)) {
    116             var ampm = 'AM',
    117             currentHours = hour = value.getHours();
    118             if (this.getAmpm()) {
    119                 if (currentHours > 12) {
    120                     ampm = "PM";
    121                     hour -= 12;
    122                 } else if (currentHours == 12) {
    123                     ampm = "PM";
    124                 } else if (currentHours == 0) {
    125                     hour = 12;
    126                 }
    127             }
    128             value = {
    129                 day: value.getDate(),
    130                 month: value.getMonth() + 1,
    131                 year: value.getFullYear(),
    132                 hour: hour,
    133                 minute: value.getMinutes(),
    134                 ampm: ampm
    135             };
    136         }
    137 
    138         this.callParent([value, animated]);
    139         this.onSlotPick();
    140     },
    141     //获取值
    142     getValue: function (useDom) {
    143         var values = {},
    144             items = this.getItems().items,
    145             ln = items.length,
    146             daysInMonth, day, month, year, hour, minute, item, i;
    147 
    148         for (i = 0; i < ln; i++) {
    149             item = items[i];
    150             if (item instanceof Ext.picker.Slot) {
    151                 values[item.getName()] = item.getValue(useDom);
    152             }
    153         }
    154 
    155         //if all the slots return null, we should not return a date
    156         if (values.year === null && values.month === null && values.day === null && values.hour === null && values.minute === null) {
    157             return null;
    158         }
    159 
    160         year = Ext.isNumber(values.year) ? values.year : 1;
    161         month = Ext.isNumber(values.month) ? values.month : 1;
    162         day = Ext.isNumber(values.day) ? values.day : 1;
    163         hour = Ext.isNumber(values.hour) ? values.hour : 1;
    164         minute = Ext.isNumber(values.minute) ? values.minute : 1;
    165 
    166         if (month && year && month && day) {
    167             daysInMonth = this.getDaysInMonth(month, year);
    168         }
    169         day = (daysInMonth) ? Math.min(day, daysInMonth) : day;
    170         if (values.ampm && values.ampm == "PM" && hour < 12) {
    171             hour = hour + 12;
    172         }
    173         if (values.ampm && values.ampm == "AM" && hour == 12) {
    174             hour = 0;
    175         }
    176         return new Date(year, month - 1, day, hour, minute);
    177     },
    178 
    179     /**
    180     * Updates the yearFrom configuration
    181     */
    182     updateYearFrom: function () {
    183         if (this.initialized) {
    184             this.createSlots();
    185         }
    186     },
    187 
    188     /**
    189     * Updates the yearTo configuration
    190     */
    191     updateYearTo: function () {
    192         if (this.initialized) {
    193             this.createSlots();
    194         }
    195     },
    196 
    197     /**
    198     * Updates the monthText configuration
    199     */
    200     updateMonthText: function (newMonthText, oldMonthText) {
    201         var innerItems = this.getInnerItems,
    202             ln = innerItems.length,
    203             item, i;
    204 
    205         //loop through each of the current items and set the title on the correct slice
    206         if (this.initialized) {
    207             for (i = 0; i < ln; i++) {
    208                 item = innerItems[i];
    209 
    210                 if ((typeof item.title == "string" && item.title == oldMonthText) || (item.title.html == oldMonthText)) {
    211                     item.setTitle(newMonthText);
    212                 }
    213             }
    214         }
    215     },
    216 
    217     /**
    218     * Updates the {@link #dayText} configuration.
    219     */
    220     updateDayText: function (newDayText, oldDayText) {
    221         var innerItems = this.getInnerItems,
    222             ln = innerItems.length,
    223             item, i;
    224 
    225         //loop through each of the current items and set the title on the correct slice
    226         if (this.initialized) {
    227             for (i = 0; i < ln; i++) {
    228                 item = innerItems[i];
    229 
    230                 if ((typeof item.title == "string" && item.title == oldDayText) || (item.title.html == oldDayText)) {
    231                     item.setTitle(newDayText);
    232                 }
    233             }
    234         }
    235     },
    236 
    237     /**
    238     * Updates the yearText configuration
    239     */
    240     updateYearText: function (yearText) {
    241         var innerItems = this.getInnerItems,
    242             ln = innerItems.length,
    243             item, i;
    244 
    245         //loop through each of the current items and set the title on the correct slice
    246         if (this.initialized) {
    247             for (i = 0; i < ln; i++) {
    248                 item = innerItems[i];
    249 
    250                 if (item.title == this.yearText) {
    251                     item.setTitle(yearText);
    252                 }
    253             }
    254         }
    255     },
    256 
    257     // @private
    258     constructor: function () {
    259         this.callParent(arguments);
    260         this.createSlots();
    261     },
    262 
    263     /**
    264     * Generates all slots for all years specified by this component, and then sets them on the component
    265     * @private
    266     */
    267     createSlots: function () {
    268         var me = this,
    269             slotOrder = me.getSlotOrder(),
    270             yearsFrom = me.getYearFrom(),
    271             yearsTo = me.getYearTo(),
    272             years = [],
    273             days = [],
    274             months = [],
    275                hours = [],
    276             minutes = [],
    277             ampm = [],
    278             reverse = yearsFrom > yearsTo,
    279             ln, i, daysInMonth;
    280 
    281         if (!this.getAmpm()) {
    282             var index = slotOrder.indexOf('ampm')
    283             if (index >= 0) {
    284                 slotOrder.splice(index);
    285             }
    286         }
    287         //填充年列表
    288         while (yearsFrom) {
    289             years.push({
    290                 text: yearsFrom,
    291                 value: yearsFrom
    292             });
    293 
    294             if (yearsFrom === yearsTo) {
    295                 break;
    296             }
    297 
    298             if (reverse) {
    299                 yearsFrom--;
    300             } else {
    301                 yearsFrom++;
    302             }
    303         }
    304         //填充天列表
    305         daysInMonth = me.getDaysInMonth(1, new Date().getFullYear());
    306 
    307         for (i = 0; i < daysInMonth; i++) {
    308             days.push({
    309                 text: i + 1,
    310                 value: i + 1
    311             });
    312         }
    313         //填充月列表
    314         for (i = 0, ln = Ext.Date.monthNames.length; i < ln; i++) {
    315             months.push({
    316                 text: Ext.Date.monthNames[i],
    317                 value: i + 1
    318             });
    319         }
    320         //填充小时列表
    321         var hourLimit = (this.getAmpm()) ? 12 : 23
    322         var hourStart = (this.getAmpm()) ? 1 : 0
    323         for (i = hourStart; i <= hourLimit; i++) {
    324             hours.push({
    325                 text: this.pad2(i),
    326                 value: i
    327             });
    328         }
    329         //填充分钟列表
    330         for (i = 0; i < 60; i += this.getMinuteInterval()) {
    331             minutes.push({
    332                 text: this.pad2(i),
    333                 value: i
    334             });
    335         }
    336         //填充上午/下午
    337         ampm.push({
    338             text: '上午',
    339             value: 'AM'
    340         }, {
    341             text: '下午',
    342             value: 'PM'
    343         });
    344 
    345         var slots = [];
    346 
    347         slotOrder.forEach(function (item) {
    348             slots.push(me.createSlot(item, days, months, years, hours, minutes, ampm));
    349         });
    350 
    351         me.setSlots(slots);
    352     },
    353 
    354     /**
    355     * Returns a slot config for a specified date.
    356     * @private
    357     */
    358     createSlot: function (name, days, months, years, hours, minutes, ampm) {
    359         switch (name) {
    360             case 'year':
    361                 return {
    362                     name: 'year',
    363                     align: 'center',
    364                     data: years,
    365                     title: this.getYearText(),
    366                     flex: 3
    367                 };
    368             case 'month':
    369                 return {
    370                     name: name,
    371                     align: 'center',
    372                     data: months,
    373                     title: this.getMonthText(),
    374                     flex: 4
    375                 };
    376             case 'day':
    377                 return {
    378                     name: 'day',
    379                     align: 'center',
    380                     data: days,
    381                      '1px',
    382                     title: this.getDayText(),
    383                     flex: 2
    384                 };
    385             case 'hour':
    386                 return {
    387                     name: 'hour',
    388                     align: 'center',
    389                     data: hours,
    390                     title: this.getHourText(),
    391                     flex: 2
    392                 };
    393             case 'minute':
    394                 return {
    395                     name: 'minute',
    396                     align: 'center',
    397                     data: minutes,
    398                     title: this.getMinuteText(),
    399                     flex: 2
    400                 };
    401             case 'ampm':
    402                 return {
    403                     name: 'ampm',
    404                     align: 'center',
    405                     data: ampm,
    406                     title: this.getAmpmText(),
    407                     flex: 2
    408                 };
    409         }
    410     },
    411 
    412     onSlotPick: function () {
    413         var value = this.getValue(true),
    414             slot = this.getDaySlot(),
    415             year = value.getFullYear(),
    416             month = value.getMonth(),
    417             days = [],
    418             daysInMonth, i;
    419         if (!value || !Ext.isDate(value) || !slot) {
    420             return;
    421         }
    422 
    423         this.callParent(arguments);
    424 
    425         //get the new days of the month for this new date
    426         daysInMonth = this.getDaysInMonth(month + 1, year);
    427         for (i = 0; i < daysInMonth; i++) {
    428             days.push({
    429                 text: i + 1,
    430                 value: i + 1
    431             });
    432         }
    433         // We don't need to update the slot days unless it has changed
    434         if (slot.getStore().getCount() == days.length) {
    435             return;
    436         }
    437 
    438         slot.getStore().setData(days);
    439 
    440         // Now we have the correct amount of days for the day slot, lets update it
    441         var store = slot.getStore(),
    442             viewItems = slot.getViewItems(),
    443             valueField = slot.getValueField(),
    444             index, item;
    445 
    446         index = store.find(valueField, value.getDate());
    447         if (index == -1) {
    448             return;
    449         }
    450 
    451         item = Ext.get(viewItems[index]);
    452 
    453         slot.selectedIndex = index;
    454         slot.scrollToItem(item);
    455         slot.setValue(slot.getValue(true));
    456 
    457 
    458     },
    459 
    460     getDaySlot: function () {
    461         var innerItems = this.getInnerItems(),
    462             ln = innerItems.length,
    463             i, slot;
    464 
    465         if (this.daySlot) {
    466             return this.daySlot;
    467         }
    468 
    469         for (i = 0; i < ln; i++) {
    470             slot = innerItems[i];
    471             if (slot.isSlot && slot.getName() == "day") {
    472                 this.daySlot = slot;
    473                 return slot;
    474             }
    475         }
    476 
    477         return null;
    478     },
    479 
    480     // @private
    481     getDaysInMonth: function (month, year) {
    482         var daysInMonth = [31, 28, 31, 30, 31, 30, 31, 31, 30, 31, 30, 31];
    483         return month == 2 && this.isLeapYear(year) ? 29 : daysInMonth[month - 1];
    484     },
    485 
    486     // @private
    487     isLeapYear: function (year) {
    488         return !!((year & 3) === 0 && (year % 100 || (year % 400 === 0 && year)));
    489     },
    490 
    491     onDoneButtonTap: function () {
    492         var oldValue = this._value,
    493             newValue = this.getValue(true),
    494             testValue = newValue;
    495 
    496         if (Ext.isDate(newValue)) {
    497             testValue = newValue.toDateString();
    498         }
    499         if (Ext.isDate(oldValue)) {
    500             oldValue = oldValue.toDateString();
    501         }
    502         if (testValue != oldValue) {
    503             this.fireEvent('change', this, newValue);
    504         }
    505         this.hide();
    506         this.inputBlocker.unblockInputs();
    507     },
    508     pad2: function (number) {
    509         return (number < 10 ? '0' : '') + number;
    510     }
    511 });

    使用方法参考:datepicker控件

    timefield:

      1 Ext.define('ux.Timefield', {
      2     extend: 'Ext.field.Text',
      3     alternateClassName: 'timefield',
      4     xtype: 'timefield',
      5     requires: [
      6         'ux.TimePicker',
      7         'Ext.DateExtras'
      8     ],
      9 
     10     /**
     11     * @event change
     12     * Fires when a date is selected
     13     * @param {Ext.field.DatePicker} this
     14     * @param {Date} newDate The new date
     15     * @param {Date} oldDate The old date
     16     */
     17 
     18     config: {
     19         ui: 'select',
     20 
     21         /**
     22         * @cfg {Object/ux.TimePicker} picker
     23         * An object that is used when creating the internal {@link ux.TimePicker} component or a direct instance of {@link ux.TimePicker}.
     24         * @accessor
     25         */
     26         picker: true,
     27 
     28         /**
     29         * @cfg {Boolean}
     30         * @hide
     31         * @accessor
     32         */
     33         clearIcon: false,
     34 
     35         /**
     36         * @cfg {Object/Date} value
     37         * Default value for the field and the internal {@link ux.TimePicker} component. Accepts an object of 'year',
     38         * 'month' and 'day' values, all of which should be numbers, or a {@link Date}.
     39         *
     40         * Example: {year: 1989, day: 1, month: 5} = 1st May 1989 or new Date()
     41         * @accessor
     42         */
     43 
     44 
     45         /**
     46         * @cfg {Boolean} destroyPickerOnHide
     47         * 完成选择时隐藏或者销毁该控件,默认销毁
     48         * @accessor
     49         */
     50         destroyPickerOnHide: false,
     51 
     52         /**
     53         * @cfg {String} dateFormat 默认时间格式
     54         * 接受任何有效的时间格式. 请参考 {@link Ext.Date}.
     55         */
     56         dateFormat: 'Y-m-d H:i',
     57 
     58         /**
     59         * @cfg {Object}
     60         * @hide
    
     61         */
     62         component: {
     63             useMask: true
     64         }
     65     },
     66 
     67     initialize: function () {
     68         var me = this,
     69             component = me.getComponent();
     70 
     71         me.callParent();
     72 
     73         component.on({
     74             scope: me,
     75             masktap: 'onMaskTap'
     76         });
     77 
     78 
     79         component.doMaskTap = Ext.emptyFn;
     80 
     81         if (Ext.browser.is.AndroidStock2) {
     82             component.input.dom.disabled = true;
     83         }
     84     },
     85 
     86     syncEmptyCls: Ext.emptyFn,
     87 
     88     applyValue: function (value) {
     89         if (!Ext.isDate(value) && !Ext.isObject(value)) {
     90             return null;
     91         }
     92 
     93         if (Ext.isObject(value)) {
     94             return new Date(value.year, value.month - 1, value.day, value.hour, value.minute);
     95         }
     96 
     97         return value;
     98     },
     99 
    100     updateValue: function (newValue, oldValue) {
    101         var me = this,
    102             picker = me._picker;
    103 
    104         if (picker && picker.isPicker) {
    105             picker.setValue(newValue);
    106         }
    107 
    108         // Ext.Date.format expects a Date
    109         if (newValue !== null) {
    110             me.getComponent().setValue(Ext.Date.format(newValue, me.getDateFormat() || Ext.util.Format.defaultDateFormat));
    111         } else {
    112             me.getComponent().setValue('');
    113         }
    114 
    115         if (newValue !== oldValue) {
    116             me.fireEvent('change', me, newValue, oldValue);
    117         }
    118     },
    119 
    120     /**
    121     * Updates the date format in the field.
    122     * @private
    123     */
    124     updateDateFormat: function (newDateFormat, oldDateFormat) {
    125         var value = this.getValue();
    126         if (newDateFormat != oldDateFormat && Ext.isDate(value)) {
    127             this.getComponent().setValue(Ext.Date.format(value, newDateFormat || Ext.util.Format.defaultDateFormat));
    128         }
    129     },
    130 
    131     /**
    132     * Returns the {@link Date} value of this field.
    133     * If you wanted a formated date
    134     * @return {Date} The date selected
    135     */
    136     getValue: function () {
    137         if (this._picker && this._picker instanceof ux.TimePicker) {
    138             return this._picker.getValue();
    139         }
    140         return this._value;
    141     },
    142 
    143     /**
    144     * Returns the value of the field formatted using the specified format. If it is not specified, it will default to
    145     * {@link #dateFormat} and then {@link Ext.util.Format#defaultDateFormat}.
    146     * @param {String} format The format to be returned.
    147     * @return {String} The formatted date.
    148     */
    149     getFormattedValue: function (format) {
    150         var value = this.getValue();
    151         return (Ext.isDate(value)) ? Ext.Date.format(value, format || this.getDateFormat() || Ext.util.Format.defaultDateFormat) : value;
    152     },
    153 
    154     applyPicker: function (picker, pickerInstance) {
    155         if (pickerInstance && pickerInstance.isPicker) {
    156             picker = pickerInstance.setConfig(picker);
    157         }
    158 
    159         return picker;
    160     },
    161 
    162     getPicker: function () {
    163         var picker = this._picker,
    164             value = this.getValue();
    165 
    166         if (picker && !picker.isPicker) {
    167             picker = Ext.factory(picker, ux.TimePicker);
    168             if (value != null) {
    169                 picker.setValue(value);
    170             }
    171         }
    172 
    173         picker.on({
    174             scope: this,
    175             change: 'onPickerChange',
    176             hide: 'onPickerHide'
    177         });
    178 
    179         this._picker = picker;
    180 
    181         return picker;
    182     },
    183 
    184     /**
    185     * @private
    186     * Listener to the tap event of the mask element. Shows the internal DatePicker component when the button has been tapped.
    187     */
    188     onMaskTap: function () {
    189         if (this.getDisabled()) {
    190             return false;
    191         }
    192 
    193         this.onFocus();
    194 
    195         return false;
    196     },
    197 
    198     /**
    199     * Called when the picker changes its value.
    200     * @param {ux.TimePicker} picker The date picker.
    201     * @param {Object} value The new value from the date picker.
    202     * @private
    203     */
    204     onPickerChange: function (picker, value) {
    205         var me = this,
    206             oldValue = me.getValue();
    207 
    208         me.setValue(value);
    209         me.fireEvent('select', me, value);
    210         me.onChange(me, value, oldValue);
    211     },
    212 
    213     /**
    214     * Override this or change event will be fired twice. change event is fired in updateValue
    215     * for this field. TOUCH-2861
    216     */
    217     onChange: Ext.emptyFn,
    218 
    219     /**
    220     * Destroys the picker when it is hidden, if
    221     * {@link Ext.field.DatePicker#destroyPickerOnHide destroyPickerOnHide} is set to `true`.
    222     * @private
    223     */
    224     onPickerHide: function () {
    225         var me = this,
    226             picker = me.getPicker();
    227 
    228         if (me.getDestroyPickerOnHide() && picker) {
    229             picker.destroy();
    230             me._picker = me.getInitialConfig().picker || true;
    231         }
    232     },
    233 
    234     reset: function () {
    235         this.setValue(this.originalValue);
    236     },
    237 
    238     onFocus: function (e) {
    239         var component = this.getComponent();
    240         this.fireEvent('focus', this, e);
    241 
    242         if (Ext.os.is.Android4) {
    243             component.input.dom.focus();
    244         }
    245         component.input.dom.blur();
    246 
    247         if (this.getReadOnly()) {
    248             return false;
    249         }
    250 
    251         this.isFocused = true;
    252 
    253         this.getPicker().show();
    254     },
    255 
    256     // @private
    257     destroy: function () {
    258         var picker = this._picker;
    259 
    260         if (picker && picker.isPicker) {
    261             picker.destroy();
    262         }
    263 
    264         this.callParent(arguments);
    265     }
    266     //<deprecated product=touch since=2.0>
    267 }, function () {
    268     this.override({
    269         getValue: function (format) {
    270             if (format) {
    271                 //<debug warn>
    272                 Ext.Logger.deprecate("format argument of the getValue method is deprecated, please use getFormattedValue instead", this);
    273                 //</debug>
    274                 return this.getFormattedValue(format);
    275             }
    276             return this.callOverridden();
    277         }
    278     });
    279 
    280     /**
    281     * @method getDatePicker
    282     * @inheritdoc Ext.field.DatePicker#getPicker
    283     * @deprecated 2.0.0 Please use #getPicker instead
    284     */
    285     Ext.deprecateMethod(this, 'getDatePicker', 'getPicker');
    286     //</deprecated>
    287 });

    使用方法参考datepickerfield控件

    效果图:

    只需要时、分选项如下:

    1             xtype: 'timefield',
    2             picker: {
    3                 slotOrder: ['hour', 'minute']
    4             },
    5             dateFormat: 'H:i',
    6             value: new Date()

    一些额外的css:

     1 /*#region pick */
     2 
     3 .x-picker-slot-title {
     4     height:auto;
     5     text-align:center;
     6     padding:0.2em 0;
     7 }
     8 .x-picker-slot .x-dataview-item {
     9     padding:0 8px !important;
    10 }
    11 .x-webkit .x-layout-box.x-horizontal > .x-layout-box-item.x-picker-slot {
    12     width:auto !important;
    13 }
    14 /*#endregion*/
  • 相关阅读:
    如何把textfield或者textview中长按出现的(全选,复制,粘贴)显示成中文
    免费真机调试 -- Xcode7
    Android性能测试工具 Emmagee
    iOS 开发 入门:使用Ad Hoc 进行用户测试
    栈与队列的区别
    iOS中ASI和AFN的区别
    iOS开发之监测网络状态
    xcode设置项目图标玻璃镜效果
    isEqual,isEqualTostring,==三者的区别
    iphone匹配邮箱的正则表达式
  • 原文地址:https://www.cnblogs.com/mlzs/p/3324677.html
Copyright © 2011-2022 走看看