/** * Project : AMDA-NG * Name : TimeTableUI.js * @class amdaUI.TimeTableUI * @extends Ext.container.Container * @brief Time Table Module UI definition (View) * @author Myriam * @version $Id: TimeTableUI.js 2075 2014-02-11 11:30:14Z elena $ */ Ext.define('amdaUI.TimeTableUI', { extend: 'Ext.container.Container', alias: 'widget.panelTimeTable', requires: [ 'Ext.ux.grid.FiltersFeature', 'Ext.ux.grid.filter.DateFilter', 'Ext.ux.grid.filter.NumericFilter', 'amdaUI.OperationsTT', 'amdaUI.StatisticalPlug', 'Ext.grid.plugin.BufferedRenderer' ], statics: { COL_TO_HIDE: 'colToHide' }, status: null, constructor: function (config) { this.init(config); this.callParent(arguments); // load object into view this.loadObject(); }, /** * set the current editing object * this method will be used on timetable edition when this win is already opened */ setObject: function (object) { // set object this.object = object; // load object into view this.loadObject(); // show the default duration column this.TTGrid.headerCt.getGridColumns(); Ext.Array.each(this.TTGrid.headerCt.getGridColumns(), function (item, index, all) { // if item is the default duration column if (item.id == amdaUI.TimeTableUI.COL_TO_HIDE + '2') { // show this column item.show(); } }); // fire the refresh event (to statistical plugin) this.fireEvent("refresh"); // global event myDesktopApp.EventManager.fireEvent("refresh"); }, /** * load object timetable into this view */ loadObject: function () { this.object.set('created', this.convertUTCDateToLocalDate(this.object.get('created'))); // load object into form this.formPanel.getForm().loadRecord(this.object); this.status = null; // var me = this; var onAfterInit = function (result, e) { if (!result || !result.success) { if (result.message) myDesktopApp.errorMsg(result.message); else myDesktopApp.errorMsg('Unknown error during cache initialisation'); return; } me.TTGrid.getSelectionModel().deselectAll(); // clear filters me.TTGrid.getStore().clearFilter(true); me.TTGrid.filters.clearFilters(); //clear sort me.TTGrid.getStore().sorters.clear(); //me.TTGrid.getStore().sorters = new Ext.util.MixedCollection(); //set cache token to the Time Table object me.object.set('cacheToken', result.token); me.TTGrid.getStore().load(); me.status = result.status; //Statistical plugin me.fireEvent("refresh"); }; if (this.object.get('fromPlugin')) { if (this.object.get('objFormat') && this.object.get('objFormat') != '') { //From uploaded file AmdaAction.initObjectCacheFromUploadedFile(this.object.get('objName'), this.object.get('objFormat'), false, onAfterInit); } else { //From tmp object (ie Search result) AmdaAction.initObjectCacheFromTmpObject(this.object.get('folderId'), this.object.get('objName'), false, onAfterInit); } } else if (this.object.get('relatedCatalogId') != '') { var pathern = this.object.get('relatedCatalogId').split('_')[0]; catType = ''; if (pathern == 'cat') catType = ' catalog'; if (pathern == 'sharedcatalog') catType = 'sharedcatalog'; AmdaAction.initObjectCacheFromCatalog(this.object.get('relatedCatalogId'), catType, onAfterInit); } else { var typeTT = myDesktopApp.getLoadedModule(myDesktopApp.dynamicModules.tt.id).linkedNode.data.nodeType; if (this.object.get('id') == '') { //Init empty cache AmdaAction.initObjectCache(false, 0, onAfterInit); } else { //From existing TT file AmdaAction.initObjectCacheFromObject(this.object.get('id'), typeTT, onAfterInit); } } }, /** * update this.object from form */ updateObject: function () { this.updateCount(); // get the basic form var basicForm = this.formPanel.getForm(); var updateStatus = true; var fieldsWithoutName = basicForm.getFields().items; Ext.Array.each(fieldsWithoutName, function (item, index, allItems) { if (item !== this.fieldName) { if (!item.isValid()) { // set update isn't allowed updateStatus = false; } } }, this); // if the update is allowed if (updateStatus) { /// real object update // update TimeTable object with the content of form basicForm.updateRecord(this.object); } // return the update status return updateStatus; }, updateCount: function () { this.object.set('nbIntervals', this.TTGrid.getStore().getTotalCount()); this.formPanel.getForm().findField('nbIntervals').setValue(this.object.get('nbIntervals')); }, /* * save method called by Save button */ saveProcess: function (toRename, onAfterSave, notDisplayMsg) { var timeTableModule = myDesktopApp.getLoadedModule(myDesktopApp.dynamicModules.tt.id); // if the name has been modified this is a creation if (timeTableModule.contextNode && (timeTableModule.contextNode.data.id == 'sharedtimeTable-treeRootNode')) { timeTableModule.linkedNode = null; timeTableModule.createLinkedNode(); timeTableModule.createObject(this.object.getJsonValues()); var ttobj = timeTableModule.linkedNode.get('object'); // synchronisation of objects this.object = ttobj; timeTableModule.linkedNode.create(); } else if (this.fclose() || this.status && (this.status.nbFiltered > 0)) /*TimeTable object has been modified*/ { if (this.object.isModified('name') || this.object.get('fromPlugin')) { // if object already has an id : it's a 'rename' of an existing TimeTable if (this.object.get('id')) { // the context Node is the parent node of current edited one var contextNode = timeTableModule.linkedNode.parentNode; // link a new node to the TimeTableModule timeTableModule.createLinkedNode(); // set the contextNode timeTableModule.linkedNode.set('contextNode', contextNode); // create a new object linked timeTableModule.createObject(this.object.getJsonValues()); var ttobj = timeTableModule.linkedNode.get('object'); // synchronisation of objects this.object = ttobj; if (toRename) timeTableModule.linkedNode.toRename = true; } timeTableModule.linkedNode.create({callback: function ($action) { if (timeTableModule.linkedNode.get('object').get('fromPlugin')) timeTableModule.linkedNode.get('object').set('fromPlugin', false); timeTableModule.linkedNode.update({callback: function () { if (onAfterSave) onAfterSave(); }, scope: this}); }, scope: this}); } else { //update timeTableModule.linkedNode.update({callback: function () { if (onAfterSave) onAfterSave(); }, scope: this}); } } }, /** * overwrite metod called by Save button */ overwriteProcess: function (btn) { if (btn == 'cancel') return; this.fieldName.clearInvalid(); this.saveProcess(true); }, addInterval: function (start, stop) { var row = this.TTGrid.getStore().getTotalCount(); var me = this; this.TTGrid.getSelectionModel().deselectAll(); AmdaAction.addCacheInterval({'start': start, 'stop': stop, 'index': row}, function (result, e) { this.status = result.status; if (!this.TTGrid.getStore().loading) { this.TTGrid.getStore().reload({ callback: function (records, options, success) { me.TTGrid.getView().bufferedRenderer.scrollTo(row, false, function () { me.TTGrid.getView().select(row); }, me); } }); } }, this); }, generateCAT: function (id) { if (this.fclose()) { Ext.Msg.confirm('Generate CAT', 'Current Time Table has been modified.\nDo you want to save it to include these changes in the generated Catalog ?', function (btn, text) { if (btn == 'yes') { // mark this.closed as true before the call to close() as that will fire the beforeclose event again if (this.object.get('id') == "") { // case of creation of catalog this.saveTT(this.createCAT(this.object.get('id'))); } else { // case existing catalog this.saveProcess(false, this.createCAT(this.object.get('id'))); } return; } }, this); } else { this.createCAT(this.object.get('id')); return; } }, createCAT: function (id) { Ext.Msg.prompt('Create catalog', 'Enter the number of columns:', function (btn, text) { if (btn == 'ok') { var catObj = Ext.create('amdaModel.Catalog'); var catNode = Ext.create('amdaModel.CatalogNode', {leaf: true}); catObj.set('relatedTimeTableId', id); creatDate = new Date(this.object.get('created')); date = Ext.Date.format(creatDate, 'Y-m-d\\TH:i:s.u'); descr = 'Generated by CDPP/Amda Time Table Module \n' + 'From Time Table: ' + this.object.get('name') + ' which was generated on: ' + date + '\n'; catObj.set('description', descr + this.object.get('description')); var nbParam = parseInt(text); if ((nbParam <= 0) || (nbParam > 100)) { nbParam = 1; } catObj.set('nbParameters', nbParam); catNode.set('object', catObj); var explorerTree = Ext.getCmp(amdaUI.ExplorerUI.RESRC_TAB.TREE_ID); var catRootNode = explorerTree.getRootNode().findChild('id', 'catalog-treeRootNode', true); amdaModel.InteractiveNode.preloadNodes(catRootNode.getRootNode(), function () { // edit newNode into Parameter Module with node as contextNode catNode.editInModule(); }); } }, this); }, saveTT: function (OnAfterSave) { if (this.updateObject()) { var basicForm = this.formPanel.getForm(); // if there's at least one record in the store of TTGrid if (this.TTGrid.getStore().getTotalCount() > 0) { // update TimeTable object which the content of form basicForm.updateRecord(this.object); var me = this; this.checkIntervalsStatusForSave(function () { //Name validation var ttModule = myDesktopApp.getLoadedModule(myDesktopApp.dynamicModules.tt.id); if (!ttModule) return; ttModule.linkedNode.isValidName(me.fieldName.getValue(), function (res) { if (!res) { me.fieldName.validFlag = 'Error during object validation'; myDesktopApp.errorMsg(me.fieldName.validFlag); me.fieldName.validate(); return; } if (!res.valid) { if (res.error) { if (res.error.search('subtree') != -1) { Ext.MessageBox.show({title: 'Warning', msg: res.error + '<br/>Do you want to overwrite it?', width: 300, buttons: Ext.MessageBox.OKCANCEL, fn: me.overwriteProcess, icon: Ext.MessageBox.WARNING, scope: me }); me.fieldName.validFlag = true; } else me.fieldName.validFlag = res.error; } else { me.fieldName.validFlag = 'Invalid object name'; myDesktopApp.errorMsg(me.fieldName.validFlag); } me.fieldName.validate(); return; } me.fieldName.validFlag = true; me.fieldName.validate(); me.saveProcess(false); }); }); } else { // warning: Ext.Msg.alert('No intervals', 'Your time table is invalid, <br>you must have at least one interval'); } } }, // Convert UTC date to client local date convertUTCDateToLocalDate: function (date) { if (date == null) { return date; }; var newDate = new Date(date.getTime() + date.getTimezoneOffset() * 60 * 1000); var offset = date.getTimezoneOffset() / 60; var hours = date.getHours(); newDate.setHours(hours - offset); return newDate; }, init: function (config) { this.object = config.object; this.fieldName = new Ext.form.field.Text({ fieldLabel: 'Name*', allowBlank: false, stripCharsRe: /(^\s+|\s+$)/g, emptyText: 'Please no spaces!', name: 'name', anchor: '100%', validateOnChange: false, validateOnBlur: false, validFlag: false, validator: function () { return this.validFlag; } }); this.formPanel = new Ext.form.Panel({ bodyStyle: {background: '#dfe8f6'}, id: 'formTimeTable', flex: 4, model: 'amdaModel.TimeTable', trackResetOnLoad: true, // reset to the last loaded record border: false, fieldDefaults: {labelWidth: 80}, items: [ this.fieldName, { xtype: 'fieldcontainer', layout: 'hbox', fieldLabel: 'Creation date', items: [ { xtype: 'datefield', width: 180, name: 'created', disabled: true, hideTrigger: true, format: 'Y/m/d H:i:s' }, {xtype: 'component', width: 20}, {xtype: 'displayfield', value: 'Intervals:', width: 50}, {xtype: 'component', width: 8}, {xtype: 'textfield', name: 'nbIntervals', disabled: true, width: 70} ] }, { xtype: 'textarea', name: 'description', fieldLabel: 'Description', anchor: '100% 50%' }, { xtype: 'textarea', name: 'history', fieldLabel: 'Operation log', anchor: '100% 30%' } ] }); var store = Ext.create('Ext.data.Store', { model: 'amdaModel.Interval', autoDestroy: false, pageSize: 200, buffered: true, autoLoad: true, purgePageCount: 0, remoteSort: true, listeners: { load: function (store, records) { // alert('nb of records in store:'+records.length ); myDesktopApp.EventManager.fireEvent('refresh'); this.TTGrid.getView().refresh(); this.TTGrid.getSelectionModel().refresh(); this.updateCount(); //Statistical plugin this.fireEvent("refresh"); }, prefetch: function (store, records, successful, operation, eOpts) { if (operation && (operation.action == 'read')) { if (operation.response && operation.response.result && operation.response.result.success) this.status = operation.response.result.status; } }, remove: function (store) { this.updateCount(); //Statistical plugin this.fireEvent("refresh"); }, add: function (store) { this.updateCount(); //Statistical plugin this.fireEvent("refresh"); }, datachanged: function (store) { this.updateCount(); //Statistical plugin this.fireEvent("refresh"); }, scope: this } }); var filters = { ftype: 'filters', encode: true, // json encode the filter query local: false, // defaults to false (remote filte filters: [ {type: 'numeric', dataIndex: 'durationDay'}, {type: 'numeric', dataIndex: 'durationHour'}, {type: 'numeric', dataIndex: 'durationMin'}, {type: 'numeric', dataIndex: 'durationSec'}, {type: 'date', dataIndex: 'start', dateFormat: 'c'}, {type: 'date', dataIndex: 'stop', dateFormat: 'c'} ] }; var cellEditing = Ext.create('Ext.grid.plugin.CellEditing', { onEditComplete: function (ed, value, startValue) { var me = this, activeColumn = me.getActiveColumn(), context = me.context, record; if (activeColumn) { record = context.record; me.setActiveEditor(null); me.setActiveColumn(null); me.setActiveRecord(null); context.value = value; if (!me.validateEdit()) { me.editing = false; return; } // Only update the record if the new value is different than the // startValue. When the view refreshes its el will gain focus if (!record.isEqual(value, startValue)) { var obj = { 'cacheId': record.get('cacheId'), 'isCatalog': false, 'data': {} }; obj['data'][activeColumn.dataIndex] = Ext.Date.format(value, 'Y-m-d\\TH:i:s.u'); //context.grid.getSelectionModel().deselectAll(); //Interval is modified on the server side me.editing = true; AmdaAction.modifyCacheInterval(obj, function (result, e) { var ttModule = myDesktopApp.getLoadedModule(myDesktopApp.dynamicModules.tt.id); if (ttModule) ttModule.getUiContent().status = result.status; if (!context.store.loading) { context.grid.getSelectionModel().deselectAll(); context.store.reload({ callback: function (records, options, success) { context.view.bufferedRenderer.scrollTo(context.rowIdx, true, function () { me.fireEvent('edit', me, context); me.editing = false; }, me); } }); } else { me.editing = false; } }, this); } else me.editing = false; } } }); this.TTGrid = Ext.create('Ext.grid.Panel', { store: store, features: [filters], columnLines: true, selModel: {pruneRemoved: false}, countDecimals: function (value) { if (Math.floor(value) === value) return 0; return value.toString().split(".")[1].length || 0; }, dateToString: function (value) { ndegits = this.countDecimals(value); if (ndegits <= 3) { return Ext.util.Format.number(value, '0.000'); } else if (value < 0.1) { return value.toExponential(3); } else { return Ext.util.Format.number(value, '0.000000'); } }, columns: [ { xtype: 'rownumberer', width: 50, renderer: function (value, metaData, record, row, col, store, gridView) { var msg = record.index + 1; if (record.get('isNew') || record.get('isModified')) { msg += ' *'; metaData.style = 'font-weight: bold' } return msg; } }, { header: 'Start Time', dataIndex: 'start', width: 120, editor: {xtype: 'datefield', allowBlank: false, hideTrigger: true, format: 'Y-m-d\\TH:i:s.u'}, renderer: function (value) { if (value != null) { if (Ext.isDate(value)) { return Ext.Date.format(value, 'Y-m-d\\TH:i:s.u'); } else { return Ext.Date.format(new Date(value), 'Y-m-d\\TH:i:s.u'); } } else { return value; } } }, { header: 'Stop Time', dataIndex: 'stop', width: 120, editor: {xtype: 'datefield', allowBlank: false, hideTrigger: true, format: 'Y-m-d\\TH:i:s.u'}, renderer: function (value) { if (value != null) { if (Ext.isDate(value)) { return Ext.Date.format(value, 'Y-m-d\\TH:i:s.u'); } else { return Ext.Date.format(new Date(value), 'Y-m-d\\TH:i:s.u'); } } else { return value; } } }, { header: 'Duration (day)', width: 120, dataIndex: 'durationDay', id: amdaUI.TimeTableUI.COL_TO_HIDE + '1', hidden: true, renderer: function (value) { return this.dateToString(value); }, listeners: { beforeshow: function () { Ext.Array.each(this.ownerCt.getGridColumns(), function (item, index, all) { // if item is a column to hide automatically if (Ext.util.Format.substr(item.id, 0, amdaUI.TimeTableUI.COL_TO_HIDE.length) == amdaUI.TimeTableUI.COL_TO_HIDE) { // if item isn't the column which is being declared and is not hidden if (item.id != amdaUI.TimeTableUI.COL_TO_HIDE + '1' && !item.isHidden()) { // hide this column item.hide(); } } }); } } }, { header: 'Duration (hour)', width: 120, dataIndex: 'durationHour', id: amdaUI.TimeTableUI.COL_TO_HIDE + '2', hidden: true, renderer: function (value) { return this.dateToString(value); }, listeners: { beforeshow: function () { Ext.Array.each(this.ownerCt.getGridColumns(), function (item, index, all) { // if item is a column to hide automatically if (Ext.util.Format.substr(item.id, 0, amdaUI.TimeTableUI.COL_TO_HIDE.length) == amdaUI.TimeTableUI.COL_TO_HIDE) { // if item isn't the column which is being declared and is not hidden if (item.id != amdaUI.TimeTableUI.COL_TO_HIDE + '2' && !item.isHidden()) { // hide this column item.hide(); } } }); } } }, { header: 'Duration (min)', width: 120, dataIndex: 'durationMin', id: amdaUI.TimeTableUI.COL_TO_HIDE + '3', renderer: function (value) { return this.dateToString(value); }, listeners: { beforeshow: function () { Ext.Array.each(this.ownerCt.getGridColumns(), function (item, index, all) { // if item is a column to hide automatically if (Ext.util.Format.substr(item.id, 0, amdaUI.TimeTableUI.COL_TO_HIDE.length) == amdaUI.TimeTableUI.COL_TO_HIDE) { // if item isn't the column which is being declared and is not hidden if (item.id != amdaUI.TimeTableUI.COL_TO_HIDE + '3' && !item.isHidden()) { // hide this column item.hide(); } } }); } } }, { header: 'Duration (sec)', width: 120, dataIndex: 'durationSec', id: amdaUI.TimeTableUI.COL_TO_HIDE + '4', hidden: true, renderer: function (value) { return Ext.util.Format.number(value, '0.000'); }, listeners: { beforeshow: function () { Ext.Array.each(this.ownerCt.getGridColumns(), function (item, index, all) { // if item is a column to hide automatically if (Ext.util.Format.substr(item.id, 0, amdaUI.TimeTableUI.COL_TO_HIDE.length) == amdaUI.TimeTableUI.COL_TO_HIDE) { // if item isn't the column which is being declared and is not hidden if (item.id != amdaUI.TimeTableUI.COL_TO_HIDE + '4' && !item.isHidden()) { // hide this column item.hide(); } } }); } } } ], frame: true, dockedItems: [{ xtype: 'toolbar', items: [{ iconCls: 'icon-add', scope: this, handler: function () { cellEditing.cancelEdit(); var selection = this.TTGrid.getView().getSelectionModel().getSelection()[0]; var row = 0; if (selection) row = store.indexOf(selection) + 1; this.TTGrid.getSelectionModel().deselectAll(); var me = this; AmdaAction.addCacheInterval({'index': row}, function (result, e) { this.status = result.status; if (!this.TTGrid.getStore().loading) { this.TTGrid.getStore().reload({ callback: function (records, options, success) { me.TTGrid.getView().bufferedRenderer.scrollTo(row, false, function () { me.TTGrid.getView().select(row); cellEditing.startEditByPosition({row: row, column: 1}); }, me); } }); } }, this); } }, { iconCls: 'icon-delete', disabled: true, itemId: 'delete', scope: this, handler: function () { var selection = this.TTGrid.getView().getSelectionModel().getSelection()[0]; if (selection) { var rowId = selection.get('cacheId'); this.TTGrid.getSelectionModel().deselectAll(); AmdaAction.removeTTCacheIntervalFromId(rowId, false, function (result, e) { this.status = result.status; if (!this.TTGrid.getStore().loading) { this.TTGrid.getStore().reload(); } }, this); } } }, '->', { text: 'Clear Filters', scope: this, handler: function () { this.TTGrid.getStore().clearFilter(true); this.TTGrid.filters.clearFilters(); } } ] }], plugins: [cellEditing, {ptype: 'bufferedrenderer'}], listeners: { scope: this, edit: function (editor, e) { if (e.record.get('stop') != null && e.record.get('start') != null) { e.record.set('durationHour', (e.record.get('stop') - e.record.get('start')) / 3600000.0); e.record.set('durationMin', (e.record.get('stop') - e.record.get('start')) / 60000.0); e.record.set('durationSec', (e.record.get('stop') - e.record.get('start')) / 1000.0); e.record.set('durationDay', (e.record.get('stop') - e.record.get('start')) / 3600000.0 / 24.0); // send refresh event to statistical plugin this.fireEvent("refresh"); } } } }); this.TTGrid.getSelectionModel().on('selectionchange', function (selModel, selections) { this.TTGrid.down('#delete').setDisabled(selections.length === 0); }, this); var myConf = { layout: 'border', defaults: {layout: 'fit', border: false}, items: [ { xtype: 'form', region: 'center', buttonAlign: 'left', bodyStyle: {background: '#dfe8f6'}, padding: '5 5 5 5', layout: {type: 'hbox', pack: 'start', align: 'stretch'}, items: [ { xtype: 'container', flex: 3.6, layout: {type: 'vbox', pack: 'start', align: 'stretch'}, items: [ this.formPanel, { xtype: 'operationsTT', parent: this, flex: 2.5, id: 'operation' } ] }, { xtype: 'container', border: false, padding: '0 0 5 15', flex: 4, layout: 'fit', items: [this.TTGrid] } ], fbar: [ { xtype: 'button', text: 'Save', width: 65, scope: this, handler: function () { this.saveTT(); } }, { xtype: 'button', text: 'Reset', width: 65, scope: this, handler: function () { var ttModule = myDesktopApp.getLoadedModule(myDesktopApp.dynamicModules.tt.id); ttModule.createLinkedNode(); ttModule.createObject(); this.setObject(ttModule.getLinkedNode().get('object')); } }, { xtype: 'button', text: 'Generate Catalog', width: 120, scope: this, handler: function () { this.generateCAT(this.object.get('id')); } } ] }, { xtype: 'panel', region: 'south', title: 'Information', collapsible: true, collapseMode: 'header', height: 100, autoHide: false, iconCls: 'icon-information', bodyStyle: 'padding:5px', loader: { autoLoad: true, url: helpDir + 'timetableHOWTO' } } ], plugins: [{ptype: 'statisticalPlugin'}] }; Ext.apply(this, Ext.apply(arguments, myConf)); }, checkIntervalsStatusForSave: function (onStatusOk) { if (this.status == null) return; if (this.status.nbValid <= 0) { myDesktopApp.errorMsg('Your time table is invalid, <br>you must have at least one valid interval'); return; } var msg = ''; if (this.status.nbInvalid > 0) msg += 'There are some invalid intervals. Only valid intervals will be saved!<br/>'; if (this.status.nbFiltered > 0) msg += 'There are some filtered intervals. Filtered intervals will not be saved!<br/>'; if (msg != '') { msg += 'Do you want to continue?'; Ext.Msg.show({ title: 'Warning!', msg: msg, buttons: Ext.Msg.OKCANCEL, fn: function (btnId) { if (btnId === 'cancel') { // cancel the save action } else { onStatusOk(); } }, scope: this, icon: Ext.Msg.WARNING }); return; } onStatusOk(); }, /** * Check if changes were made before closing window * @return true if changes */ fclose: function () { if (this.status == null) return false; var isDirty = this.formPanel.getForm().isDirty() || (this.status.isModified) || (this.status.nbModified > 0) || (this.status.nbNew > 0); return isDirty; } });