`
gogo1217
  • 浏览: 152310 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

真正的Extjs 4.2 支持时间选择的日历扩展

 
阅读更多

版权所有,转载请注明来源http://gogo1217.iteye.com,违者必究!

 

最近在做一个case的时候,客户要求能在选择年月日,时分秒在一个面板中选择;客户端JS框架使用的是ExtJs4.2。

我们知道,ExtJs4.2的Ext.picker.Date只能选择年月日,网上有部分实现,但大多数的都是基于ExtJs3修改的,而且过多的破坏了ExtJs的编写习惯。

现将我写的跟大家分享下,代码中有相关注释,目前还不支持minValue和maxValue设置。

顺便提下,官方示例中的时间区间的vtype扩展在2个日历控件都设置了value情况下会陷入无限的validate校验,从而导致浏览器崩溃。

 

1、自定义DateTime选择器:

/*
 * 带时间选择的日历选择器
 * 转载请注明来自于gogo1217.iteye.com
*/
Ext.define('Go.picker.DateTime', {
    extend: 'Ext.picker.Date',//继承于 Ext.picker.Date
    alias: 'widget.dateptimeicker',//添加xtype dateptimeicker
    okText:'确定',//确认按钮文字描述
    okTip:'确定',//确认按钮提示内容

    renderTpl: [
        '<div id="{id}-innerEl" role="grid">',
            '<div role="presentation" class="{baseCls}-header">',
                '<a id="{id}-prevEl" class="{baseCls}-prev {baseCls}-arrow" href="#" role="button" title="{prevText}" hidefocus="on" ></a>',
                '<div class="{baseCls}-month" id="{id}-middleBtnEl">{%this.renderMonthBtn(values, out)%}</div>',
                '<a id="{id}-nextEl" class="{baseCls}-next {baseCls}-arrow" href="#" role="button" title="{nextText}" hidefocus="on" ></a>',
            '</div>',
            '<table id="{id}-eventEl" class="{baseCls}-inner" cellspacing="0" role="presentation">',
                '<thead role="presentation"><tr role="presentation">',
                    '<tpl for="dayNames">',
                        '<th role="columnheader" class="{parent.baseCls}-column-header" title="{.}">',
                            '<div class="{parent.baseCls}-column-header-inner">{.:this.firstInitial}</div>',
                        '</th>',
                    '</tpl>',
                '</tr></thead>',
                '<tbody role="presentation"><tr role="presentation">',
                    '<tpl for="days">',
                        '{#:this.isEndOfWeek}',
                        '<td role="gridcell" id="{[Ext.id()]}">',
                           '<a role="presentation" hidefocus="on" class="{parent.baseCls}-date" href="#"></a>',
                        '</td>',
                    '</tpl>',
                '</tr></tbody>',
            '</table>',

            //指定时分秒渲染框架
            '<table id="{id}-timeEl" style="table-layout:auto;width:auto;margin:0 3px;" class="x-datepicker-inner" cellspacing="0">',
                '<tbody><tr>',
                    '<td>{%this.renderHourBtn(values,out)%}</td>',
                    '<td>{%this.renderMinuteBtn(values,out)%}</td>',
                    '<td>{%this.renderSecondBtn(values,out)%}</td>',
                '</tr></tbody>',
            '</table>',

            '<tpl if="showToday">',
                //添加一个确认按钮渲染
                '<div id="{id}-footerEl" role="presentation" class="{baseCls}-footer">{%this.renderOkBtn(values, out)%}{%this.renderTodayBtn(values, out)%}</div>',
            '</tpl>',
        '</div>',
        {
            firstInitial: function(value) {
                return Ext.picker.Date.prototype.getDayInitial(value);
            },
            isEndOfWeek: function(value) {
                // convert from 1 based index to 0 based
                // by decrementing value once.
                value--;
                var end = value % 7 === 0 && value !== 0;
                return end ? '</tr><tr role="row">' : '';
            },
            renderTodayBtn: function(values, out) {
                Ext.DomHelper.generateMarkup(values.$comp.todayBtn.getRenderTree(), out);
            },
            renderMonthBtn: function(values, out) {
                Ext.DomHelper.generateMarkup(values.$comp.monthBtn.getRenderTree(), out);
            },

            //指定渲染方法调用
            renderHourBtn: function(values, out) {
                Ext.DomHelper.generateMarkup(values.$comp.hourBtn.getRenderTree(), out);//根据组件获得组件的html输出
            },
            renderMinuteBtn: function(values, out) {
                Ext.DomHelper.generateMarkup(values.$comp.minuteBtn.getRenderTree(), out);
            },
            renderSecondBtn: function(values, out) {
                Ext.DomHelper.generateMarkup(values.$comp.secondBtn.getRenderTree(), out);
            },
            renderOkBtn: function(values, out) {
                Ext.DomHelper.generateMarkup(values.$comp.okBtn.getRenderTree(), out);
            }
        }
    ],

    beforeRender: function () {
        var me = this,_$Number=Ext.form.field.Number;
        //在组件渲染之前,将自定义添加的时、分、秒和确认按钮进行初始化
        //组件宽度可能需要调整下,根据使用的theme不同,宽度需要调整
        me.hourBtn=new _$Number({
            minValue:0,
            maxValue:23,
            step:1,
            width:55
        });
        me.minuteBtn=new _$Number({
            minValue:0,
            maxValue:59,
            step:1,
            width:70,
            labelWidth:10,
            fieldLabel:'&nbsp;'
        });
        me.secondBtn=new _$Number({
            minValue:0,
            maxValue:59,
            step:1,
            width:70,
            labelWidth:10,
            fieldLabel:'&nbsp;'//在组件之前渲染 ':'
        });

        me.okBtn = new Ext.button.Button({
            ownerCt: me,
            ownerLayout: me.getComponentLayout(),
            text: me.okText,
            tooltip: me.okTip,
            tooltipType:'title',
            handler:me.okHandler,//确认按钮的事件委托
            scope: me
        });
        me.callParent();
    },
    
    finishRenderChildren: function () {
        var me = this;
        //组件渲染完成后,需要调用子元素的finishRender,从而获得事件绑定
        me.hourBtn.finishRender();
        me.minuteBtn.finishRender();
        me.secondBtn.finishRender();
        me.okBtn.finishRender();
        me.callParent();
    },

    /**
     * 确认 按钮触发的调用
     */
    okHandler : function(){
        var me = this,
            btn = me.okBtn;

        if(btn && !btn.disabled){
            me.setValue(this.getValue());
            me.fireEvent('select', me, me.value);
            me.onSelect();
        }
        return me;
    },

    /**
     * 覆盖了父类的方法,因为父类中是根据时间的getTime判断的,因此需要对时、分、秒分别值为0才能保证当前值的日期选择
     * @private
     * @param {Date} date The new date
     */
    selectedUpdate: function(date){
        this.callParent([Ext.Date.clearTime(date,true)]);
    },

    /**
     * 更新picker的显示内容,需要同时更新时、分、秒输入框的值
     * @private
     * @param {Date} date The new date
     * @param {Boolean} forceRefresh True to force a full refresh
     */
    update : function(date, forceRefresh){
        var me = this;
        me.hourBtn.setValue(date.getHours());
        me.minuteBtn.setValue(date.getMinutes());
        me.secondBtn.setValue(date.getSeconds());

        return this.callParent(arguments);
    },

    /**
     * 从picker选中后,赋值时,需要从时、分、秒也获得当前值
     * datetimefield也会调用这个方法对picker初始化,因此添加一个isfixed参数。
     * @param {Date} date The new date
     * @param {Boolean} isfixed True 时,忽略从时分秒中获取值
    */
    setValue : function(date, isfixed){
        var me = this;
        if(isfixed!==true){
            date.setHours(me.hourBtn.getValue());
            date.setMinutes(me.minuteBtn.getValue());
            date.setSeconds(me.secondBtn.getValue());
        }
        me.value=date;
        me.update(me.value);
        return me;
    },

    // @private
    // @inheritdoc
    beforeDestroy : function() {
        var me = this;

        if (me.rendered) {
            //销毁组件时,也需要销毁自定义的控件
            Ext.destroy(
                me.hourBtn,
                me.minuteBtn,
                me.secondBtn,
                me.okBtn
            );
        }
        me.callParent();
    }
},
function() {
    var proto = this.prototype,
        date = Ext.Date;

    proto.monthNames = date.monthNames;
    proto.dayNames   = date.dayNames;
    proto.format     = date.defaultFormat;
});

 

2、自定义DateTime Field输入控件。

/**
 * 带时间的日期输入控件
 * 转载请注明来自于gogo1217.iteye.com
 */
Ext.define('Go.form.field.DateTime', {
    extend:'Ext.form.field.Date',
    alias: 'widget.datetimefield',
    requires: ['Go.picker.DateTime'],

    /**
     * @cfg {String} format
     * The default date format string which can be overriden for localization support. The format must be valid
     * according to {@link Ext.Date#parse}.
     */
    format : "Y-m-d H:i:s",
 
    /**
     * @cfg {String} altFormats
     * Multiple date formats separated by "|" to try when parsing a user input value and it does not match the defined
     * format.
     */
    altFormats : "Y-m-d H:i:s",

    createPicker: function() {
        var me = this,
            format = Ext.String.format;

        //修改picker为自定义picker
        return new Go.picker.DateTime({
            pickerField: me,
            ownerCt: me.ownerCt,
            renderTo: document.body,
            floating: true,
            hidden: true,
            focusOnShow: true,
            minDate: me.minValue,
            maxDate: me.maxValue,
            disabledDatesRE: me.disabledDatesRE,
            disabledDatesText: me.disabledDatesText,
            disabledDays: me.disabledDays,
            disabledDaysText: me.disabledDaysText,
            format: me.format,
            showToday: me.showToday,
            startDay: me.startDay,
            minText: format(me.minText, me.formatDate(me.minValue)),
            maxText: format(me.maxText, me.formatDate(me.maxValue)),
            listeners: {
                scope: me,
                select: me.onSelect
            },
            keyNavConfig: {
                esc: function() {
                    me.collapse();
                }
            }
        });
    },

    /**
     * @private
     */
    onExpand: function() {
        var value = this.getValue();

        //多传一个参数,从而避免时分秒被忽略。
        this.picker.setValue(Ext.isDate(value) ? value : new Date(), true);
    }
});

 3、测试示例代码:

<html>
<head>
    <title>Ext JS 4.2 Examples</title>
    <meta http-equiv="Context-Type" content="text/html;chartset=utf-8">

    <link rel="stylesheet" type="text/css" href="resources/ext-theme-neptune/ext-theme-neptune-all.css" />
    <script type="text/javascript" src="ext-all.js"></script>
    <script type="text/javascript" src="locale/ext-lang-zh_CN.js"></script>
    <script type="text/javascript">
       //启动ExtJs4动态加载特性
    	Ext.Loader.setConfig({
    		enabled:true,
    		disableCaching:false,
    		paths:{
    			'Go':'Go/'
    		}
    	});
    	Ext.onReady(function(){
    		Ext.create('Go.form.field.DateTime',{
                renderTo:Ext.getBody(),
    			fieldLabel:'日期选择器',
    			value:'2013-04-27 12:12:12',
    			format:'Y-m-d H:i:s'
    		});
    	});
    </script>
</head>
<body>
</body>
</html>

 3、效果:

 

4、附代码:下载 

 

  • 大小: 29.5 KB
8
0
分享到:
评论
20 楼 beinnerstrong 2017-01-16  
楼主,你好,乱码的问题原来是引入js文件顺序的问题,应该先引入中文包。。。。另外一个问题想问问楼主,这样做好的一个组件,怎么给它引入到系统中啊???
19 楼 beinnerstrong 2017-01-15  
楼主,你好!我下载了ext4.2.1,然后把ext包跟你这三个文件放到同一级目录下,修改了引用路径,最后用Firefox调试怎么有乱码呢?乱码还不少!
18 楼 zjhdreams 2015-01-12  
有bug,就是在选中时间后,回显时如果选择其他的日期,再换时间,回显就出问题了,而且datetime空间好像没一个月的日期都是选中的
17 楼 跃逸冰颠 2014-11-27  

附件文件放到哪里可以运行?test.html无法运行
16 楼 我就是小娜娜 2014-04-08  
真好用,要是能加上minvalue 和maxvalue就好了呢,赞赞赞
15 楼 yuwang115 2014-03-24  
正好要用到,非常感谢!
14 楼 okooo00 2013-12-03  
测试可用,十分感谢,如果可以做到点确定选时间而不是选择日期就选的时间就好了,感觉确定按钮有点鸡肋,我去掉了,哈哈
13 楼 ming7879 2013-10-21  
lastanimals 写道
我这是进到ext-all-debug.js里去报了一个TypeError: name is undefined错误


这个是因为没预加载楼主的组件,把组件required一下

Ext.require([ 
    		'Ext.grid.*', 
	    	'Ext.data.*', 
	    	'Ext.util.*', 
	    	'Ext.state.*',
	    	[color=red]'Go.form.field.DateTime'[/color]
    	]);

12 楼 lastanimals 2013-09-13  
我这是进到ext-all-debug.js里去报了一个TypeError: name is undefined错误
11 楼 gogo1217 2013-08-29  
请使用4.2.1试试,这个错误是什么操作后发生的。
我本机测试是没有任何问题的。
lvpeiqiang 写道
TypeError: d[w] is undefined

10 楼 lvpeiqiang 2013-08-28  
TypeError: d[w] is undefined
9 楼 gogo1217 2013-08-21  
IE几?
cherish77 写道
ie用不了 是不是我这边哪里没弄对?您那边可以不?求教、

8 楼 cherish77 2013-08-21  
ie用不了 是不是我这边哪里没弄对?您那边可以不?求教、
7 楼 sunhualin11y 2013-07-02  
sunhualin11y 写道
4.2版本下用不了

搞了三天终于解决了问题。楼主,按照你的思路自己弄写了一个。你的控件真用不了。不过还是要谢谢你。
6 楼 sunhualin11y 2013-07-02  
gogo1217 写道
不会的,你引入的ExtJS的路径对不对,如果有错误用调试工具,如firebug看看。
sunhualin11y 写道
4.2版本下用不了


路径应该是对的。控件能显示出来,但是一点击日历图标,就报“SCRIPT5007: 无法设置属性“innerHTML”的值: 对象为 null 或未定义 ” 的错误
5 楼 gogo1217 2013-07-01  
不会的,你引入的ExtJS的路径对不对,如果有错误用调试工具,如firebug看看。
sunhualin11y 写道
4.2版本下用不了

4 楼 sunhualin11y 2013-06-29  
4.2版本下用不了
3 楼 gogo1217 2013-05-09  
不好意思,在动态加载那里,写错了,
Ext.Loader.setConfig({ 
            enabled:true, 
            disableCaching:false, 
            paths:{ 
                'Go':'Go/' 
            } 
        }); 
原有的代码enable 改为enabled,path为paths;正文已经修改,附件已经重传。
2 楼 j1017631563 2013-05-04  
    
1 楼 zhanglu_king 2013-04-28  
[img][/img]

相关推荐

    EXTjs4.2中文版

    同时,EXTjs4.2提供了强大的布局管理器,支持多种布局模式,如fit、form、border等,使界面布局更加灵活。 在数据处理方面,EXTjs4.2加强了数据绑定和模型(Model)的概念,允许开发者方便地处理和展示后台数据。它...

    ExtJS4.2 tree 级联选择

    ExtJS4.2 Tree 级联选择是一个用于构建用户界面的功能,特别是在处理层次结构数据时非常有用。在ExtJS库中,Tree组件允许我们展示和操作数据以树形结构显示,而级联选择功能则意味着当用户选择一个节点时,它的所有...

    SpringMVC+ExtJs4.2实例

    在本实践项目“SpringMVC+ExtJs4.2实例”中,我们将深入探讨如何将SpringMVC框架与ExtJs4.2前端框架相结合,构建一个功能完善的Web应用程序。这个项目旨在展示如何利用这两个强大的技术栈来实现数据的动态交互和用户...

    MVC设计模式实战ExtJS4.2高级组件+SSH2在线投稿系统

    01.教程简介_ExtJS4.2简介_SSH2基本框架搭建 02.编写几个通用的service方法、设计数据库 03.搭建ExtJS的MVC框架 04.主界面的搭建、登录功能和菜单树的生成 05.创建菜单树、前台保存用户信息 06.菜单树响应事件、我的...

    Extjs4.2 Grid filter Demo 表格过滤实验

    这是利用sencha cmd 生成的GridFilterDemo工程中的app和build文件夹,其余文件过大并且与主题无关,因此未包含。具体方法,请参看我的博客: 《Extjs4.2 Grid Filter Feature 表格过滤特性》

    ExtJs4.2下拉树(修改版)

    ExtJs4.2没有直接提供下拉树这个组件,但是有例子可以用,文件位置:ext-4.2.1.883\examples\ux\TreePicker.js 但是它有点小毛病吧:默认显示了根节点;达到最小高度时再展开节点,高度不能自动调整。 所以我做了一...

    基于Extjs 4.2的通用权限管理系统,通用后台模板,EF+MVC+Extjs 4.2

    WMC2.0-Client.zip是一个基于Extjs4.2的开发框架,其实是个只有大框架的,并没有其他功能,您可能会骂我标题党“通用权限管理系统,通用后台模板”,呵呵,其实不是这样的。 整个WMC系统分为WMC2.0-Server服务端...

    EXTJS4.2学习入门教程

    EXTJS4.2学习入门教程 EXTJS4.2学习入门教程 EXTJS4.2学习入门教程

    extjs 4.2 开发 参考书

    在压缩包中的`ext-4.2.1.883`文件可能包含了ExtJS 4.2.1的源代码和资源文件,这将有助于你深入研究每个组件的内部工作原理,调试代码,甚至自定义和扩展框架功能。通过结合书籍内容和源码学习,将能更全面地掌握...

    extjs 4.2 多文件上传

    用Ext编写的多文件上传组件,已封装。 支持多文件上传,文件下载,文件删除,

    extjs4.2官方文档

    Extjs官方文档 帮助你更好的学习Extjs,同事这里面的代码是最完整,最规范的。

    Extjs4.2中文教程

    Extjs4.2入门教程详解,及API文档。

    ExtJs4.2 Window常用方法

    根据提供的文件信息,本文将详细解释ExtJs 4.2中Window组件的一些常用配置属性以及方法,帮助读者更好地理解和使用这些功能。 ### ExtJs 4.2 Window 组件概述 ExtJs 是一个基于 JavaScript 的开源框架,用于创建...

    ExtJs4.2正式版

    ExtJs4.2正式版

    extjs4.2 desktop mvc

    EXTJS 4.2 Desktop MVC 是一个基于EXTJS 4.2版本的桌面应用程序框架,它结合了MVC(Model-View-Controller)设计模式,为开发者提供了构建富客户端桌面应用的强大工具。EXTJS是一个流行的JavaScript库,专门用于创建...

    extjs 4.2 jsb2

    extjs 4.2 jsb2 4.2没有自带jsb2文件

    nodejs+extjs4.2+mysql

    标题 "nodejs+extjs4.2+mysql" 暗示了这是一个使用 Node.js、ExtJS 4.2 和 MySQL 数据库构建的项目。这个项目的核心是利用这些技术搭建了一个基本的框架,使得开发者可以方便地在此基础上添加自己的业务逻辑和功能。...

    ExtJS4.2入门案例

    ExtJS4.2入门案例 博客:http://blog.csdn.net/coco2d_x2014/article/details/52986835

Global site tag (gtag.js) - Google Analytics