You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
753 lines
59 KiB
753 lines
59 KiB
/******/ (() => { // webpackBootstrap
|
|
/******/ var __webpack_modules__ = ({
|
|
|
|
/***/ "./src/chat/chat.js":
|
|
/*!**************************!*\
|
|
!*** ./src/chat/chat.js ***!
|
|
\**************************/
|
|
/***/ ((module, __unused_webpack_exports, __webpack_require__) => {
|
|
|
|
/* Licensed under the Apache License, Version 2.0 (the "License") http://www.apache.org/licenses/LICENSE-2.0 */
|
|
const OmUtil = __webpack_require__(/*! ../main/omutils */ "../main/omutils");
|
|
const Settings = __webpack_require__(/*! ../main/settings */ "../main/settings");
|
|
const CSSEmoticon = __webpack_require__(/*! ./cssemoticons */ "./src/chat/cssemoticons.js");
|
|
|
|
const msgIdPrefix = 'chat-msg-id-'
|
|
, closedSize = 20
|
|
, closedSizePx = closedSize + "px"
|
|
, doneTypingInterval = 5000 //time in ms, 5 second for example
|
|
, SEND_ENTER = 'enter', SEND_CTRL = 'ctrl'
|
|
, audio = new Audio('./public/chat_message.mp3')
|
|
;
|
|
let chatPanel, ctrlBlk, tabs, openedHeight = "345px", openedWidth = "300px", allPrefix = "All"
|
|
, roomPrefix = "Room ", typingTimer, roomMode = false
|
|
, editor = $('#chatMessage .wysiwyg-editor'), muted = false, sendOn, DEF_SEND
|
|
, userId, inited = false, newMsgNotification
|
|
;
|
|
|
|
function __setCssVar(key, _val) {
|
|
const val = ('' + _val).endsWith('px') ? _val : _val + 'px';
|
|
if (roomMode) {
|
|
if (typeof(Room) == 'object' && typeof(Room.setCssVar) === 'function') {
|
|
Room.setCssVar(key, val);
|
|
}
|
|
} else {
|
|
OmUtil.setCssVar(key, val);
|
|
}
|
|
}
|
|
function __setCssWidth(val) {
|
|
__setCssVar('--chat-width', val)
|
|
}
|
|
function __setCssHeight(val) {
|
|
__setCssVar('--chat-height', val)
|
|
}
|
|
function _load() {
|
|
const s = Settings.load();
|
|
if (typeof(s.chat) === 'undefined') {
|
|
s.chat = {
|
|
muted: false
|
|
, sendOn: DEF_SEND
|
|
};
|
|
Settings.save(s)
|
|
}
|
|
muted = s.chat.muted === true;
|
|
sendOn = s.chat.sendOn === SEND_ENTER ? SEND_ENTER : SEND_CTRL;
|
|
return s;
|
|
}
|
|
function _updateAudioBtn() {
|
|
const btn = $('#chat .audio');
|
|
btn.removeClass('sound' + (muted ? '' : '-mute')).addClass('sound' + (muted ? '-mute' : ''))
|
|
.attr('title', btn.data(muted ? 'sound-enabled' : 'sound-muted'));
|
|
}
|
|
function _updateSendBtn() {
|
|
const btn = $('#chat .send-btn')
|
|
, ctrl = sendOn === SEND_CTRL;
|
|
if (ctrl) {
|
|
btn.addClass('send-ctrl');
|
|
editor.off('keydown', _sendOnEnter).keydown('Ctrl+return', _sendOnEnter);
|
|
} else {
|
|
btn.removeClass('send-ctrl');
|
|
editor.off('keydown', _sendOnEnter).keydown('return', _sendOnEnter);
|
|
}
|
|
btn.attr('title', btn.data(ctrl ? 'send-ctrl' : 'send-enter'));
|
|
}
|
|
function _sendOnEnter() {
|
|
$('#chat .send').trigger('click');
|
|
}
|
|
function doneTyping () {
|
|
typingTimer = null;
|
|
chatActivity('typing_stop', $('.room-block .room-container').data('room-id'));
|
|
}
|
|
function _emtClick() {
|
|
_editorAppend($(this).data('emt'));
|
|
}
|
|
function initToolbar() {
|
|
const emots = CSSEmoticon.emoticons;
|
|
const rowSize = 20, emotMenuList = $('#emotMenuList');
|
|
emotMenuList.html('');
|
|
let row;
|
|
for (let i = 0; i < emots.length; ++i) {
|
|
if (i % rowSize === 0) {
|
|
row = $('<tr></tr>');
|
|
emotMenuList.append(row);
|
|
}
|
|
row.append($('<td>').append(
|
|
$('<div>').addClass('emt').html(CSSEmoticon.emoticonize(emots[i]))
|
|
.data('emt', emots[i]).click(_emtClick)
|
|
));
|
|
}
|
|
const emtBtn = $('#emoticons');
|
|
emtBtn.html('');
|
|
emtBtn.append(' ' + CSSEmoticon.emoticonize(':)'));
|
|
const a = $('#chat .audio');
|
|
const sbtn = $('#chat .send-btn');
|
|
{ //scope
|
|
_load();
|
|
_updateAudioBtn();
|
|
_updateSendBtn()
|
|
}
|
|
a.off().click(function() {
|
|
const s = _load();
|
|
muted = s.chat.muted = !s.chat.muted;
|
|
_updateAudioBtn();
|
|
Settings.save(s);
|
|
});
|
|
sbtn.off().click(function() {
|
|
const s = _load();
|
|
sendOn = s.chat.sendOn = s.chat.sendOn !== SEND_CTRL ? SEND_CTRL : SEND_ENTER;
|
|
_updateSendBtn();
|
|
Settings.save(s);
|
|
});
|
|
$('#chat #hyperlink').parent().find('button').off().click(function() {
|
|
_insertLink();
|
|
});
|
|
CSSEmoticon.animate();
|
|
}
|
|
function isClosed() {
|
|
return chatPanel.hasClass('closed');
|
|
}
|
|
function activateTab(id) {
|
|
if (isClosed()) {
|
|
tabs.find('.nav.nav-tabs .nav-link').each(function() {
|
|
const self = $(this)
|
|
, tabId = self.attr('aria-controls')
|
|
, tab = $('#' + tabId);
|
|
|
|
if (tabId === id) {
|
|
self.addClass('active');
|
|
tab.addClass('active');
|
|
self.attr('aria-selected', true);
|
|
} else {
|
|
self.removeClass('active');
|
|
tab.removeClass('active');
|
|
self.attr('aria-selected', false);
|
|
}
|
|
});
|
|
} else {
|
|
$('#chatTabs li a[aria-controls="' + id + '"]').tab('show');
|
|
}
|
|
$('#activeChatTab').val(id).trigger('change');
|
|
}
|
|
function _reinit(opts) {
|
|
userId = opts.userId;
|
|
allPrefix = opts.all;
|
|
roomPrefix = opts.room;
|
|
DEF_SEND = opts.sendOnEnter === true ? SEND_ENTER : SEND_CTRL;
|
|
sendOn = DEF_SEND;
|
|
chatPanel = $('#chatPanel');
|
|
clearTimeout(chatPanel.data('timeout'));
|
|
ctrlBlk = $('#chatPopup .control.block');
|
|
newMsgNotification = ctrlBlk.data('new-msg');
|
|
editor = $('#chatMessage .wysiwyg-editor');
|
|
initToolbar();
|
|
tabs = $("#chatTabs");
|
|
tabs.off().on('shown.bs.tab', function (e) {
|
|
const ct = $(e.target).attr('aria-controls');
|
|
_scrollDown($('#' + ct));
|
|
$('#activeChatTab').val(ct).trigger('change');
|
|
});
|
|
tabs.delegate(".btn.close-chat", "click", function() {
|
|
const panelId = $(this).closest("a").attr("aria-controls");
|
|
_removeTab(panelId);
|
|
$('#chatTabs li:last-child a').tab('show');
|
|
});
|
|
if (roomMode) {
|
|
_removeResize();
|
|
} else {
|
|
ctrlBlk.attr('title', '');
|
|
chatPanel.removeClass('room opened').addClass('closed')
|
|
.off('mouseenter mouseleave').resizable({
|
|
handles: 'n, ' + (Settings.isRtl ? 'e' : 'w')
|
|
, disabled: isClosed()
|
|
, minHeight: 195
|
|
, minWidth: 260
|
|
, stop: function(_, ui) {
|
|
chatPanel.css({'top': '', 'left': ''});
|
|
openedHeight = ui.size.height + 'px';
|
|
__setCssHeight(openedHeight);
|
|
__setCssWidth(ui.size.width);
|
|
}
|
|
});
|
|
__setCssHeight(closedSize);
|
|
}
|
|
ctrlBlk.off().click(_toggle);
|
|
$('#chatMessage').off().on('input propertychange paste', function () {
|
|
const room = $('.room-block .room-container');
|
|
if (room.length) {
|
|
if (!!typingTimer) {
|
|
clearTimeout(typingTimer);
|
|
} else {
|
|
chatActivity('typing_start', room.data('room-id'));
|
|
}
|
|
typingTimer = setTimeout(doneTyping, doneTypingInterval);
|
|
}
|
|
});
|
|
$('#chat .chat-toolbar .link-field').off().on('keypress', function(evt) {
|
|
if (evt.keyCode === 13) {
|
|
$(this).parent().find('button').trigger('click');
|
|
}
|
|
return evt.keyCode !== 13;
|
|
});
|
|
inited = true;
|
|
}
|
|
function _removeTab(id) {
|
|
$('#chatTabs li a[aria-controls="' + id + '"]').parent().remove();
|
|
$('#' + id).remove();
|
|
}
|
|
function _addTab(id, label) {
|
|
if (!inited) {
|
|
_reinit({});
|
|
}
|
|
if ($('#chat').length < 1 || $('#' + id).length) {
|
|
return;
|
|
}
|
|
if (!label) {
|
|
label = id === "chatTab-all" ? allPrefix : roomPrefix + id.substr(9);
|
|
}
|
|
const link = $('<a class="nav-link" data-bs-toggle="tab" role="tab">')
|
|
.attr('aria-controls', id)
|
|
.attr('href', '#' + id).text(label)
|
|
, li = $('<li class="nav-item">').append(link);
|
|
if (id.indexOf("chatTab-u") === 0) {
|
|
link.append(OmUtil.tmpl('#chat-close-block'));
|
|
}
|
|
tabs.find('.nav.nav-tabs').append(li);
|
|
const msgArea = OmUtil.tmpl('#chat-msg-area-template', id);
|
|
tabs.find('.tab-content').append(msgArea);
|
|
msgArea.append($('<div class="clear icons actions float-start">').addClass('short')
|
|
.append(OmUtil.tmpl('#chat-actions-short-template')));
|
|
msgArea.append($('<div class="clear icons actions float-start">').addClass('short-mod')
|
|
.append(OmUtil.tmpl('#chat-actions-short-template'))
|
|
.append(OmUtil.tmpl('#chat-actions-accept-template')));
|
|
msgArea.append($('<div class="clear icons actions float-start">').addClass('full')
|
|
.append(OmUtil.tmpl('#chat-actions-short-template'))
|
|
.append(OmUtil.tmpl('#chat-actions-others-template').children().clone()));
|
|
msgArea.append($('<div class="clear icons actions float-start">').addClass('full-mod')
|
|
.append(OmUtil.tmpl('#chat-actions-short-template'))
|
|
.append(OmUtil.tmpl('#chat-actions-others-template').children().clone())
|
|
.append(OmUtil.tmpl('#chat-actions-accept-template')));
|
|
const actions = __hideActions();
|
|
actions.find('.user').off().click(function() {
|
|
const e = $(this).parent();
|
|
showUserInfo(e.data("userId"));
|
|
});
|
|
actions.find('.add').off().click(function() {
|
|
const e = $(this).parent();
|
|
addContact(e.data("userId"));
|
|
});
|
|
actions.find('.new-email').off().click(function() {
|
|
const e = $(this).parent();
|
|
privateMessage(e.data("userId"));
|
|
});
|
|
actions.find('.invite').off().click(function() {
|
|
const e = $(this).parent();
|
|
inviteUser(e.data("userId"));
|
|
});
|
|
actions.find('.accept').off().click(function() {
|
|
const e = $(this).parent()
|
|
, msgId = e.data('msgId');
|
|
chatActivity('accept', e.data('roomId'), msgId);
|
|
__hideActions();
|
|
$('#chat-msg-id-' + msgId).remove();
|
|
});
|
|
activateTab(id);
|
|
}
|
|
function __hideActions() {
|
|
return $('#chat .tab-content .messageArea .icons').hide();
|
|
}
|
|
function __getActions(row) {
|
|
return row.closest('.messageArea').find('.actions.' + row.data('actions'));
|
|
}
|
|
function _addMessage(m) {
|
|
if ($('#chat').length > 0 && m && m.type === "chat") {
|
|
let msg, cm, notify = false;
|
|
while (!!(cm = m.msg.pop())) {
|
|
let area = $('#' + cm.scope);
|
|
if (cm.from.id !== userId && (isClosed() || !area.is(':visible'))) {
|
|
notify = true;
|
|
}
|
|
const actions = ('full' === cm.actions ? 'full' : 'short') + (cm.needModeration ? '-mod' : '');
|
|
msg = OmUtil.tmpl('#chat-msg-template', msgIdPrefix + cm.id)
|
|
const row = msg.find('.user-row')
|
|
.data('userId', cm.from.id)
|
|
.data('actions', actions)
|
|
.mouseenter(function() {
|
|
__hideActions();
|
|
__getActions($(this))
|
|
.data('userId', $(this).data('userId'))
|
|
.data('roomId', $(this).data('roomId'))
|
|
.data('msgId', $(this).data('msgId'))
|
|
.css('top', ($(this).closest('.msg-row')[0].offsetTop + 20) + 'px')
|
|
.show();
|
|
});
|
|
if (cm.needModeration) {
|
|
row.parent().addClass('need-moderation');
|
|
row.data('roomId', cm.scope.substring(9))
|
|
.data('msgId', cm.id);
|
|
}
|
|
area.mouseleave(function() {
|
|
__hideActions();
|
|
});
|
|
msg.find('.from').data('user-id', cm.from.id).html(cm.from.displayName || cm.from.name);
|
|
msg.find('.time').html(cm.time).attr('title', cm.sent);
|
|
if (!area.length) {
|
|
_addTab(cm.scope, cm.scopeName);
|
|
area = $('#' + cm.scope);
|
|
}
|
|
if (m.mode === "accept") {
|
|
$('#chat-msg-id-' + cm.id).remove();
|
|
}
|
|
const btm = area[0].scrollHeight - (area.scrollTop() + area.innerHeight()) < 3; //approximately equal
|
|
if (area.data('lastDate') !== cm.date) {
|
|
area.append(OmUtil.tmpl('#chat-date-template').html(cm.date).mouseenter(function() {
|
|
__hideActions();
|
|
}));
|
|
area.data('lastDate', cm.date);
|
|
}
|
|
area.append(msg);
|
|
msg.find('.user-row')[0].style.cssText = `
|
|
background-image: url(${(!!cm.from.img ? cm.from.img : './profile/' + cm.from.id + '?anticache=' + Date.now())});
|
|
background-position-x: ${Settings.isRtl ? 'right' : 'left'};
|
|
`;
|
|
|
|
msg.find('.msg').html(CSSEmoticon.emoticonize(!!cm.message ? cm.message : ""));
|
|
if (btm) {
|
|
_scrollDown(area);
|
|
}
|
|
}
|
|
if (notify) {
|
|
ctrlBlk.addClass('bg-warning');
|
|
if (chatPanel.is(':visible') && !muted) {
|
|
OmUtil.notify(newMsgNotification, 'new_chat_msg', () => {
|
|
// impossible to use Notification API from iFrame
|
|
audio.play()
|
|
.then(function() {
|
|
// Automatic playback started!
|
|
}).catch(function() {
|
|
// Automatic playback failed.
|
|
});
|
|
});
|
|
}
|
|
}
|
|
CSSEmoticon.animate();
|
|
}
|
|
}
|
|
function _setOpened() {
|
|
__setCssWidth(openedWidth);
|
|
chatPanel.resizable({
|
|
handles: (Settings.isRtl ? 'e' : 'w')
|
|
, minWidth: 165
|
|
, stop: function(_, ui) {
|
|
chatPanel.css({'left': '', 'width': '', 'height': ''});
|
|
openedWidth = ui.size.width + 'px';
|
|
__setCssWidth(openedWidth);
|
|
}
|
|
});
|
|
}
|
|
function _removeResize() {
|
|
if (chatPanel.resizable('instance') !== undefined) {
|
|
chatPanel.resizable('destroy');
|
|
}
|
|
}
|
|
function _open(handler) {
|
|
if (isClosed()) {
|
|
ctrlBlk.removeClass('bg-warning');
|
|
let opts;
|
|
if (roomMode) {
|
|
opts = {width: openedWidth};
|
|
} else {
|
|
opts = {height: openedHeight};
|
|
chatPanel.resizable("option", "disabled", false);
|
|
}
|
|
chatPanel.removeClass('closed').animate(opts, 1000, function() {
|
|
__hideActions();
|
|
chatPanel.removeClass('closed');
|
|
chatPanel.css({'height': '', 'width': ''});
|
|
if (typeof(handler) === 'function') {
|
|
handler();
|
|
}
|
|
ctrlBlk.attr('title', ctrlBlk.data('ttl-undock'));
|
|
if (roomMode) {
|
|
_setOpened();
|
|
if (typeof(window.WbArea) === 'object') {
|
|
window.WbArea.resize();
|
|
}
|
|
} else {
|
|
__setCssHeight(openedHeight);
|
|
}
|
|
_setAreaHeight();
|
|
});
|
|
}
|
|
}
|
|
function _close(handler) {
|
|
if (!isClosed()) {
|
|
let opts;
|
|
if (roomMode) {
|
|
opts = {width: closedSizePx};
|
|
} else {
|
|
opts = {height: closedSizePx};
|
|
chatPanel.resizable("option", "disabled", true);
|
|
}
|
|
chatPanel.animate(opts, 1000, function() {
|
|
chatPanel.addClass('closed').css({'height': '', 'width': ''});
|
|
if (roomMode) {
|
|
__setCssWidth(closedSizePx);
|
|
_removeResize();
|
|
if (typeof(window.WbArea) === 'object') {
|
|
window.WbArea.resize();
|
|
}
|
|
} else {
|
|
__setCssHeight(closedSizePx);
|
|
}
|
|
if (typeof(handler) === 'function') {
|
|
handler();
|
|
}
|
|
ctrlBlk.attr('title', ctrlBlk.data('ttl-dock'));
|
|
});
|
|
}
|
|
}
|
|
function _toggle() {
|
|
if (isClosed()) {
|
|
_open();
|
|
} else {
|
|
_close();
|
|
}
|
|
}
|
|
function _editorAppend(_emoticon) {
|
|
editor.html(editor.html() + ' ' + _emoticon + ' ').trigger('change');
|
|
}
|
|
function _clean() {
|
|
editor.html('').trigger('change');
|
|
}
|
|
function _setRoomMode(_mode) {
|
|
roomMode = _mode;
|
|
if (inited && !roomMode) {
|
|
// remove all private chats on room exit
|
|
$('li[aria-controls^="chatTab-u"]').remove();
|
|
$('div[id^="chatTab-u"]').remove();
|
|
}
|
|
_reinit({userId: userId, all: allPrefix, room: roomPrefix, sendOnEnter: sendOn === SEND_ENTER});
|
|
}
|
|
function _scrollDown(area) {
|
|
area.animate({
|
|
scrollTop: area[0].scrollHeight
|
|
}, 300);
|
|
}
|
|
function _setAreaHeight() {
|
|
$('#chat .messageArea').each(function() {
|
|
_scrollDown($(this));
|
|
});
|
|
}
|
|
function _insertLink() {
|
|
const text = $('#chat #hyperlink').parent().find('input').val();
|
|
if ('' === text) {
|
|
return;
|
|
}
|
|
let url = text.trim();
|
|
if ('' === url) {
|
|
return;
|
|
}
|
|
if (!/^(https?:)?\/\//i.test(url)) {
|
|
url = 'http://' + url;
|
|
}
|
|
const a = $('<div>').append($('<a></a>').attr('target', '_blank').attr('href', url).text(url)).html();
|
|
if (window.getSelection) {
|
|
const sel = window.getSelection();
|
|
if (sel.rangeCount) {
|
|
const range = sel.getRangeAt(0);
|
|
if ($(range.startContainer).parents('.wysiwyg-editor').length > 0) {
|
|
range.deleteContents();
|
|
range.insertNode(a);
|
|
} else {
|
|
_editorAppend(a);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
function _typingActivity(uid, active) {
|
|
if (typeof(Room) !== 'object') {
|
|
return;
|
|
}
|
|
const c = Room.getClient(uid);
|
|
if (!c) {
|
|
return;
|
|
}
|
|
const u = c.find('.typing-activity');
|
|
if (!u) {
|
|
return;
|
|
}
|
|
if (active) {
|
|
u.addClass("typing");
|
|
} else {
|
|
u.removeClass("typing");
|
|
}
|
|
}
|
|
|
|
$(function() {
|
|
Wicket.Event.subscribe("/websocket/message", function(_, msg) {
|
|
try {
|
|
if (msg instanceof Blob) {
|
|
return; //ping
|
|
}
|
|
const m = JSON.parse(msg);
|
|
if (m) {
|
|
switch(m.type) {
|
|
case "chat":
|
|
if ('clean' === m.action) {
|
|
$('#' + m.scope).html('');
|
|
} else {
|
|
_addMessage(m);
|
|
}
|
|
break;
|
|
case "typing":
|
|
_typingActivity(m.uid, m.active);
|
|
break;
|
|
}
|
|
}
|
|
} catch (err) {
|
|
//no-op
|
|
}
|
|
});
|
|
function _cancelAskNotification() {
|
|
$(document).off('click', _askNotification)
|
|
}
|
|
function _askNotification() {
|
|
OmUtil.requestNotifyPermission(_cancelAskNotification, _cancelAskNotification);
|
|
}
|
|
$(document).on('click', _askNotification);
|
|
});
|
|
|
|
module.exports = {
|
|
SEND_ENTER: SEND_ENTER
|
|
, SEND_CTRL: SEND_CTRL
|
|
|
|
, reinit: _reinit
|
|
, removeTab: _removeTab
|
|
, addTab: _addTab
|
|
, addMessage: _addMessage
|
|
, open: _open
|
|
, setOpened: function() {
|
|
_open(function() {
|
|
_setOpened();
|
|
});
|
|
}
|
|
, close: _close
|
|
, toggle: _toggle
|
|
, setRoomMode: _setRoomMode
|
|
, clean: _clean
|
|
, reload: () => {
|
|
_load();
|
|
_updateAudioBtn();
|
|
_updateSendBtn();
|
|
}
|
|
, validate: function() {
|
|
return !!editor && editor.text().trim().length > 0;
|
|
}
|
|
};
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "./src/chat/cssemoticons.js":
|
|
/*!**********************************!*\
|
|
!*** ./src/chat/cssemoticons.js ***!
|
|
\**********************************/
|
|
/***/ ((module) => {
|
|
|
|
/*
|
|
* Based on
|
|
*
|
|
* jQuery CSSEmoticons plugin 0.2.9
|
|
*
|
|
* Copyright (c) 2010 Steve Schwartz (JangoSteve)
|
|
*
|
|
* Dual licensed under the MIT and GPL licenses:
|
|
* http://www.opensource.org/licenses/mit-license.php
|
|
* http://www.gnu.org/licenses/gpl.html
|
|
*
|
|
* Date: Sun Oct 22 1:00:00 2010 -0500
|
|
*/
|
|
const escapeCharacters = [")", "(", "*", "[", "]", "{", "}", "|", "^", "<", ">", "\\", "?", "+", "=", "."]
|
|
, specialRegex = new RegExp('(\\' + escapeCharacters.join('|\\') + ')', 'g')
|
|
// One of these characters must be present before the matched emoticon, or the matched emoticon must be the first character in the container HTML
|
|
// This is to ensure that the characters in the middle of HTML properties or URLs are not matched as emoticons
|
|
// Below matches ^ (first character in container HTML), \s (whitespace like space or tab), or \0 (NULL character)
|
|
// (<\\S+.*>) matches <\\S+.*> (matches an HTML tag like <span> or <div>), but haven't quite gotten it working yet, need to push this fix now
|
|
, preMatch = '(^|[\\s\\0])'
|
|
, emoticons = []
|
|
, matchers = []
|
|
, defaults = {animate: true, delay: 500, exclude: 'pre,code,.no-emoticons'};
|
|
|
|
function createMatcher(m) {
|
|
const str = m.text.replace(specialRegex, '\\$1');
|
|
m.regexp = new RegExp(preMatch + '(' + str + ')', 'g');
|
|
return m;
|
|
}
|
|
function addMatchers(arr) {
|
|
for (let i = 0; i < arr.length; ++i) {
|
|
const o = arr[i]
|
|
let m = typeof(o) === 'object' ? JSON.parse(JSON.stringify(o)) : {text: o, cssClass: ' '};
|
|
emoticons.push(m.text);
|
|
|
|
matchers.push(createMatcher(m));
|
|
if (m.text.indexOf('=') > -1) {
|
|
m = JSON.parse(JSON.stringify(m));
|
|
m.text = m.text.replace(/=/g, '=').replace(/[+]/g, '+');
|
|
matchers.push(createMatcher(m));
|
|
}
|
|
if (m.text.indexOf('\'') > -1) {
|
|
m = JSON.parse(JSON.stringify(m));
|
|
m.text = m.text.replace(/'/g, ''');
|
|
matchers.push(createMatcher(m));
|
|
}
|
|
}
|
|
}
|
|
addMatchers([
|
|
":-)", ":o)", ":c)", ":^)", ":-D", ":-(", ":-9", ";-)", ":-P", ":-p", ":-Þ", ":-b", ":-O", ":-/", ":-X", ":-#", ":'(", "B-)", "8-)", ";*(", ":-*", ":-\\",
|
|
"?-)" // <== This is my own invention, it's a smiling pirate (with an eye-patch)!
|
|
]);
|
|
addMatchers([ // separate these out so that we can add a letter-spacing between the characters for better proportions
|
|
":)", ":]", "=]", "=)", "8)", ":}", ":D", ":(", ":[", ":{", "=(", ";)", ";]", ";D", ":P", ":p", "=P", "=p", ":b", ":Þ", ":O", ":/", "=/", ":S", ":#", ":X", "B)", ":|", ":\\", "=\\", ":*", ":>", ":<"
|
|
]);
|
|
addMatchers([ // emoticons to be treated with a special class, hash specifies the additional class to add, along with standard css-emoticon class
|
|
{text: ">:)", cssClass: "red-emoticon small-emoticon spaced-emoticon"},
|
|
{text: ">;)", cssClass: "red-emoticon small-emoticon spaced-emoticon"},
|
|
{text: ">:(", cssClass: "red-emoticon small-emoticon spaced-emoticon"},
|
|
{text: ">: )", cssClass: "red-emoticon small-emoticon"},
|
|
{text: ">; )", cssClass: "red-emoticon small-emoticon"},
|
|
{text: ">: (", cssClass: "red-emoticon small-emoticon"},
|
|
{text: ";(", cssClass: "red-emoticon spaced-emoticon"},
|
|
{text: "<3", cssClass: "pink-emoticon counter-rotated"},
|
|
{text: "O_O", cssClass: "no-rotate"},
|
|
{text: "o_o", cssClass: "no-rotate"},
|
|
{text: "0_o", cssClass: "no-rotate"},
|
|
{text: "O_o", cssClass: "no-rotate"},
|
|
{text: "T_T", cssClass: "no-rotate"},
|
|
{text: "^_^", cssClass: "no-rotate"},
|
|
{text: "O:)", cssClass: "small-emoticon spaced-emoticon"},
|
|
{text: "O: )", cssClass: "small-emoticon"},
|
|
{text: "8D", cssClass: "small-emoticon spaced-emoticon"},
|
|
{text: "XD", cssClass: "small-emoticon spaced-emoticon"},
|
|
{text: "xD", cssClass: "small-emoticon spaced-emoticon"},
|
|
{text: "=D", cssClass: "small-emoticon spaced-emoticon"},
|
|
{text: "8O", cssClass: "small-emoticon spaced-emoticon"},
|
|
{text: "[+=..]", cssClass: "no-rotate nintendo-controller"}
|
|
]);
|
|
|
|
module.exports = {
|
|
emoticons: emoticons
|
|
, matchers: matchers
|
|
, defaults: defaults
|
|
, emoticonize: function(str, options) {
|
|
const opts = $.extend({}, defaults, options);
|
|
|
|
let cssClass = 'css-emoticon';
|
|
if (opts.animate) {
|
|
cssClass += ' un-transformed-emoticon animated-emoticon';
|
|
}
|
|
for (let i = 0; i < matchers.length; ++i) {
|
|
const m = matchers[i];
|
|
const css = cssClass + " " + m.cssClass;
|
|
str = str.replace(m.regexp, "$1<span class='" + css + "'>$2</span>");
|
|
}
|
|
return str;
|
|
}
|
|
, animate: function(options) {
|
|
const opts = $.extend({}, defaults, options);
|
|
// animate emoticons
|
|
if (opts.animate) {
|
|
setTimeout(function () {
|
|
$('.un-transformed-emoticon').removeClass('un-transformed-emoticon');
|
|
}, opts.delay);
|
|
}
|
|
}
|
|
};
|
|
|
|
|
|
/***/ }),
|
|
|
|
/***/ "../main/omutils":
|
|
/*!*************************!*\
|
|
!*** external "OmUtil" ***!
|
|
\*************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = OmUtil;
|
|
|
|
/***/ }),
|
|
|
|
/***/ "../main/settings":
|
|
/*!***************************!*\
|
|
!*** external "Settings" ***!
|
|
\***************************/
|
|
/***/ ((module) => {
|
|
|
|
"use strict";
|
|
module.exports = Settings;
|
|
|
|
/***/ })
|
|
|
|
/******/ });
|
|
/************************************************************************/
|
|
/******/ // The module cache
|
|
/******/ var __webpack_module_cache__ = {};
|
|
/******/
|
|
/******/ // The require function
|
|
/******/ function __webpack_require__(moduleId) {
|
|
/******/ // Check if module is in cache
|
|
/******/ var cachedModule = __webpack_module_cache__[moduleId];
|
|
/******/ if (cachedModule !== undefined) {
|
|
/******/ return cachedModule.exports;
|
|
/******/ }
|
|
/******/ // Create a new module (and put it into the cache)
|
|
/******/ var module = __webpack_module_cache__[moduleId] = {
|
|
/******/ // no module.id needed
|
|
/******/ // no module.loaded needed
|
|
/******/ exports: {}
|
|
/******/ };
|
|
/******/
|
|
/******/ // Execute the module function
|
|
/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__);
|
|
/******/
|
|
/******/ // Return the exports of the module
|
|
/******/ return module.exports;
|
|
/******/ }
|
|
/******/
|
|
/************************************************************************/
|
|
var __webpack_exports__ = {};
|
|
// This entry need to be wrapped in an IIFE because it need to be isolated against other modules in the chunk.
|
|
(() => {
|
|
/*!***************************!*\
|
|
!*** ./src/chat/index.js ***!
|
|
\***************************/
|
|
/* Licensed under the Apache License, Version 2.0 (the "License") http://www.apache.org/licenses/LICENSE-2.0 */
|
|
Object.assign(window, {
|
|
Chat: __webpack_require__(/*! ./chat */ "./src/chat/chat.js")
|
|
});
|
|
|
|
})();
|
|
|
|
/******/ })()
|
|
;
|
|
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"chat.js","mappings":";;;;;;;;;AAAA;AACA,eAAe,mBAAO,CAAC,wCAAiB;AACxC,iBAAiB,mBAAO,CAAC,0CAAkB;AAC3C,oBAAoB,mBAAO,CAAC,kDAAgB;;AAE5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,kBAAkB;AACnC;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,GAAG;AACH,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,oBAAoB,sBAAsB;AAC1C;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,mFAAmF;AACnF;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA,4BAA4B,uFAAuF;AACnH,6BAA6B;AAC7B;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,OAAO;AACP;AACA,OAAO;AACP,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,kBAAkB,sCAAsC;AACxD;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX,IAAI;AACJ,WAAW;AACX;AACA;AACA;AACA;AACA;AACA,kBAAkB,0BAA0B;AAC5C;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA,WAAW;AACX,IAAI;AACJ,WAAW;AACX;AACA;AACA;AACA,qCAAqC,0BAA0B;AAC/D;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,UAAU,qFAAqF;AAC/F;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,KAAK;AACL;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA,YAAY;AACZ;AACA;AACA;AACA;AACA;AACA;AACA;AACA,QAAQ;AACR;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA,EAAE;AACF;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CAAC;;AAED;AACA;AACA;;AAEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,GAAG;AACH;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;;;;;;;;;AC5iBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,qDAAqD,KAAK;AAC1D;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,eAAe;;AAEf;AACA;AACA;AACA;AACA;AACA;AACA,iBAAiB,gBAAgB;AACjC;AACA,oEAAoE;AACpE;;AAEA;AACA;AACA;AACA,uCAAuC,yBAAyB;AAChE;AACA;AACA;AACA;AACA,uCAAuC;AACvC;AACA;AACA;AACA;AACA;AACA,oDAAoD,oFAAoF;AACxI;AACA;AACA;AACA,kCAAkC,wBAAwB,WAAW,MAAM,MAAM,kHAAkH,SAAS;AAC5M;AACA;AACA,EAAE,WAAW,6DAA6D;AAC1E,EAAE,YAAY,4DAA4D;AAC1E,EAAE,WAAW,6DAA6D;AAC1E,EAAE,WAAW,8CAA8C;AAC3D,EAAE,aAAa,4CAA4C;AAC3D,EAAE,WAAW,8CAA8C;AAC3D,EAAE,QAAQ,6CAA6C;AACvD,EAAE,WAAW,8CAA8C;AAC3D,EAAE,mCAAmC;AACrC,EAAE,mCAAmC;AACrC,EAAE,mCAAmC;AACrC,EAAE,mCAAmC;AACrC,EAAE,mCAAmC;AACrC,EAAE,mCAAmC;AACrC,EAAE,wDAAwD;AAC1D,EAAE,yCAAyC;AAC3C,EAAE,uDAAuD;AACzD,EAAE,uDAAuD;AACzD,EAAE,uDAAuD;AACzD,EAAE,uDAAuD;AACzD,EAAE,uDAAuD;AACzD,EAAE;AACF;;AAEA;AACA;AACA;AACA;AACA;AACA,0BAA0B;;AAE1B;AACA;AACA;AACA;AACA,kBAAkB,qBAAqB;AACvC;AACA;AACA;AACA;AACA;AACA;AACA;AACA,0BAA0B;AAC1B;AACA;AACA;AACA;AACA,IAAI;AACJ;AACA;AACA;;;;;;;;;;;;AC3GA;;;;;;;;;;;ACAA;;;;;;UCAA;UACA;;UAEA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;UACA;;UAEA;UACA;;UAEA;UACA;UACA;;;;;;;;;ACtBA;AACA;AACA,OAAO,mBAAO,CAAC,kCAAQ;AACvB,CAAC","sources":["webpack://om-frontend/./src/chat/chat.js","webpack://om-frontend/./src/chat/cssemoticons.js","webpack://om-frontend/external var \"OmUtil\"","webpack://om-frontend/external var \"Settings\"","webpack://om-frontend/webpack/bootstrap","webpack://om-frontend/./src/chat/index.js"],"sourcesContent":["/* Licensed under the Apache License, Version 2.0 (the \"License\") http://www.apache.org/licenses/LICENSE-2.0 */\nconst OmUtil = require('../main/omutils');\nconst Settings = require('../main/settings');\nconst CSSEmoticon = require('./cssemoticons');\n\nconst msgIdPrefix = 'chat-msg-id-'\n\t, closedSize = 20\n\t, closedSizePx = closedSize + \"px\"\n\t, doneTypingInterval = 5000 //time in ms, 5 second for example\n\t, SEND_ENTER = 'enter', SEND_CTRL = 'ctrl'\n\t, audio = new Audio('./public/chat_message.mp3')\n\t;\nlet chatPanel, ctrlBlk, tabs, openedHeight = \"345px\", openedWidth = \"300px\", allPrefix = \"All\"\n\t, roomPrefix = \"Room \", typingTimer, roomMode = false\n\t, editor = $('#chatMessage .wysiwyg-editor'), muted = false, sendOn, DEF_SEND\n\t, userId, inited = false, newMsgNotification\n\t;\n\nfunction __setCssVar(key, _val) {\n\tconst val = ('' + _val).endsWith('px') ? _val : _val + 'px';\n\tif (roomMode) {\n\t\tif (typeof(Room) == 'object' && typeof(Room.setCssVar) === 'function') {\n\t\t\tRoom.setCssVar(key, val);\n\t\t}\n\t} else {\n\t\tOmUtil.setCssVar(key, val);\n\t}\n}\nfunction __setCssWidth(val) {\n\t__setCssVar('--chat-width', val)\n}\nfunction __setCssHeight(val) {\n\t__setCssVar('--chat-height', val)\n}\nfunction _load() {\n\tconst s = Settings.load();\n\tif (typeof(s.chat) === 'undefined') {\n\t\ts.chat = {\n\t\t\tmuted: false\n\t\t\t, sendOn: DEF_SEND\n\t\t};\n\t\tSettings.save(s)\n\t}\n\tmuted = s.chat.muted === true;\n\tsendOn = s.chat.sendOn === SEND_ENTER ? SEND_ENTER : SEND_CTRL;\n\treturn s;\n}\nfunction _updateAudioBtn() {\n\tconst btn = $('#chat .audio');\n\tbtn.removeClass('sound' + (muted ? '' : '-mute')).addClass('sound' + (muted ? '-mute' : ''))\n\t\t\t.attr('title', btn.data(muted ? 'sound-enabled' : 'sound-muted'));\n}\nfunction _updateSendBtn() {\n\tconst btn = $('#chat .send-btn')\n\t\t, ctrl = sendOn === SEND_CTRL;\n\tif (ctrl) {\n\t\tbtn.addClass('send-ctrl');\n\t\teditor.off('keydown', _sendOnEnter).keydown('Ctrl+return', _sendOnEnter);\n\t} else {\n\t\tbtn.removeClass('send-ctrl');\n\t\teditor.off('keydown', _sendOnEnter).keydown('return', _sendOnEnter);\n\t}\n\tbtn.attr('title', btn.data(ctrl ? 'send-ctrl' : 'send-enter'));\n}\nfunction _sendOnEnter() {\n\t$('#chat .send').trigger('click');\n}\nfunction doneTyping () {\n\ttypingTimer = null;\n\tchatActivity('typing_stop', $('.room-block .room-container').data('room-id'));\n}\nfunction _emtClick() {\n\t_editorAppend($(this).data('emt'));\n}\nfunction initToolbar() {\n\tconst emots = CSSEmoticon.emoticons;\n\tconst rowSize = 20, emotMenuList = $('#emotMenuList');\n\temotMenuList.html('');\n\tlet row;\n\tfor (let i = 0; i < emots.length; ++i) {\n\t\tif (i % rowSize === 0) {\n\t\t\trow = $('<tr></tr>');\n\t\t\temotMenuList.append(row);\n\t\t}\n\t\trow.append($('<td>').append(\n\t\t\t\t$('<div>').addClass('emt').html(CSSEmoticon.emoticonize(emots[i]))\n\t\t\t\t\t.data('emt', emots[i]).click(_emtClick)\n\t\t\t));\n\t}\n\tconst emtBtn = $('#emoticons');\n\temtBtn.html('');\n\temtBtn.append(' ' + CSSEmoticon.emoticonize(':)'));\n\tconst a = $('#chat .audio');\n\tconst sbtn = $('#chat .send-btn');\n\t{ //scope\n\t\t_load();\n\t\t_updateAudioBtn();\n\t\t_updateSendBtn()\n\t}\n\ta.off().click(function() {\n\t\tconst s = _load();\n\t\tmuted = s.chat.muted = !s.chat.muted;\n\t\t_updateAudioBtn();\n\t\tSettings.save(s);\n\t});\n\tsbtn.off().click(function() {\n\t\tconst s = _load();\n\t\tsendOn = s.chat.sendOn = s.chat.sendOn !== SEND_CTRL ? SEND_CTRL : SEND_ENTER;\n\t\t_updateSendBtn();\n\t\tSettings.save(s);\n\t});\n\t$('#chat #hyperlink').parent().find('button').off().click(function() {\n\t\t_insertLink();\n\t});\n\tCSSEmoticon.animate();\n}\nfunction isClosed() {\n\treturn chatPanel.hasClass('closed');\n}\nfunction activateTab(id) {\n\tif (isClosed()) {\n\t\ttabs.find('.nav.nav-tabs .nav-link').each(function() {\n\t\t\tconst self = $(this)\n\t\t\t\t, tabId = self.attr('aria-controls')\n\t\t\t\t, tab = $('#' + tabId);\n\n\t\t\tif (tabId === id) {\n\t\t\t\tself.addClass('active');\n\t\t\t\ttab.addClass('active');\n\t\t\t\tself.attr('aria-selected', true);\n\t\t\t} else {\n\t\t\t\tself.removeClass('active');\n\t\t\t\ttab.removeClass('active');\n\t\t\t\tself.attr('aria-selected', false);\n\t\t\t}\n\t\t});\n\t} else {\n\t\t$('#chatTabs li a[aria-controls=\"' + id + '\"]').tab('show');\n\t}\n\t$('#activeChatTab').val(id).trigger('change');\n}\nfunction _reinit(opts) {\n\tuserId = opts.userId;\n\tallPrefix = opts.all;\n\troomPrefix = opts.room;\n\tDEF_SEND = opts.sendOnEnter === true ? SEND_ENTER : SEND_CTRL;\n\tsendOn = DEF_SEND;\n\tchatPanel = $('#chatPanel');\n\tclearTimeout(chatPanel.data('timeout'));\n\tctrlBlk = $('#chatPopup .control.block');\n\tnewMsgNotification = ctrlBlk.data('new-msg');\n\teditor = $('#chatMessage .wysiwyg-editor');\n\tinitToolbar();\n\ttabs = $(\"#chatTabs\");\n\ttabs.off().on('shown.bs.tab', function (e) {\n\t\tconst ct = $(e.target).attr('aria-controls');\n\t\t_scrollDown($('#' + ct));\n\t\t$('#activeChatTab').val(ct).trigger('change');\n\t});\n\ttabs.delegate(\".btn.close-chat\", \"click\", function() {\n\t\tconst panelId = $(this).closest(\"a\").attr(\"aria-controls\");\n\t\t_removeTab(panelId);\n\t\t$('#chatTabs li:last-child a').tab('show');\n\t});\n\tif (roomMode) {\n\t\t_removeResize();\n\t} else {\n\t\tctrlBlk.attr('title', '');\n\t\tchatPanel.removeClass('room opened').addClass('closed')\n\t\t\t.off('mouseenter mouseleave').resizable({\n\t\t\t\thandles: 'n, ' + (Settings.isRtl ? 'e' : 'w')\n\t\t\t\t, disabled: isClosed()\n\t\t\t\t, minHeight: 195\n\t\t\t\t, minWidth: 260\n\t\t\t\t, stop: function(_, ui) {\n\t\t\t\t\tchatPanel.css({'top': '', 'left': ''});\n\t\t\t\t\topenedHeight = ui.size.height + 'px';\n\t\t\t\t\t__setCssHeight(openedHeight);\n\t\t\t\t\t__setCssWidth(ui.size.width);\n\t\t\t\t}\n\t\t\t});\n\t\t__setCssHeight(closedSize);\n\t}\n\tctrlBlk.off().click(_toggle);\n\t$('#chatMessage').off().on('input propertychange paste', function () {\n\t\tconst room = $('.room-block .room-container');\n\t\tif (room.length) {\n\t\t\tif (!!typingTimer) {\n\t\t\t\tclearTimeout(typingTimer);\n\t\t\t} else {\n\t\t\t\tchatActivity('typing_start', room.data('room-id'));\n\t\t\t}\n\t\t\ttypingTimer = setTimeout(doneTyping, doneTypingInterval);\n\t\t}\n\t});\n\t$('#chat .chat-toolbar .link-field').off().on('keypress', function(evt) {\n\t\tif (evt.keyCode === 13) {\n\t\t\t$(this).parent().find('button').trigger('click');\n\t\t}\n\t\treturn evt.keyCode !== 13;\n\t});\n\tinited = true;\n}\nfunction _removeTab(id) {\n\t$('#chatTabs li a[aria-controls=\"' + id + '\"]').parent().remove();\n\t$('#' + id).remove();\n}\nfunction _addTab(id, label) {\n\tif (!inited) {\n\t\t_reinit({});\n\t}\n\tif ($('#chat').length < 1 || $('#' + id).length) {\n\t\treturn;\n\t}\n\tif (!label) {\n\t\tlabel = id === \"chatTab-all\" ? allPrefix : roomPrefix + id.substr(9);\n\t}\n\tconst link = $('<a class=\"nav-link\" data-bs-toggle=\"tab\" role=\"tab\">')\n\t\t.attr('aria-controls', id)\n\t\t.attr('href', '#' + id).text(label)\n\t\t, li = $('<li class=\"nav-item\">').append(link);\n\tif (id.indexOf(\"chatTab-u\") === 0) {\n\t\tlink.append(OmUtil.tmpl('#chat-close-block'));\n\t}\n\ttabs.find('.nav.nav-tabs').append(li);\n\tconst msgArea = OmUtil.tmpl('#chat-msg-area-template', id);\n\ttabs.find('.tab-content').append(msgArea);\n\tmsgArea.append($('<div class=\"clear icons actions float-start\">').addClass('short')\n\t\t\t.append(OmUtil.tmpl('#chat-actions-short-template')));\n\tmsgArea.append($('<div class=\"clear icons actions float-start\">').addClass('short-mod')\n\t\t\t.append(OmUtil.tmpl('#chat-actions-short-template'))\n\t\t\t.append(OmUtil.tmpl('#chat-actions-accept-template')));\n\tmsgArea.append($('<div class=\"clear icons actions float-start\">').addClass('full')\n\t\t\t.append(OmUtil.tmpl('#chat-actions-short-template'))\n\t\t\t.append(OmUtil.tmpl('#chat-actions-others-template').children().clone()));\n\tmsgArea.append($('<div class=\"clear icons actions float-start\">').addClass('full-mod')\n\t\t\t.append(OmUtil.tmpl('#chat-actions-short-template'))\n\t\t\t.append(OmUtil.tmpl('#chat-actions-others-template').children().clone())\n\t\t\t.append(OmUtil.tmpl('#chat-actions-accept-template')));\n\tconst actions = __hideActions();\n\tactions.find('.user').off().click(function() {\n\t\tconst e = $(this).parent();\n\t\tshowUserInfo(e.data(\"userId\"));\n\t});\n\tactions.find('.add').off().click(function() {\n\t\tconst e = $(this).parent();\n\t\taddContact(e.data(\"userId\"));\n\t});\n\tactions.find('.new-email').off().click(function() {\n\t\tconst e = $(this).parent();\n\t\tprivateMessage(e.data(\"userId\"));\n\t});\n\tactions.find('.invite').off().click(function() {\n\t\tconst e = $(this).parent();\n\t\tinviteUser(e.data(\"userId\"));\n\t});\n\tactions.find('.accept').off().click(function() {\n\t\tconst e = $(this).parent()\n\t\t\t, msgId = e.data('msgId');\n\t\tchatActivity('accept', e.data('roomId'), msgId);\n\t\t__hideActions();\n\t\t$('#chat-msg-id-' + msgId).remove();\n\t});\n\tactivateTab(id);\n}\nfunction __hideActions() {\n\treturn $('#chat .tab-content .messageArea .icons').hide();\n}\nfunction __getActions(row) {\n\treturn row.closest('.messageArea').find('.actions.' + row.data('actions'));\n}\nfunction _addMessage(m) {\n\tif ($('#chat').length > 0 && m && m.type === \"chat\") {\n\t\tlet msg, cm, notify = false;\n\t\twhile (!!(cm = m.msg.pop())) {\n\t\t\tlet area = $('#' + cm.scope);\n\t\t\tif (cm.from.id !== userId && (isClosed() || !area.is(':visible'))) {\n\t\t\t\tnotify = true;\n\t\t\t}\n\t\t\tconst actions = ('full' === cm.actions ? 'full' : 'short') + (cm.needModeration ? '-mod' : '');\n\t\t\tmsg = OmUtil.tmpl('#chat-msg-template', msgIdPrefix + cm.id)\n\t\t\tconst row = msg.find('.user-row')\n\t\t\t\t.data('userId', cm.from.id)\n\t\t\t\t.data('actions', actions)\n\t\t\t\t.mouseenter(function() {\n\t\t\t\t\t__hideActions();\n\t\t\t\t\t__getActions($(this))\n\t\t\t\t\t\t.data('userId', $(this).data('userId'))\n\t\t\t\t\t\t.data('roomId', $(this).data('roomId'))\n\t\t\t\t\t\t.data('msgId', $(this).data('msgId'))\n\t\t\t\t\t\t.css('top', ($(this).closest('.msg-row')[0].offsetTop + 20) + 'px')\n\t\t\t\t\t\t.show();\n\t\t\t\t});\n\t\t\tif (cm.needModeration) {\n\t\t\t\trow.parent().addClass('need-moderation');\n\t\t\t\trow.data('roomId', cm.scope.substring(9))\n\t\t\t\t\t.data('msgId', cm.id);\n\t\t\t}\n\t\t\tarea.mouseleave(function() {\n\t\t\t\t__hideActions();\n\t\t\t});\n\t\t\tmsg.find('.from').data('user-id', cm.from.id).html(cm.from.displayName || cm.from.name);\n\t\t\tmsg.find('.time').html(cm.time).attr('title', cm.sent);\n\t\t\tif (!area.length) {\n\t\t\t\t_addTab(cm.scope, cm.scopeName);\n\t\t\t\tarea = $('#' + cm.scope);\n\t\t\t}\n\t\t\tif (m.mode === \"accept\") {\n\t\t\t\t$('#chat-msg-id-' + cm.id).remove();\n\t\t\t}\n\t\t\tconst btm = area[0].scrollHeight - (area.scrollTop() + area.innerHeight()) < 3; //approximately equal\n\t\t\tif (area.data('lastDate') !== cm.date) {\n\t\t\t\tarea.append(OmUtil.tmpl('#chat-date-template').html(cm.date).mouseenter(function() {\n\t\t\t\t\t__hideActions();\n\t\t\t\t}));\n\t\t\t\tarea.data('lastDate', cm.date);\n\t\t\t}\n\t\t\tarea.append(msg);\n\t\t\tmsg.find('.user-row')[0].style.cssText = `\n\t\t\t\tbackground-image: url(${(!!cm.from.img ? cm.from.img : './profile/' + cm.from.id + '?anticache=' + Date.now())});\n\t\t\t\tbackground-position-x: ${Settings.isRtl ? 'right' : 'left'};\n\t\t\t`;\n\n\t\t\tmsg.find('.msg').html(CSSEmoticon.emoticonize(!!cm.message ? cm.message : \"\"));\n\t\t\tif (btm) {\n\t\t\t\t_scrollDown(area);\n\t\t\t}\n\t\t}\n\t\tif (notify) {\n\t\t\tctrlBlk.addClass('bg-warning');\n\t\t\tif (chatPanel.is(':visible') && !muted) {\n\t\t\t\tOmUtil.notify(newMsgNotification, 'new_chat_msg', () => {\n\t\t\t\t\t// impossible to use Notification API from iFrame\n\t\t\t\t\taudio.play()\n\t\t\t\t\t\t.then(function() {\n\t\t\t\t\t\t\t// Automatic playback started!\n\t\t\t\t\t\t}).catch(function() {\n\t\t\t\t\t\t\t// Automatic playback failed.\n\t\t\t\t\t\t});\n\t\t\t\t});\n\t\t\t}\n\t\t}\n\t\tCSSEmoticon.animate();\n\t}\n}\nfunction _setOpened() {\n\t__setCssWidth(openedWidth);\n\tchatPanel.resizable({\n\t\thandles: (Settings.isRtl ? 'e' : 'w')\n\t\t, minWidth: 165\n\t\t, stop: function(_, ui) {\n\t\t\tchatPanel.css({'left': '', 'width': '', 'height': ''});\n\t\t\topenedWidth = ui.size.width + 'px';\n\t\t\t__setCssWidth(openedWidth);\n\t\t}\n\t});\n}\nfunction _removeResize() {\n\tif (chatPanel.resizable('instance') !== undefined) {\n\t\tchatPanel.resizable('destroy');\n\t}\n}\nfunction _open(handler) {\n\tif (isClosed()) {\n\t\tctrlBlk.removeClass('bg-warning');\n\t\tlet opts;\n\t\tif (roomMode) {\n\t\t\topts = {width: openedWidth};\n\t\t} else {\n\t\t\topts = {height: openedHeight};\n\t\t\tchatPanel.resizable(\"option\", \"disabled\", false);\n\t\t}\n\t\tchatPanel.removeClass('closed').animate(opts, 1000, function() {\n\t\t\t__hideActions();\n\t\t\tchatPanel.removeClass('closed');\n\t\t\tchatPanel.css({'height': '', 'width': ''});\n\t\t\tif (typeof(handler) === 'function') {\n\t\t\t\thandler();\n\t\t\t}\n\t\t\tctrlBlk.attr('title', ctrlBlk.data('ttl-undock'));\n\t\t\tif (roomMode) {\n\t\t\t\t_setOpened();\n\t\t\t\tif (typeof(window.WbArea) === 'object') {\n\t\t\t\t\twindow.WbArea.resize();\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t__setCssHeight(openedHeight);\n\t\t\t}\n\t\t\t_setAreaHeight();\n\t\t});\n\t}\n}\nfunction _close(handler) {\n\tif (!isClosed()) {\n\t\tlet opts;\n\t\tif (roomMode) {\n\t\t\topts = {width: closedSizePx};\n\t\t} else {\n\t\t\topts = {height: closedSizePx};\n\t\t\tchatPanel.resizable(\"option\", \"disabled\", true);\n\t\t}\n\t\tchatPanel.animate(opts, 1000, function() {\n\t\t\tchatPanel.addClass('closed').css({'height': '', 'width': ''});\n\t\t\tif (roomMode) {\n\t\t\t\t__setCssWidth(closedSizePx);\n\t\t\t\t_removeResize();\n\t\t\t\tif (typeof(window.WbArea) === 'object') {\n\t\t\t\t\twindow.WbArea.resize();\n\t\t\t\t}\n\t\t\t} else {\n\t\t\t\t__setCssHeight(closedSizePx);\n\t\t\t}\n\t\t\tif (typeof(handler) === 'function') {\n\t\t\t\thandler();\n\t\t\t}\n\t\t\tctrlBlk.attr('title', ctrlBlk.data('ttl-dock'));\n\t\t});\n\t}\n}\nfunction _toggle() {\n\tif (isClosed()) {\n\t\t_open();\n\t} else {\n\t\t_close();\n\t}\n}\nfunction _editorAppend(_emoticon) {\n\teditor.html(editor.html() + ' ' + _emoticon + ' ').trigger('change');\n}\nfunction _clean() {\n\teditor.html('').trigger('change');\n}\nfunction _setRoomMode(_mode) {\n\troomMode = _mode;\n\tif (inited && !roomMode) {\n\t\t// remove all private chats on room exit\n\t\t$('li[aria-controls^=\"chatTab-u\"]').remove();\n\t\t$('div[id^=\"chatTab-u\"]').remove();\n\t}\n\t_reinit({userId: userId, all: allPrefix, room: roomPrefix, sendOnEnter: sendOn === SEND_ENTER});\n}\nfunction _scrollDown(area) {\n\tarea.animate({\n\t\tscrollTop: area[0].scrollHeight\n\t}, 300);\n}\nfunction _setAreaHeight() {\n\t$('#chat .messageArea').each(function() {\n\t\t_scrollDown($(this));\n\t});\n}\nfunction _insertLink() {\n\tconst text = $('#chat #hyperlink').parent().find('input').val();\n\tif ('' === text) {\n\t\treturn;\n\t}\n\tlet url = text.trim();\n\tif ('' === url) {\n\t\treturn;\n\t}\n\tif (!/^(https?:)?\\/\\//i.test(url)) {\n\t\turl = 'http://' + url;\n\t}\n\tconst a = $('<div>').append($('<a></a>').attr('target', '_blank').attr('href', url).text(url)).html();\n\tif (window.getSelection) {\n\t\tconst sel = window.getSelection();\n\t\tif (sel.rangeCount) {\n\t\t\tconst range = sel.getRangeAt(0);\n\t\t\tif ($(range.startContainer).parents('.wysiwyg-editor').length > 0) {\n\t\t\t\trange.deleteContents();\n\t\t\t\trange.insertNode(a);\n\t\t\t} else {\n\t\t\t\t_editorAppend(a);\n\t\t\t}\n\t\t}\n\t}\n}\nfunction _typingActivity(uid, active) {\n\tif (typeof(Room) !== 'object') {\n\t\treturn;\n\t}\n\tconst c = Room.getClient(uid);\n\tif (!c) {\n\t\treturn;\n\t}\n\tconst u = c.find('.typing-activity');\n\tif (!u) {\n\t\treturn;\n\t}\n\tif (active) {\n\t\tu.addClass(\"typing\");\n\t} else {\n\t\tu.removeClass(\"typing\");\n\t}\n}\n\n$(function() {\n\tWicket.Event.subscribe(\"/websocket/message\", function(_, msg) {\n\t\ttry {\n\t\t\tif (msg instanceof Blob) {\n\t\t\t\treturn; //ping\n\t\t\t}\n\t\t\tconst m = JSON.parse(msg);\n\t\t\tif (m) {\n\t\t\t\tswitch(m.type) {\n\t\t\t\t\tcase \"chat\":\n\t\t\t\t\t\tif ('clean' === m.action) {\n\t\t\t\t\t\t\t$('#' + m.scope).html('');\n\t\t\t\t\t\t} else {\n\t\t\t\t\t\t\t_addMessage(m);\n\t\t\t\t\t\t}\n\t\t\t\t\t\tbreak;\n\t\t\t\t\tcase \"typing\":\n\t\t\t\t\t\t_typingActivity(m.uid, m.active);\n\t\t\t\t\t\tbreak;\n\t\t\t\t}\n\t\t\t}\n\t\t} catch (err) {\n\t\t\t//no-op\n\t\t}\n\t});\n\tfunction _cancelAskNotification() {\n\t\t$(document).off('click', _askNotification)\n\t}\n\tfunction _askNotification() {\n\t\tOmUtil.requestNotifyPermission(_cancelAskNotification, _cancelAskNotification);\n\t}\n\t$(document).on('click', _askNotification);\n});\n\nmodule.exports = {\n\tSEND_ENTER: SEND_ENTER\n\t, SEND_CTRL: SEND_CTRL\n\n\t, reinit: _reinit\n\t, removeTab: _removeTab\n\t, addTab: _addTab\n\t, addMessage: _addMessage\n\t, open: _open\n\t, setOpened: function() {\n\t\t_open(function() {\n\t\t\t_setOpened();\n\t\t});\n\t}\n\t, close: _close\n\t, toggle: _toggle\n\t, setRoomMode: _setRoomMode\n\t, clean: _clean\n\t, reload: () => {\n\t\t_load();\n\t\t_updateAudioBtn();\n\t\t_updateSendBtn();\n\t}\n\t, validate: function() {\n\t\treturn !!editor && editor.text().trim().length > 0;\n\t}\n};\n","/*\n * Based on\n *\n * jQuery CSSEmoticons plugin 0.2.9\n *\n * Copyright (c) 2010 Steve Schwartz (JangoSteve)\n *\n * Dual licensed under the MIT and GPL licenses:\n *   http://www.opensource.org/licenses/mit-license.php\n *   http://www.gnu.org/licenses/gpl.html\n *\n * Date: Sun Oct 22 1:00:00 2010 -0500\n */\nconst escapeCharacters = [\")\", \"(\", \"*\", \"[\", \"]\", \"{\", \"}\", \"|\", \"^\", \"<\", \">\", \"\\\\\", \"?\", \"+\", \"=\", \".\"]\n\t, specialRegex = new RegExp('(\\\\' + escapeCharacters.join('|\\\\') + ')', 'g')\n\t// One of these characters must be present before the matched emoticon, or the matched emoticon must be the first character in the container HTML\n\t//  This is to ensure that the characters in the middle of HTML properties or URLs are not matched as emoticons\n\t//  Below matches ^ (first character in container HTML), \\s (whitespace like space or tab), or \\0 (NULL character)\n\t// (<\\\\S+.*>) matches <\\\\S+.*> (matches an HTML tag like <span> or <div>), but haven't quite gotten it working yet, need to push this fix now\n\t, preMatch = '(^|[\\\\s\\\\0])'\n\t, emoticons = []\n\t, matchers = []\n\t, defaults = {animate: true, delay: 500, exclude: 'pre,code,.no-emoticons'};\n\nfunction createMatcher(m) {\n\tconst str = m.text.replace(specialRegex, '\\\\$1');\n\tm.regexp = new RegExp(preMatch + '(' + str + ')', 'g');\n\treturn m;\n}\nfunction addMatchers(arr) {\n\tfor (let i = 0; i < arr.length; ++i) {\n\t\tconst o = arr[i]\n\t\tlet m = typeof(o) === 'object' ? JSON.parse(JSON.stringify(o)) : {text: o, cssClass: ' '};\n\t\temoticons.push(m.text);\n\n\t\tmatchers.push(createMatcher(m));\n\t\tif (m.text.indexOf('=') > -1) {\n\t\t\tm = JSON.parse(JSON.stringify(m));\n\t\t\tm.text = m.text.replace(/=/g, '&#61;').replace(/[+]/g, '&#43;');\n\t\t\tmatchers.push(createMatcher(m));\n\t\t}\n\t\tif (m.text.indexOf('\\'') > -1) {\n\t\t\tm = JSON.parse(JSON.stringify(m));\n\t\t\tm.text = m.text.replace(/'/g, '&#39;');\n\t\t\tmatchers.push(createMatcher(m));\n\t\t}\n\t}\n}\naddMatchers([\n\t\":-)\", \":o)\", \":c)\", \":^)\", \":-D\", \":-(\", \":-9\", \";-)\", \":-P\", \":-p\", \":-Þ\", \":-b\", \":-O\", \":-/\", \":-X\", \":-#\", \":'(\", \"B-)\", \"8-)\", \";*(\", \":-*\", \":-\\\\\",\n\t\"?-)\" // <== This is my own invention, it's a smiling pirate (with an eye-patch)!\n]);\naddMatchers([ // separate these out so that we can add a letter-spacing between the characters for better proportions\n\t\":)\", \":]\", \"=]\", \"=)\", \"8)\", \":}\", \":D\", \":(\", \":[\", \":{\", \"=(\", \";)\", \";]\", \";D\", \":P\", \":p\", \"=P\", \"=p\", \":b\", \":Þ\", \":O\", \":/\", \"=/\", \":S\", \":#\", \":X\", \"B)\", \":|\", \":\\\\\", \"=\\\\\", \":*\", \":&gt;\", \":&lt;\"\n]);\naddMatchers([ // emoticons to be treated with a special class, hash specifies the additional class to add, along with standard css-emoticon class\n\t{text: \"&gt;:)\", cssClass: \"red-emoticon small-emoticon spaced-emoticon\"},\n\t{text: \"&gt;;)\", cssClass: \"red-emoticon small-emoticon spaced-emoticon\"},\n\t{text: \"&gt;:(\", cssClass: \"red-emoticon small-emoticon spaced-emoticon\"},\n\t{text: \"&gt;: )\", cssClass: \"red-emoticon small-emoticon\"},\n\t{text: \"&gt;; )\", cssClass: \"red-emoticon small-emoticon\"},\n\t{text: \"&gt;: (\", cssClass: \"red-emoticon small-emoticon\"},\n\t{text: \";(\", cssClass: \"red-emoticon spaced-emoticon\"},\n\t{text: \"&lt;3\", cssClass: \"pink-emoticon counter-rotated\"},\n\t{text: \"O_O\", cssClass: \"no-rotate\"},\n\t{text: \"o_o\", cssClass: \"no-rotate\"},\n\t{text: \"0_o\", cssClass: \"no-rotate\"},\n\t{text: \"O_o\", cssClass: \"no-rotate\"},\n\t{text: \"T_T\", cssClass: \"no-rotate\"},\n\t{text: \"^_^\", cssClass: \"no-rotate\"},\n\t{text: \"O:)\", cssClass: \"small-emoticon spaced-emoticon\"},\n\t{text: \"O: )\", cssClass: \"small-emoticon\"},\n\t{text: \"8D\", cssClass: \"small-emoticon spaced-emoticon\"},\n\t{text: \"XD\", cssClass: \"small-emoticon spaced-emoticon\"},\n\t{text: \"xD\", cssClass: \"small-emoticon spaced-emoticon\"},\n\t{text: \"=D\", cssClass: \"small-emoticon spaced-emoticon\"},\n\t{text: \"8O\", cssClass: \"small-emoticon spaced-emoticon\"},\n\t{text: \"[+=..]\", cssClass: \"no-rotate nintendo-controller\"}\n]);\n\nmodule.exports = {\n\temoticons: emoticons\n\t, matchers: matchers\n\t, defaults: defaults\n\t, emoticonize: function(str, options) {\n\t\tconst opts = $.extend({}, defaults, options);\n\n\t\tlet cssClass = 'css-emoticon';\n\t\tif (opts.animate) {\n\t\t\tcssClass += ' un-transformed-emoticon animated-emoticon';\n\t\t}\n\t\tfor (let i = 0; i < matchers.length; ++i) {\n\t\t\tconst m = matchers[i];\n\t\t\tconst css = cssClass + \" \" + m.cssClass;\n\t\t\tstr = str.replace(m.regexp, \"$1<span class='\" + css + \"'>$2</span>\");\n\t\t}\n\t\treturn str;\n\t}\n\t, animate: function(options) {\n\t\tconst opts = $.extend({}, defaults, options);\n\t\t// animate emoticons\n\t\tif (opts.animate) {\n\t\t\tsetTimeout(function () {\n\t\t\t\t$('.un-transformed-emoticon').removeClass('un-transformed-emoticon');\n\t\t\t}, opts.delay);\n\t\t}\n\t}\n};\n","module.exports = OmUtil;","module.exports = Settings;","// The module cache\nvar __webpack_module_cache__ = {};\n\n// The require function\nfunction __webpack_require__(moduleId) {\n\t// Check if module is in cache\n\tvar cachedModule = __webpack_module_cache__[moduleId];\n\tif (cachedModule !== undefined) {\n\t\treturn cachedModule.exports;\n\t}\n\t// Create a new module (and put it into the cache)\n\tvar module = __webpack_module_cache__[moduleId] = {\n\t\t// no module.id needed\n\t\t// no module.loaded needed\n\t\texports: {}\n\t};\n\n\t// Execute the module function\n\t__webpack_modules__[moduleId](module, module.exports, __webpack_require__);\n\n\t// Return the exports of the module\n\treturn module.exports;\n}\n\n","/* Licensed under the Apache License, Version 2.0 (the \"License\") http://www.apache.org/licenses/LICENSE-2.0 */\nObject.assign(window, {\n\tChat: require('./chat')\n});\n"],"names":[],"sourceRoot":""}
|