2016-04-01 13:17:09 +00:00
|
|
|
/*
|
|
|
|
*
|
|
|
|
* (c) Copyright Ascensio System Limited 2010-2016
|
|
|
|
*
|
|
|
|
* 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 Lubanas st. 125a-25, Riga, Latvia,
|
|
|
|
* EU, LV-1021.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*
|
|
|
|
*/
|
2016-03-11 00:48:53 +00:00
|
|
|
/**
|
|
|
|
* StatusBar View
|
|
|
|
*
|
|
|
|
* Created by Maxim Kadushkin on 27 March 2014
|
|
|
|
* Copyright (c) 2014 Ascensio System SIA. All rights reserved.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
define([
|
|
|
|
'text!spreadsheeteditor/main/app/template/StatusBar.template',
|
|
|
|
'tip',
|
|
|
|
'common/main/lib/component/TabBar',
|
|
|
|
'common/main/lib/component/Menu',
|
|
|
|
'common/main/lib/component/Window',
|
|
|
|
'common/main/lib/component/ThemeColorPalette'
|
|
|
|
],
|
|
|
|
function(template){
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
if (SSE.Views.Statusbar) {
|
|
|
|
var RenameDialog = SSE.Views.Statusbar.RenameDialog;
|
|
|
|
var CopyDialog = SSE.Views.Statusbar.CopyDialog;
|
|
|
|
}
|
|
|
|
|
|
|
|
SSE.Views.Statusbar = Common.UI.BaseView.extend(_.extend({
|
|
|
|
el: '#statusbar',
|
|
|
|
template: _.template(template),
|
|
|
|
|
|
|
|
storeUsers: undefined,
|
|
|
|
|
|
|
|
tplUser: ['<li id="status-chat-user-<%= user.get("id") %>" class="<% if (!user.get("online")) { %> offline <% } if (user.get("view")) {%> viewmode <% } %>">',
|
|
|
|
'<div class="color" style="background-color: <%= user.get("color") %>;" >',
|
|
|
|
'<label class="name"><%= scope.getUserName(user.get("username")) %></label>',
|
|
|
|
'</div>',
|
|
|
|
'</li>'].join(''),
|
|
|
|
|
|
|
|
templateUserList: _.template('<ul>' +
|
|
|
|
'<% _.each(users, function(item) { %>' +
|
|
|
|
'<%= _.template(usertpl, {user: item, scope: scope}) %>' +
|
|
|
|
'<% }); %>' +
|
|
|
|
'</ul>'),
|
|
|
|
|
|
|
|
events: function() {
|
|
|
|
return {
|
|
|
|
'click #status-btn-tabfirst': _.bind(this.onBtnTabScroll, this, 'first'),
|
|
|
|
'click #status-btn-tabback': _.bind(this.onBtnTabScroll, this, 'backward'),
|
|
|
|
'click #status-btn-tabnext': _.bind(this.onBtnTabScroll, this, 'forward'),
|
|
|
|
'click #status-btn-tablast': _.bind(this.onBtnTabScroll, this, 'last')
|
|
|
|
};
|
|
|
|
},
|
|
|
|
|
|
|
|
api: undefined,
|
|
|
|
|
|
|
|
initialize: function (options) {
|
|
|
|
_.extend(this, options);
|
|
|
|
},
|
|
|
|
|
|
|
|
render: function () {
|
|
|
|
var me = this;
|
|
|
|
$(this.el).html(this.template({
|
|
|
|
scope: this
|
|
|
|
}));
|
|
|
|
|
|
|
|
this.editMode = false;
|
|
|
|
|
|
|
|
this.btnZoomDown = new Common.UI.Button({
|
|
|
|
el: $('#status-btn-zoomdown',this.el),
|
|
|
|
hint: this.tipZoomOut+' (Ctrl+-)',
|
|
|
|
hintAnchor: 'top'
|
|
|
|
});
|
|
|
|
|
|
|
|
this.btnZoomUp = new Common.UI.Button({
|
|
|
|
el: $('#status-btn-zoomup',this.el),
|
|
|
|
hint: this.tipZoomIn+' (Ctrl++)',
|
|
|
|
hintAnchor: 'top-right'
|
|
|
|
});
|
|
|
|
|
|
|
|
this.btnScrollFirst = new Common.UI.Button({
|
|
|
|
el: $('#status-btn-tabfirst',this.el),
|
|
|
|
hint: this.tipFirst,
|
|
|
|
disabled: true,
|
|
|
|
hintAnchor: 'top'
|
|
|
|
});
|
|
|
|
|
|
|
|
this.btnScrollBack = new Common.UI.Button({
|
|
|
|
el: $('#status-btn-tabback',this.el),
|
|
|
|
hint: this.tipPrev,
|
|
|
|
disabled: true,
|
|
|
|
hintAnchor: 'top'
|
|
|
|
});
|
|
|
|
|
|
|
|
this.btnScrollNext = new Common.UI.Button({
|
|
|
|
el: $('#status-btn-tabnext',this.el),
|
|
|
|
hint: this.tipNext,
|
|
|
|
disabled: true,
|
|
|
|
hintAnchor: 'top'
|
|
|
|
});
|
|
|
|
|
|
|
|
this.btnScrollLast = new Common.UI.Button({
|
|
|
|
el: $('#status-btn-tablast',this.el),
|
|
|
|
hint: this.tipLast,
|
|
|
|
disabled: true,
|
|
|
|
hintAnchor: 'top'
|
|
|
|
});
|
|
|
|
|
|
|
|
this.btnAddWorksheet = new Common.UI.Button({
|
|
|
|
el: $('#status-btn-addtab',this.el),
|
|
|
|
hint: this.tipAddTab,
|
|
|
|
disabled: true,
|
|
|
|
hintAnchor: 'top'
|
|
|
|
});
|
|
|
|
|
|
|
|
this.cntZoom = new Common.UI.Button({
|
|
|
|
el: $('.cnt-zoom',this.el),
|
|
|
|
hint: this.tipZoomFactor,
|
|
|
|
hintAnchor: 'top'
|
|
|
|
});
|
|
|
|
this.cntZoom.cmpEl.on({
|
|
|
|
'show.bs.dropdown': function () {
|
|
|
|
_.defer(function(){
|
|
|
|
me.cntZoom.cmpEl.find('ul').focus();
|
|
|
|
}, 100);
|
|
|
|
},
|
|
|
|
'hide.bs.dropdown': function () {
|
|
|
|
_.defer(function(){
|
|
|
|
me.api.asc_enableKeyEvents(true);
|
|
|
|
}, 100);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
this.zoomMenu = new Common.UI.Menu({
|
|
|
|
style: 'margin-top:-5px;',
|
|
|
|
menuAlign: 'bl-tl',
|
|
|
|
items: [
|
|
|
|
{ caption: "50%", value: 50 },
|
|
|
|
{ caption: "75%", value: 75 },
|
|
|
|
{ caption: "100%", value: 100 },
|
|
|
|
{ caption: "125%", value: 125 },
|
|
|
|
{ caption: "150%", value: 150 },
|
|
|
|
{ caption: "175%", value: 175 },
|
|
|
|
{ caption: "200%", value: 200 }
|
|
|
|
]
|
|
|
|
});
|
|
|
|
this.zoomMenu.render($('.cnt-zoom',this.el));
|
|
|
|
this.zoomMenu.cmpEl.attr({tabindex: -1});
|
|
|
|
|
|
|
|
this.labelZoom = $('#status-label-zoom',this.$el);
|
|
|
|
|
|
|
|
/** coauthoring begin **/
|
|
|
|
this.panelUsersList = $('#status-users-list', this.el);
|
|
|
|
this.storeUsers.bind({
|
|
|
|
add : _.bind(this._onAddUser, this),
|
|
|
|
change : _.bind(this._onUsersChanged, this),
|
|
|
|
reset : _.bind(this._onResetUsers, this)
|
|
|
|
});
|
|
|
|
|
|
|
|
this.panelUsers = $('#status-users-ct', this.el);
|
|
|
|
this.panelUsers.on('shown.bs.dropdown', function () {
|
|
|
|
me.panelUsersList.scroller.update({minScrollbarLength : 40, alwaysVisibleY: true});
|
|
|
|
});
|
|
|
|
|
|
|
|
this.panelUsersBlock = this.panelUsers.find('#status-users-block');
|
|
|
|
this.panelUsersBlock.tooltip({
|
|
|
|
title: this.tipAccessRights,
|
|
|
|
html: true,
|
|
|
|
placement: 'top'
|
|
|
|
});
|
|
|
|
this.panelUsersBlock.on('click', _.bind(this.onUsersClick, this));
|
|
|
|
|
|
|
|
this.lblUserCount = this.panelUsers.find('#status-users-count');
|
|
|
|
|
|
|
|
this.lblChangeRights = this.panelUsers.find('#status-change-rights');
|
|
|
|
this.lblChangeRights.on('click', _.bind(this.onUsersClick, this));
|
|
|
|
|
|
|
|
this.$el.find('#status-users-menu').on('click', function() {
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
/** coauthoring end **/
|
|
|
|
|
|
|
|
this.tabBarBox = $('#status-sheets-bar-box', this.el);
|
|
|
|
this.tabbar = new Common.UI.TabBar({
|
|
|
|
el: '#status-sheets-bar',
|
|
|
|
placement: 'bottom',
|
|
|
|
draggable: false
|
|
|
|
}).render();
|
|
|
|
|
|
|
|
this.tabbar.on({
|
|
|
|
'tab:invisible' : _.bind(this.onTabInvisible, this),
|
|
|
|
'tab:changed' : _.bind(this.onSheetChanged, this),
|
|
|
|
//'tab:manual' : _.bind(this.onAddTabClick, this),
|
|
|
|
'tab:contextmenu' : _.bind(this.onTabMenu, this),
|
|
|
|
'tab:dblclick' : _.bind(function () {
|
2016-04-05 11:52:34 +00:00
|
|
|
if (me.editMode && (me.rangeSelectionMode !== Asc.c_oAscSelectionDialogType.Chart) &&
|
|
|
|
(me.rangeSelectionMode !== Asc.c_oAscSelectionDialogType.FormatTable)) {
|
2016-03-11 00:48:53 +00:00
|
|
|
me.fireEvent('sheet:changename');
|
|
|
|
}
|
|
|
|
}, this),
|
|
|
|
'tab:move' : _.bind(function (tabIndex, index) {
|
|
|
|
me.tabBarScroll = {scrollLeft: me.tabbar.scrollX};
|
|
|
|
|
|
|
|
if (_.isUndefined(index) || tabIndex === index) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (tabIndex < index) {
|
|
|
|
++index;
|
|
|
|
}
|
|
|
|
|
|
|
|
me.fireEvent('sheet:move', [false, true, tabIndex, index]);
|
|
|
|
|
|
|
|
}, this)
|
|
|
|
});
|
|
|
|
|
|
|
|
var menuHiddenItems = new Common.UI.Menu({
|
2016-11-10 07:19:17 +00:00
|
|
|
maxHeight: 260,
|
2016-03-11 00:48:53 +00:00
|
|
|
menuAlign: 'tl-tr'
|
2016-11-10 07:19:17 +00:00
|
|
|
}).on('render:after', function(mnu) {
|
|
|
|
this.scroller = new Common.UI.Scroller({
|
|
|
|
el: $(this.el).find('.dropdown-menu '),
|
|
|
|
useKeyboard: this.enableKeyEvents && !this.handleSelect,
|
|
|
|
minScrollbarLength : 40,
|
|
|
|
alwaysVisibleY: true
|
|
|
|
});
|
|
|
|
}).on('show:after', function () {
|
|
|
|
this.scroller.update({alwaysVisibleY: true});
|
2016-03-11 00:48:53 +00:00
|
|
|
});
|
|
|
|
menuHiddenItems.on('item:click', function(obj,item,e) {
|
|
|
|
me.fireEvent('show:hidden', [me, item.value]);
|
|
|
|
});
|
|
|
|
|
|
|
|
var menuColorItems = new Common.UI.Menu({
|
|
|
|
menuAlign: 'tl-tr',
|
|
|
|
cls: 'color-tab',
|
|
|
|
items: [
|
2016-12-05 08:06:53 +00:00
|
|
|
{ template: _.template('<div id="id-tab-menu-color" style="width: 169px; height: 220px; margin: 10px;"></div>') },
|
2016-03-11 00:48:53 +00:00
|
|
|
{ template: _.template('<a id="id-tab-menu-new-color" style="padding-left:12px;">' + me.textNewColor + '</a>') }
|
|
|
|
]
|
|
|
|
});
|
|
|
|
|
|
|
|
function dummyCmp() {
|
|
|
|
return {
|
|
|
|
isDummy : true,
|
|
|
|
on : function() {}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
me.mnuTabColor = dummyCmp();
|
|
|
|
this.tabMenu = new Common.UI.Menu({
|
|
|
|
menuAlign: 'bl-tl',
|
|
|
|
items: [
|
|
|
|
{caption: this.itemInsert, value: 'ins'},
|
|
|
|
{caption: this.itemDelete, value: 'del'},
|
|
|
|
{caption: this.itemRename, value: 'ren'},
|
|
|
|
{caption: this.itemCopy, value: 'copy'},
|
|
|
|
{caption: this.itemMove, value: 'move'},
|
|
|
|
{caption: this.itemHide, value: 'hide'},
|
|
|
|
{
|
|
|
|
caption: this.itemHidden,
|
|
|
|
menu: menuHiddenItems
|
|
|
|
},
|
|
|
|
{
|
|
|
|
caption: this.itemTabColor,
|
|
|
|
menu: menuColorItems
|
|
|
|
}
|
|
|
|
]
|
|
|
|
}).on('render:after', function(btn) {
|
|
|
|
var colorVal = $('<div class="btn-color-value-line"></div>');
|
|
|
|
$('button:first-child', btn.cmpEl).append(colorVal);
|
|
|
|
colorVal.css('background-color', btn.currentColor || 'transparent');
|
|
|
|
|
|
|
|
me.mnuTabColor = new Common.UI.ThemeColorPalette({
|
|
|
|
el: $('#id-tab-menu-color'),
|
2016-08-08 14:13:39 +00:00
|
|
|
transparent: true
|
2016-03-11 00:48:53 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
me.mnuTabColor.on('select', function(picker, color) {
|
|
|
|
me.fireEvent('sheet:setcolor', [color]);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2016-11-18 08:23:29 +00:00
|
|
|
this.tabbar.$el.append('<div class="dropdown-toggle" data-toggle="dropdown" style="width:0; height:0;"/>');
|
2016-03-11 00:48:53 +00:00
|
|
|
this.tabMenu.render(this.tabbar.$el);
|
2016-11-18 08:23:29 +00:00
|
|
|
this.tabMenu.cmpEl.attr({tabindex: -1});
|
2016-03-11 00:48:53 +00:00
|
|
|
this.tabMenu.on('show:after', _.bind(this.onTabMenuAfterShow, this));
|
|
|
|
this.tabMenu.on('hide:after', _.bind(this.onTabMenuAfterHide, this));
|
|
|
|
this.tabMenu.on('item:click', _.bind(this.onTabMenuClick, this));
|
|
|
|
|
|
|
|
this.boxMath = $('#status-math-box', this.el);
|
|
|
|
this.labelSum = $('#status-math-sum', this.boxMath);
|
|
|
|
this.labelCount = $('#status-math-count', this.boxMath);
|
|
|
|
this.labelAverage = $('#status-math-average', this.boxMath);
|
|
|
|
this.boxMath.hide();
|
|
|
|
|
|
|
|
this.boxZoom = $('#status-zoom-box', this.el);
|
|
|
|
this.boxZoom.find('.separator').css('border-left-color','transparent');
|
|
|
|
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
setApi: function(api) {
|
|
|
|
this.api = api;
|
|
|
|
this.api.asc_registerCallback('asc_onSheetsChanged', _.bind(this.update, this));
|
|
|
|
/** coauthoring begin **/
|
|
|
|
this.api.asc_registerCallback('asc_onAuthParticipantsChanged', _.bind(this.onApiUsersChanged, this));
|
|
|
|
this.api.asc_registerCallback('asc_onParticipantsChanged', _.bind(this.onApiUsersChanged, this));
|
|
|
|
this.api.asc_coAuthoringGetUsers();
|
|
|
|
/** coauthoring end **/
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
setMode: function(mode) {
|
|
|
|
this.mode = _.extend({}, this.mode, mode);
|
|
|
|
// this.$el.find('.el-edit')[mode.isEdit?'show':'hide']();
|
|
|
|
this.btnAddWorksheet.setVisible(this.mode.isEdit);
|
|
|
|
this.btnAddWorksheet.setDisabled(this.mode.isDisconnected);
|
|
|
|
this.lblChangeRights[(!this.mode.isOffline && this.mode.sharingSettingsUrl&&this.mode.sharingSettingsUrl.length)?'show':'hide']();
|
|
|
|
this.panelUsers[(!this.mode.isOffline && this.mode.sharingSettingsUrl&&this.mode.sharingSettingsUrl.length)?'show':'hide']();
|
|
|
|
this.updateTabbarBorders();
|
|
|
|
},
|
|
|
|
|
|
|
|
setVisible: function(visible) {
|
|
|
|
visible ? this.show(): this.hide();
|
|
|
|
},
|
|
|
|
|
|
|
|
update: function() {
|
|
|
|
var me = this;
|
|
|
|
|
|
|
|
this.tabbar.empty(true);
|
|
|
|
this.tabMenu.items[6].menu.removeAll();
|
|
|
|
this.tabMenu.items[6].hide();
|
|
|
|
this.btnAddWorksheet.setDisabled(true);
|
|
|
|
|
|
|
|
if (this.api) {
|
|
|
|
var wc = this.api.asc_getWorksheetsCount(), i = -1;
|
|
|
|
var hidentems = [], items = [], tab, locked;
|
|
|
|
var sindex = this.api.asc_getActiveWorksheetIndex();
|
|
|
|
|
|
|
|
while (++i < wc) {
|
|
|
|
locked = me.api.asc_isWorksheetLockedOrDeleted(i);
|
|
|
|
tab = {
|
|
|
|
sheetindex : i,
|
|
|
|
active : sindex == i,
|
|
|
|
label : me.api.asc_getWorksheetName(i),
|
|
|
|
// reorderable : !locked,
|
|
|
|
cls : locked ? 'coauth-locked':'',
|
|
|
|
isLockTheDrag : locked
|
|
|
|
};
|
|
|
|
|
|
|
|
this.api.asc_isWorksheetHidden(i)? hidentems.push(tab) : items.push(tab);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (hidentems.length) {
|
|
|
|
hidentems.forEach(function(item){
|
|
|
|
me.tabMenu.items[6].menu.addItem(new Common.UI.MenuItem({
|
|
|
|
style: 'white-space: pre-wrap',
|
|
|
|
caption: Common.Utils.String.htmlEncode(item.label),
|
|
|
|
value: item.sheetindex
|
|
|
|
}));
|
|
|
|
});
|
|
|
|
this.tabMenu.items[6].show();
|
|
|
|
}
|
|
|
|
|
|
|
|
this.tabbar.add(items);
|
|
|
|
|
|
|
|
if (!_.isUndefined(this.tabBarScroll)) {
|
|
|
|
this.tabbar.$bar.scrollLeft(this.tabBarScroll.scrollLeft);
|
|
|
|
this.tabBarScroll = undefined;
|
|
|
|
}
|
|
|
|
if (!this.tabbar.isTabVisible(sindex))
|
|
|
|
this.tabbar.setTabVisible(sindex);
|
|
|
|
|
|
|
|
this.btnAddWorksheet.setDisabled(me.mode.isDisconnected || me.api.asc_isWorkbookLocked());
|
|
|
|
$('#status-label-zoom').text(Common.Utils.String.format(this.zoomText, Math.floor((this.api.asc_getZoom() +.005)*100)));
|
|
|
|
|
|
|
|
me.fireEvent('sheet:changed', [me, sindex]);
|
|
|
|
me.fireEvent('sheet:updateColors', [true]);
|
|
|
|
Common.NotificationCenter.trigger('comments:updatefilter', {property: 'uid', value: new RegExp('^(doc_|sheet' + me.api.asc_getActiveWorksheetId() + '_)')}, false);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
setMathInfo: function(info) {
|
|
|
|
if (info.count>1) {
|
|
|
|
if (!this.boxMath.is(':visible')) this.boxMath.show();
|
|
|
|
this.labelCount.text(this.textCount + ': ' + info.count);
|
|
|
|
this.labelSum.text((info.sum && info.sum.length) ? (this.textSum + ': ' + info.sum) : '');
|
|
|
|
this.labelAverage.text((info.average && info.average.length) ? (this.textAverage + ': ' + info.average) : '');
|
|
|
|
} else {
|
|
|
|
if (this.boxMath.is(':visible')) this.boxMath.hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
var me = this;
|
|
|
|
_.delay(function(){
|
|
|
|
me.onTabInvisible(undefined, me.tabbar.checkInvisible(true));
|
|
|
|
},30);
|
|
|
|
},
|
|
|
|
|
|
|
|
/** coauthoring begin **/
|
|
|
|
onUsersClick: function() {
|
|
|
|
this.panelUsers.removeClass('open');
|
|
|
|
this.fireEvent('click:users', this);
|
|
|
|
},
|
|
|
|
|
|
|
|
onApiUsersChanged: function(users) {
|
|
|
|
var length = 0;
|
|
|
|
_.each(users, function(item){
|
|
|
|
if (!item.asc_getView())
|
|
|
|
length++;
|
|
|
|
});
|
|
|
|
|
|
|
|
this.panelUsers[(length>1 || this.mode.sharingSettingsUrl&&this.mode.sharingSettingsUrl.length)?'show':'hide']();
|
|
|
|
this.updateTabbarBorders();
|
|
|
|
|
|
|
|
this.lblUserCount.css({
|
|
|
|
'font-size': (length > 1 ? '11px' : '14px'),
|
|
|
|
'font-weight': (length > 1 ? 'bold' : 'normal'),
|
|
|
|
'margin-top': (length > 1 ? '0' : '-1px')
|
|
|
|
});
|
|
|
|
this.lblUserCount.text(length > 1 ? length : '+');
|
|
|
|
$('#status-users-icon').css('margin-bottom', length > 1 ? '0' : '1px');
|
|
|
|
|
|
|
|
var usertip = this.panelUsersBlock.data('bs.tooltip');
|
|
|
|
if (usertip) {
|
|
|
|
usertip.options.title = (length > 1) ? this.tipViewUsers : this.tipAccessRights;
|
|
|
|
usertip.setContent();
|
|
|
|
}
|
|
|
|
(length > 1) ? this.panelUsersBlock.attr('data-toggle', 'dropdown') : this.panelUsersBlock.removeAttr('data-toggle');
|
2016-10-19 11:16:46 +00:00
|
|
|
this.panelUsersBlock.toggleClass('dropdown-toggle', length > 1);
|
2016-03-11 00:48:53 +00:00
|
|
|
(length > 1) ? this.panelUsersBlock.off('click') : this.panelUsersBlock.on('click', _.bind(this.onUsersClick, this));
|
|
|
|
},
|
|
|
|
|
|
|
|
_onAddUser: function(m, c, opts) {
|
|
|
|
if (this.panelUsersList) {
|
|
|
|
this.panelUsersList.find('ul').append(_.template(this.tplUser, {user: m, scope: this}));
|
|
|
|
this.panelUsersList.scroller.update({minScrollbarLength : 40, alwaysVisibleY: true});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_onUsersChanged: function(m) {
|
|
|
|
if (m.changed.online != undefined && this.panelUsersList) {
|
|
|
|
this.panelUsersList.find('#status-chat-user-'+ m.get('id'))[m.changed.online?'removeClass':'addClass']('offline');
|
|
|
|
this.panelUsersList.scroller.update({minScrollbarLength : 40, alwaysVisibleY: true});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
_onResetUsers: function(c, opts) {
|
|
|
|
if (this.panelUsersList) {
|
|
|
|
this.panelUsersList.html(this.templateUserList({users: c.models, usertpl: this.tplUser, scope: this}));
|
|
|
|
this.panelUsersList.scroller = new Common.UI.Scroller({
|
|
|
|
el : $('#status-users-list ul'),
|
|
|
|
useKeyboard : true,
|
|
|
|
minScrollbarLength : 40,
|
|
|
|
alwaysVisibleY: true
|
|
|
|
});
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
getUserName: function (username) {
|
|
|
|
return Common.Utils.String.htmlEncode(username);
|
|
|
|
},
|
|
|
|
/** coauthoring end **/
|
|
|
|
|
|
|
|
onSheetChanged: function(o, index, tab) {
|
|
|
|
this.api.asc_showWorksheet(tab.sheetindex);
|
|
|
|
|
|
|
|
if (this.hasTabInvisible && !this.tabbar.isTabVisible(index)) {
|
|
|
|
this.tabbar.setTabVisible(index);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.fireEvent('sheet:changed', [this, tab.sheetindex]);
|
|
|
|
this.fireEvent('sheet:updateColors', [true]);
|
|
|
|
|
|
|
|
Common.NotificationCenter.trigger('comments:updatefilter',
|
|
|
|
{
|
|
|
|
property: 'uid',
|
|
|
|
value: new RegExp('^(doc_|sheet' + this.api.asc_getActiveWorksheetId() + '_)')
|
|
|
|
},
|
|
|
|
false // hide popover
|
|
|
|
);
|
|
|
|
},
|
|
|
|
|
|
|
|
onTabMenu: function (o, index, tab) {
|
2016-04-05 11:52:34 +00:00
|
|
|
if (this.mode.isEdit && !this.isEditFormula && (this.rangeSelectionMode !== Asc.c_oAscSelectionDialogType.Chart) &&
|
2016-11-03 12:00:48 +00:00
|
|
|
(this.rangeSelectionMode !== Asc.c_oAscSelectionDialogType.FormatTable) &&
|
|
|
|
!this.mode.isDisconnected ) {
|
2016-03-11 00:48:53 +00:00
|
|
|
if (tab && tab.sheetindex >= 0) {
|
|
|
|
var rect = tab.$el.get(0).getBoundingClientRect(),
|
|
|
|
childPos = tab.$el.offset(),
|
|
|
|
parentPos = tab.$el.parent().offset();
|
|
|
|
|
|
|
|
if (!tab.isActive()) this.tabbar.setActive(tab);
|
|
|
|
|
|
|
|
var issheetlocked = this.api.asc_isWorksheetLockedOrDeleted(tab.sheetindex),
|
|
|
|
isdoclocked = this.api.asc_isWorkbookLocked();
|
|
|
|
|
|
|
|
this.tabMenu.items[0].setDisabled(isdoclocked);
|
|
|
|
this.tabMenu.items[1].setDisabled(issheetlocked);
|
|
|
|
this.tabMenu.items[2].setDisabled(issheetlocked);
|
|
|
|
this.tabMenu.items[3].setDisabled(issheetlocked);
|
|
|
|
this.tabMenu.items[4].setDisabled(issheetlocked);
|
|
|
|
this.tabMenu.items[5].setDisabled(issheetlocked);
|
|
|
|
this.tabMenu.items[6].setDisabled(isdoclocked);
|
|
|
|
this.tabMenu.items[7].setDisabled(issheetlocked);
|
|
|
|
|
|
|
|
this.api.asc_closeCellEditor();
|
|
|
|
this.api.asc_enableKeyEvents(false);
|
|
|
|
|
|
|
|
this.tabMenu.atposition = (function () {
|
|
|
|
return {
|
|
|
|
top : rect.top,
|
|
|
|
left: rect.left - parentPos.left - 2
|
|
|
|
};
|
|
|
|
})();
|
|
|
|
|
|
|
|
this.tabMenu.hide();
|
|
|
|
this.tabMenu.show();
|
2016-11-21 10:26:12 +00:00
|
|
|
var menu = this.tabMenu;
|
|
|
|
_.defer(function(){
|
|
|
|
menu.cmpEl.focus();
|
|
|
|
}, 10);
|
2016-03-11 00:48:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onTabMenuAfterShow: function (obj) {
|
|
|
|
if (obj.atposition) {
|
|
|
|
obj.setOffset(obj.atposition.left);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.enableKeyEvents = true;
|
|
|
|
},
|
|
|
|
|
|
|
|
onTabMenuAfterHide: function () {
|
|
|
|
if (!_.isUndefined(this.enableKeyEvents)) {
|
|
|
|
if (this.api) {
|
|
|
|
this.api.asc_enableKeyEvents(this.enableKeyEvents);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.enableKeyEvents = undefined;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onTabMenuClick: function (o, item) {
|
|
|
|
if (item && this.api) {
|
|
|
|
this.enableKeyEvents = (item.value === 'ins' || item.value === 'hide');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
onTabInvisible: function(obj, opts) {
|
|
|
|
if (this.btnScrollFirst.isDisabled() !== (!opts.first)) {
|
|
|
|
this.btnScrollFirst.setDisabled(!opts.first);
|
|
|
|
this.btnScrollBack.setDisabled(!opts.first);
|
|
|
|
}
|
|
|
|
if (this.btnScrollNext.isDisabled() !== (!opts.last)) {
|
|
|
|
this.btnScrollNext.setDisabled(!opts.last);
|
|
|
|
this.btnScrollLast.setDisabled(!opts.last);
|
|
|
|
}
|
|
|
|
this.hasTabInvisible = opts.first || opts.last;
|
|
|
|
},
|
|
|
|
|
|
|
|
onBtnTabScroll: function(action, e) {
|
|
|
|
this.tabbar.setTabVisible(action);
|
|
|
|
},
|
|
|
|
|
|
|
|
updateTabbarBorders: function() {
|
|
|
|
var right = parseInt(this.boxZoom.css('width')), visible = false;
|
|
|
|
if (this.boxMath.is(':visible')) {
|
|
|
|
right += parseInt(this.boxMath.css('width'));
|
|
|
|
visible = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.panelUsers.is(':visible')) {
|
|
|
|
right += parseInt(this.panelUsers.css('width'));
|
|
|
|
visible = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.boxZoom.find('.separator').css('border-left-color',visible?'':'transparent');
|
|
|
|
this.tabBarBox.css('right', right + 'px');
|
|
|
|
},
|
|
|
|
|
|
|
|
changeViewMode: function (edit) {
|
|
|
|
if (edit) {
|
|
|
|
this.tabBarBox.css('left', '152px');
|
|
|
|
} else {
|
|
|
|
this.tabBarBox.css('left', '');
|
|
|
|
}
|
|
|
|
|
|
|
|
this.tabbar.options.draggable = edit;
|
|
|
|
this.editMode = edit;
|
|
|
|
},
|
|
|
|
|
|
|
|
tipZoomIn : 'Zoom In',
|
|
|
|
tipZoomOut : 'Zoom Out',
|
|
|
|
tipZoomFactor : 'Magnification',
|
|
|
|
tipFirst : 'First Sheet',
|
|
|
|
tipLast : 'Last Sheet',
|
|
|
|
tipPrev : 'Previous Sheet',
|
|
|
|
tipNext : 'Next Sheet',
|
|
|
|
tipAddTab : 'Add Worksheet',
|
|
|
|
itemInsert : 'Insert',
|
|
|
|
itemDelete : 'Delete',
|
|
|
|
itemRename : 'Rename',
|
|
|
|
itemCopy : 'Copy',
|
|
|
|
itemMove : 'Move',
|
|
|
|
itemHide : 'Hide',
|
|
|
|
itemHidden : 'Hidden',
|
|
|
|
itemTabColor : 'Tab Color',
|
|
|
|
textNoColor : 'No Color',
|
|
|
|
textNewColor : 'Add New Custom Color',
|
|
|
|
zoomText : 'Zoom {0}%',
|
|
|
|
textSum : 'SUM',
|
|
|
|
textCount : 'COUNT',
|
|
|
|
textAverage : 'AVERAGE',
|
|
|
|
tipUsers : 'Document is currently being edited by several users.',
|
|
|
|
tipAccessRights : 'Manage document access rights',
|
|
|
|
tipViewUsers : 'View users and manage document access rights',
|
|
|
|
txAccessRights : 'Change access rights'
|
|
|
|
|
|
|
|
}, SSE.Views.Statusbar || {}));
|
|
|
|
|
|
|
|
SSE.Views.Statusbar.RenameDialog = Common.UI.Window.extend(_.extend({
|
|
|
|
options: {
|
|
|
|
header: false,
|
|
|
|
width: 280,
|
|
|
|
cls: 'modal-dlg'
|
|
|
|
},
|
|
|
|
|
|
|
|
template: '<div class="box">' +
|
|
|
|
'<div class="input-row">' +
|
|
|
|
'<label><%= label %></label>' +
|
|
|
|
'</div>' +
|
|
|
|
'<div class="input-row" id="txt-sheet-name" />' +
|
|
|
|
'</div>' +
|
|
|
|
'<div class="footer right">' +
|
|
|
|
'<button class="btn normal dlg-btn primary" result="ok" style="margin-right: 10px;"><%= btns.ok %></button>'+
|
|
|
|
'<button class="btn normal dlg-btn" result="cancel"><%= btns.cancel %></button>'+
|
|
|
|
'</div>',
|
|
|
|
|
|
|
|
initialize : function(options) {
|
|
|
|
_.extend(this.options, options || {}, {
|
|
|
|
label: this.labelSheetName,
|
|
|
|
btns: {ok: this.okButtonText, cancel: this.cancelButtonText}
|
|
|
|
});
|
|
|
|
this.options.tpl = _.template(this.template, this.options);
|
|
|
|
|
|
|
|
Common.UI.Window.prototype.initialize.call(this, this.options);
|
|
|
|
},
|
|
|
|
|
|
|
|
render: function() {
|
|
|
|
Common.UI.Window.prototype.render.call(this);
|
|
|
|
|
|
|
|
var $window = this.getChild();
|
|
|
|
$window.find('.dlg-btn').on('click', _.bind(this.onBtnClick, this));
|
|
|
|
|
|
|
|
this.txtName = new Common.UI.InputField({
|
|
|
|
el: $window.find('#txt-sheet-name'),
|
|
|
|
style: 'width:100%;',
|
|
|
|
value: this.options.current,
|
|
|
|
allowBlank: false,
|
|
|
|
maxLength: 31,
|
|
|
|
validation: _.bind(this.nameValidator, this)
|
|
|
|
});
|
|
|
|
|
|
|
|
if (this.txtName) {
|
|
|
|
this.txtName.$el.on('keypress', 'input[type=text]', _.bind(this.onNameKeyPress, this));
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
show: function(x,y) {
|
|
|
|
Common.UI.Window.prototype.show.apply(this, arguments);
|
|
|
|
var edit = this.txtName.$el.find('input');
|
|
|
|
_.delay(function(me){
|
|
|
|
edit.focus();
|
|
|
|
edit.select();
|
|
|
|
}, 100, this);
|
|
|
|
},
|
|
|
|
|
|
|
|
onBtnClick: function(event) {
|
|
|
|
this.doClose(event.currentTarget.attributes['result'].value);
|
|
|
|
},
|
|
|
|
|
|
|
|
doClose: function(res) {
|
|
|
|
if (res == 'ok') {
|
|
|
|
if (this.txtName.checkValidate() !== true) {
|
|
|
|
_.delay(function(me){
|
|
|
|
me.txtName.focus();
|
|
|
|
}, 100, this);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (this.options.handler) {
|
|
|
|
this.options.handler.call(this, res, this.txtName.getValue());
|
|
|
|
}
|
|
|
|
|
|
|
|
this.close();
|
|
|
|
},
|
|
|
|
|
|
|
|
onNameKeyPress: function(e) {
|
|
|
|
if (e.keyCode == Common.UI.Keys.RETURN) {
|
|
|
|
this.doClose('ok');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
nameValidator: function(value) {
|
|
|
|
var items = this.options.names;
|
|
|
|
if (!items && this.options.api) {
|
|
|
|
var api = this.options.api,
|
|
|
|
sindex = api.asc_getActiveWorksheetIndex(),
|
|
|
|
wc = api.asc_getWorksheetsCount();
|
|
|
|
items = [];
|
|
|
|
while (wc--) {
|
|
|
|
if (sindex !== wc) {
|
|
|
|
items.push(api.asc_getWorksheetName(wc).toLowerCase());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (items) {
|
|
|
|
var testval = value.toLowerCase();
|
|
|
|
for (var i = items.length - 1; i >= 0; --i) {
|
|
|
|
if (items[i] === testval) {
|
|
|
|
return this.errNameExists;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (value.length > 2 && value[0]=='"' && value[value.length-1]=='"') return true;
|
|
|
|
if (!/[:\\\/\*\?\[\]\']/.test(value)) return true;
|
|
|
|
|
|
|
|
return this.errNameWrongChar;
|
|
|
|
},
|
|
|
|
|
|
|
|
errNameExists : 'Worksheet with such name already exist.',
|
|
|
|
errNameWrongChar: 'A sheet name cannot contains characters: \\, \/, *, ?, [, ], :',
|
|
|
|
labelSheetName : 'Sheet Name'
|
|
|
|
}, RenameDialog||{}));
|
|
|
|
|
|
|
|
SSE.Views.Statusbar.CopyDialog = Common.UI.Window.extend(_.extend({
|
|
|
|
options: {
|
|
|
|
width: 270,
|
|
|
|
height: 300,
|
|
|
|
cls: 'modal-dlg'
|
|
|
|
},
|
|
|
|
|
|
|
|
template: '<div class="box">' +
|
|
|
|
'<div class="input-row">' +
|
|
|
|
'<label><%= label %></label>' +
|
|
|
|
'</div>' +
|
|
|
|
'<div id="status-list-names" style="height: 170px;"/>' +
|
|
|
|
'</div>' +
|
|
|
|
'<div class="footer center">' +
|
|
|
|
'<button class="btn normal dlg-btn primary" result="ok" style="margin-right: 10px;"><%= btns.ok %></button>'+
|
|
|
|
'<button class="btn normal dlg-btn" result="cancel"><%= btns.cancel %></button>'+
|
|
|
|
'</div>',
|
|
|
|
|
|
|
|
initialize : function(options) {
|
|
|
|
_.extend(this.options, options || {}, {
|
|
|
|
label: options.ismove ? this.textMoveBefore : this.textCopyBefore,
|
|
|
|
btns: {ok: this.okButtonText, cancel: this.cancelButtonText}
|
|
|
|
});
|
|
|
|
this.options.tpl = _.template(this.template, this.options);
|
|
|
|
|
|
|
|
Common.UI.Window.prototype.initialize.call(this, this.options);
|
|
|
|
},
|
|
|
|
|
|
|
|
render: function() {
|
|
|
|
Common.UI.Window.prototype.render.call(this);
|
|
|
|
|
|
|
|
var $window = this.getChild();
|
|
|
|
$window.find('.dlg-btn').on('click', _.bind(this.onBtnClick, this));
|
|
|
|
|
|
|
|
var pages = [];
|
|
|
|
this.options.names.forEach(function(item){
|
|
|
|
pages.push(new Common.UI.DataViewModel(item));
|
|
|
|
}, this);
|
|
|
|
|
|
|
|
if (pages.length) {
|
|
|
|
pages.push(new Common.UI.DataViewModel({
|
|
|
|
value : this.options.ismove ? this.itemMoveToEnd : this.itemCopyToEnd,
|
|
|
|
inindex : -255
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
|
|
|
|
this.listNames = new Common.UI.ListView({
|
|
|
|
el: $('#status-list-names', $window),
|
|
|
|
store: new Common.UI.DataViewStore(pages),
|
|
|
|
itemTemplate: _.template('<div id="<%= id %>" class="list-item" style="pointer-events:none;"><%= Common.Utils.String.htmlEncode(value) %></div>')
|
|
|
|
});
|
|
|
|
|
|
|
|
this.listNames.selectByIndex(0);
|
|
|
|
this.listNames.on('entervalue', _.bind(this.onPrimary, this));
|
|
|
|
this.listNames.on('item:dblclick', _.bind(this.onPrimary, this));
|
|
|
|
|
|
|
|
this.mask = $('.modals-mask');
|
|
|
|
this.mask.on('mousedown',_.bind(this.onUpdateFocus, this));
|
|
|
|
},
|
|
|
|
|
|
|
|
show: function(x,y) {
|
|
|
|
Common.UI.Window.prototype.show.apply(this, arguments);
|
|
|
|
|
|
|
|
_.delay(function(me){
|
|
|
|
me.listNames.$el.find('.listview').focus();
|
|
|
|
}, 100, this);
|
|
|
|
},
|
|
|
|
|
|
|
|
hide: function () {
|
|
|
|
Common.UI.Window.prototype.hide.apply(this, arguments);
|
|
|
|
|
|
|
|
this.mask.off('mousedown',_.bind(this.onUpdateFocus, this));
|
|
|
|
},
|
|
|
|
|
|
|
|
onBtnClick: function(event) {
|
|
|
|
var active = this.listNames.getSelectedRec();
|
|
|
|
|
|
|
|
if (this.options.handler) {
|
|
|
|
this.options.handler.call(this,
|
|
|
|
event.currentTarget.attributes['result'].value, active[0].get('inindex'));
|
|
|
|
}
|
|
|
|
|
|
|
|
this.close();
|
|
|
|
},
|
|
|
|
|
|
|
|
onPrimary: function() {
|
|
|
|
if (this.options.handler) {
|
|
|
|
this.options.handler.call(this, 'ok', this.listNames.getSelectedRec()[0].get('inindex'));
|
|
|
|
}
|
|
|
|
|
|
|
|
this.close();
|
|
|
|
},
|
|
|
|
|
|
|
|
onUpdateFocus: function () {
|
|
|
|
_.delay(function(me){
|
|
|
|
me.listNames.$el.find('.listview').focus();
|
|
|
|
}, 100, this);
|
|
|
|
},
|
|
|
|
|
|
|
|
itemCopyToEnd : '(Copy to end)',
|
|
|
|
itemMoveToEnd : '(Move to end)',
|
|
|
|
textCopyBefore : 'Copy before sheet',
|
|
|
|
textMoveBefore : 'Move before sheet'
|
|
|
|
}, CopyDialog||{}));
|
|
|
|
|
|
|
|
}
|
|
|
|
);
|