/* * * (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 * */ /** * ReviewPopover.js * * View * * Created by Julia Radzhabova on 06.06.2018 * Copyright (c) 2018 Ascensio System SIA. All rights reserved. * */ if (Common === undefined) var Common = {}; Common.Views = Common.Views || {}; define([ 'text!common/main/lib/template/CommentsPopover.template', 'text!common/main/lib/template/ReviewChangesPopover.template', 'common/main/lib/util/utils', 'common/main/lib/component/Button', 'common/main/lib/component/ComboBox', 'common/main/lib/component/DataView', 'common/main/lib/component/Layout', 'common/main/lib/component/Window' ], function (commentsTemplate, reviewTemplate) { 'use strict'; function replaceWords(template, words) { var word, value, tpl = template; for (word in words) { if (undefined !== word) { value = words[word]; tpl = tpl.replace(new RegExp(word, 'g'), value); } } return tpl; } Common.Views.ReviewPopover = Common.UI.Window.extend(_.extend({ // Window initialize: function (options) { var _options = {}; _.extend(_options, { closable: false, width: 265, height: 120, header: false, modal: false, alias: 'Common.Views.ReviewPopover' }, options); this.template = options.template || [ '
', '
', '
', '
', '
', '
', '
' ].join(''); this.commentsStore = options.commentsStore; this.reviewStore = options.reviewStore; this.canRequestUsers = options.canRequestUsers; this.canRequestSendNotify = options.canRequestSendNotify; this.mentionShare = options.mentionShare; this.api = options.api; this.externalUsers = []; this._state = {commentsVisible: false, reviewVisible: false}; _options.tpl = _.template(this.template)(_options); this.arrow = {margin: 20, width: 10, height: 30}; this.sdkBounds = {width: 0, height: 0, padding: 10, paddingTop: 20}; Common.UI.Window.prototype.initialize.call(this, _options); if (this.canRequestUsers) { Common.Gateway.on('setusers', _.bind(this.setUsers, this)); Common.NotificationCenter.on('mentions:clearusers', _.bind(this.clearUsers, this)); } return this; }, render: function (comments, review) { Common.UI.Window.prototype.render.call(this); var me = this, window = this.$window; window.css({ height: '', minHeight: '', overflow: 'hidden', position: 'absolute', zIndex: '990' }); var body = window.find('.body'); if (body) { body.css('position', 'relative'); } var CommentsPopoverDataView = Common.UI.DataView.extend((function () { var parentView = me; return { options: { handleSelect: false, allowScrollbar: false, template: _.template('
') }, getTextBox: function () { var text = $(this.el).find('textarea'); return (text && text.length) ? text : undefined; }, setFocusToTextBox: function (blur) { var text = $(this.el).find('textarea'); if (blur) { text.blur(); } else { if (text && text.length) { var val = text.val(); text.focus(); text.val(''); text.val(val); } } }, getActiveTextBoxVal: function () { var text = $(this.el).find('textarea'); return (text && text.length) ? text.val().trim() : ''; }, disableTextBoxButton: function(textboxEl) { var button = $(textboxEl.siblings('#id-comments-change-popover')[0]); if(textboxEl.val().length > 0) { button.removeAttr('disabled'); button.removeClass('disabled'); } else { button.attr('disabled', true); button.addClass('disabled'); } }, autoHeightTextBox: function () { var view = this, textBox = this.$el.find('textarea'), domTextBox = null, minHeight = 55, lineHeight = 0, scrollPos = 0, oldHeight = 0, newHeight = 0; function updateTextBoxHeight() { scrollPos = parentView.scroller.getScrollTop(); if (domTextBox.scrollHeight > domTextBox.clientHeight) { if (domTextBox.clientHeight + 2 < parseInt($(domTextBox).css('max-height'))) { // 2 = border of textarea textBox.css({height: (domTextBox.scrollHeight + lineHeight) + 'px'}); parentView.calculateSizeOfContent(); } } else { oldHeight = domTextBox.clientHeight; if (oldHeight >= minHeight) { textBox.css({height: minHeight + 'px'}); if (domTextBox.scrollHeight > domTextBox.clientHeight) { newHeight = Math.max(domTextBox.scrollHeight + lineHeight, minHeight); textBox.css({height: newHeight + 'px'}); } parentView.calculateSizeOfContent(); parentView.setLeftTop(me.arrowPosX, me.arrowPosY, me.leftX); parentView.calculateSizeOfContent(); } } parentView.scroller.scrollTop(scrollPos); parentView.autoScrollToEditButtons(); } function onTextareaInput(event) { updateTextBoxHeight(); view.disableTextBoxButton($(event.target)); } if (textBox && textBox.length && parentView.scroller) { domTextBox = textBox.get(0); view.disableTextBoxButton(textBox); if (domTextBox) { lineHeight = parseInt(textBox.css('lineHeight'), 10) * 0.25; updateTextBoxHeight(); textBox.bind('input propertychange', onTextareaInput) } } this.textBox = textBox; }, clearTextBoxBind: function () { if (this.textBox) { this.textBox.unbind('input propertychange'); this.textBox = undefined; } } } })()); if (CommentsPopoverDataView) { if (this.commentsView) { this.commentsView.render($('#id-comments-popover')); this.commentsView.onResetItems(); } else { this.commentsView = new CommentsPopoverDataView({ el: $('#id-comments-popover'), itemTemplate: _.template(replaceWords(commentsTemplate, { textAddReply: me.textAddReply, textMentionReply: me.canRequestSendNotify ? (me.mentionShare ? me.textMention : me.textMentionNotify) : me.textAddReply, textAdd: me.textAdd, textCancel: me.textCancel, textEdit: me.textEdit, textReply: me.textReply, textClose: me.textClose, maxCommLength: Asc.c_oAscMaxCellOrCommentLength, textMentionComment: me.canRequestSendNotify ? (me.mentionShare ? me.textMention : me.textMentionNotify) : me.textEnterComment }) ) }); var addtooltip = function (dataview, view, record) { if (view.tipsArray) { view.tipsArray.forEach(function (item) { item.remove(); }); } var arr = [], btns = $(view.el).find('.btn-resolve:not(.comment-resolved)'); btns.tooltip({title: me.textResolve, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); btns = $(view.el).find('.comment-resolved'); btns.tooltip({title: me.textOpenAgain, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); btns = $(view.el).find('.i-comment-resolved'); btns.tooltip({title: me.textViewResolved, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); btns = $(view.el).find('.btn-edit'); btns.tooltip({title: me.txtEditTip, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); btns = $(view.el).find('.btn-delete'); btns.tooltip({title: me.txtDeleteTip, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); view.tipsArray = arr; this.autoHeightTextBox(); }; var onCommentsViewMouseOver = function () { me._isMouseOver = true; }; var onCommentsViewMouseOut = function () { me._isMouseOver = false; }; this.commentsView.on('item:add', addtooltip); this.commentsView.on('item:remove', addtooltip); this.commentsView.on('item:change', addtooltip); this.commentsView.cmpEl.on('mouseover', onCommentsViewMouseOver).on('mouseout', onCommentsViewMouseOut); this.commentsView.setStore(me.commentsStore); this.commentsView.onResetItems(); this.commentsView.on('item:click', function (picker, item, record, e) { var btn, showEditBox, showReplyBox, commentId, replyId, hideAddReply; function readdresolves() { me.update(); } btn = $(e.target); if (btn) { showEditBox = record.get('editTextInPopover'); showReplyBox = record.get('showReplyInPopover'); hideAddReply = record.get('hideAddReply'); commentId = record.get('uid'); replyId = btn.attr('data-value'); if (record.get('hint')) { me.fireEvent('comment:disableHint', [record]); if(!record.get('fullInfoInHint')) return; } if (btn.hasClass('btn-edit')) { var tip = btn.data('bs.tooltip'); if (tip) tip.dontShow = true; if (!_.isUndefined(replyId)) { me.fireEvent('comment:closeEditing', [commentId]); me.fireEvent('comment:editReply', [commentId, replyId, true]); this.replyId = replyId; this.autoHeightTextBox(); me.calculateSizeOfContent(); me.setLeftTop(me.arrowPosX, me.arrowPosY, me.leftX); me.calculateSizeOfContent(); readdresolves(); me.hookTextBox(); me.autoScrollToEditButtons(); this.setFocusToTextBox(); } else { if (!showEditBox) { me.fireEvent('comment:closeEditing'); record.set('editTextInPopover', true); me.fireEvent('comment:show', [commentId]); this.autoHeightTextBox(); me.calculateSizeOfContent(); me.setLeftTop(me.arrowPosX, me.arrowPosY, me.leftX); me.calculateSizeOfContent(); readdresolves(); me.hookTextBox(); me.autoScrollToEditButtons(); this.setFocusToTextBox(); } } } else if (btn.hasClass('btn-delete')) { var tip = btn.data('bs.tooltip'); if (tip) tip.dontShow = true; if (!_.isUndefined(replyId)) { me.fireEvent('comment:removeReply', [commentId, replyId]); me.calculateSizeOfContent(); me.setLeftTop(me.arrowPosX, me.arrowPosY, me.leftX); me.calculateSizeOfContent(); } else { me.fireEvent('comment:remove', [commentId]); } me.fireEvent('comment:closeEditing'); readdresolves(); } else if (btn.hasClass('user-reply')) { me.fireEvent('comment:closeEditing'); record.set('showReplyInPopover', true); me.calculateSizeOfContent(); me.setLeftTop(me.arrowPosX, me.arrowPosY, me.leftX); me.calculateSizeOfContent(); readdresolves(); this.autoHeightTextBox(); me.hookTextBox(); me.autoScrollToEditButtons(); this.setFocusToTextBox(); } else if (btn.hasClass('btn-reply', false)) { if (showReplyBox) { this.clearTextBoxBind(); me.fireEvent('comment:addReply', [commentId, this.getActiveTextBoxVal()]); me.fireEvent('comment:closeEditing'); me.calculateSizeOfContent(); readdresolves(); } } else if (btn.hasClass('btn-close', false)) { me.fireEvent('comment:closeEditing', [commentId]); me.calculateSizeOfContent(); me.fireEvent('comment:show', [commentId]); readdresolves(); } else if (btn.hasClass('btn-inner-edit', false)) { if (record.get('dummy')) { var commentVal = this.getActiveTextBoxVal(); me.clearDummyText(); if (commentVal.length > 0) me.fireEvent('comment:addDummyComment', [commentVal]); else { var text = me.$window.find('textarea:not(.user-message)'); if (text && text.length) setTimeout(function () { text.focus(); }, 10); } return; } this.clearTextBoxBind(); if (!_.isUndefined(this.replyId)) { me.fireEvent('comment:changeReply', [commentId, this.replyId, this.getActiveTextBoxVal()]); this.replyId = undefined; me.fireEvent('comment:closeEditing'); } else if (showEditBox) { me.fireEvent('comment:change', [commentId, this.getActiveTextBoxVal()]); me.fireEvent('comment:closeEditing'); me.calculateSizeOfContent(); } readdresolves(); } else if (btn.hasClass('btn-inner-close', false)) { if (record.get('dummy')) { me.clearDummyText(); me.hide(); return; } if (hideAddReply && this.getActiveTextBoxVal().length > 0) { me.saveText(); record.set('hideAddReply', false); this.getTextBox().val(me.textVal); this.autoHeightTextBox(); } else { this.clearTextBoxBind(); me.fireEvent('comment:closeEditing', [commentId]); } this.replyId = undefined; me.calculateSizeOfContent(); me.setLeftTop(me.arrowPosX, me.arrowPosY, me.leftX); me.calculateSizeOfContent(); readdresolves(); } else if (btn.hasClass('btn-resolve')) { var tip = btn.data('bs.tooltip'); if (tip) tip.dontShow = true; me.fireEvent('comment:resolve', [commentId]); readdresolves(); } } }); this.emailMenu = new Common.UI.Menu({ maxHeight: 200, cyclic: false, items: [] }).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}); me.$window.css({zIndex: '1001'}); }).on('hide:after', function () { me.$window.css({zIndex: '990'}); }); me.on({ 'show': function () { me.commentsView.autoHeightTextBox(); me.$window.find('textarea').keydown(function (event) { if (event.keyCode == Common.UI.Keys.ESC) { me.hide(true); // clear text in dummy comment } }); }, 'animate:before': function () { var text = me.$window.find('textarea'); if (text && text.length){ text.focus(); me.commentsView.disableTextBoxButton(text); } } }); } } var ReviewPopoverDataView = Common.UI.DataView.extend((function() { return { options : { handleSelect: false, scrollable: true, template: _.template('
'+ '
' ) } } })()); if (ReviewPopoverDataView) { if (this.reviewChangesView) { this.reviewChangesView.render($('#id-review-popover')); this.reviewChangesView.onResetItems(); } else { this.reviewChangesView = new ReviewPopoverDataView({ el: $('#id-review-popover'), itemTemplate: _.template(reviewTemplate) }); var addtooltip = function (dataview, view, record) { if (view.tipsArray) { view.tipsArray.forEach(function (item) { item.remove(); }); } var arr = [], btns = $(view.el).find('.btn-goto'); btns.tooltip({title: me.textFollowMove, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); btns = $(view.el).find('.btn-accept'); btns.tooltip({title: me.txtAccept, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); btns = $(view.el).find('.btn-reject'); btns.tooltip({title: me.txtReject, placement: 'cursor'}); btns.each(function (idx, item) { arr.push($(item).data('bs.tooltip').tip()); }); view.tipsArray = arr; }; this.reviewChangesView.on('item:add', addtooltip); this.reviewChangesView.on('item:remove', addtooltip); this.reviewChangesView.on('item:change', addtooltip); this.reviewChangesView.on('item:click', function (picker, item, record, e) { var btn = $(e.target); if (btn) { if (btn.hasClass('btn-accept')) { var tip = btn.data('bs.tooltip'); if (tip) tip.dontShow = true; me.fireEvent('reviewchange:accept', [record.get('changedata')]); } else if (btn.hasClass('btn-reject')) { var tip = btn.data('bs.tooltip'); if (tip) tip.dontShow = true; me.fireEvent('reviewchange:reject', [record.get('changedata')]); } else if (btn.hasClass('btn-delete')) { me.fireEvent('reviewchange:delete', [record.get('changedata')]); } else if (btn.hasClass('btn-goto')) { var tip = btn.data('bs.tooltip'); if (tip) tip.dontShow = true; me.fireEvent('reviewchange:goto', [record.get('changedata')]); } } }); this.reviewChangesView.setStore(me.reviewStore); this.reviewChangesView.onResetItems(); } } if (_.isUndefined(this.scroller)) { this.scroller = new Common.UI.Scroller({ el: window.find('#id-popover'), minScrollbarLength : 40, wheelSpeed: 10, alwaysVisibleY: true }); } }, showComments: function (animate, loadText, focus, showText) { this.options.animate = animate; var me = this, textBox = this.commentsView.getTextBox(); if (loadText && this.textVal) { textBox && textBox.val(this.textVal); } if (showText && showText.length) { textBox && textBox.val(showText); } this.show(animate); this.hookTextBox(); this._state.commentsVisible = true; }, showReview: function (animate, lock, lockuser) { this.show(animate); // this.reviewChangesView.cmpEl.find('.lock-area').toggleClass('hidden', !lock); // this.reviewChangesView.cmpEl.find('.lock-author').toggleClass('hidden', !lock || _.isEmpty(lockuser)).text(lockuser); this._state.reviewVisible = true; }, show: function (animate, loadText, focus, showText) { this.options.animate = animate; Common.UI.Window.prototype.show.call(this); if (this.scroller) { this.scroller.update({minScrollbarLength: 40, alwaysVisibleY: true}); } }, hideComments: function () { if (this.handlerHide) { this.handlerHide(); } this.hideTips(); this._state.commentsVisible = false; if (!this._state.reviewVisible) this.hide(); else this.calculateSizeOfContent(); }, hideReview: function () { if (this.handlerHide) { this.handlerHide(); } this.hideTips(); this._state.reviewVisible = false; if (!this._state.commentsVisible) this.hide(); else this.calculateSizeOfContent(); }, hide: function () { if (this.handlerHide) { this.handlerHide.apply(this, arguments); } this.hideTips(); Common.UI.Window.prototype.hide.call(this); if (!_.isUndefined(this.e) && this.e.keyCode == Common.UI.Keys.ESC) { this.e.preventDefault(); this.e.stopImmediatePropagation(); this.e = undefined; } }, // CommentsPopover update: function (needRender) { if (this.commentsView && needRender) this.commentsView.onResetItems(); if (this.scroller) { this.scroller.update({minScrollbarLength: 40, alwaysVisibleY: true}); } }, isVisible: function () { return (this.$window && this.$window.is(':visible')); }, setLeftTop: function (posX, posY, leftX, loadInnerValues, retainContent) { if (!this.$window) this.render(); if (loadInnerValues) { posX = this.arrowPosX; posY = this.arrowPosY; leftX = this.leftX; } if (_.isUndefined(posX) && _.isUndefined(posY)) return; this.arrowPosX = posX; this.arrowPosY = posY; this.leftX = leftX; var commentsView = $('#id-popover'), arrowView = $('#id-comments-arrow'), editorView = $('#editor_sdk'), editorBounds = null, sdkBoundsHeight = 0, sdkBoundsTop = 0, sdkBoundsLeft = 0, sdkPanelRight = '', sdkPanelRightWidth = 0, sdkPanelLeft = '', sdkPanelLeftWidth = 0, sdkPanelThumbs = '', // for PE sdkPanelThumbsWidth = 0, // for PE sdkPanelTop = '', sdkPanelHeight = 0, leftPos = 0, windowWidth = 0, outerHeight = 0, topPos = 0, sdkBoundsTopPos = 0; if (commentsView && arrowView && editorView && editorView.get(0)) { editorBounds = editorView.get(0).getBoundingClientRect(); if (editorBounds) { sdkBoundsHeight = editorBounds.height - this.sdkBounds.padding * 2; this.$window.css({maxHeight: sdkBoundsHeight + 'px'}); this.sdkBounds.width = editorBounds.width; this.sdkBounds.height = editorBounds.height; // LEFT CORNER if (!_.isUndefined(posX)) { sdkPanelRight = $('#id_vertical_scroll'); if (sdkPanelRight.length) { sdkPanelRightWidth = (sdkPanelRight.css('display') !== 'none') ? sdkPanelRight.width() : 0; } else { sdkPanelRight = $('#ws-v-scrollbar'); if (sdkPanelRight.length) { sdkPanelRightWidth = (sdkPanelRight.css('display') !== 'none') ? sdkPanelRight.width() : 0; } } this.sdkBounds.width -= sdkPanelRightWidth; sdkPanelLeft = $('#id_panel_left'); if (sdkPanelLeft.length) { sdkPanelLeftWidth = (sdkPanelLeft.css('display') !== 'none') ? sdkPanelLeft.width() : 0; } sdkPanelThumbs = $('#id_panel_thumbnails'); if (sdkPanelThumbs.length) { sdkPanelThumbsWidth = (sdkPanelThumbs.css('display') !== 'none') ? sdkPanelThumbs.width() : 0; this.sdkBounds.width -= sdkPanelThumbsWidth; } leftPos = Math.min(sdkBoundsLeft + posX + this.arrow.width, sdkBoundsLeft + this.sdkBounds.width - this.$window.outerWidth() - 25); leftPos = Math.max(sdkBoundsLeft + sdkPanelLeftWidth + this.arrow.width, leftPos); arrowView.removeClass('right top bottom').addClass('left'); arrowView.css({left: ''}); if (!_.isUndefined(leftX)) { windowWidth = this.$window.outerWidth(); if (windowWidth) { if ((posX + windowWidth > this.sdkBounds.width - this.arrow.width + 5) && (this.leftX > windowWidth)) { leftPos = this.leftX - windowWidth + sdkBoundsLeft - this.arrow.width; arrowView.removeClass('left').addClass('right'); } else { leftPos = sdkBoundsLeft + posX + this.arrow.width; } } } this.$window.css('left', leftPos + 'px'); } // TOP CORNER if (!_.isUndefined(posY)) { sdkPanelTop = $('#id_panel_top'); sdkBoundsTopPos = sdkBoundsTop; if (sdkPanelTop.length) { sdkPanelHeight = (sdkPanelTop.css('display') !== 'none') ? sdkPanelTop.height() : 0; sdkBoundsTopPos += this.sdkBounds.paddingTop; } else { sdkPanelTop = $('#ws-h-scrollbar'); if (sdkPanelTop.length) { sdkPanelHeight = (sdkPanelTop.css('display') !== 'none') ? sdkPanelTop.height() : 0; } } this.sdkBounds.height -= sdkPanelHeight; outerHeight = this.$window.outerHeight(); topPos = Math.min(sdkBoundsTop + sdkBoundsHeight - outerHeight, this.arrowPosY + sdkBoundsTop - this.arrow.height); topPos = Math.max(topPos, sdkBoundsTopPos); if (parseInt(arrowView.css('top')) + this.arrow.height > outerHeight) { arrowView.css({top: (outerHeight-this.arrow.height) + 'px'}); } this.$window.css('top', topPos + 'px'); } } } if (!retainContent || this.isOverCursor()) this.calculateSizeOfContent(); }, calculateSizeOfContent: function (testVisible) { if (testVisible && !this.$window.is(':visible')) return; this.$window.css({overflow: 'hidden'}); var arrowView = $('#id-comments-arrow'), commentsView = $('#id-popover'), contentBounds = null, editorView = null, editorBounds = null, sdkBoundsHeight = 0, sdkBoundsTop = 0, sdkBoundsLeft = 0, sdkPanelTop = '', sdkPanelHeight = 0, arrowPosY = 0, arrowPosX = 0, windowHeight = 0, outerHeight = 0, topPos = 0, sdkBoundsTopPos = 0; if (commentsView && arrowView && commentsView.get(0)) { var scrollPos = this.scroller.getScrollTop(); commentsView.css({height: '100%'}); contentBounds = commentsView.get(0).getBoundingClientRect(); if (contentBounds) { editorView = $('#editor_sdk'); if (editorView && editorView.get(0)) { editorBounds = editorView.get(0).getBoundingClientRect(); if (editorBounds) { sdkBoundsHeight = editorBounds.height - this.sdkBounds.padding * 2; sdkBoundsTopPos = sdkBoundsTop; windowHeight = this.$window.outerHeight(); // TOP CORNER sdkPanelTop = $('#id_panel_top'); if (sdkPanelTop.length) { sdkPanelHeight = (sdkPanelTop.css('display') !== 'none') ? sdkPanelTop.height() : 0; sdkBoundsTopPos += this.sdkBounds.paddingTop; } else { sdkPanelTop = $('#ws-h-scrollbar'); if (sdkPanelTop.length) { sdkPanelHeight = (sdkPanelTop.css('display') !== 'none') ? sdkPanelTop.height() : 0; } } outerHeight = Math.max(commentsView.outerHeight(), this.$window.outerHeight()); var movePos = this.isOverCursor(); if (movePos) { var leftPos = parseInt(this.$window.css('left')) - this.arrow.width, newTopDown = movePos[1][1] + sdkPanelHeight + this.arrow.width,// try move down newTopUp = movePos[0][1] + sdkPanelHeight - this.arrow.width, // try move up isMoveDown = false; if (newTopDown + outerHeight>sdkBoundsTop + sdkBoundsHeight) { var diffDown = sdkBoundsTop + sdkBoundsHeight - newTopDown; if (newTopUp - outerHeight 0) { if (contentBounds.top + outerHeight > sdkBoundsHeight + sdkBoundsTop || contentBounds.height === 0) { topPos = Math.min(sdkBoundsTop + sdkBoundsHeight - outerHeight, this.arrowPosY + sdkBoundsTop - this.arrow.height); topPos = Math.max(topPos, sdkBoundsTopPos); this.$window.css({top: topPos + 'px'}); } } arrowPosY = Math.max(this.arrow.margin, this.arrowPosY - (sdkBoundsHeight - outerHeight) - this.arrow.height); arrowPosY = Math.min(arrowPosY, outerHeight - this.arrow.margin - this.arrow.height); arrowView.css({top: arrowPosY + 'px', left: ''}); arrowView.removeClass('top bottom right'); arrowView.addClass('left'); } } } } } this.$window.css({overflow: ''}); if (this.scroller) { this.scroller.update({minScrollbarLength: 40, alwaysVisibleY: true}); } }, isOverCursor: function() { if (!this.api.asc_GetSelectionBounds) return; var p = this.api.asc_GetSelectionBounds(), isCursor = Math.abs(p[0][0] - p[1][0])<0.1 && Math.abs(p[0][1] - p[1][1])<0.1 && Math.abs(p[2][0] - p[3][0])<0.1 && Math.abs(p[2][1] - p[3][1])<0.1, sdkPanelLeft = $('#id_panel_left'), sdkPanelLeftWidth = 0; if (sdkPanelLeft.length) sdkPanelLeftWidth = (sdkPanelLeft.css('display') !== 'none') ? sdkPanelLeft.width() : 0; var x0 = p[0][0] + sdkPanelLeftWidth, y0 = p[0][1], x1 = p[isCursor ? 2 : 1][0] + sdkPanelLeftWidth, y1 = p[isCursor ? 2 : 1][1]; var leftPos = parseInt(this.$window.css('left')) - this.arrow.width, windowWidth = this.$window.outerWidth() + this.arrow.width, topPos = parseInt(this.$window.css('top')), windowHeight = this.$window.outerHeight(); if (x0>leftPos && x0topPos && y0leftPos && x1topPos && y1y1) ? [x0, y0] : [x1, y1],// try move down newUp = (y0=0; i--) { if (val.charCodeAt(i) == 32 /*space*/ || val.charCodeAt(i) == 13 /*enter*/ || val.charCodeAt(i) == 10 /*new line*/ || val.charCodeAt(i) == 9 /*tab*/) { left = i+1; break; } } for (var i=start; i<=right; i++) { if (val.charCodeAt(i) == 32 || val.charCodeAt(i) == 13 || val.charCodeAt(i) == 10 || val.charCodeAt(i) == 9) { right = i-1; break; } } var str = val.substring(left, right+1), res = str.match(/^(?:[@]|[+](?!1))(\S*)/); if (res && res.length>1) { str = res[1]; // send to show email menu me.onEmailListMenu(str, left, right); } else me.onEmailListMenu(); // hide email menu }); } }, hideTips: function () { if (this.commentsView) _.each(this.commentsView.dataViewItems, function (item) { if (item.tipsArray) { item.tipsArray.forEach(function (item) { item.hide(); }); } }, this); if (this.reviewChangesView) _.each(this.reviewChangesView.dataViewItems, function (item) { if (item.tipsArray) { item.tipsArray.forEach(function (item) { item.hide(); }); } }, this); if (this.emailMenu && this.emailMenu.rendered) this.emailMenu.cmpEl.css('display', 'none'); }, isCommentsViewMouseOver: function () { return this._isMouseOver; }, setReviewStore: function(store) { this.reviewStore = store; if (this.reviewChangesView) this.reviewChangesView.setStore(this.reviewStore); }, setCommentsStore: function(store) { this.commentsStore = store; if (this.commentsView) this.commentsView.setStore(this.commentsStore); }, setUsers: function(data) { this.externalUsers = data.users || []; this.isUsersLoading = false; this._state.emailSearch && this.onEmailListMenu(this._state.emailSearch.str, this._state.emailSearch.left, this._state.emailSearch.right); this._state.emailSearch = null; }, clearUsers: function() { this.externalUsers = []; }, getPopover: function(options) { if (!this.popover) this.popover = new Common.Views.ReviewPopover(options); return this.popover; }, autoScrollToEditButtons: function () { var button = $('#id-comments-change-popover'), // TODO: add to cache btnBounds = null, contentBounds = this.$window[0].getBoundingClientRect(), moveY = 0, padding = 7; if (button.length) { btnBounds = button.get(0).getBoundingClientRect(); if (btnBounds && contentBounds) { moveY = contentBounds.bottom - (btnBounds.bottom + padding); if (moveY < 0) { this.scroller.scrollTop(this.scroller.getScrollTop() - moveY); } } } }, onEmailListMenu: function(str, left, right, show) { var me = this, users = me.externalUsers, menu = me.emailMenu; if (users.length<1) { this._state.emailSearch = { str: str, left: left, right: right }; if (this.isUsersLoading) return; this.isUsersLoading = true; Common.Gateway.requestUsers(); return; } if (typeof str == 'string') { var menuContainer = me.$window.find(Common.Utils.String.format('#menu-container-{0}', menu.id)), textbox = this.commentsView.getTextBox(), textboxDom = textbox ? textbox[0] : null, showPoint = textboxDom ? [textboxDom.offsetLeft, textboxDom.offsetTop + textboxDom.clientHeight + 3] : [0, 0]; if (!menu.rendered) { // Prepare menu container if (menuContainer.length < 1) { menuContainer = $(Common.Utils.String.format('', menu.id)); me.$window.append(menuContainer); } menu.render(menuContainer); menu.cmpEl.css('min-width', textboxDom ? textboxDom.clientWidth : 220); menu.cmpEl.attr({tabindex: "-1"}); menu.on('hide:after', function(){ setTimeout(function(){ var tb = me.commentsView.getTextBox(); tb && tb.focus(); }, 10); }); } for (var i = 0; i < menu.items.length; i++) { menu.removeItem(menu.items[i]); i--; } if (users.length>0) { str = str.toLowerCase(); if (str.length>0) { users = _.filter(users, function(item) { return (item.email && 0 === item.email.toLowerCase().indexOf(str) || item.name && 0 === item.name.toLowerCase().indexOf(str)) }); } var tpl = _.template('' + '
<%= Common.Utils.String.htmlEncode(caption) %>
' + '
<%= Common.Utils.String.htmlEncode(options.value) %>
' + '
'), divider = false; _.each(users, function(menuItem, index) { if (divider && !menuItem.hasAccess) { divider = false; menu.addItem(new Common.UI.MenuItem({caption: '--'})); } if (menuItem.email && menuItem.name) { var mnu = new Common.UI.MenuItem({ caption : menuItem.name, value : menuItem.email, template : tpl }).on('click', function(item, e) { me.insertEmailToTextbox(item.options.value, left, right); }); menu.addItem(mnu); if (menuItem.hasAccess) divider = true; } }); } if (menu.items.length>0) { menuContainer.css({left: showPoint[0], top : showPoint[1]}); menu.menuAlignEl = textbox; menu.show(); menu.cmpEl.css('display', ''); menu.alignPosition('bl-tl', -5); menu.scroller.update({alwaysVisibleY: true}); } else { menu.rendered && menu.cmpEl.css('display', 'none'); } } else { menu.rendered && menu.cmpEl.css('display', 'none'); } }, insertEmailToTextbox: function(str, left, right) { var textBox = this.commentsView.getTextBox(); if (!textBox) return; var val = textBox.val(); textBox.val(val.substring(0, left) + '+' + str + ' ' + val.substring(right+1, val.length)); setTimeout(function(){ textBox[0].selectionStart = textBox[0].selectionEnd = left + str.length + 2; }, 10); }, textAddReply : 'Add Reply', textAdd : "Add", textCancel : 'Cancel', textEdit : 'Edit', textReply : 'Reply', textClose : 'Close', textResolve : 'Resolve', textOpenAgain : "Open Again", textFollowMove : 'Follow Move', textMention : '+mention will provide access to the document and send an email', textMentionNotify : '+mention will notify the user via email', textEnterComment : 'Enter your comment here', textViewResolved : 'You have not permission for reopen comment', txtAccept: 'Accept', txtReject: 'Reject', txtEditTip: 'Edit', txtDeleteTip: 'Delete' }, Common.Views.ReviewPopover || {})) });