/* * * (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"><%= 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(); 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) { 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(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.txtMessage.bind('input propertychange', _.bind(this.updateHeightTextBox, this)); }, 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 || {})) });