123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866 |
- /**
-
- @Name : layDate 5.0.9 日期时间控件
- @Author: 贤心
- @Site:http://www.layui.com/laydate/
- @License:MIT
-
- */
- ;!function(){
- "use strict";
- var isLayui = window.layui && layui.define, ready = {
- getPath: function(){
- var jsPath = document.currentScript ? document.currentScript.src : function(){
- var js = document.scripts
- ,last = js.length - 1
- ,src;
- for(var i = last; i > 0; i--){
- if(js[i].readyState === 'interactive'){
- src = js[i].src;
- break;
- }
- }
- return src || js[last].src;
- }();
- return jsPath.substring(0, jsPath.lastIndexOf('/') + 1);
- }()
-
- //获取节点的style属性值
- ,getStyle: function(node, name){
- var style = node.currentStyle ? node.currentStyle : window.getComputedStyle(node, null);
- return style[style.getPropertyValue ? 'getPropertyValue' : 'getAttribute'](name);
- }
-
- //载入CSS配件
- ,link: function(href, fn, cssname){
-
- //未设置路径,则不主动加载css
- if(!laydate.path) return;
-
- var head = document.getElementsByTagName("head")[0], link = document.createElement('link');
- if(typeof fn === 'string') cssname = fn;
- var app = (cssname || href).replace(/\.|\//g, '');
- var id = 'layuicss-'+ app, timeout = 0;
-
- link.rel = 'stylesheet';
- link.href = laydate.path + href;
- link.id = id;
-
- if(!document.getElementById(id)){
- head.appendChild(link);
- }
-
- if(typeof fn !== 'function') return;
-
- //轮询css是否加载完毕
- (function poll() {
- if(++timeout > 8 * 1000 / 100){
- return window.console && console.error('laydate.css: Invalid');
- };
- parseInt(ready.getStyle(document.getElementById(id), 'width')) === 1989 ? fn() : setTimeout(poll, 100);
- }());
- }
- }
- ,laydate = {
- v: '5.0.9'
- ,config: {} //全局配置项
- ,index: (window.laydate && window.laydate.v) ? 100000 : 0
- ,path: ready.getPath
-
- //设置全局项
- ,set: function(options){
- var that = this;
- that.config = lay.extend({}, that.config, options);
- return that;
- }
-
- //主体CSS等待事件
- ,ready: function(fn){
- var cssname = 'laydate', ver = ''
- ,path = (isLayui ? 'modules/laydate/' : 'theme/') + 'default/laydate.css?v='+ laydate.v + ver;
- isLayui ? layui.addcss(path, fn, cssname) : ready.link(path, fn, cssname);
- return this;
- }
- }
-
- //操作当前实例
- ,thisDate = function(){
- var that = this;
- return {
- //提示框
- hint: function(content){
- that.hint.call(that, content);
- }
- ,config: that.config
- };
- }
- //字符常量
- ,MOD_NAME = 'laydate', ELEM = '.layui-laydate', THIS = 'layui-this', SHOW = 'layui-show', HIDE = 'layui-hide', DISABLED = 'laydate-disabled', TIPS_OUT = '开始日期超出了结束日期<br>建议重新选择', LIMIT_YEAR = [100, 200000]
-
- ,ELEM_STATIC = 'layui-laydate-static', ELEM_LIST = 'layui-laydate-list', ELEM_SELECTED = 'laydate-selected', ELEM_HINT = 'layui-laydate-hint', ELEM_PREV = 'laydate-day-prev', ELEM_NEXT = 'laydate-day-next', ELEM_FOOTER = 'layui-laydate-footer', ELEM_CONFIRM = '.laydate-btns-confirm', ELEM_TIME_TEXT = 'laydate-time-text', ELEM_TIME_BTN = '.laydate-btns-time'
-
- //组件构造器
- ,Class = function(options){
- var that = this;
- that.index = ++laydate.index;
- that.config = lay.extend({}, that.config, laydate.config, options);
- laydate.ready(function(){
- that.init();
- });
- }
-
- //DOM查找
- ,lay = function(selector){
- return new LAY(selector);
- }
-
- //DOM构造器
- ,LAY = function(selector){
- var index = 0
- ,nativeDOM = typeof selector === 'object' ? [selector] : (
- this.selector = selector
- ,document.querySelectorAll(selector || null)
- );
- for(; index < nativeDOM.length; index++){
- this.push(nativeDOM[index]);
- }
- };
-
-
- /*
- lay对象操作
- */
-
- LAY.prototype = [];
- LAY.prototype.constructor = LAY;
-
- //普通对象深度扩展
- lay.extend = function(){
- var ai = 1, args = arguments
- ,clone = function(target, obj){
- target = target || (obj.constructor === Array ? [] : {});
- for(var i in obj){
- //如果值为对象,则进入递归,继续深度合并
- target[i] = (obj[i] && (obj[i].constructor === Object))
- ? clone(target[i], obj[i])
- : obj[i];
- }
- return target;
- }
- args[0] = typeof args[0] === 'object' ? args[0] : {};
- for(; ai < args.length; ai++){
- if(typeof args[ai] === 'object'){
- clone(args[0], args[ai])
- }
- }
- return args[0];
- };
-
- //ie版本
- lay.ie = function(){
- var agent = navigator.userAgent.toLowerCase();
- return (!!window.ActiveXObject || "ActiveXObject" in window) ? (
- (agent.match(/msie\s(\d+)/) || [])[1] || '11' //由于ie11并没有msie的标识
- ) : false;
- }();
-
- //中止冒泡
- lay.stope = function(e){
- e = e || window.event;
- e.stopPropagation
- ? e.stopPropagation()
- : e.cancelBubble = true;
- };
-
- //对象遍历
- lay.each = function(obj, fn){
- var key
- ,that = this;
- if(typeof fn !== 'function') return that;
- obj = obj || [];
- if(obj.constructor === Object){
- for(key in obj){
- if(fn.call(obj[key], key, obj[key])) break;
- }
- } else {
- for(key = 0; key < obj.length; key++){
- if(fn.call(obj[key], key, obj[key])) break;
- }
- }
- return that;
- };
-
- //数字前置补零
- lay.digit = function(num, length, end){
- var str = '';
- num = String(num);
- length = length || 2;
- for(var i = num.length; i < length; i++){
- str += '0';
- }
- return num < Math.pow(10, length) ? str + (num|0) : num;
- };
-
- //创建元素
- lay.elem = function(elemName, attr){
- var elem = document.createElement(elemName);
- lay.each(attr || {}, function(key, value){
- elem.setAttribute(key, value);
- });
- return elem;
- };
-
- //追加字符
- LAY.addStr = function(str, new_str){
- str = str.replace(/\s+/, ' ');
- new_str = new_str.replace(/\s+/, ' ').split(' ');
- lay.each(new_str, function(ii, item){
- if(!new RegExp('\\b'+ item + '\\b').test(str)){
- str = str + ' ' + item;
- }
- });
- return str.replace(/^\s|\s$/, '');
- };
-
- //移除值
- LAY.removeStr = function(str, new_str){
- str = str.replace(/\s+/, ' ');
- new_str = new_str.replace(/\s+/, ' ').split(' ');
- lay.each(new_str, function(ii, item){
- var exp = new RegExp('\\b'+ item + '\\b')
- if(exp.test(str)){
- str = str.replace(exp, '');
- }
- });
- return str.replace(/\s+/, ' ').replace(/^\s|\s$/, '');
- };
-
- //查找子元素
- LAY.prototype.find = function(selector){
- var that = this;
- var index = 0, arr = []
- ,isObject = typeof selector === 'object';
-
- this.each(function(i, item){
- var nativeDOM = isObject ? [selector] : item.querySelectorAll(selector || null);
- for(; index < nativeDOM.length; index++){
- arr.push(nativeDOM[index]);
- }
- that.shift();
- });
-
- if(!isObject){
- that.selector = (that.selector ? that.selector + ' ' : '') + selector
- }
-
- lay.each(arr, function(i, item){
- that.push(item);
- });
-
- return that;
- };
-
- //DOM遍历
- LAY.prototype.each = function(fn){
- return lay.each.call(this, this, fn);
- };
-
- //添加css类
- LAY.prototype.addClass = function(className, type){
- return this.each(function(index, item){
- item.className = LAY[type ? 'removeStr' : 'addStr'](item.className, className)
- });
- };
-
- //移除css类
- LAY.prototype.removeClass = function(className){
- return this.addClass(className, true);
- };
-
- //是否包含css类
- LAY.prototype.hasClass = function(className){
- var has = false;
- this.each(function(index, item){
- if(new RegExp('\\b'+ className +'\\b').test(item.className)){
- has = true;
- }
- });
- return has;
- };
-
- //添加或获取属性
- LAY.prototype.attr = function(key, value){
- var that = this;
- return value === undefined ? function(){
- if(that.length > 0) return that[0].getAttribute(key);
- }() : that.each(function(index, item){
- item.setAttribute(key, value);
- });
- };
-
- //移除属性
- LAY.prototype.removeAttr = function(key){
- return this.each(function(index, item){
- item.removeAttribute(key);
- });
- };
-
- //设置HTML内容
- LAY.prototype.html = function(html){
- return this.each(function(index, item){
- item.innerHTML = html;
- });
- };
-
- //设置值
- LAY.prototype.val = function(value){
- return this.each(function(index, item){
- item.value = value;
- });
- };
-
- //追加内容
- LAY.prototype.append = function(elem){
- return this.each(function(index, item){
- typeof elem === 'object'
- ? item.appendChild(elem)
- : item.innerHTML = item.innerHTML + elem;
- });
- };
-
- //移除内容
- LAY.prototype.remove = function(elem){
- return this.each(function(index, item){
- elem ? item.removeChild(elem) : item.parentNode.removeChild(item);
- });
- };
-
- //事件绑定
- LAY.prototype.on = function(eventName, fn){
- return this.each(function(index, item){
- item.attachEvent ? item.attachEvent('on' + eventName, function(e){
- e.target = e.srcElement;
- fn.call(item, e);
- }) : item.addEventListener(eventName, fn, false);
- });
- };
-
- //解除事件
- LAY.prototype.off = function(eventName, fn){
- return this.each(function(index, item){
- item.detachEvent
- ? item.detachEvent('on'+ eventName, fn)
- : item.removeEventListener(eventName, fn, false);
- });
- };
-
-
- /*
- 组件操作
- */
-
-
- //是否闰年
- Class.isLeapYear = function(year){
- return (year % 4 === 0 && year % 100 !== 0) || year % 400 === 0;
- };
-
- //默认配置
- Class.prototype.config = {
- type: 'date' //控件类型,支持:year/month/date/time/datetime
- ,range: false //是否开启范围选择,即双控件
- ,format: 'yyyy-MM-dd' //默认日期格式
- ,value: null //默认日期,支持传入new Date(),或者符合format参数设定的日期格式字符
- ,min: '1900-1-1' //有效最小日期,年月日必须用“-”分割,时分秒必须用“:”分割。注意:它并不是遵循 format 设定的格式。
- ,max: '2099-12-31' //有效最大日期,同上
- ,trigger: 'focus' //呼出控件的事件
- ,show: false //是否直接显示,如果设置true,则默认直接显示控件
- ,showBottom: true //是否显示底部栏
- ,btns: ['clear', 'now', 'confirm'] //右下角显示的按钮,会按照数组顺序排列
- ,lang: 'cn' //语言,只支持cn/en,即中文和英文
- ,theme: 'default' //主题
- ,position: null //控件定位方式定位, 默认absolute,支持:fixed/absolute/static
- ,calendar: false //是否开启公历重要节日,仅支持中文版
- ,mark: {} //日期备注,如重要事件或活动标记
- ,zIndex: null //控件层叠顺序
- ,done: null //控件选择完毕后的回调,点击清空/现在/确定也均会触发
- ,change: null //日期时间改变后的回调
- };
-
- //多语言
- Class.prototype.lang = function(){
- var that = this
- ,options = that.config
- ,text = {
- cn: {
- weeks: ['日', '一', '二', '三', '四', '五', '六']
- ,time: ['时', '分', '秒']
- ,timeTips: '选择时间'
- ,startTime: '开始时间'
- ,endTime: '结束时间'
- ,dateTips: '返回日期'
- ,month: ['一', '二', '三', '四', '五', '六', '七', '八', '九', '十', '十一', '十二']
- ,tools: {
- confirm: '确定'
- ,clear: '清空'
- ,now: '现在'
- }
- }
- ,en: {
- weeks: ['Su', 'Mo', 'Tu', 'We', 'Th', 'Fr', 'Sa']
- ,time: ['Hours', 'Minutes', 'Seconds']
- ,timeTips: 'Select Time'
- ,startTime: 'Start Time'
- ,endTime: 'End Time'
- ,dateTips: 'Select Date'
- ,month: ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', 'Jul', 'Aug', 'Sep', 'Oct', 'Nov', 'Dec']
- ,tools: {
- confirm: 'Confirm'
- ,clear: 'Clear'
- ,now: 'Now'
- }
- }
- };
- return text[options.lang] || text['cn'];
- };
-
- //初始准备
- Class.prototype.init = function(){
- var that = this
- ,options = that.config
- ,dateType = 'yyyy|y|MM|M|dd|d|HH|H|mm|m|ss|s'
- ,isStatic = options.position === 'static'
- ,format = {
- year: 'yyyy'
- ,month: 'yyyy-MM'
- ,date: 'yyyy-MM-dd'
- ,time: 'HH:mm:ss'
- ,datetime: 'yyyy-MM-dd HH:mm:ss'
- };
-
- options.elem = lay(options.elem);
- options.eventElem = lay(options.eventElem);
-
- if(!options.elem[0]) return;
-
- //日期范围分隔符
- if(options.range === true) options.range = '-';
-
- //根据不同type,初始化默认format
- if(options.format === format.date){
- options.format = format[options.type];
- }
-
- //将日期格式转化成数组
- that.format = options.format.match(new RegExp(dateType + '|.', 'g')) || [];
-
- //生成正则表达式
- that.EXP_IF = '';
- that.EXP_SPLIT = '';
- lay.each(that.format, function(i, item){
- var EXP = new RegExp(dateType).test(item)
- ? '\\d{'+ function(){
- if(new RegExp(dateType).test(that.format[i === 0 ? i + 1 : i - 1]||'')){
- if(/^yyyy|y$/.test(item)) return 4;
- return item.length;
- }
- if(/^yyyy$/.test(item)) return '1,4';
- if(/^y$/.test(item)) return '1,308';
- return '1,2';
- }() +'}'
- : '\\' + item;
- that.EXP_IF = that.EXP_IF + EXP;
- that.EXP_SPLIT = that.EXP_SPLIT + '(' + EXP + ')';
- });
- that.EXP_IF = new RegExp('^'+ (
- options.range ?
- that.EXP_IF + '\\s\\'+ options.range + '\\s' + that.EXP_IF
- : that.EXP_IF
- ) +'$');
- that.EXP_SPLIT = new RegExp('^'+ that.EXP_SPLIT +'$', '');
-
- //如果不是input|textarea元素,则默认采用click事件
- if(!that.isInput(options.elem[0])){
- if(options.trigger === 'focus'){
- options.trigger = 'click';
- }
- }
-
- //设置唯一KEY
- if(!options.elem.attr('lay-key')){
- options.elem.attr('lay-key', that.index);
- options.eventElem.attr('lay-key', that.index);
- }
-
- //记录重要日期
- options.mark = lay.extend({}, (options.calendar && options.lang === 'cn') ? {
- '0-1-1': '元旦'
- ,'0-2-14': '情人'
- ,'0-3-8': '妇女'
- ,'0-3-12': '植树'
- ,'0-4-1': '愚人'
- ,'0-5-1': '劳动'
- ,'0-5-4': '青年'
- ,'0-6-1': '儿童'
- ,'0-9-10': '教师'
- ,'0-9-18': '国耻'
- ,'0-10-1': '国庆'
- ,'0-12-25': '圣诞'
- } : {}, options.mark);
-
- //获取限制内日期
- lay.each(['min', 'max'], function(i, item){
- var ymd = [], hms = [];
- if(typeof options[item] === 'number'){ //如果为数字
- var day = options[item]
- ,time = new Date().getTime()
- ,STAMP = 86400000 //代表一天的时间戳
- ,thisDate = new Date(
- day ? (
- day < STAMP ? time + day*STAMP : day //如果数字小于一天的时间戳,则数字为天数,否则为时间戳
- ) : time
- );
- ymd = [thisDate.getFullYear(), thisDate.getMonth() + 1, thisDate.getDate()];
- day < STAMP || (hms = [thisDate.getHours(), thisDate.getMinutes(), thisDate.getSeconds()]);
- } else {
- ymd = (options[item].match(/\d+-\d+-\d+/) || [''])[0].split('-');
- hms = (options[item].match(/\d+:\d+:\d+/) || [''])[0].split(':');
- }
- options[item] = {
- year: ymd[0] | 0 || new Date().getFullYear()
- ,month: ymd[1] ? (ymd[1] | 0) - 1 : new Date().getMonth()
- ,date: ymd[2] | 0 || new Date().getDate()
- ,hours: hms[0] | 0
- ,minutes: hms[1] | 0
- ,seconds: hms[2] | 0
- };
- });
-
- that.elemID = 'layui-laydate'+ options.elem.attr('lay-key');
-
- if(options.show || isStatic) that.render();
- isStatic || that.events();
-
- //默认赋值
- if(options.value){
- if(options.value.constructor === Date){
- that.setValue(that.parse(0, that.systemDate(options.value)));
- } else {
- that.setValue(options.value);
- }
- }
- };
-
- //控件主体渲染
- Class.prototype.render = function(){
- var that = this
- ,options = that.config
- ,lang = that.lang()
- ,isStatic = options.position === 'static'
-
- //主面板
- ,elem = that.elem = lay.elem('div', {
- id: that.elemID
- ,'class': [
- 'layui-laydate'
- ,options.range ? ' layui-laydate-range' : ''
- ,isStatic ? (' '+ ELEM_STATIC) : ''
- ,options.theme && options.theme !== 'default' && !/^#/.test(options.theme) ? (' laydate-theme-' + options.theme) : ''
- ].join('')
- })
-
- //主区域
- ,elemMain = that.elemMain = []
- ,elemHeader = that.elemHeader = []
- ,elemCont = that.elemCont = []
- ,elemTable = that.table = []
- //底部区域
- ,divFooter = that.footer = lay.elem('div', {
- 'class': ELEM_FOOTER
- });
-
- if(options.zIndex) elem.style.zIndex = options.zIndex;
-
- //单双日历区域
- lay.each(new Array(2), function(i){
- if(!options.range && i > 0){
- return true;
- }
- //头部区域
- var divHeader = lay.elem('div', {
- 'class': 'layui-laydate-header'
- })
-
- //左右切换
- ,headerChild = [function(){ //上一年
- var elem = lay.elem('i', {
- 'class': 'layui-icon laydate-icon laydate-prev-y'
- });
- elem.innerHTML = '';
- return elem;
- }(), function(){ //上一月
- var elem = lay.elem('i', {
- 'class': 'layui-icon laydate-icon laydate-prev-m'
- });
- elem.innerHTML = '';
- return elem;
- }(), function(){ //年月选择
- var elem = lay.elem('div', {
- 'class': 'laydate-set-ym'
- }), spanY = lay.elem('span'), spanM = lay.elem('span');
- elem.appendChild(spanY);
- elem.appendChild(spanM);
- return elem;
- }(), function(){ //下一月
- var elem = lay.elem('i', {
- 'class': 'layui-icon laydate-icon laydate-next-m'
- });
- elem.innerHTML = '';
- return elem;
- }(), function(){ //下一年
- var elem = lay.elem('i', {
- 'class': 'layui-icon laydate-icon laydate-next-y'
- });
- elem.innerHTML = '';
- return elem;
- }()]
-
- //日历内容区域
- ,divContent = lay.elem('div', {
- 'class': 'layui-laydate-content'
- })
- ,table = lay.elem('table')
- ,thead = lay.elem('thead'), theadTr = lay.elem('tr');
-
- //生成年月选择
- lay.each(headerChild, function(i, item){
- divHeader.appendChild(item);
- });
-
- //生成表格
- thead.appendChild(theadTr);
- lay.each(new Array(6), function(i){ //表体
- var tr = table.insertRow(0);
- lay.each(new Array(7), function(j){
- if(i === 0){
- var th = lay.elem('th');
- th.innerHTML = lang.weeks[j];
- theadTr.appendChild(th);
- }
- tr.insertCell(j);
- });
- });
- table.insertBefore(thead, table.children[0]); //表头
- divContent.appendChild(table);
-
- elemMain[i] = lay.elem('div', {
- 'class': 'layui-laydate-main laydate-main-list-'+ i
- });
-
- elemMain[i].appendChild(divHeader);
- elemMain[i].appendChild(divContent);
-
- elemHeader.push(headerChild);
- elemCont.push(divContent);
- elemTable.push(table);
- });
-
- //生成底部栏
- lay(divFooter).html(function(){
- var html = [], btns = [];
- if(options.type === 'datetime'){
- html.push('<span lay-type="datetime" class="laydate-btns-time">'+ lang.timeTips +'</span>');
- }
- lay.each(options.btns, function(i, item){
- var title = lang.tools[item] || 'btn';
- if(options.range && item === 'now') return;
- if(isStatic && item === 'clear') title = options.lang === 'cn' ? '重置' : 'Reset';
- btns.push('<span lay-type="'+ item +'" class="laydate-btns-'+ item +'">'+ title +'</span>');
- });
- html.push('<div class="laydate-footer-btns">'+ btns.join('') +'</div>');
- return html.join('');
- }());
-
- //插入到主区域
- lay.each(elemMain, function(i, main){
- elem.appendChild(main);
- });
- options.showBottom && elem.appendChild(divFooter);
-
- //生成自定义主题
- if(/^#/.test(options.theme)){
- var style = lay.elem('style')
- ,styleText = [
- '#{{id}} .layui-laydate-header{background-color:{{theme}};}'
- ,'#{{id}} .layui-this{background-color:{{theme}} !important;}'
- ].join('').replace(/{{id}}/g, that.elemID).replace(/{{theme}}/g, options.theme);
-
- if('styleSheet' in style){
- style.setAttribute('type', 'text/css');
- style.styleSheet.cssText = styleText;
- } else {
- style.innerHTML = styleText;
- }
-
- lay(elem).addClass('laydate-theme-molv');
- elem.appendChild(style);
- }
-
- //移除上一个控件
- that.remove(Class.thisElemDate);
-
- //如果是静态定位,则插入到指定的容器中,否则,插入到body
- isStatic ? options.elem.append(elem) : (
- document.body.appendChild(elem)
- ,that.position() //定位
- );
-
- that.checkDate().calendar(); //初始校验
- that.changeEvent(); //日期切换
-
- Class.thisElemDate = that.elemID;
- typeof options.ready === 'function' && options.ready(lay.extend({}, options.dateTime, {
- month: options.dateTime.month + 1
- }));
- };
-
- //控件移除
- Class.prototype.remove = function(prev){
- var that = this
- ,options = that.config
- ,elem = lay('#'+ (prev || that.elemID));
- if(!elem.hasClass(ELEM_STATIC)){
- that.checkDate(function(){
- elem.remove();
- });
- }
- return that;
- };
-
- //定位算法
- Class.prototype.position = function(){
- var that = this
- ,options = that.config
- ,elem = that.bindElem || options.elem[0]
- ,rect = elem.getBoundingClientRect() //绑定元素的坐标
- ,elemWidth = that.elem.offsetWidth //控件的宽度
- ,elemHeight = that.elem.offsetHeight //控件的高度
-
- //滚动条高度
- ,scrollArea = function(type){
- type = type ? 'scrollLeft' : 'scrollTop';
- return document.body[type] | document.documentElement[type];
- }
- ,winArea = function(type){
- return document.documentElement[type ? 'clientWidth' : 'clientHeight']
- }, margin = 5, left = rect.left, top = rect.bottom;
-
- //如果右侧超出边界
- if(left + elemWidth + margin > winArea('width')){
- left = winArea('width') - elemWidth - margin;
- }
-
- //如果底部超出边界
- if(top + elemHeight + margin > winArea()){
- top = rect.top > elemHeight //顶部是否有足够区域显示完全
- ? rect.top - elemHeight
- : winArea() - elemHeight;
- top = top - margin*2;
- }
-
- if(options.position){
- that.elem.style.position = options.position;
- }
- that.elem.style.left = left + (options.position === 'fixed' ? 0 : scrollArea(1)) + 'px';
- that.elem.style.top = top + (options.position === 'fixed' ? 0 : scrollArea()) + 'px';
- };
-
- //提示
- Class.prototype.hint = function(content){
- var that = this
- ,options = that.config
- ,div = lay.elem('div', {
- 'class': ELEM_HINT
- });
-
- div.innerHTML = content || '';
- lay(that.elem).find('.'+ ELEM_HINT).remove();
- that.elem.appendChild(div);
- clearTimeout(that.hinTimer);
- that.hinTimer = setTimeout(function(){
- lay(that.elem).find('.'+ ELEM_HINT).remove();
- }, 3000);
- };
-
- //获取递增/减后的年月
- Class.prototype.getAsYM = function(Y, M, type){
- type ? M-- : M++;
- if(M < 0){
- M = 11;
- Y--;
- }
- if(M > 11){
- M = 0;
- Y++;
- }
- return [Y, M];
- };
-
- //系统消息
- Class.prototype.systemDate = function(newDate){
- var thisDate = newDate || new Date();
- return {
- year: thisDate.getFullYear() //年
- ,month: thisDate.getMonth() //月
- ,date: thisDate.getDate() //日
- ,hours: newDate ? newDate.getHours() : 0 //时
- ,minutes: newDate ? newDate.getMinutes() : 0 //分
- ,seconds: newDate ? newDate.getSeconds() : 0 //秒
- }
- };
-
- //日期校验
- Class.prototype.checkDate = function(fn){
- var that = this
- ,thisDate = new Date()
- ,options = that.config
- ,dateTime = options.dateTime = options.dateTime || that.systemDate()
- ,thisMaxDate, error
-
- ,elem = that.bindElem || options.elem[0]
- ,valType = that.isInput(elem) ? 'val' : 'html'
- ,value = that.isInput(elem) ? elem.value : (options.position === 'static' ? '' : elem.innerHTML)
-
- //校验日期有效数字
- ,checkValid = function(dateTime){
- if(dateTime.year > LIMIT_YEAR[1]) dateTime.year = LIMIT_YEAR[1], error = true; //不能超过20万年
- if(dateTime.month > 11) dateTime.month = 11, error = true;
- if(dateTime.hours > 23) dateTime.hours = 0, error = true;
- if(dateTime.minutes > 59) dateTime.minutes = 0, dateTime.hours++, error = true;
- if(dateTime.seconds > 59) dateTime.seconds = 0, dateTime.minutes++, error = true;
-
- //计算当前月的最后一天
- thisMaxDate = laydate.getEndDate(dateTime.month + 1, dateTime.year);
- if(dateTime.date > thisMaxDate) dateTime.date = thisMaxDate, error = true;
- }
-
- //获得初始化日期值
- ,initDate = function(dateTime, value, index){
- var startEnd = ['startTime', 'endTime'];
- value = (value.match(that.EXP_SPLIT) || []).slice(1);
- index = index || 0;
- if(options.range){
- that[startEnd[index]] = that[startEnd[index]] || {};
- }
- lay.each(that.format, function(i, item){
- var thisv = parseFloat(value[i]);
- if(value[i].length < item.length) error = true;
- if(/yyyy|y/.test(item)){ //年
- if(thisv < LIMIT_YEAR[0]) thisv = LIMIT_YEAR[0], error = true; //年不能低于100年
- dateTime.year = thisv;
- } else if(/MM|M/.test(item)){ //月
- if(thisv < 1) thisv = 1, error = true;
- dateTime.month = thisv - 1;
- } else if(/dd|d/.test(item)){ //日
- if(thisv < 1) thisv = 1, error = true;
- dateTime.date = thisv;
- } else if(/HH|H/.test(item)){ //时
- if(thisv < 1) thisv = 0, error = true;
- dateTime.hours = thisv;
- options.range && (that[startEnd[index]].hours = thisv);
- } else if(/mm|m/.test(item)){ //分
- if(thisv < 1) thisv = 0, error = true;
- dateTime.minutes = thisv;
- options.range && (that[startEnd[index]].minutes = thisv);
- } else if(/ss|s/.test(item)){ //秒
- if(thisv < 1) thisv = 0, error = true;
- dateTime.seconds = thisv;
- options.range && (that[startEnd[index]].seconds = thisv);
- }
- });
- checkValid(dateTime)
- };
-
- if(fn === 'limit') return checkValid(dateTime), that;
-
- value = value || options.value;
- if(typeof value === 'string'){
- value = value.replace(/\s+/g, ' ').replace(/^\s|\s$/g, '');
- }
-
- //如果点击了开始,单未选择结束就关闭,则重新选择开始
- if(that.startState && !that.endState){
- delete that.startState;
- that.endState = true;
- };
- if(typeof value === 'string' && value){
- if(that.EXP_IF.test(value)){ //校验日期格式
- if(options.range){
- value = value.split(' '+ options.range +' ');
- that.startDate = that.startDate || that.systemDate();
- that.endDate = that.endDate || that.systemDate();
- options.dateTime = lay.extend({}, that.startDate);
- lay.each([that.startDate, that.endDate], function(i, item){
- initDate(item, value[i], i);
- });
- } else {
- initDate(dateTime, value)
- }
- } else {
- that.hint('日期格式不合法<br>必须遵循下述格式:<br>'+ (
- options.range ? (options.format + ' '+ options.range +' ' + options.format) : options.format
- ) + '<br>已为你重置');
- error = true;
- }
- } else if(value && value.constructor === Date){ //如果值为日期对象时
- options.dateTime = that.systemDate(value);
- } else {
- options.dateTime = that.systemDate();
- delete that.startState;
- delete that.endState;
- delete that.startDate;
- delete that.endDate;
- delete that.startTime;
- delete that.endTime;
- }
- checkValid(dateTime);
- if(error && value){
- that.setValue(
- options.range ? (that.endDate ? that.parse() : '') : that.parse()
- );
- }
- fn && fn();
- return that;
- };
-
- //公历重要日期与自定义备注
- Class.prototype.mark = function(td, YMD){
- var that = this
- ,mark, options = that.config;
- lay.each(options.mark, function(key, title){
- var keys = key.split('-');
- if((keys[0] == YMD[0] || keys[0] == 0) //每年的每月
- && (keys[1] == YMD[1] || keys[1] == 0) //每月的每日
- && keys[2] == YMD[2]){ //特定日
- mark = title || YMD[2];
- }
- });
- mark && td.html('<span class="laydate-day-mark">'+ mark +'</span>');
-
- return that;
- };
-
- //无效日期范围的标记
- Class.prototype.limit = function(elem, date, index, time){
- var that = this
- ,options = that.config, timestrap = {}
- ,dateTime = options[index > 41 ? 'endDate' : 'dateTime']
- ,isOut, thisDateTime = lay.extend({}, dateTime, date || {});
- lay.each({
- now: thisDateTime
- ,min: options.min
- ,max: options.max
- }, function(key, item){
- timestrap[key] = that.newDate(lay.extend({
- year: item.year
- ,month: item.month
- ,date: item.date
- }, function(){
- var hms = {};
- lay.each(time, function(i, keys){
- hms[keys] = item[keys];
- });
- return hms;
- }())).getTime(); //time:是否比较时分秒
- });
-
- isOut = timestrap.now < timestrap.min || timestrap.now > timestrap.max;
- elem && elem[isOut ? 'addClass' : 'removeClass'](DISABLED);
- return isOut;
- };
-
- //日历表
- Class.prototype.calendar = function(value){
- var that = this
- ,options = that.config
- ,dateTime = value || options.dateTime
- ,thisDate = new Date(), startWeek, prevMaxDate, thisMaxDate
- ,lang = that.lang()
-
- ,isAlone = options.type !== 'date' && options.type !== 'datetime'
- ,index = value ? 1 : 0
- ,tds = lay(that.table[index]).find('td')
- ,elemYM = lay(that.elemHeader[index][2]).find('span');
-
- if(dateTime.year < LIMIT_YEAR[0]) dateTime.year = LIMIT_YEAR[0], that.hint('最低只能支持到公元'+ LIMIT_YEAR[0] +'年');
- if(dateTime.year > LIMIT_YEAR[1]) dateTime.year = LIMIT_YEAR[1], that.hint('最高只能支持到公元'+ LIMIT_YEAR[1] +'年');
-
- //记录初始值
- if(!that.firstDate){
- that.firstDate = lay.extend({}, dateTime);
- }
-
- //计算当前月第一天的星期
- thisDate.setFullYear(dateTime.year, dateTime.month, 1);
- startWeek = thisDate.getDay();
-
- prevMaxDate = laydate.getEndDate(dateTime.month || 12, dateTime.year); //计算上个月的最后一天
- thisMaxDate = laydate.getEndDate(dateTime.month + 1, dateTime.year); //计算当前月的最后一天
-
- //赋值日
- lay.each(tds, function(index, item){
- var YMD = [dateTime.year, dateTime.month], st = 0;
- item = lay(item);
- item.removeAttr('class');
- if(index < startWeek){
- st = prevMaxDate - startWeek + index;
- item.addClass('laydate-day-prev');
- YMD = that.getAsYM(dateTime.year, dateTime.month, 'sub');
- } else if(index >= startWeek && index < thisMaxDate + startWeek){
- st = index - startWeek;
- if(!options.range){
- st + 1 === dateTime.date && item.addClass(THIS);
- }
- } else {
- st = index - thisMaxDate - startWeek;
- item.addClass('laydate-day-next');
- YMD = that.getAsYM(dateTime.year, dateTime.month);
- }
- YMD[1]++;
- YMD[2] = st + 1;
- item.attr('lay-ymd', YMD.join('-')).html(YMD[2]);
- that.mark(item, YMD).limit(item, {
- year: YMD[0]
- ,month: YMD[1] - 1
- ,date: YMD[2]
- }, index);
- });
-
- //同步头部年月
- lay(elemYM[0]).attr('lay-ym', dateTime.year + '-' + (dateTime.month + 1));
- lay(elemYM[1]).attr('lay-ym', dateTime.year + '-' + (dateTime.month + 1));
-
- if(options.lang === 'cn'){
- lay(elemYM[0]).attr('lay-type', 'year').html(dateTime.year + '年')
- lay(elemYM[1]).attr('lay-type', 'month').html((dateTime.month + 1) + '月');
- } else {
- lay(elemYM[0]).attr('lay-type', 'month').html(lang.month[dateTime.month]);
- lay(elemYM[1]).attr('lay-type', 'year').html(dateTime.year);
- }
- //初始默认选择器
- if(isAlone){
- if(options.range){
- value ? that.endDate = (that.endDate || {
- year: dateTime.year + (options.type === 'year' ? 1 : 0)
- ,month: dateTime.month + (options.type === 'month' ? 0 : -1)
- }) : (that.startDate = that.startDate || {
- year: dateTime.year
- ,month: dateTime.month
- });
- if(value){
- that.listYM = [
- [that.startDate.year, that.startDate.month + 1]
- ,[that.endDate.year, that.endDate.month + 1]
- ];
- that.list(options.type, 0).list(options.type, 1);
- //同步按钮可点状态
- options.type === 'time' ? that.setBtnStatus('时间'
- ,lay.extend({}, that.systemDate(), that.startTime)
- ,lay.extend({}, that.systemDate(), that.endTime)
- ) : that.setBtnStatus(true);
- }
- }
- if(!options.range){
- that.listYM = [[dateTime.year, dateTime.month + 1]];
- that.list(options.type, 0);
- }
- }
-
- //赋值双日历
- if(options.range && !value){
- var EYM = that.getAsYM(dateTime.year, dateTime.month)
- that.calendar(lay.extend({}, dateTime, {
- year: EYM[0]
- ,month: EYM[1]
- }));
- }
-
- //通过检测当前有效日期,来设定确定按钮是否可点
- if(!options.range) that.limit(lay(that.footer).find(ELEM_CONFIRM), null, 0, ['hours', 'minutes', 'seconds']);
-
- //标记选择范围
- if(options.range && value && !isAlone) that.stampRange();
- return that;
- };
- //生成年月时分秒列表
- Class.prototype.list = function(type, index){
- var that = this
- ,options = that.config
- ,dateTime = options.dateTime
- ,lang = that.lang()
- ,isAlone = options.range && options.type !== 'date' && options.type !== 'datetime' //独立范围选择器
-
- ,ul = lay.elem('ul', {
- 'class': ELEM_LIST + ' ' + ({
- year: 'laydate-year-list'
- ,month: 'laydate-month-list'
- ,time: 'laydate-time-list'
- })[type]
- })
- ,elemHeader = that.elemHeader[index]
- ,elemYM = lay(elemHeader[2]).find('span')
- ,elemCont = that.elemCont[index || 0]
- ,haveList = lay(elemCont).find('.'+ ELEM_LIST)[0]
- ,isCN = options.lang === 'cn'
- ,text = isCN ? '年' : ''
-
- ,listYM = that.listYM[index] || {}
- ,hms = ['hours', 'minutes', 'seconds']
- ,startEnd = ['startTime', 'endTime'][index];
- if(listYM[0] < 1) listYM[0] = 1;
-
- if(type === 'year'){ //年列表
- var yearNum, startY = yearNum = listYM[0] - 7;
- if(startY < 1) startY = yearNum = 1;
- lay.each(new Array(15), function(i){
- var li = lay.elem('li', {
- 'lay-ym': yearNum
- }), ymd = {year: yearNum};
- yearNum == listYM[0] && lay(li).addClass(THIS);
- li.innerHTML = yearNum + text;
- ul.appendChild(li);
- if(yearNum < that.firstDate.year){
- ymd.month = options.min.month;
- ymd.date = options.min.date;
- } else if(yearNum >= that.firstDate.year){
- ymd.month = options.max.month;
- ymd.date = options.max.date;
- }
- that.limit(lay(li), ymd, index);
- yearNum++;
- });
- lay(elemYM[isCN ? 0 : 1]).attr('lay-ym', (yearNum - 8) + '-' + listYM[1])
- .html((startY + text) + ' - ' + (yearNum - 1 + text));
- } else if(type === 'month'){ //月列表
- lay.each(new Array(12), function(i){
- var li = lay.elem('li', {
- 'lay-ym': i
- }), ymd = {year: listYM[0], month: i};
- i + 1 == listYM[1] && lay(li).addClass(THIS);
- li.innerHTML = lang.month[i] + (isCN ? '月' : '');
- ul.appendChild(li);
- if(listYM[0] < that.firstDate.year){
- ymd.date = options.min.date;
- } else if(listYM[0] >= that.firstDate.year){
- ymd.date = options.max.date;
- }
- that.limit(lay(li), ymd, index);
- });
- lay(elemYM[isCN ? 0 : 1]).attr('lay-ym', listYM[0] + '-' + listYM[1])
- .html(listYM[0] + text);
- } else if(type === 'time'){ //时间列表
- //检测时分秒状态是否在有效日期时间范围内
- var setTimeStatus = function(){
- lay(ul).find('ol').each(function(i, ol){
- lay(ol).find('li').each(function(ii, li){
- that.limit(lay(li), [{
- hours: ii
- }, {
- hours: that[startEnd].hours
- ,minutes: ii
- }, {
- hours: that[startEnd].hours
- ,minutes: that[startEnd].minutes
- ,seconds: ii
- }][i], index, [['hours'], ['hours', 'minutes'], ['hours', 'minutes', 'seconds']][i]);
- });
- });
- if(!options.range) that.limit(lay(that.footer).find(ELEM_CONFIRM), that[startEnd], 0, ['hours', 'minutes', 'seconds']);
- };
- if(options.range){
- if(!that[startEnd]) that[startEnd] = {
- hours: 0
- ,minutes: 0
- ,seconds: 0
- };
- } else {
- that[startEnd] = dateTime;
- }
- lay.each([24, 60, 60], function(i, item){
- var li = lay.elem('li'), childUL = ['<p>'+ lang.time[i] +'</p><ol>'];
- lay.each(new Array(item), function(ii){
- childUL.push('<li'+ (that[startEnd][hms[i]] === ii ? ' class="'+ THIS +'"' : '') +'>'+ lay.digit(ii, 2) +'</li>');
- });
- li.innerHTML = childUL.join('') + '</ol>';
- ul.appendChild(li);
- });
- setTimeStatus();
- }
-
- //插入容器
- if(haveList) elemCont.removeChild(haveList);
- elemCont.appendChild(ul);
-
- //年月
- if(type === 'year' || type === 'month'){
- //显示切换箭头
- lay(that.elemMain[index]).addClass('laydate-ym-show');
-
- //选中
- lay(ul).find('li').on('click', function(){
- var ym = lay(this).attr('lay-ym') | 0;
- if(lay(this).hasClass(DISABLED)) return;
-
- if(index === 0){
- dateTime[type] = ym;
- if(isAlone) that.startDate[type] = ym;
- that.limit(lay(that.footer).find(ELEM_CONFIRM), null, 0);
- } else { //范围选择
- if(isAlone){ //非date/datetime类型
- that.endDate[type] = ym;
- } else { //date/datetime类型
- var YM = type === 'year'
- ? that.getAsYM(ym, listYM[1] - 1, 'sub')
- : that.getAsYM(listYM[0], ym, 'sub');
- lay.extend(dateTime, {
- year: YM[0]
- ,month: YM[1]
- });
- }
- }
-
- if(options.type === 'year' || options.type === 'month'){
- lay(ul).find('.'+ THIS).removeClass(THIS);
- lay(this).addClass(THIS);
-
- //如果为年月选择器,点击了年列表,则切换到月选择器
- if(options.type === 'month' && type === 'year'){
- that.listYM[index][0] = ym;
- isAlone && (that[['startDate', 'endDate'][index]].year = ym);
- that.list('month', index);
- }
- } else {
- that.checkDate('limit').calendar();
- that.closeList();
- }
- that.setBtnStatus(); //同步按钮可点状态
- options.range || that.done(null, 'change');
- lay(that.footer).find(ELEM_TIME_BTN).removeClass(DISABLED);
- });
- } else {
- var span = lay.elem('span', {
- 'class': ELEM_TIME_TEXT
- }), scroll = function(){ //滚动条定位
- lay(ul).find('ol').each(function(i){
- var ol = this
- ,li = lay(ol).find('li')
- ol.scrollTop = 30*(that[startEnd][hms[i]] - 2);
- if(ol.scrollTop <= 0){
- li.each(function(ii, item){
- if(!lay(this).hasClass(DISABLED)){
- ol.scrollTop = 30*(ii - 2);
- return true;
- }
- });
- }
- });
- }, haveSpan = lay(elemHeader[2]).find('.'+ ELEM_TIME_TEXT);
- scroll()
- span.innerHTML = options.range ? [lang.startTime,lang.endTime][index] : lang.timeTips
- lay(that.elemMain[index]).addClass('laydate-time-show');
- if(haveSpan[0]) haveSpan.remove();
- elemHeader[2].appendChild(span);
- lay(ul).find('ol').each(function(i){
- var ol = this;
- //选择时分秒
- lay(ol).find('li').on('click', function(){
- var value = this.innerHTML | 0;
- if(lay(this).hasClass(DISABLED)) return;
- if(options.range){
- that[startEnd][hms[i]] = value;
- } else {
- dateTime[hms[i]] = value;
- }
- lay(ol).find('.'+ THIS).removeClass(THIS);
- lay(this).addClass(THIS);
- setTimeStatus();
- scroll();
- (that.endDate || options.type === 'time') && that.done(null, 'change');
-
- //同步按钮可点状态
- that.setBtnStatus();
- });
- });
- }
-
- return that;
- };
-
- //记录列表切换后的年月
- Class.prototype.listYM = [];
-
- //关闭列表
- Class.prototype.closeList = function(){
- var that = this
- ,options = that.config;
-
- lay.each(that.elemCont, function(index, item){
- lay(this).find('.'+ ELEM_LIST).remove();
- lay(that.elemMain[index]).removeClass('laydate-ym-show laydate-time-show');
- });
- lay(that.elem).find('.'+ ELEM_TIME_TEXT).remove();
- };
-
- //检测结束日期是否超出开始日期
- Class.prototype.setBtnStatus = function(tips, start, end){
- var that = this
- ,options = that.config
- ,isOut, elemBtn = lay(that.footer).find(ELEM_CONFIRM)
- ,isAlone = options.range && options.type !== 'date' && options.type !== 'time';
- if(isAlone){
- start = start || that.startDate;
- end = end || that.endDate;
- isOut = that.newDate(start).getTime() > that.newDate(end).getTime();
-
- //如果不在有效日期内,直接禁用按钮,否则比较开始和结束日期
- (that.limit(null, start) || that.limit(null, end))
- ? elemBtn.addClass(DISABLED)
- : elemBtn[isOut ? 'addClass' : 'removeClass'](DISABLED);
-
- //是否异常提示
- if(tips && isOut) that.hint(
- typeof tips === 'string' ? TIPS_OUT.replace(/日期/g, tips) : TIPS_OUT
- );
- }
- };
-
- //转义为规定格式的日期字符
- Class.prototype.parse = function(state, date){
- var that = this
- ,options = that.config
- ,dateTime = date || (state
- ? lay.extend({}, that.endDate, that.endTime)
- : (options.range ? lay.extend({}, that.startDate, that.startTime) : options.dateTime))
- ,format = that.format.concat();
- //转义为规定格式
- lay.each(format, function(i, item){
- if(/yyyy|y/.test(item)){ //年
- format[i] = lay.digit(dateTime.year, item.length);
- } else if(/MM|M/.test(item)){ //月
- format[i] = lay.digit(dateTime.month + 1, item.length);
- } else if(/dd|d/.test(item)){ //日
- format[i] = lay.digit(dateTime.date, item.length);
- } else if(/HH|H/.test(item)){ //时
- format[i] = lay.digit(dateTime.hours, item.length);
- } else if(/mm|m/.test(item)){ //分
- format[i] = lay.digit(dateTime.minutes, item.length);
- } else if(/ss|s/.test(item)){ //秒
- format[i] = lay.digit(dateTime.seconds, item.length);
- }
- });
-
- //返回日期范围字符
- if(options.range && !state){
- return format.join('') + ' '+ options.range +' ' + that.parse(1);
- }
-
- return format.join('');
- };
-
- //创建指定日期时间对象
- Class.prototype.newDate = function(dateTime){
- dateTime = dateTime || {};
- return new Date(
- dateTime.year || 1
- ,dateTime.month || 0
- ,dateTime.date || 1
- ,dateTime.hours || 0
- ,dateTime.minutes || 0
- ,dateTime.seconds || 0
- );
- };
-
- //赋值
- Class.prototype.setValue = function(value){
- var that = this
- ,options = that.config
- ,elem = that.bindElem || options.elem[0]
- ,valType = that.isInput(elem) ? 'val' : 'html'
-
- options.position === 'static' || lay(elem)[valType](value || '');
- return this;
- };
-
- //标记范围内的日期
- Class.prototype.stampRange = function(){
- var that = this
- ,options = that.config
- ,startTime, endTime
- ,tds = lay(that.elem).find('td');
-
- if(options.range && !that.endDate) lay(that.footer).find(ELEM_CONFIRM).addClass(DISABLED);
- if(!that.endDate) return;
- startTime = that.newDate({
- year: that.startDate.year
- ,month: that.startDate.month
- ,date: that.startDate.date
- }).getTime();
-
- endTime = that.newDate({
- year: that.endDate.year
- ,month: that.endDate.month
- ,date: that.endDate.date
- }).getTime();
-
- if(startTime > endTime) return that.hint(TIPS_OUT);
-
- lay.each(tds, function(i, item){
- var ymd = lay(item).attr('lay-ymd').split('-')
- ,thisTime = that.newDate({
- year: ymd[0]
- ,month: ymd[1] - 1
- ,date: ymd[2]
- }).getTime();
- lay(item).removeClass(ELEM_SELECTED + ' ' + THIS);
- if(thisTime === startTime || thisTime === endTime){
- lay(item).addClass(
- lay(item).hasClass(ELEM_PREV) || lay(item).hasClass(ELEM_NEXT)
- ? ELEM_SELECTED
- : THIS
- );
- }
- if(thisTime > startTime && thisTime < endTime){
- lay(item).addClass(ELEM_SELECTED);
- }
- });
- };
-
- //执行done/change回调
- Class.prototype.done = function(param, type){
- var that = this
- ,options = that.config
- ,start = lay.extend({}, that.startDate ? lay.extend(that.startDate, that.startTime) : options.dateTime)
- ,end = lay.extend({}, lay.extend(that.endDate, that.endTime))
-
- lay.each([start, end], function(i, item){
- if(!('month' in item)) return;
- lay.extend(item, {
- month: item.month + 1
- });
- });
-
- param = param || [that.parse(), start, end];
- typeof options[type || 'done'] === 'function' && options[type || 'done'].apply(options, param);
-
- return that;
- };
-
- //选择日期
- Class.prototype.choose = function(td){
- var that = this
- ,options = that.config
- ,dateTime = options.dateTime
- ,tds = lay(that.elem).find('td')
- ,YMD = td.attr('lay-ymd').split('-')
-
- ,setDateTime = function(one){
- var thisDate = new Date();
-
- //同步dateTime
- one && lay.extend(dateTime, YMD);
-
- //记录开始日期
- if(options.range){
- that.startDate ? lay.extend(that.startDate, YMD) : (
- that.startDate = lay.extend({}, YMD, that.startTime)
- );
- that.startYMD = YMD;
- }
- };
-
- YMD = {
- year: YMD[0] | 0
- ,month: (YMD[1] | 0) - 1
- ,date: YMD[2] | 0
- };
-
- if(td.hasClass(DISABLED)) return;
- //范围选择
- if(options.range){
-
- lay.each(['startTime', 'endTime'], function(i, item){
- that[item] = that[item] || {
- hours: 0
- ,minutes: 0
- ,seconds: 0
- };
- });
-
- if(that.endState){ //重新选择
- setDateTime();
- delete that.endState;
- delete that.endDate;
- that.startState = true;
- tds.removeClass(THIS + ' ' + ELEM_SELECTED);
- td.addClass(THIS);
- } else if(that.startState){ //选中截止
- td.addClass(THIS);
-
- that.endDate ? lay.extend(that.endDate, YMD) : (
- that.endDate = lay.extend({}, YMD, that.endTime)
- );
-
- //判断是否顺时或逆时选择
- if(that.newDate(YMD).getTime() < that.newDate(that.startYMD).getTime()){
- var startDate = lay.extend({}, that.endDate, {
- hours: that.startDate.hours
- ,minutes: that.startDate.minutes
- ,seconds: that.startDate.seconds
- });
- lay.extend(that.endDate, that.startDate, {
- hours: that.endDate.hours
- ,minutes: that.endDate.minutes
- ,seconds: that.endDate.seconds
- });
- that.startDate = startDate;
- }
-
- options.showBottom || that.done();
- that.stampRange(); //标记范围内的日期
- that.endState = true;
- that.done(null, 'change');
- } else { //选中开始
- td.addClass(THIS);
- setDateTime();
- that.startState = true;
- }
- lay(that.footer).find(ELEM_CONFIRM)[that.endDate ? 'removeClass' : 'addClass'](DISABLED);
- } else if(options.position === 'static'){ //直接嵌套的选中
- setDateTime(true);
- that.calendar().done().done(null, 'change');
- } else if(options.type === 'date'){
- setDateTime(true);
- that.setValue(that.parse()).remove().done();
- } else if(options.type === 'datetime'){
- setDateTime(true);
- that.calendar().done(null, 'change');
- }
- };
-
- //底部按钮
- Class.prototype.tool = function(btn, type){
- var that = this
- ,options = that.config
- ,dateTime = options.dateTime
- ,isStatic = options.position === 'static'
- ,active = {
- //选择时间
- datetime: function(){
- if(lay(btn).hasClass(DISABLED)) return;
- that.list('time', 0);
- options.range && that.list('time', 1);
- lay(btn).attr('lay-type', 'date').html(that.lang().dateTips);
- }
-
- //选择日期
- ,date: function(){
- that.closeList();
- lay(btn).attr('lay-type', 'datetime').html(that.lang().timeTips);
- }
-
- //清空、重置
- ,clear: function(){
- that.setValue('').remove();
- isStatic && (
- lay.extend(dateTime, that.firstDate)
- ,that.calendar()
- )
- options.range && (
- delete that.startState
- ,delete that.endState
- ,delete that.endDate
- ,delete that.startTime
- ,delete that.endTime
- );
- that.done(['', {}, {}]);
- }
-
- //现在
- ,now: function(){
- var thisDate = new Date();
- lay.extend(dateTime, that.systemDate(), {
- hours: thisDate.getHours()
- ,minutes: thisDate.getMinutes()
- ,seconds: thisDate.getSeconds()
- });
- that.setValue(that.parse()).remove();
- isStatic && that.calendar();
- that.done();
- }
-
- //确定
- ,confirm: function(){
- if(options.range){
- if(!that.endDate) return that.hint('请先选择日期范围');
- if(lay(btn).hasClass(DISABLED)) return that.hint(
- options.type === 'time' ? TIPS_OUT.replace(/日期/g, '时间') : TIPS_OUT
- );
- } else {
- if(lay(btn).hasClass(DISABLED)) return that.hint('不在有效日期或时间范围内');
- }
- that.done();
- that.setValue(that.parse()).remove()
- }
- };
- active[type] && active[type]();
- };
-
- //统一切换处理
- Class.prototype.change = function(index){
- var that = this
- ,options = that.config
- ,dateTime = options.dateTime
- ,isAlone = options.range && (options.type === 'year' || options.type === 'month')
-
- ,elemCont = that.elemCont[index || 0]
- ,listYM = that.listYM[index]
- ,addSubYeay = function(type){
- var startEnd = ['startDate', 'endDate'][index]
- ,isYear = lay(elemCont).find('.laydate-year-list')[0]
- ,isMonth = lay(elemCont).find('.laydate-month-list')[0];
-
- //切换年列表
- if(isYear){
- listYM[0] = type ? listYM[0] - 15 : listYM[0] + 15;
- that.list('year', index);
- }
-
- if(isMonth){ //切换月面板中的年
- type ? listYM[0]-- : listYM[0]++;
- that.list('month', index);
- }
-
- if(isYear || isMonth){
- lay.extend(dateTime, {
- year: listYM[0]
- });
- if(isAlone) that[startEnd].year = listYM[0];
- options.range || that.done(null, 'change');
- that.setBtnStatus();
- options.range || that.limit(lay(that.footer).find(ELEM_CONFIRM), {
- year: listYM[0]
- });
- }
- return isYear || isMonth;
- };
-
- return {
- prevYear: function(){
- if(addSubYeay('sub')) return;
- dateTime.year--;
- that.checkDate('limit').calendar();
- options.range || that.done(null, 'change');
- }
- ,prevMonth: function(){
- var YM = that.getAsYM(dateTime.year, dateTime.month, 'sub');
- lay.extend(dateTime, {
- year: YM[0]
- ,month: YM[1]
- });
- that.checkDate('limit').calendar();
- options.range || that.done(null, 'change');
- }
- ,nextMonth: function(){
- var YM = that.getAsYM(dateTime.year, dateTime.month);
- lay.extend(dateTime, {
- year: YM[0]
- ,month: YM[1]
- });
- that.checkDate('limit').calendar();
- options.range || that.done(null, 'change');
- }
- ,nextYear: function(){
- if(addSubYeay()) return;
- dateTime.year++
- that.checkDate('limit').calendar();
- options.range || that.done(null, 'change');
- }
- };
- };
-
- //日期切换事件
- Class.prototype.changeEvent = function(){
- var that = this
- ,options = that.config;
- //日期选择事件
- lay(that.elem).on('click', function(e){
- lay.stope(e);
- });
-
- //年月切换
- lay.each(that.elemHeader, function(i, header){
- //上一年
- lay(header[0]).on('click', function(e){
- that.change(i).prevYear();
- });
-
- //上一月
- lay(header[1]).on('click', function(e){
- that.change(i).prevMonth();
- });
-
- //选择年月
- lay(header[2]).find('span').on('click', function(e){
- var othis = lay(this)
- ,layYM = othis.attr('lay-ym')
- ,layType = othis.attr('lay-type');
-
- if(!layYM) return;
-
- layYM = layYM.split('-');
- that.listYM[i] = [layYM[0] | 0, layYM[1] | 0];
- that.list(layType, i);
- lay(that.footer).find(ELEM_TIME_BTN).addClass(DISABLED);
- });
- //下一月
- lay(header[3]).on('click', function(e){
- that.change(i).nextMonth();
- });
-
- //下一年
- lay(header[4]).on('click', function(e){
- that.change(i).nextYear();
- });
- });
-
- //点击日期
- lay.each(that.table, function(i, table){
- var tds = lay(table).find('td');
- tds.on('click', function(){
- that.choose(lay(this));
- });
- });
-
- //点击底部按钮
- lay(that.footer).find('span').on('click', function(){
- var type = lay(this).attr('lay-type');
- that.tool(this, type);
- });
- };
-
- //是否输入框
- Class.prototype.isInput = function(elem){
- return /input|textarea/.test(elem.tagName.toLocaleLowerCase());
- };
- //绑定的元素事件处理
- Class.prototype.events = function(){
- var that = this
- ,options = that.config
- //绑定呼出控件事件
- ,showEvent = function(elem, bind){
- elem.on(options.trigger, function(){
- bind && (that.bindElem = this);
- that.render();
- });
- };
-
- if(!options.elem[0] || options.elem[0].eventHandler) return;
-
- showEvent(options.elem, 'bind');
- showEvent(options.eventElem);
-
- //绑定关闭控件事件
- lay(document).on('click', function(e){
- if(e.target === options.elem[0]
- || e.target === options.eventElem[0]
- || e.target === lay(options.closeStop)[0]){
- return;
- }
- that.remove();
- }).on('keydown', function(e){
- if(e.keyCode === 13){
- if(lay('#'+ that.elemID)[0] && that.elemID === Class.thisElem){
- e.preventDefault();
- lay(that.footer).find(ELEM_CONFIRM)[0].click();
- }
- }
- });
-
- //自适应定位
- lay(window).on('resize', function(){
- if(!that.elem || !lay(ELEM)[0]){
- return false;
- }
- that.position();
- });
-
- options.elem[0].eventHandler = true;
- };
-
- //核心接口
- laydate.render = function(options){
- var inst = new Class(options);
- return thisDate.call(inst);
- };
-
- //得到某月的最后一天
- laydate.getEndDate = function(month, year){
- var thisDate = new Date();
- //设置日期为下个月的第一天
- thisDate.setFullYear(
- year || thisDate.getFullYear()
- ,month || (thisDate.getMonth() + 1)
- ,1);
- //减去一天,得到当前月最后一天
- return new Date(thisDate.getTime() - 1000*60*60*24).getDate();
- };
-
- //暴露lay
- window.lay = window.lay || lay;
-
- //加载方式
- isLayui ? (
- laydate.ready()
- ,layui.define(function(exports){ //layui加载
- laydate.path = layui.cache.dir;
- exports(MOD_NAME, laydate);
- })
- ) : (
- (typeof define === 'function' && define.amd) ? define(function(){ //requirejs加载
- return laydate;
- }) : function(){ //普通script标签加载
- laydate.ready();
- window.laydate = laydate
- }()
- );
- }();
|