web-apps/apps/common/main/lib/view/Chat.js
2022-11-10 00:07:33 +03:00

435 lines
18 KiB
JavaScript

/*
*
* (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
*
*/
/**
* Chat.js
*
* View
*
* Created by Maxim Kadushkin on 27 February 2014
* Copyright (c) 2018 Ascensio System SIA. All rights reserved.
*
*/
if (Common === undefined)
var Common = {};
Common.Views = Common.Views || {};
define([
'text!common/main/lib/template/Chat.template',
'common/main/lib/util/utils',
'common/main/lib/component/BaseView',
'common/main/lib/component/Layout'
], function (template) {
'use strict';
Common.Views.Chat = Common.UI.BaseView.extend(_.extend({
el: '#left-panel-chat',
template: _.template(template),
storeUsers: undefined,
storeMessages: undefined,
tplUser: ['<li id="<%= user.get("iid") %>"<% if (!user.get("online")) { %> class="offline"<% } %>>',
'<div class="name"><div class="color" style="background-color: <%= user.get("color") %>;" ></div><%= scope.getUserName(user.get("username")) %>',
'</div>',
'</li>'].join(''),
templateUserList: _.template('<ul>' +
'<% for (originalId in users) { %>' +
'<%= _.template(usertpl)({user: users[originalId][0], scope: scope}) %>' +
'<% } %>' +
'</ul>'),
tplMsg: ['<li>',
'<% if (msg.get("type")==1) { %>',
'<div class="message service" data-can-copy="true"><%= msg.get("message") %></div>',
'<% } else { %>',
'<div class="user-name" data-can-copy="true">',
'<div class="color" style="display: inline-block; background-color: <% if (msg.get("usercolor")!==null) { %><%=msg.get("usercolor")%><% } else { %> #cfcfcf <% } %>; " ></div><%= scope.getUserName(msg.get("username")) %>',
'</div>',
'<label class="message user-select" data-can-copy="true" tabindex="-1" oo_editor_input="true"><%= msg.get("message") %></label>',
'<% } %>',
'</li>'].join(''),
templateMsgList: _.template('<ul>' +
'<% _.each(messages, function(item) { %>' +
'<%= _.template(msgtpl)({msg: item, scope: scope}) %>' +
'<% }); %>' +
'</ul>'),
events: {
},
usersBoxHeight: 72,
messageBoxHeight: 70,
addMessageBoxHeight: 110,
initialize: function(options) {
_.extend(this, options);
Common.UI.BaseView.prototype.initialize.call(this, arguments);
this.storeUsers.bind({
add : _.bind(this._onResetUsers, this),
change : _.bind(this._onResetUsers, this),
reset : _.bind(this._onResetUsers, this)
});
this.storeMessages.bind({
add : _.bind(this._onAddMessage, this),
reset : _.bind(this._onResetMessages, this)
});
},
render: function(el) {
el = el || this.el;
$(el).html(this.template({scope: this, maxMsgLength: Asc.c_oAscMaxCellOrCommentLength}));
this.panelBox = $('#chat-box', this.el);
this.panelUsers = $('#chat-users', this.el);
this.panelMessages = $('#chat-messages', this.el);
this.txtMessage = $('#chat-msg-text', this.el);
this.panelOptions = $('#chat-options', this.el);
this.panelUsers.scroller = new Common.UI.Scroller({
el : $('#chat-users'),
useKeyboard : true,
minScrollbarLength : 25
});
this.panelMessages.scroller = new Common.UI.Scroller({
el : $('#chat-messages'),
includePadding : true,
useKeyboard : true,
minScrollbarLength : 40
});
$('#chat-msg-btn-add', this.el).on('click', _.bind(this._onBtnAddMessage, this));
this.txtMessage.on('keydown', _.bind(this._onKeyDown, this));
this.setupLayout();
this.trigger('render:after', this);
return this;
},
focus: function() {
var me = this;
_.defer(function(){
me.txtMessage.focus();
}, 100);
this.updateLayout(true);
this.setupAutoSizingTextBox();
},
_onKeyDown: function(event) {
if (event.keyCode == Common.UI.Keys.RETURN) {
if ((event.ctrlKey || event.metaKey) && !event.altKey) {
this._onBtnAddMessage(event);
}
} else
if (event.keyCode == Common.UI.Keys.ESC && !Common.UI.HintManager.isHintVisible()) {
this.hide();
}
},
_onResetUsers: function(c, opts) {
if (this.panelUsers) {
this.panelUsers.html(this.templateUserList({users: this.storeUsers.chain().filter(function(item){return item.get('online');}).groupBy(function(item) {return item.get('idOriginal');}).value(),
usertpl: this.tplUser, scope: this}));
this.panelUsers.scroller.update({minScrollbarLength : 25, alwaysVisibleY: true});
}
},
_onAddMessage: function(m, c, opts) {
if (this.panelMessages) {
var content = this.panelMessages.find('ul');
if (content && content.length) {
this._prepareMessage(m);
content.append(_.template(this.tplMsg)({msg: m, scope: this}));
// scroll to end
this.panelMessages.scroller.update({minScrollbarLength : 40, alwaysVisibleY: true});
this.panelMessages.scroller.scrollTop(content.get(0).getBoundingClientRect().height);
}
}
},
_onResetMessages: function(c, opts) {
if (this.panelMessages) {
var user, color;
c.each(function(msg){
this._prepareMessage(msg);
}, this);
this.panelMessages.html(this.templateMsgList({messages: c.models, msgtpl: this.tplMsg, scope: this}));
this.panelMessages.scroller.update({minScrollbarLength : 40, alwaysVisibleY: true});
}
},
_onBtnAddMessage: function(e) {
if (this.txtMessage) {
this.fireEvent('message:add', [this, this.txtMessage.val().trim()]);
this.txtMessage.val('');
this.focus();
}
},
_prepareMessage: function(m) {
var user = this.storeUsers.findOriginalUser(m.get('userid'));
m.set({
usercolor : user ? user.get('color') : null,
message : this._pickLink(m.get('message'))
}, {silent:true});
},
_pickLink: function(message) {
var arr = [], offset, len;
message.replace(Common.Utils.ipStrongRe, function(subStr) {
var result = /[\.,\?\+;:=!\(\)]+$/.exec(subStr);
if (result)
subStr = subStr.substring(0, result.index);
offset = arguments[arguments.length-2];
arr.push({start: offset, end: subStr.length+offset, str: '<a href="' + subStr + '" target="_blank" data-can-copy="true">' + subStr + '</a>'});
return '';
});
if (message.length<1000 || message.search(/\S{255,}/)<0)
message.replace(Common.Utils.hostnameStrongRe, function(subStr) {
var result = /[\.,\?\+;:=!\(\)]+$/.exec(subStr);
if (result)
subStr = subStr.substring(0, result.index);
var ref = (! /(((^https?)|(^ftp)):\/\/)/i.test(subStr) ) ? ('http://' + subStr) : subStr;
offset = arguments[arguments.length-2];
len = subStr.length;
var elem = _.find(arr, function(item){
return ( (offset>=item.start) && (offset<item.end) ||
(offset<=item.start) && (offset+len>item.start));
});
if (!elem)
arr.push({start: offset, end: len+offset, str: '<a href="' + ref + '" target="_blank" data-can-copy="true">' + subStr + '</a>'});
return '';
});
message.replace(Common.Utils.emailStrongRe, function(subStr) {
var ref = (! /((^mailto:)\/\/)/i.test(subStr) ) ? ('mailto:' + subStr) : subStr;
offset = arguments[arguments.length-2];
len = subStr.length;
var elem = _.find(arr, function(item){
return ( (offset>=item.start) && (offset<item.end) ||
(offset<=item.start) && (offset+len>item.start));
});
if (!elem)
arr.push({start: offset, end: len+offset, str: '<a href="' + ref + '">' + subStr + '</a>'});
return '';
});
arr = _.sortBy(arr, function(item){ return item.start; });
var str_res = (arr.length>0) ? ( Common.Utils.String.htmlEncode(message.substring(0, arr[0].start)) + arr[0].str) : Common.Utils.String.htmlEncode(message);
for (var i=1; i<arr.length; i++) {
str_res += (Common.Utils.String.htmlEncode(message.substring(arr[i-1].end, arr[i].start)) + arr[i].str);
}
if (arr.length>0) {
str_res += Common.Utils.String.htmlEncode(message.substring(arr[i-1].end, message.length));
}
return str_res;
},
getUserName: function (username) {
return Common.Utils.String.htmlEncode(AscCommon.UserInfoParser.getParsedName(username));
},
hide: function () {
Common.UI.BaseView.prototype.hide.call(this,arguments);
this.fireEvent('hide', this );
this.textBoxAutoSizeLocked = undefined;
},
setupLayout: function () {
var me = this, parent = $(me.el), items = this.panelBox.find(' > .layout-item');
me.layout = new Common.UI.VBoxLayout({
box: this.panelBox,
items: [
{el: items[0], rely: true, behaviour: 'splitter',
resize: {
hidden: false,
autohide: false,
fmin: (function () {
return me.usersBoxHeight;
}),
fmax: (function () {
return me.panelBox.height() * 0.5 - me.messageBoxHeight;
})
}},
{el: items[1], rely: true, behaviour: 'splitter',
resize: {
hidden: false,
autohide: false,
fmin: (function () {
return Math.max(me.messageBoxHeight + me.usersBoxHeight, me.panelBox.height() * 0.5);
}),
fmax: (function () {
return me.panelBox.height() - me.addMessageBoxHeight;
})
}},
{el: items[2], stretch: true}
]
});
me.layout.on('layout:resizedrag', function(resizer) {
me.updateScrolls();
me.usersCachedHeigt = me.panelUsers.height() + 8 + 1; // resizeHeight * 2 + 1
if (!resizer.index) {
me.textBoxAutoSizeLocked = true;
}
}, this);
$(window).on('resize', function() {
if (parent.css('display') !== 'none') {
me.updateLayout();
}
});
this.updateLayout();
// default sizes
var height = this.panelBox.height();
this.layout.setResizeValue(0, this.usersBoxHeight);
this.layout.setResizeValue(1,
Math.max (this.addMessageBoxHeight,
Math.max (height * 0.5, height - me.panelOptions.height() - 4)));
// text box setup autosize input text
this.setupAutoSizingTextBox();
this.disableTextBoxButton($(this.txtMessage));
this.txtMessage.bind('input propertychange', _.bind(this.onTextareaInput, this));
},
onTextareaInput: function(event) {
this.updateHeightTextBox(event);
this.disableTextBoxButton($(event.target));
},
disableTextBoxButton: function(textboxEl) {
var button = $(textboxEl.siblings('#chat-msg-btn-add')[0]);
if(textboxEl.val().length > 0) {
button.removeAttr('disabled');
button.removeClass('disabled');
} else {
button.attr('disabled', true);
button.addClass('disabled');
}
},
updateLayout: function (applyUsersAutoSizig) {
var me = this;
var height = this.panelBox.height();
me.layout.setResizeValue(1,
Math.max (me.addMessageBoxHeight,
Math.max (height * 0.5, height - me.panelOptions.height() - 4)));
if (applyUsersAutoSizig) {
var oldHeight = this.panelUsers.css('height');
this.panelUsers.css('height', '1px');
var content = this.panelUsers.get(0).scrollHeight;
me.layout.setResizeValue(0, Math.max(me.usersBoxHeight,
Math.min(content+2, Math.floor(height * 0.5) - me.messageBoxHeight)));
} else {
me.layout.setResizeValue(0, Math.max(me.usersBoxHeight,
Math.min(me.usersCachedHeigt + 2, Math.floor(height * 0.5) - me.messageBoxHeight)));
}
me.updateScrolls();
me.updateHeightTextBox(null);
},
setupAutoSizingTextBox: function () {
this.lineHeight = 0;
this.minHeight = 44;
this.lineHeight = parseInt(this.txtMessage.css('lineHeight'), 10) * 1.25; // TODO: need fix
this.updateHeightTextBox(true);
},
updateHeightTextBox: function (event) {
var textBox = this.txtMessage,
controlHeight, contentHeight, height,
textBoxMinHeightIndent = 36 + 4; // 4px - autosize line height + big around border
height = this.panelBox.height();
if (event && 0 == textBox.val().length) {
this.layout.setResizeValue(1, Math.max(this.addMessageBoxHeight, height - this.addMessageBoxHeight));
this.textBoxAutoSizeLocked = undefined;
this.updateScrolls();
return;
}
if (!_.isUndefined(this.textBoxAutoSizeLocked))
return;
controlHeight = textBox.height();
contentHeight = textBox.get(0).scrollHeight;
// calculate text content height
textBox.css({height: this.minHeight + 'px'});
controlHeight = textBox.height();
contentHeight = Math.max(textBox.get(0).scrollHeight + this.lineHeight, 1);
textBox.css({height: '100%'});
height = this.panelBox.height();
if (this.layout.setResizeValue(1, Math.max(this.addMessageBoxHeight, Math.min(height - contentHeight - textBoxMinHeightIndent, height - this.addMessageBoxHeight))))
this.updateScrolls(); // update when resize position changed
},
updateScrolls: function () {
if (this.panelUsers && this.panelUsers.scroller && this.panelMessages && this.panelMessages.scroller) {
this.panelUsers.scroller.update({minScrollbarLength: 25, alwaysVisibleY: true});
this.panelMessages.scroller.update({minScrollbarLength: 40, alwaysVisibleY: true});
}
},
textSend: "Send"
}, Common.Views.Chat || {}))
});