' +
@@ -59,12 +85,25 @@ SSE.ApplicationView = new(function(){
function getTools(name) {
return $btnTools.parent().find(name);
}
+ function updateCellInfo(info) {
+ if (info) {
+ this.$cellname.val(typeof(info)=='string' ? info : info.asc_getName());
+ }
+ }
+ function cellNameDisabled(disabled){
+ (disabled) ? this.$cellname.attr('disabled', 'disabled') : this.$cellname.removeAttr('disabled');
+ this.btnNamedRanges.setDisabled(disabled);
+ }
return {
create: createView
, tools: {
get: getTools
},
+ cell: {
+ updateInfo: updateCellInfo,
+ nameDisabled: cellNameDisabled
+ },
txtDownload: 'Download',
txtPrint: 'Print',
diff --git a/test/spreadsheeteditor/main/js/CellEditorController.js b/test/spreadsheeteditor/main/js/CellEditorController.js
new file mode 100644
index 0000000000..d9481bd18b
--- /dev/null
+++ b/test/spreadsheeteditor/main/js/CellEditorController.js
@@ -0,0 +1,338 @@
+/*
+ *
+ * (c) Copyright Ascensio System SIA 2010-2019
+ *
+ * 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
+ *
+*/
+/**
+ * CellEditor.js
+ *
+ * CellEditor Controller
+ *
+ * Created by Maxim Kadushkin on 08 April 2014
+ * Copyright (c) 2018 Ascensio System SIA. All rights reserved.
+ *
+ */
+
+define([
+ 'test/unit-tests/common/main/lib/core/application'
+
+], function (Viewport) {
+ 'use strict';
+
+ SSE.Controllers.CellEditor = Backbone.Controller.extend({
+ views: [
+ 'CellEditorView'
+ ],
+
+ events: function() {
+ return {
+ 'keyup input#ce-cell-name': _.bind(this.onCellName,this),
+ 'keyup textarea#ce-cell-content': _.bind(this.onKeyupCellEditor,this),
+ 'blur textarea#ce-cell-content': _.bind(this.onBlurCellEditor,this),
+ 'click button#ce-btn-expand': _.bind(this.expandEditorField,this),
+ 'click button#ce-func-label': _.bind(this.onInsertFunction, this)
+ };
+ },
+
+ initialize: function() {
+ this.addListeners({
+ 'CellEditor': {},
+ 'Viewport': {
+ 'layout:resizedrag': _.bind(this.onLayoutResize, this)
+ },
+ 'Common.Views.Header': {
+ 'formulabar:hide': function (state) {
+ this.editor.setVisible(!state);
+ common.localStorage.setBool('sse-hidden-formula', state);
+ Common.NotificationCenter.trigger('layout:changed', 'celleditor', state?'hidden':'showed');
+ }.bind(this)
+ }
+ });
+ },
+
+ setApi: function(api) {
+ this.api = api;
+
+ this.api.isCEditorFocused = false;
+ this.api.asc_registerCallback('asc_onSelectionNameChanged', _.bind(this.onApiCellSelection, this));
+ this.api.asc_registerCallback('asc_onEditCell', _.bind(this.onApiEditCell, this));
+ this.api.asc_registerCallback('asc_onCoAuthoringDisconnect', _.bind(this.onApiDisconnect,this));
+ Common.NotificationCenter.on('api:disconnect', _.bind(this.onApiDisconnect, this));
+ Common.NotificationCenter.on('cells:range', _.bind(this.onCellsRange, this));
+ this.api.asc_registerCallback('asc_onLockDefNameManager', _.bind(this.onLockDefNameManager, this));
+ this.api.asc_registerCallback('asc_onInputKeyDown', _.bind(this.onInputKeyDown, this));
+
+ return this;
+ },
+
+ setMode: function(mode) {
+ this.mode = mode;
+
+ this.editor.$btnfunc[this.mode.isEdit?'removeClass':'addClass']('disabled');
+ this.editor.btnNamedRanges.setVisible(this.mode.isEdit && !this.mode.isEditDiagram && !this.mode.isEditMailMerge);
+
+ if ( this.mode.isEdit ) {
+ this.api.asc_registerCallback('asc_onSelectionChanged', _.bind(this.onApiSelectionChanged, this));
+ }
+ },
+
+ onInputKeyDown: function(e) {
+ if (Common.UI.Keys.UP === e.keyCode || Common.UI.Keys.DOWN === e.keyCode ||
+ Common.UI.Keys.TAB === e.keyCode || Common.UI.Keys.RETURN === e.keyCode || Common.UI.Keys.ESC === e.keyCode ||
+ Common.UI.Keys.LEFT === e.keyCode || Common.UI.Keys.RIGHT === e.keyCode) {
+ var menu = $('#menu-formula-selection'); // for formula menu
+ if (menu.hasClass('open'))
+ menu.find('.dropdown-menu').trigger('keydown', e);
+ }
+ },
+
+ onLaunch: function() {
+ this.editor = this.getView('CellEditor');
+ this.bindViewEvents(this.editor, this.events);
+
+ this.editor.$el.parent().find('.after').css({zIndex: '4'}); // for spreadsheets - bug 23127
+
+ var val = common.localStorage.getItem('sse-celleditor-height');
+ this.editor.keep_height = (val!==null && parseInt(val)>0) ? parseInt(val) : 74;
+ if (common.localStorage.getBool('sse-celleditor-expand')) {
+ this.editor.$el.height(this.editor.keep_height);
+ this.onLayoutResize(undefined, 'cell:edit');
+ }
+
+ this.editor.btnNamedRanges.menu.on('item:click', _.bind(this.onNamedRangesMenu, this))
+ .on('show:before', _.bind(this.onNameBeforeShow, this));
+ this.namedrange_locked = false;
+ },
+
+ onApiEditCell: function(state) {
+ if (this.viewmode) return; // signed file
+
+ if (state == Asc.c_oAscCellEditorState.editStart){
+ this.api.isCellEdited = true;
+ this.editor.cellNameDisabled(true);
+ } else if (state == Asc.c_oAscCellEditorState.editInCell) {
+ this.api.isCEditorFocused = 'clear';
+ } else if (state == Asc.c_oAscCellEditorState.editEnd) {
+ this.api.isCellEdited = false;
+ this.api.isCEditorFocused = false;
+ this.editor.cellNameDisabled(false);
+ }
+ this.editor.$btnfunc.toggleClass('disabled', state == Asc.c_oAscCellEditorState.editText);
+ },
+
+ onApiCellSelection: function(info) {
+ this.editor.updateCellInfo(info);
+ },
+
+ onApiSelectionChanged: function(info) {
+ if (this.viewmode) return; // signed file
+
+ var seltype = info.asc_getSelectionType(),
+ coauth_disable = (!this.mode.isEditMailMerge && !this.mode.isEditDiagram) ? (info.asc_getLocked() === true || info.asc_getLockedTable() === true || info.asc_getLockedPivotTable()===true) : false;
+
+ var is_chart_text = seltype == Asc.c_oAscSelectionType.RangeChartText,
+ is_chart = seltype == Asc.c_oAscSelectionType.RangeChart,
+ is_shape_text = seltype == Asc.c_oAscSelectionType.RangeShapeText,
+ is_shape = seltype == Asc.c_oAscSelectionType.RangeShape,
+ is_image = seltype == Asc.c_oAscSelectionType.RangeImage || seltype == Asc.c_oAscSelectionType.RangeSlicer,
+ is_mode_2 = is_shape_text || is_shape || is_chart_text || is_chart;
+
+ this.editor.$btnfunc.toggleClass('disabled', is_image || is_mode_2 || coauth_disable);
+ },
+
+ onApiDisconnect: function() {
+ this.mode.isEdit = false;
+
+ var controller = this.getApplication().getController('FormulaDialog');
+ if (controller) {
+ controller.hideDialog();
+ }
+
+ if (!this.mode.isEdit) {
+ $('#ce-func-label', this.editor.el).addClass('disabled');
+ this.editor.btnNamedRanges.setVisible(false);
+ }
+ },
+
+ onCellsRange: function(status) {
+ this.editor.cellNameDisabled(status != Asc.c_oAscSelectionDialogType.None);
+ this.editor.$btnfunc.toggleClass('disabled', status != Asc.c_oAscSelectionDialogType.None);
+ },
+
+ onLayoutResize: function(o, r) {
+ if (r == 'cell:edit') {
+ if (Math.floor(this.editor.$el.height()) > 19) {
+ if (!this.editor.$btnexpand.hasClass('btn-collapse')) {
+ this.editor.$el.addClass('expanded');
+ this.editor.$btnexpand['addClass']('btn-collapse');
+ }
+
+ o && common.localStorage.setItem('sse-celleditor-height', this.editor.$el.height());
+ o && common.localStorage.setBool('sse-celleditor-expand', true);
+ } else {
+ this.editor.$el.removeClass('expanded');
+ this.editor.$btnexpand['removeClass']('btn-collapse');
+ o && common.localStorage.setBool('sse-celleditor-expand', false);
+ }
+ }
+ },
+
+ onCellName: function(e) {
+ if (e.keyCode == Common.UI.Keys.RETURN){
+ var name = this.editor.$cellname.val();
+ if (name && name.length) {
+ this.api.asc_findCell(name);
+ }
+
+ Common.NotificationCenter.trigger('edit:complete', this.editor);
+ }
+ },
+
+ onBlurCellEditor: function() {
+ if (this.api.isCEditorFocused == 'clear')
+ this.api.isCEditorFocused = undefined;
+ else if (this.api.isCellEdited)
+ this.api.isCEditorFocused = true;
+// if (Common.Utils.isIE && !$('#menu-formula-selection').hasClass('open')) {// for formula menu
+// this.getApplication().getController('DocumentHolder').documentHolder.focus();
+// }
+ },
+
+ onKeyupCellEditor: function(e) {
+ if(e.keyCode == Common.UI.Keys.RETURN && !e.altKey){
+ this.api.isCEditorFocused = 'clear';
+ }
+ },
+
+ expandEditorField: function() {
+ if ( Math.floor(this.editor.$el.height()) > 19) {
+ this.editor.keep_height = this.editor.$el.height();
+ this.editor.$el.height(19);
+ this.editor.$el.removeClass('expanded');
+ this.editor.$btnexpand['removeClass']('btn-collapse');
+ common.localStorage.setBool('sse-celleditor-expand', false);
+ } else {
+ this.editor.$el.height(this.editor.keep_height);
+ this.editor.$el.addClass('expanded');
+ this.editor.$btnexpand['addClass']('btn-collapse');
+ common.localStorage.setBool('sse-celleditor-expand', true);
+ }
+
+ Common.NotificationCenter.trigger('layout:changed', 'celleditor');
+ Common.NotificationCenter.trigger('edit:complete', this.editor, {restorefocus:true});
+ },
+
+ onInsertFunction: function() {
+ if (this.viewmode) return; // signed file
+
+ if ( this.mode.isEdit && !this.editor.$btnfunc['hasClass']('disabled')) {
+ var controller = this.getApplication().getController('FormulaDialog');
+ if (controller) {
+ $('#ce-func-label', this.editor.el).blur();
+ controller.showDialog();
+ }
+ }
+ },
+
+ onNamedRangesMenu: function(menu, item) {
+ var me = this;
+ if (item.options.value=='manager') {
+ var wc = this.api.asc_getWorksheetsCount(),
+ i = -1,
+ items = [], sheetNames = [];
+ while (++i < wc) {
+ if (!this.api.asc_isWorksheetHidden(i)) {
+ sheetNames[i] = this.api.asc_getWorksheetName(i);
+ items.push({displayValue: sheetNames[i], value: i});
+ }
+ }
+
+ (new SSE.Views.NameManagerDlg({
+ api: this.api,
+ handler: function(result) {
+ Common.NotificationCenter.trigger('edit:complete', me.editor);
+ },
+ locked: this.namedrange_locked,
+ sheets: items,
+ sheetNames: sheetNames,
+ ranges: this.api.asc_getDefinedNames(Asc.c_oAscGetDefinedNamesList.All),
+ props : this.api.asc_getDefaultDefinedName(),
+ sort : this.rangeListSort
+ })).on('close', function(win){
+ me.rangeListSort = win.getSettings();
+ }).show();
+ } else {
+ this.api.asc_findCell(item.caption);
+ Common.NotificationCenter.trigger('edit:complete', this.editor);
+ }
+ },
+
+ onNameBeforeShow: function() {
+ var names = this.api.asc_getDefinedNames(Asc.c_oAscGetDefinedNamesList.WorksheetWorkbook, true),
+ rangesMenu = this.editor.btnNamedRanges.menu,
+ prev_name='';
+
+ rangesMenu.removeItems(2, rangesMenu.items.length-1);
+ names.sort(function(item1, item2) {
+ var n1 = item1.asc_getName(true).toLowerCase(),
+ n2 = item2.asc_getName(true).toLowerCase();
+ if (n1==n2) return 0;
+ return (n12);
+ },
+
+ onLockDefNameManager: function(state) {
+ this.namedrange_locked = (state == Asc.c_oAscDefinedNameReason.LockDefNameManager);
+ },
+
+ SetDisabled: function(disabled) {
+ this.editor.$btnfunc[!disabled && this.mode.isEdit ?'removeClass':'addClass']('disabled');
+ this.editor.btnNamedRanges.setVisible(!disabled && this.mode.isEdit && !this.mode.isEditDiagram && !this.mode.isEditMailMerge);
+ },
+
+ setPreviewMode: function(mode) {
+ if (this.viewmode === mode) return;
+ this.viewmode = mode;
+ this.editor.$btnfunc[!mode && this.mode.isEdit?'removeClass':'addClass']('disabled');
+ this.editor.cellNameDisabled(mode && !(this.mode.isEdit && !this.mode.isEditDiagram && !this.mode.isEditMailMerge));
+ }
+ });
+});
\ No newline at end of file
diff --git a/test/spreadsheeteditor/main/js/CellEditorView.js b/test/spreadsheeteditor/main/js/CellEditorView.js
new file mode 100644
index 0000000000..25a908a19f
--- /dev/null
+++ b/test/spreadsheeteditor/main/js/CellEditorView.js
@@ -0,0 +1,106 @@
+/*
+ *
+ * (c) Copyright Ascensio System SIA 2010-2019
+ *
+ * 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
+ *
+*/
+/**
+ * CellEdit.js
+ *
+ * Created by Maxim Kadushkin on 04 April 2014
+ * Copyright (c) 2018 Ascensio System SIA. All rights reserved.
+ *
+ */
+
+define([
+ 'text!spreadsheeteditor/main/template/CellEditor.template',
+ 'test/unit-tests/common/main/lib/component/BaseView'
+], function (template) {
+ 'use strict';
+
+ SSE.Views.CellEditor = Common.UI.BaseView.extend(_.extend({
+ template: _.template(template),
+
+ initialize: function (options) {
+ Common.UI.BaseView.prototype.initialize.call(this, options);
+ },
+
+ render: function () {
+ $(this.el).html(this.template());
+
+ this.btnNamedRanges = new Common.UI.Button({
+ parentEl: $('#ce-cell-name-menu'),
+ menu : new Common.UI.Menu({
+ style : 'min-width: 70px;max-width:400px;',
+ maxHeight: 250,
+ items: [
+ { caption: this.textManager, value: 'manager' },
+ { caption: '--' }
+ ]
+ })
+ });
+ this.btnNamedRanges.setVisible(false);
+ this.btnNamedRanges.menu.setOffset(-81);
+
+ this.$cellname = $('#ce-cell-name', this.el);
+ this.$btnexpand = $('#ce-btn-expand', this.el);
+ this.$btnfunc = $('#ce-func-label', this.el);
+
+ var me = this;
+ this.$cellname.on('focus', function(e){
+ var txt = me.$cellname[0];
+ txt.selectionStart = 0;
+ txt.selectionEnd = txt.value.length;
+ txt.scrollLeft = txt.scrollWidth;
+ });
+
+ this.$btnfunc.addClass('disabled');
+ this.$btnfunc.tooltip({
+ title : this.tipFormula,
+ placement : 'cursor'
+ });
+
+ return this;
+ },
+
+ updateCellInfo: function(info) {
+ if (info) {
+ this.$cellname.val(typeof(info)=='string' ? info : info.asc_getName());
+ }
+ },
+
+ cellNameDisabled: function(disabled){
+ (disabled) ? this.$cellname.attr('disabled', 'disabled') : this.$cellname.removeAttr('disabled');
+ this.btnNamedRanges.setDisabled(disabled);
+ },
+
+ tipFormula: 'Insert Function',
+ textManager: 'Manager'
+ }, SSE.Views.CellEditor || {}));
+});
diff --git a/test/spreadsheeteditor/main/resources/less/celleditor.less b/test/spreadsheeteditor/main/resources/less/celleditor.less
new file mode 100644
index 0000000000..75dbb5e845
--- /dev/null
+++ b/test/spreadsheeteditor/main/resources/less/celleditor.less
@@ -0,0 +1,159 @@
+// Common styles
+@import "../../../../unit-tests/common/main/resources/less/colors-table-ie-fix.less";
+@import "../../../../unit-tests/common/main/resources/less/variables.less";
+@import "../../../../unit-tests/common/main/resources/less/colors-table.less";
+#cell-editing-box {
+ --pixel-ratio-factor: 1;
+ border-bottom: solid @scaled-one-px-value-ie @border-toolbar-ie;
+ border-bottom: solid @scaled-one-px-value @border-toolbar;
+ border-left: solid @scaled-one-px-value-ie @border-toolbar-ie;
+ border-left: solid @scaled-one-px-value @border-toolbar;
+ min-height: 20px;
+ background-color: @background-toolbar-ie;
+ background-color: @background-toolbar;
+
+ .ce-group-name {
+ float: left;
+ height: 20px;
+ //border-bottom: @scaled-one-px-value-ie solid @border-toolbar-ie;
+ //border-bottom: @scaled-one-px-value solid @border-toolbar;
+ background-color: @background-toolbar-ie;
+ background-color: @background-toolbar;
+
+ #ce-cell-name {
+ width: 100px;
+ height: 19px;
+ height: calc(19px + (1px - 1px/var(--pixel-ratio-factor,1)));
+ padding: 0px 19px 0 4px;
+ vertical-align: top;
+ display: inline-block;
+ border: 0 none;
+ border-right: @scaled-one-px-value-ie solid @border-toolbar-ie;
+ border-right: @scaled-one-px-value solid @border-toolbar;
+ transition: none;
+ -webkit-transition: none;
+ box-shadow: 0 @scaled-one-px-value-ie 0 0 @border-toolbar-ie;
+ box-shadow: 0 @scaled-one-px-value 0 0 @border-toolbar;
+
+ &[disabled] {
+ color: @border-preview-select-ie;
+ color: @border-preview-select;
+ opacity: 0.5;
+ }
+ }
+
+ #ce-cell-name-menu {
+ display: inline-block;
+ position: absolute;
+ left: 80px;
+ background-color: @background-toolbar-ie;
+ background-color: @background-toolbar;
+
+ button {
+ background-color: @background-normal-ie;
+ background-color: @background-normal;
+ height: 19px;
+
+ &.disabled {
+ opacity: 0.5;
+ }
+
+ &:active:not(.disabled),
+ &.active:not(.disabled){
+ .caret {
+ border-color: @icon-normal;
+ }
+ }
+ }
+
+ .dropdown-menu a {
+ overflow: hidden;
+ text-overflow: ellipsis;
+ }
+ }
+
+ #ce-func-label {
+ height: 20px;
+ margin-left: -3px;
+ .border-radius(0);
+
+ &.disabled {
+ opacity: 0.4;
+ }
+ }
+ }
+
+ &:not(.expanded) {
+ #ce-func-label {
+ border-bottom: @scaled-one-px-value-ie solid @border-toolbar-ie;
+ border-bottom: @scaled-one-px-value solid @border-toolbar;
+ }
+ }
+
+ .ce-group-expand {
+ float: right;
+ //height: 20px;
+ height: 100%;
+ background-color: @background-normal-ie;
+ background-color: @background-normal;
+ }
+
+ .ce-group-content {
+ margin: 0 16px 0 120px;
+ height: 100%;
+ border-left: @scaled-one-px-value-ie solid @border-toolbar-ie;
+ border-left: @scaled-one-px-value solid @border-toolbar;
+
+ #ce-cell-content {
+ height: 100%;
+ resize: none;
+ min-height: 19px;
+ border: 0 none;
+ font-size: 12px;
+ line-height: 18px;
+ padding-bottom: 0;
+
+ &[disabled] {
+ color: @border-preview-select-ie;
+ color: @border-preview-select;
+ opacity: 0.5;
+ }
+ }
+ }
+
+ #ce-cell-name, #ce-cell-content {
+ border-radius: 0;
+ }
+
+ &+.layout-resizer {
+ border-top: 0 none;
+ border-bottom: 0 none;
+
+ &.move {
+ border-top: @scaled-one-px-value-ie solid @border-toolbar-ie;
+ border-top: @scaled-one-px-value solid @border-toolbar;
+ border-bottom: @scaled-one-px-value-ie solid @border-toolbar-ie;
+ border-bottom: @scaled-one-px-value solid @border-toolbar;
+ opacity: 0.4;
+ }
+ }
+}
+
+#ce-btn-expand {
+ width: 16px;
+ height: 18px;
+ .border-radius(0);
+ background: transparent;
+ padding: 0 2px 0;
+
+ .caret {
+ transition: transform .2s;
+ border-color: @icon-normal;
+ }
+
+ &.btn-collapse {
+ .caret {
+ transform: rotate(45deg);
+ }
+ }
+}
diff --git a/test/spreadsheeteditor/main/template/CellEditor.template b/test/spreadsheeteditor/main/template/CellEditor.template
new file mode 100644
index 0000000000..1c1abe44c7
--- /dev/null
+++ b/test/spreadsheeteditor/main/template/CellEditor.template
@@ -0,0 +1,11 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/test/unit-tests/common/main/lib/component/BaseView.js b/test/unit-tests/common/main/lib/component/BaseView.js
new file mode 100644
index 0000000000..bab307eab4
--- /dev/null
+++ b/test/unit-tests/common/main/lib/component/BaseView.js
@@ -0,0 +1,117 @@
+/*
+ *
+ * (c) Copyright Ascensio System SIA 2010-2019
+ *
+ * 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
+ *
+*/
+/**
+ * BaseView.js
+ *
+ * Created by Alexander Yuzhin on 1/17/14
+ * Copyright (c) 2018 Ascensio System SIA. All rights reserved.
+ *
+ */
+
+if (Common === undefined)
+ var Common = {};
+
+define([
+ 'backbone'
+], function (Backbone) {
+ 'use strict';
+
+ Common.UI = _.extend(Common.UI || {}, {
+ Keys : {
+ BACKSPACE: 8,
+ TAB: 9,
+ RETURN: 13,
+ SHIFT: 16,
+ CTRL: 17,
+ ALT: 18,
+ ESC: 27,
+ LEFT: 37,
+ UP: 38,
+ RIGHT: 39,
+ DOWN: 40,
+ DELETE: 46,
+ HOME: 36,
+ END: 35,
+ SPACE: 32,
+ PAGEUP: 33,
+ PAGEDOWN: 34,
+ INSERT: 45,
+ EQUALITY_FF:61,
+ NUM_PLUS: 107,
+ NUM_MINUS: 109,
+ F1: 112,
+ F2: 113,
+ F3: 114,
+ F4: 115,
+ F5: 116,
+ F6: 117,
+ F7: 118,
+ F8: 119,
+ F9: 120,
+ F10: 121,
+ F11: 122,
+ F12: 123,
+ MINUS_FF: 173,
+ EQUALITY: 187,
+ MINUS: 189
+ },
+
+ BaseView: Backbone.View.extend({
+ isSuspendEvents: false,
+
+ initialize : function(options) {
+ this.options = this.options ? _({}).extend(this.options, options) : options;
+ },
+
+ setVisible: function(visible) {
+ return this[visible ? 'show': 'hide']();
+ },
+
+ isVisible: function() {
+ return $(this.el).is(":visible");
+ },
+
+ suspendEvents: function() {
+ this.isSuspendEvents = true;
+ },
+
+ resumeEvents: function() {
+ this.isSuspendEvents = false;
+ }
+ }),
+
+ getId: function(prefix) {
+ return _.uniqueId(prefix || "asc-gen");
+ }
+ });
+});
\ No newline at end of file
diff --git a/test/unit-tests/common/main/lib/component/Button.js b/test/unit-tests/common/main/lib/component/Button.js
new file mode 100644
index 0000000000..941a930b75
--- /dev/null
+++ b/test/unit-tests/common/main/lib/component/Button.js
@@ -0,0 +1,774 @@
+/*
+ *
+ * (c) Copyright Ascensio System SIA 2010-2019
+ *
+ * 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
+ *
+*/
+/**
+ * Button.js
+ *
+ * Created by Alexander Yuzhin on 1/20/14
+ * Copyright (c) 2018 Ascensio System SIA. All rights reserved.
+ *
+ */
+
+/**
+ * Using template
+ *
+ * A simple button with text:
+ *
+ *
+ * A simple button with icon:
+ *
+ *
+ * A button with menu:
+ *
+ *
+ *
+ *
+ *
+ *
+ * A split button:
+ *
+ *
+ *
+ *
+ *
+ *
+ *
+ * A useful classes of button size
+ *
+ * - `'small'`
+ * - `'normal'`
+ * - `'large'`
+ * - `'huge'`
+ *
+ * A useful classes of button type
+ *
+ * - `'default'`
+ * - `'active'`
+ *
+ *
+ * Buttons can also be toggled. To enable this, you simple set the {@link #enableToggle} property to `true`.
+ *
+ * Example usage:
+ * new Common.UI.Button({
+ * el: $('#id'),
+ * enableToggle: true
+ * });
+ *
+ *
+ * @property {Boolean} disabled
+ * True if this button is disabled. Read-only.
+ *
+ * disabled: false,
+ *
+ *
+ * @property {Boolean} pressed
+ * True if this button is pressed (only if enableToggle = true). Read-only.
+ *
+ * pressed: false,
+ *
+ *
+ * @cfg {Boolean} [allowDepress=true]
+ * False to not allow a pressed Button to be depressed. Only valid when {@link #enableToggle} is true.
+ *
+ * @cfg {String/Object} hint
+ * The tooltip for the button - can be a string to be used as bootstrap tooltip
+ *
+ */
+
+if (Common === undefined)
+ var Common = {};
+
+define([
+ 'unit-tests/common/main/lib/component/BaseView',
+ 'unit-tests/common/main/lib/component/ToggleManager'
+], function () {
+ 'use strict';
+
+ window.createButtonSet = function() {
+ function ButtonsArray(args) {};
+ ButtonsArray.prototype = new Array;
+ ButtonsArray.prototype.constructor = ButtonsArray;
+
+ var _disabled = false;
+
+ ButtonsArray.prototype.add = function(button) {
+ button.setDisabled(_disabled);
+ this.push(button);
+ };
+
+ ButtonsArray.prototype.setDisabled = function(disable) {
+ // if ( _disabled != disable ) //bug when disable buttons outside the group
+ {
+ _disabled = disable;
+
+ this.forEach( function(button) {
+ button.setDisabled(disable);
+ });
+ }
+ };
+
+ ButtonsArray.prototype.toggle = function(state, suppress) {
+ this.forEach(function(button) {
+ button.toggle(state, suppress);
+ });
+ };
+
+ ButtonsArray.prototype.pressed = function() {
+ return this.some(function(button) {
+ return button.pressed;
+ });
+ };
+
+ ButtonsArray.prototype.contains = function(id) {
+ return this.some(function(button) {
+ return button.id == id;
+ });
+ };
+
+ ButtonsArray.prototype.concat = function () {
+ var args = Array.prototype.slice.call(arguments);
+ var result = Array.prototype.slice.call(this);
+
+ args.forEach(function(sub){
+ if (sub instanceof Array )
+ Array.prototype.push.apply(result, sub);
+ else if (sub)
+ result.push(sub);
+ });
+
+ return result;
+ };
+
+ var _out_array = Object.create(ButtonsArray.prototype);
+ for ( var i in arguments ) {
+ _out_array.add(arguments[i]);
+ }
+
+ return _out_array;
+ };
+
+ var templateBtnIcon =
+ '<% if ( iconImg ) { %>' +
+ '' +
+ '<% } else { %>' +
+ '<% if (/svgicon/.test(iconCls)) {' +
+ 'print(\'\');' +
+ '} else ' +
+ 'print(\'\'); %>' +
+ '<% } %>';
+
+ var templateHugeCaption =
+ '';
+
+ var templateHugeMenuCaption =
+ '