事件是指一個特定的動做,這個動做能夠針對HTML元素的,如keydown,keyup,mouseover, mouseout等,也能夠是對於其它自定義的動做,如對Ajax異步請求的響應等。在ExtJS中,該如何處理呢?瀏覽器
1. 處理HTML元素的標準事件app
HTML元素的標準事件是指mouseover、mousedown、click、blur、focus、change等可以直接對HTML元素髮生的事件。在ExtJS中,這些事件的處理能夠用以下的代碼:異步
註冊一個事件處理函數使用: Ext.get('myElement').on('click', myHandler, myScope) myElement是要註冊的元素的ID,click是事件的名稱(注意,和HTML元素中的聲明onide
XXX不一樣,這裏不須要on),myHandler 是處理函數的函數名稱,myScope是一個可選的參數,指定處理函數綁定的對象,也就是處理函數的做用域,若是不提供這個參數,則是默認的 window。函數
撤銷一個事件處理函數: Ext.get('myElement').un('click', myHandler, myScope) 參數的意義同上。ui
ExtJS會根據不一樣的瀏覽器進行相應的處理,根本不須要理會用戶用的是什麼瀏覽器。根據事件的不一樣,傳給處理函數的參數也會不一樣,這個只能參考ExtJS的文檔了,必要時還得參考源代碼。this
2. 處理自定義事件spa
在ExtJS中使用自定義事件,須要從Ext.util.Observable繼承,示例代碼以下:code
Employee = function(name){ this.name = name; this.addEvents({ "fired" : true, "quit" : true }); } Ext.extend(Employee, Ext.util.Observable, { ... });
在這段代碼中,定義了一個Employee類,定義了fired和quit兩個事件。如何觸發這兩個事件呢,基類Ext.util.Observable提供了觸發自定義事件的方法fireEvent(eventName, arg1, arg2, ... argn), eventName是要觸發的時間的名稱(不區分大小寫),後面的參數arg1,arg2等是要傳給事件處理函數的參數。用上面的Employee類作示例,觸發quit事件:component
this.fireEvent('quit', this);
這行代碼將觸發quit事件,並將Empolyee類的實例傳給quit事件的處理函數,quit事件的訂閱能夠採用以下代碼:
function myHandler1(empolyee){ ... } function myHandler2(empolyee){ ... } var emp = new Empolyee('tom'); emp.on('quit', myHandler1); emp.on('quit', myHandler2);
在上面的代碼中,爲quit事件註冊了兩個處理函數(myHandler1與myHandler),當quit事件被激發時,將會依次調用myHandler1和myHandler2兩個函數。
值得注意的是,無論是HTML元素的標準事件仍是自定義事件,若是爲某個時間註冊了多個處理函數,如前面的例子,若是myHandler1返回false 的話,則會取消在myHandler1以後註冊的處理函數的執行,即該事件被取消,從而中止繼續執行該事件的處理函數,而這個返回值false會做爲事件 激發的結果,返回給empolyee,即:
var result = this.fireEvent('quit', this); if (result === false) { alert('event canceled'); //這裏表示事件被某個處理函數取消 } else { alert('event complete'); // 這裏表示事件執行完畢 }
經過Ext的自定義事件的機制,能夠實現一對多的觀察者模式,也能夠實現一對一的綁定模式,這一點,在ExtJS的開發中是很重要的。
參考官方相關實現類:
/** * @class Ext.calendar.form.EventWindow * @extends Ext.Window * <p>A custom window containing a basic edit form used for quick editing of events.</p> * <p>This window also provides custom events specific to the calendar so that other calendar components can be easily * notified when an event has been edited via this component.</p> * @constructor * @param {Object} config The config object */ Ext.define('Ext.calendar.form.EventWindow', { extend: 'Ext.window.Window', alias: 'widget.eventeditwindow', requires: [ 'Ext.form.Panel', 'Ext.calendar.util.Date', 'Ext.calendar.data.EventModel', 'Ext.calendar.data.EventMappings' ], constructor: function(config) { var formPanelCfg = { xtype: 'form', fieldDefaults: { msgTarget: 'side', labelWidth: 65 }, frame: false, bodyStyle: 'background:transparent;padding:5px 10px 10px;', bodyBorder: false, border: false, items: [{ itemId: 'title', name: Ext.calendar.data.EventMappings.Title.name, fieldLabel: 'Title', xtype: 'textfield', allowBlank: false, emptyText: 'Event Title', anchor: '100%' }, { xtype: 'daterangefield', itemId: 'date-range', name: 'dates', anchor: '100%', fieldLabel: 'When' }] }; if (config.calendarStore) { this.calendarStore = config.calendarStore; delete config.calendarStore; formPanelCfg.items.push({ xtype: 'calendarpicker', itemId: 'calendar', name: Ext.calendar.data.EventMappings.CalendarId.name, anchor: '100%', store: this.calendarStore }); } this.callParent([Ext.apply({ titleTextAdd: 'Add Event', titleTextEdit: 'Edit Event', width: 600, autocreate: true, border: true, closeAction: 'hide', modal: false, resizable: false, buttonAlign: 'left', savingMessage: 'Saving changes...', deletingMessage: 'Deleting event...', layout: 'fit', defaultFocus: 'title', onEsc: function(key, event) { event.target.blur(); // Remove the focus to avoid doing the validity checks when the window is shown again. this.onCancel(); }, fbar: [{ xtype: 'tbtext', text: '<a href="#" id="tblink">Edit Details...</a>' }, '->', { itemId: 'delete-btn', text: 'Delete Event', disabled: false, handler: this.onDelete, scope: this, minWidth: 150, hideMode: 'offsets' }, { text: 'Save', disabled: false, handler: this.onSave, scope: this }, { text: 'Cancel', disabled: false, handler: this.onCancel, scope: this }], items: formPanelCfg }, config)]); }, // private newId: 10000, // private initComponent: function() { this.callParent(); this.formPanel = this.items.items[0]; this.addEvents({ /** * @event eventadd * Fires after a new event is added * @param {Ext.calendar.form.EventWindow} this * @param {Ext.calendar.EventRecord} rec The new {@link Ext.calendar.EventRecord record} that was added */ eventadd: true, /** * @event eventupdate * Fires after an existing event is updated * @param {Ext.calendar.form.EventWindow} this * @param {Ext.calendar.EventRecord} rec The new {@link Ext.calendar.EventRecord record} that was updated */ eventupdate: true, /** * @event eventdelete * Fires after an event is deleted * @param {Ext.calendar.form.EventWindow} this * @param {Ext.calendar.EventRecord} rec The new {@link Ext.calendar.EventRecord record} that was deleted */ eventdelete: true, /** * @event eventcancel * Fires after an event add/edit operation is canceled by the user and no store update took place * @param {Ext.calendar.form.EventWindow} this * @param {Ext.calendar.EventRecord} rec The new {@link Ext.calendar.EventRecord record} that was canceled */ eventcancel: true, /** * @event editdetails * Fires when the user selects the option in this window to continue editing in the detailed edit form * (by default, an instance of {@link Ext.calendar.EventEditForm}. Handling code should hide this window * and transfer the current event record to the appropriate instance of the detailed form by showing it * and calling {@link Ext.calendar.EventEditForm#loadRecord loadRecord}. * @param {Ext.calendar.form.EventWindow} this * @param {Ext.calendar.EventRecord} rec The {@link Ext.calendar.EventRecord record} that is currently being edited */ editdetails: true }); }, // private afterRender: function() { this.callParent(); this.el.addCls('ext-cal-event-win'); Ext.get('tblink').on('click', this.onEditDetailsClick, this); this.titleField = this.down('#title'); this.dateRangeField = this.down('#date-range'); this.calendarField = this.down('#calendar'); this.deleteButton = this.down('#delete-btn'); }, // private onEditDetailsClick: function(e){ e.stopEvent(); this.updateRecord(this.activeRecord, true); this.fireEvent('editdetails', this, this.activeRecord, this.animateTarget); }, /** * Shows the window, rendering it first if necessary, or activates it and brings it to front if hidden. * @param {Ext.data.Record/Object} o Either a {@link Ext.data.Record} if showing the form * for an existing event in edit mode, or a plain object containing a StartDate property (and * optionally an EndDate property) for showing the form in add mode. * @param {String/Element} animateTarget (optional) The target element or id from which the window should * animate while opening (defaults to null with no animation) * @return {Ext.Window} this */ show: function(o, animateTarget) { // Work around the CSS day cell height hack needed for initial render in IE8/strict: var me = this, anim = (Ext.isIE8 && Ext.isStrict) ? null: animateTarget, M = Ext.calendar.data.EventMappings; this.callParent([anim, function(){ me.titleField.focus(true); }]); this.deleteButton[o.data && o.data[M.EventId.name] ? 'show': 'hide'](); var rec, f = this.formPanel.form; if (o.data) { rec = o; this.setTitle(rec.phantom ? this.titleTextAdd : this.titleTextEdit); f.loadRecord(rec); } else { this.setTitle(this.titleTextAdd); var start = o[M.StartDate.name], end = o[M.EndDate.name] || Ext.calendar.util.Date.add(start, {hours: 1}); rec = Ext.create('Ext.calendar.data.EventModel'); rec.data[M.StartDate.name] = start; rec.data[M.EndDate.name] = end; rec.data[M.IsAllDay.name] = !!o[M.IsAllDay.name] || start.getDate() != Ext.calendar.util.Date.add(end, {millis: 1}).getDate(); f.reset(); f.loadRecord(rec); } if (this.calendarStore) { this.calendarField.setValue(rec.data[M.CalendarId.name]); } this.dateRangeField.setValue(rec.data); this.activeRecord = rec; return this; }, // private roundTime: function(dt, incr) { incr = incr || 15; var m = parseInt(dt.getMinutes(), 10); return dt.add('mi', incr - (m % incr)); }, // private onCancel: function() { this.cleanup(true); this.fireEvent('eventcancel', this); }, // private cleanup: function(hide) { if (this.activeRecord && this.activeRecord.dirty) { this.activeRecord.reject(); } delete this.activeRecord; if (hide === true) { // Work around the CSS day cell height hack needed for initial render in IE8/strict: //var anim = afterDelete || (Ext.isIE8 && Ext.isStrict) ? null : this.animateTarget; this.hide(); } }, // private updateRecord: function(record, keepEditing) { var fields = record.fields, values = this.formPanel.getForm().getValues(), name, M = Ext.calendar.data.EventMappings, obj = {}; fields.each(function(f) { name = f.name; if (name in values) { obj[name] = values[name]; } }); var dates = this.dateRangeField.getValue(); obj[M.StartDate.name] = dates[0]; obj[M.EndDate.name] = dates[1]; obj[M.IsAllDay.name] = dates[2]; record.beginEdit(); record.set(obj); if (!keepEditing) { record.endEdit(); } return this; }, // private onSave: function(){ if(!this.formPanel.form.isValid()){ return; } if(!this.updateRecord(this.activeRecord)){ this.onCancel(); return; } this.fireEvent(this.activeRecord.phantom ? 'eventadd' : 'eventupdate', this, this.activeRecord, this.animateTarget); // Clear phantom and modified states. this.activeRecord.commit(); }, // private onDelete: function(){ this.fireEvent('eventdelete', this, this.activeRecord, this.animateTarget); } });