123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471 |
- /**
- * 日期时间插件
- * varstion 1.0.5
- * by Houfeng
- * Houfeng@DCloud.io
- */
- (function($, document) {
- //创建 DOM
- $.dom = function(str) {
- if (typeof(str) !== 'string') {
- if ((str instanceof Array) || (str[0] && str.length)) {
- return [].slice.call(str);
- } else {
- return [str];
- }
- }
- if (!$.__create_dom_div__) {
- $.__create_dom_div__ = document.createElement('div');
- }
- $.__create_dom_div__.innerHTML = str;
- return [].slice.call($.__create_dom_div__.childNodes);
- };
- var domBuffer = '<div class="mui-dtpicker" data-type="datetime">\
- <div class="mui-dtpicker-header">\
- <button data-id="btn-cancel" class="mui-btn">取消</button>\
- <button data-id="btn-ok" class="mui-btn mui-btn-blue">确定</button>\
- </div>\
- <div class="mui-dtpicker-title"><h5 data-id="title-y">年</h5><h5 data-id="title-m">月</h5><h5 data-id="title-d">日</h5><h5 data-id="title-h">时</h5><h5 data-id="title-i">分</h5></div>\
- <div class="mui-dtpicker-body">\
- <div data-id="picker-y" class="mui-picker">\
- <div class="mui-picker-inner">\
- <div class="mui-pciker-rule mui-pciker-rule-ft"></div>\
- <ul class="mui-pciker-list">\
- </ul>\
- <div class="mui-pciker-rule mui-pciker-rule-bg"></div>\
- </div>\
- </div>\
- <div data-id="picker-m" class="mui-picker">\
- <div class="mui-picker-inner">\
- <div class="mui-pciker-rule mui-pciker-rule-ft"></div>\
- <ul class="mui-pciker-list">\
- </ul>\
- <div class="mui-pciker-rule mui-pciker-rule-bg"></div>\
- </div>\
- </div>\
- <div data-id="picker-d" class="mui-picker">\
- <div class="mui-picker-inner">\
- <div class="mui-pciker-rule mui-pciker-rule-ft"></div>\
- <ul class="mui-pciker-list">\
- </ul>\
- <div class="mui-pciker-rule mui-pciker-rule-bg"></div>\
- </div>\
- </div>\
- <div data-id="picker-h" class="mui-picker">\
- <div class="mui-picker-inner">\
- <div class="mui-pciker-rule mui-pciker-rule-ft"></div>\
- <ul class="mui-pciker-list">\
- </ul>\
- <div class="mui-pciker-rule mui-pciker-rule-bg"></div>\
- </div>\
- </div>\
- <div data-id="picker-i" class="mui-picker">\
- <div class="mui-picker-inner">\
- <div class="mui-pciker-rule mui-pciker-rule-ft"></div>\
- <ul class="mui-pciker-list">\
- </ul>\
- <div class="mui-pciker-rule mui-pciker-rule-bg"></div>\
- </div>\
- </div>\
- </div>\
- </div>';
- //plugin
- var DtPicker = $.DtPicker = $.Class.extend({
- init: function(options) {
- var self = this;
- var _picker = $.dom(domBuffer)[0];
- document.body.appendChild(_picker);
- $('[data-id*="picker"]', _picker).picker();
- var ui = self.ui = {
- picker: _picker,
- mask: $.createMask(),
- ok: $('[data-id="btn-ok"]', _picker)[0],
- cancel: $('[data-id="btn-cancel"]', _picker)[0],
- y: $('[data-id="picker-y"]', _picker)[0],
- m: $('[data-id="picker-m"]', _picker)[0],
- d: $('[data-id="picker-d"]', _picker)[0],
- h: $('[data-id="picker-h"]', _picker)[0],
- i: $('[data-id="picker-i"]', _picker)[0],
- labels: $('[data-id*="title-"]', _picker),
- };
- ui.cancel.addEventListener('tap', function() {
- self.hide();
- }, false);
- ui.ok.addEventListener('tap', function() {
- var rs = self.callback(self.getSelected());
- if (rs !== false) {
- self.hide();
- }
- }, false);
- ui.y.addEventListener('change', function(e) { //目前的change事件容易导致级联触发
- if (self.options.beginMonth || self.options.endMonth) {
- self._createMonth();
- } else {
- self._createDay();
- }
- }, false);
- ui.m.addEventListener('change', function(e) {
- self._createDay();
- }, false);
- ui.d.addEventListener('change', function(e) {
- if (self.options.beginMonth || self.options.endMonth) { //仅提供了beginDate时,触发day,hours,minutes的change
- self._createHours();
- }
- }, false);
- ui.h.addEventListener('change', function(e) {
- if (self.options.beginMonth || self.options.endMonth) {
- self._createMinutes();
- }
- }, false);
- ui.mask[0].addEventListener('tap', function() {
- self.hide();
- }, false);
- self._create(options);
- //防止滚动穿透
- self.ui.picker.addEventListener($.EVENT_START, function(event) {
- event.preventDefault();
- }, false);
- self.ui.picker.addEventListener($.EVENT_MOVE, function(event) {
- event.preventDefault();
- }, false);
- },
- getSelected: function() {
- var self = this;
- var ui = self.ui;
- var type = self.options.type;
- var selected = {
- type: type,
- y: ui.y.picker.getSelectedItem(),
- m: ui.m.picker.getSelectedItem(),
- d: ui.d.picker.getSelectedItem(),
- h: ui.h.picker.getSelectedItem(),
- i: ui.i.picker.getSelectedItem(),
- toString: function() {
- return this.value;
- }
- };
- switch (type) {
- case 'datetime':
- selected.value = selected.y.value + '-' + selected.m.value + '-' + selected.d.value + ' ' + selected.h.value + ':' + selected.i.value;
- selected.text = selected.y.text + '-' + selected.m.text + '-' + selected.d.text + ' ' + selected.h.text + ':' + selected.i.text;
- break;
- case 'date':
- selected.value = selected.y.value + '-' + selected.m.value + '-' + selected.d.value;
- selected.text = selected.y.text + '-' + selected.m.text + '-' + selected.d.text;
- break;
- case 'time':
- selected.value = selected.h.value + ':' + selected.i.value;
- selected.text = selected.h.text + ':' + selected.i.text;
- break;
- case 'month':
- selected.value = selected.y.value + '-' + selected.m.value;
- selected.text = selected.y.text + '-' + selected.m.text;
- break;
- case 'hour':
- selected.value = selected.y.value + '-' + selected.m.value + '-' + selected.d.value + ' ' + selected.h.value;
- selected.text = selected.y.text + '-' + selected.m.text + '-' + selected.d.text + ' ' + selected.h.text;
- break;
- }
- return selected;
- },
- setSelectedValue: function(value) {
- var self = this;
- var ui = self.ui;
- var parsedValue = self._parseValue(value);
- //TODO 嵌套过多,因为picker的change时间是异步(考虑到性能)的,所以为了保证change之后再setSelected,目前使用回调处理
- ui.y.picker.setSelectedValue(parsedValue.y, 0, function() {
- ui.m.picker.setSelectedValue(parsedValue.m, 0, function() {
- ui.d.picker.setSelectedValue(parsedValue.d, 0, function() {
- ui.h.picker.setSelectedValue(parsedValue.h, 0, function() {
- ui.i.picker.setSelectedValue(parsedValue.i, 0);
- });
- });
- });
- });
- },
- isLeapYear: function(year) {
- return (year % 4 == 0 && year % 100 != 0) || (year % 400 == 0);
- },
- _inArray: function(array, item) {
- for (var index in array) {
- var _item = array[index];
- if (_item === item) return true;
- }
- return false;
- },
- getDayNum: function(year, month) {
- var self = this;
- if (self._inArray([1, 3, 5, 7, 8, 10, 12], month)) {
- return 31;
- } else if (self._inArray([4, 6, 9, 11], month)) {
- return 30;
- } else if (self.isLeapYear(year)) {
- return 29;
- } else {
- return 28;
- }
- },
- _fill: function(num) {
- num = num.toString();
- if (num.length < 2) {
- num = 0 + num;
- }
- return num;
- },
- _isBeginYear: function() {
- return this.options.beginYear === parseInt(this.ui.y.picker.getSelectedValue());
- },
- _isBeginMonth: function() {
- return this.options.beginMonth && this._isBeginYear() && this.options.beginMonth === parseInt(this.ui.m.picker.getSelectedValue());
- },
- _isBeginDay: function() {
- return this._isBeginMonth() && this.options.beginDay === parseInt(this.ui.d.picker.getSelectedValue());
- },
- _isBeginHours: function() {
- return this._isBeginDay() && this.options.beginHours === parseInt(this.ui.h.picker.getSelectedValue());
- },
- _isEndYear: function() {
- return this.options.endYear === parseInt(this.ui.y.picker.getSelectedValue());
- },
- _isEndMonth: function() {
- return this.options.endMonth && this._isEndYear() && this.options.endMonth === parseInt(this.ui.m.picker.getSelectedValue());
- },
- _isEndDay: function() {
- return this._isEndMonth() && this.options.endDay === parseInt(this.ui.d.picker.getSelectedValue());
- },
- _isEndHours: function() {
- return this._isEndDay() && this.options.endHours === parseInt(this.ui.h.picker.getSelectedValue());
- },
- _createYear: function(current) {
- var self = this;
- var options = self.options;
- var ui = self.ui;
- //生成年列表
- var yArray = [];
- if (options.customData.y) {
- yArray = options.customData.y;
- } else {
- var yBegin = options.beginYear;
- var yEnd = options.endYear;
- for (var y = yBegin; y <= yEnd; y++) {
- yArray.push({
- text: y + '',
- value: y
- });
- }
- }
- ui.y.picker.setItems(yArray);
- //ui.y.picker.setSelectedValue(current);
- },
- _createMonth: function(current) {
- var self = this;
- var options = self.options;
- var ui = self.ui;
- //生成月列表
- var mArray = [];
- if (options.customData.m) {
- mArray = options.customData.m;
- } else {
- var m = options.beginMonth && self._isBeginYear() ? options.beginMonth : 1;
- var maxMonth = options.endMonth && self._isEndYear() ? options.endMonth : 12;
- for (; m <= maxMonth; m++) {
- var val = self._fill(m);
- mArray.push({
- text: val,
- value: val
- });
- }
- }
- ui.m.picker.setItems(mArray);
- //ui.m.picker.setSelectedValue(current);
- },
- _createDay: function(current) {
- var self = this;
- var options = self.options;
- var ui = self.ui;
- //生成日列表
- var dArray = [];
- if (options.customData.d) {
- dArray = options.customData.d;
- } else {
- var d = self._isBeginMonth() ? options.beginDay : 1;
- var maxDay = self._isEndMonth() ? options.endDay : self.getDayNum(parseInt(this.ui.y.picker.getSelectedValue()), parseInt(this.ui.m.picker.getSelectedValue()));
- for (; d <= maxDay; d++) {
- var val = self._fill(d);
- dArray.push({
- text: val,
- value: val
- });
- }
- }
- ui.d.picker.setItems(dArray);
- current = current || ui.d.picker.getSelectedValue();
- //ui.d.picker.setSelectedValue(current);
- },
- _createHours: function(current) {
- var self = this;
- var options = self.options;
- var ui = self.ui;
- //生成时列表
- var hArray = [];
- if (options.customData.h) {
- hArray = options.customData.h;
- } else {
- var h = self._isBeginDay() ? options.beginHours : 0;
- var maxHours = self._isEndDay() ? options.endHours : 23;
- for (; h <= maxHours; h++) {
- var val = self._fill(h);
- hArray.push({
- text: val,
- value: val
- });
- }
- }
- ui.h.picker.setItems(hArray);
- //ui.h.picker.setSelectedValue(current);
- },
- _createMinutes: function(current) {
- var self = this;
- var options = self.options;
- var ui = self.ui;
- //生成分列表
- var iArray = [];
- if (options.customData.i) {
- iArray = options.customData.i;
- } else {
- var i = self._isBeginHours() ? options.beginMinutes : 0;
- var maxMinutes = self._isEndHours() ? options.endMinutes : 59;
- for (; i <= maxMinutes; i++) {
- var val = self._fill(i);
- iArray.push({
- text: val,
- value: val
- });
- }
- }
- ui.i.picker.setItems(iArray);
- //ui.i.picker.setSelectedValue(current);
- },
- _setLabels: function() {
- var self = this;
- var options = self.options;
- var ui = self.ui;
- ui.labels.each(function(i, label) {
- label.innerText = options.labels[i];
- });
- },
- _setButtons: function() {
- var self = this;
- var options = self.options;
- var ui = self.ui;
- ui.cancel.innerText = options.buttons[0];
- ui.ok.innerText = options.buttons[1];
- },
- _parseValue: function(value) {
- var self = this;
- var rs = {};
- if (value) {
- var parts = value.replace(":", "-").replace(" ", "-").split("-");
- rs.y = parts[0];
- rs.m = parts[1];
- rs.d = parts[2];
- rs.h = parts[3];
- rs.i = parts[4];
- } else {
- var now = new Date();
- rs.y = now.getFullYear();
- rs.m = now.getMonth() + 1;
- rs.d = now.getDate();
- rs.h = now.getHours();
- rs.i = now.getMinutes();
- }
- return rs;
- },
- _create: function(options) {
- var self = this;
- options = options || {};
- options.labels = options.labels || ['年', '月', '日', '时', '分'];
- options.buttons = options.buttons || ['取消', '确定'];
- options.type = options.type || 'datetime';
- options.customData = options.customData || {};
- self.options = options;
- var now = new Date();
- var beginDate = options.beginDate;
- if (beginDate instanceof Date && !isNaN(beginDate.valueOf())) { //设定了开始日期
- options.beginYear = beginDate.getFullYear();
- options.beginMonth = beginDate.getMonth() + 1;
- options.beginDay = beginDate.getDate();
- options.beginHours = beginDate.getHours();
- options.beginMinutes = beginDate.getMinutes();
- }
- var endDate = options.endDate;
- if (endDate instanceof Date && !isNaN(endDate.valueOf())) { //设定了结束日期
- options.endYear = endDate.getFullYear();
- options.endMonth = endDate.getMonth() + 1;
- options.endDay = endDate.getDate();
- options.endHours = endDate.getHours();
- options.endMinutes = endDate.getMinutes();
- }
- options.beginYear = options.beginYear || (now.getFullYear() - 5);
- options.endYear = options.endYear || (now.getFullYear() + 5);
- var ui = self.ui;
- //设定label
- self._setLabels();
- self._setButtons();
- //设定类型
- ui.picker.setAttribute('data-type', options.type);
- //生成
- self._createYear();
- self._createMonth();
- self._createDay();
- self._createHours();
- self._createMinutes();
- //设定默认值
- self.setSelectedValue(options.value);
- },
- //显示
- show: function(callback) {
- var self = this;
- var ui = self.ui;
- self.callback = callback || $.noop;
- ui.mask.show();
- document.body.classList.add($.className('dtpicker-active-for-page'));
- ui.picker.classList.add($.className('active'));
- //处理物理返回键
- self.__back = $.back;
- $.back = function() {
- self.hide();
- };
- },
- hide: function() {
- var self = this;
- if (self.disposed) return;
- var ui = self.ui;
- ui.picker.classList.remove($.className('active'));
- ui.mask.close();
- document.body.classList.remove($.className('dtpicker-active-for-page'));
- //处理物理返回键
- $.back = self.__back;
- },
- dispose: function() {
- var self = this;
- self.hide();
- setTimeout(function() {
- self.ui.picker.parentNode.removeChild(self.ui.picker);
- for (var name in self) {
- self[name] = null;
- delete self[name];
- };
- self.disposed = true;
- }, 300);
- }
- });
- })(mui, document);
|