/* * * (c) Copyright Ascensio System SIA 2010-2020 * * This program is a free software product. You can redistribute it and/or * modify it under the terms of the GNU Affero General Public License (AGPL) * version 3 as published by the Free Software Foundation. In accordance with * Section 7(a) of the GNU AGPL its Section 15 shall be amended to the effect * that Ascensio System SIA expressly excludes the warranty of non-infringement * of any third-party rights. * * This program is distributed WITHOUT ANY WARRANTY; without even the implied * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. For * details, see the GNU AGPL at: http://www.gnu.org/licenses/agpl-3.0.html * * You can contact Ascensio System SIA at 20A-12 Ernesta Birznieka-Upisha * street, Riga, Latvia, EU, LV-1050. * * The interactive user interfaces in modified source and object code versions * of the Program must display Appropriate Legal Notices, as required under * Section 5 of the GNU AGPL version 3. * * Pursuant to Section 7(b) of the License you must retain the original Product * logo when distributing the program. Pursuant to Section 7(e) we decline to * grant you any rights under trademark law for use of our trademarks. * * All the Product's GUI elements, including illustrations and icon sets, as * well as technical writing content are licensed under the terms of the * Creative Commons Attribution-ShareAlike 4.0 International. See the License * terms at http://creativecommons.org/licenses/by-sa/4.0/legalcode * */ /** * ChartDataDialog.js * * Created by Julia Radzhabova on 06.07.2020 * Copyright (c) 2020 Ascensio System SIA. All rights reserved. * */ define([ 'common/main/lib/util/utils', 'common/main/lib/component/MetricSpinner', 'common/main/lib/component/ComboBox', 'common/main/lib/component/ListView', 'common/main/lib/view/AdvancedSettingsWindow' ], function () { 'use strict'; SSE.Views.ChartDataDialog = Common.Views.AdvancedSettingsWindow.extend(_.extend({ options: { contentWidth: 370, height: 490 }, initialize : function(options) { var me = this; _.extend(this.options, { title: this.textTitle, template: [ '
', '
', '
', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '', '
', '', '
', '
', '
', '', '
', '
', '', '', '', '
', '
', '
', '
', '
', '', '
', '', '
', '
', '', '
', '
', '
', '
', '
' ].join('') }, options); this.handler = options.handler; Common.Views.AdvancedSettingsWindow.prototype.initialize.call(this, this.options); this._noApply = true; this._changedProps = null; this.api = this.options.api; this.chartSettings = this.options.chartSettings; this.dataRangeValid = ''; this.dataDirect = 0; this.currentChartType = Asc.c_oAscChartTypeSettings.barNormal; }, render: function() { Common.Views.AdvancedSettingsWindow.prototype.render.call(this); var me = this; this.txtDataRange = new Common.UI.InputFieldBtn({ el : $('#chart-dlg-txt-range'), name : 'range', style : 'width: 100%;', btnHint : this.textSelectData, allowBlank : true, validateOnChange: true }); this.txtDataRange.on('button:click', _.bind(this.onSelectData_simple, this)); this.txtDataRange.on('changed:after', function(input, newValue, oldValue, e) { if (newValue !==me.dataRangeValid) { if (me.isRangeValid()) { me.dataRangeValid = newValue; me.txtDataRange.checkValidate(); me.chartSettings.putRange(me.dataRangeValid); me.api.asc_editChartDrawingObject(me.chartSettings); } else { me.txtDataRange.setValue(me.dataRangeValid); me.txtDataRange.checkValidate(); } } }); // Chart data this.seriesList = new Common.UI.ListView({ el: $('#chart-dlg-series-list', this.$window), store: new Common.UI.DataViewStore(), emptyText: '', scrollAlwaysVisible: true }); this.seriesList.onKeyDown = _.bind(this.onListKeyDown, this, 'series'); this.seriesList.on('item:select', _.bind(this.onSelectSeries, this)); this.seriesList.store.comparator = function(rec) { return rec.get("order"); }; this.btnAdd = new Common.UI.Button({ el: $('#chart-dlg-btn-add') }); this.btnAdd.on('click', _.bind(this.onAddSeries, this, false)); this.btnDelete = new Common.UI.Button({ el: $('#chart-dlg-btn-delete') }); this.btnDelete.on('click', _.bind(this.onDeleteSeries, this)); this.btnEdit = new Common.UI.Button({ el: $('#chart-dlg-btn-edit') }); this.btnEdit.on('click', _.bind(this.onEditSeries, this, false)); this.btnUp = new Common.UI.Button({ parentEl: $('#chart-dlg-btn-up'), cls: 'btn-toolbar', iconCls: 'caret-up', hint: this.textUp }); this.btnUp.on('click', _.bind(this.onMoveClick, this, true)); this.btnDown = new Common.UI.Button({ parentEl: $('#chart-dlg-btn-down'), cls: 'btn-toolbar', iconCls: 'caret-down', hint: this.textDown }); this.btnDown.on('click', _.bind(this.onMoveClick, this, false)); this.btnSwitch = new Common.UI.Button({ el: $('#chart-dlg-btn-switch') }); // this.btnSwitch.on('click', _.bind(this.onSwitch, this)); this.categoryList = new Common.UI.ListView({ el: $('#chart-dlg-category-list', this.$window), store: new Common.UI.DataViewStore(), emptyText: '', scrollAlwaysVisible: true }); this.categoryList.onKeyDown = _.bind(this.onListKeyDown, this, 'category'); this.btnEditCategory = new Common.UI.Button({ el: $('#chart-dlg-btn-category-edit') }); this.btnEditCategory.on('click', _.bind(this.onEditCategory, this, false)); this.afterRender(); }, afterRender: function() { this._setDefaults(this.chartSettings); }, show: function() { Common.Views.AdvancedSettingsWindow.prototype.show.apply(this, arguments); var me = this; _.delay(function(){ me.txtDataRange.cmpEl.find('input').focus(); },50); }, close: function () { this.api.asc_onCloseChartFrame(); Common.Views.AdvancedSettingsWindow.prototype.close.apply(this, arguments); }, _setDefaults: function (props) { var me = this; if (props ){ this.chartSettings = props; this.currentChartType = props.getType(); var value = props.getRange(); this.txtDataRange.setValue((value) ? value : ''); this.dataRangeValid = value; this.txtDataRange.validation = function(value) { if (_.isEmpty(value)) { return true; } var isvalid = me.api.asc_checkDataRange(Asc.c_oAscSelectionDialogType.Chart, value, false); return (isvalid==Asc.c_oAscError.ID.DataRangeError) ? me.textInvalidRange : true; }; this.dataDirect = props.getInColumns() ? 1 : 0; this.updateSeriesList(props.getSeries(), 0); var categories = props.getCatValues(), arr = []; var store = this.categoryList.store; for (var i = 0, len = categories.length; i < len; i++) { var item = categories[i], rec = new Common.UI.DataViewModel(); rec.set({ value: item }); arr.push(rec); } store.reset(arr); (len>0) && this.categoryList.selectByIndex(0); } this.updateButtons(); this.updateCatButtons(); }, getSettings: function () { this.chartSettings.putRange(this.txtDataRange.getValue()); return { chartSettings: this.chartSettings}; }, onDlgBtnClick: function(event) { var me = this; var state = (typeof(event) == 'object') ? event.currentTarget.attributes['result'].value : event; if (state == 'ok') { this.handler && this.handler.call(this, state, (state == 'ok') ? this.getSettings() : undefined); } this.close(); }, onPrimary: function() { this.onDlgBtnClick('ok'); return false; }, isRangeValid: function() { var isvalid; if (!_.isEmpty(this.txtDataRange.getValue())) { isvalid = this.api.asc_checkDataRange(Asc.c_oAscSelectionDialogType.Chart, this.txtDataRange.getValue(), true, this.dataDirect===0, this.currentChartType); if (isvalid == Asc.c_oAscError.ID.No) return true; } else return true; if (isvalid == Asc.c_oAscError.ID.StockChartError) { Common.UI.warning({msg: this.errorStockChart}); } else if (isvalid == Asc.c_oAscError.ID.MaxDataSeriesError) { Common.UI.warning({msg: this.errorMaxRows}); } else if (isvalid == Asc.c_oAscError.ID.MaxDataPointsError) Common.UI.warning({msg: this.errorMaxPoints}); else this.txtDataRange.cmpEl.find('input').focus(); return false; }, onSelectData_simple: function() { var me = this; if (me.api) { var props = me.chartSettings; var handlerDlg = function(dlg, result) { if (result == 'ok' && me.isRangeValid()) { me.dataRangeValid = dlg.getSettings(); me.txtDataRange.setValue(me.dataRangeValid); me.txtDataRange.checkValidate(); props.putRange(me.dataRangeValid); me.api.asc_setSelectionDialogMode(Asc.c_oAscSelectionDialogType.None); me.api.asc_editChartDrawingObject(props); } }; var win = new SSE.Views.CellRangeDialog({ handler: handlerDlg }).on('close', function() { me.show(); }); var xy = me.$window.offset(); me.hide(); win.show(xy.left + 160, xy.top + 125); win.setSettings({ api : me.api, range : (!_.isEmpty(me.txtDataRange.getValue()) && (me.txtDataRange.checkValidate()==true)) ? me.txtDataRange.getValue() : me.dataRangeValid, type : Asc.c_oAscSelectionDialogType.Chart }); } }, onSelectData: function() { var me = this; if (me.api) { var props = me.chartSettings, handlerDlg = function(dlg, result) { if (result == 'ok' && me.isRangeValid()) { props.putRange(dlg.getSettings()); me.api.asc_setSelectionDialogMode(Asc.c_oAscSelectionDialogType.None); me.api.asc_editChartDrawingObject(props); } }, validation = function(value) { var isvalid; if (!_.isEmpty(value)) { isvalid = me.api.asc_checkDataRange(Asc.c_oAscSelectionDialogType.Chart, value, true, !props.getInColumns(), me.currentChartType); if (isvalid == Asc.c_oAscError.ID.No) return true; } else return ''; if (isvalid == Asc.c_oAscError.ID.StockChartError) { return this.errorStockChart; } else if (isvalid == Asc.c_oAscError.ID.MaxDataSeriesError) { return this.errorMaxRows; } return this.txtInvalidRange; }; var win = new SSE.Views.CellRangeDialog({ handler: handlerDlg }).on('close', function() { me.show(); // me.api.asc_onCloseChartFrame(); }); var xy = me.$window.offset(); me.hide(); win.show(xy.left + 160, xy.top + 125); win.setSettings({ api : me.api, range : props.getRange(), validation: validation, type : Asc.c_oAscSelectionDialogType.Chart }); } }, onListKeyDown: function (type, e, data) { var record = null, listView = (type=='series') ? this.seriesList : this.categoryList; if (listView.disabled) return; if (_.isUndefined(undefined)) data = e; if (type=='series' && data.keyCode==Common.UI.Keys.DELETE && !this.btnDelete.isDisabled()) { // this.onDeleteSeries(); } else { Common.UI.DataView.prototype.onKeyDown.call(listView, e, data); } }, onSelectSeries: function(lisvView, itemView, record) { this.updateCatButtons(record.get('series').asc_IsScatter()); this.updateMoveButtons(); }, updateButtons: function() { // this.btnAdd.setDisabled(this.seriesList.store.length>63); this.btnEdit.setDisabled(this.seriesList.store.length<1); this.btnDelete.setDisabled(this.seriesList.store.length<1); this.updateMoveButtons(); }, updateCatButtons: function(isScatter) { if (isScatter===undefined) { var rec = this.seriesList.getSelectedRec(); rec && (isScatter = rec.get('series').asc_IsScatter()); } this.btnEditCategory.setDisabled(this.categoryList.store.length<1 || !!isScatter); }, updateMoveButtons: function() { var rec = this.seriesList.getSelectedRec(), index = rec ? this.seriesList.store.indexOf(rec) : -1; this.btnUp.setDisabled(index<1); this.btnDown.setDisabled(index<0 || index==this.seriesList.store.length-1); }, onAddSeries: function() { var rec = (this.seriesList.store.length>0) ? this.seriesList.store.at(this.seriesList.store.length-1) : null, isScatter = false; rec && (isScatter = rec.get('series').asc_IsScatter()); var handlerDlg = function(dlg, result) { if (result == 'ok') { var changedValue = dlg.getSettings(); if (isScatter) { this.chartSettings.addScatterSeries(changedValue.name, changedValue.valuesX, changedValue.valuesY); } else { this.chartSettings.addSeries(changedValue.name, changedValue.values); } this.updateSeriesList(this.chartSettings.getSeries(), this.seriesList.store.length-1); this.updateButtons(); } }; this.changeDataRange(1, {isScatter: isScatter}, true, handlerDlg); }, onDeleteSeries: function() { var rec = this.seriesList.getSelectedRec(); if (rec) { var order = rec.get('order'); // this.chartSettings.deleteSeries(rec.get('index')); this.updateSeriesList(this.chartSettings.getSeries(), order); } this.updateButtons(); }, onEditSeries: function() { var rec = this.seriesList.getSelectedRec(); if (rec) { var series = rec.get('series'), isScatter = series.asc_IsScatter(); var handlerDlg = function(dlg, result) { if (result == 'ok') { var changedValue = dlg.getSettings(); } }; this.changeDataRange(1, {series: series, name: series.asc_getName(), isScatter: isScatter, values: isScatter ? null : series.asc_getValues(), valuesX: !isScatter ? null : series.asc_getXValues(), valuesY: !isScatter ? null : series.asc_getYValues() }, false, handlerDlg); } }, onEditCategory: function() { var handlerDlg = function(dlg, result) { if (result == 'ok') { var changedValue = dlg.getSettings(); } }; this.changeDataRange(0, {category: '', values: this.chartSettings.getCatValues()}, false, handlerDlg); }, changeDataRange: function(type, props, add, handlerDlg) { var me = this; var win = new SSE.Views.ChartDataRangeDialog({ type: type, //series isScatter: !!props.isScatter, handler: handlerDlg }).on('close', function() { me.show(); }); var xy = me.$window.offset(); me.hide(); win.show(xy.left + 160, xy.top + 125); win.setSettings({ api : me.api, props : props }); }, onMoveClick: function(up) { var store = this.seriesList.store, length = store.length, rec = this.seriesList.getSelectedRec(); if (rec) { var index = store.indexOf(rec), order = rec.get('order'), newindex = up ? Math.max(0, index-1) : Math.min(length-1, index+1), newrec = store.at(newindex), neworder = newrec.get('order'); store.add(store.remove(rec), {at: newindex}); rec.set('order', neworder); newrec.set('order', order); // this.chartSettings.changeSeriesOrder(rec.get('index'), neworder, newrec.get('index'), order); this.seriesList.selectRecord(rec); this.seriesList.scrollToRecord(rec); } this.updateMoveButtons(); }, updateSeriesList: function(series, index) { var arr = []; var store = this.seriesList.store; for (var i = 0, len = series.length; i < len; i++) { var item = series[i], rec = new Common.UI.DataViewModel(); rec.set({ value: item.getSeriesName(), index: item.asc_getIdx(), order: item.asc_getOrder(), series: item }); arr.push(rec); } store.reset(arr); (len>0) && this.seriesList.selectByIndex(Math.min(index || 0, store.length-1)); }, textTitle: 'Chart Data', txtEmpty: 'This field is required', textInvalidRange: 'ERROR! Invalid cells range', textSelectData: 'Select data', errorMaxRows: 'ERROR! The maximum number of data series per chart is 255.', errorStockChart: 'Incorrect row order. To build a stock chart place the data on the sheet in the following order:
opening price, max price, min price, closing price.', errorMaxPoints: 'ERROR! The maximum number of points in series per chart is 4096.', textSeries: 'Legend Entries (Series)', textAdd: 'Add', textEdit: 'Edit', textDelete: 'Remove', textSwitch: 'Switch Row/Column', textCategory: 'Horizontal (Category) Axis Labels', textUp: 'Up', textDown: 'Down', textData: 'Data' }, SSE.Views.ChartDataDialog || {})) });