mirror of
https://github.com/NodeBB/NodeBB.git
synced 2025-10-26 16:46:12 +01:00
feat: allow chat rooms in widgets
This commit is contained in:
@@ -108,7 +108,7 @@
|
|||||||
"nodebb-theme-lavender": "7.1.8",
|
"nodebb-theme-lavender": "7.1.8",
|
||||||
"nodebb-theme-peace": "2.2.6",
|
"nodebb-theme-peace": "2.2.6",
|
||||||
"nodebb-theme-persona": "13.3.25",
|
"nodebb-theme-persona": "13.3.25",
|
||||||
"nodebb-widget-essentials": "7.0.18",
|
"nodebb-widget-essentials": "7.0.19",
|
||||||
"nodemailer": "6.9.13",
|
"nodemailer": "6.9.13",
|
||||||
"nprogress": "0.2.0",
|
"nprogress": "0.2.0",
|
||||||
"passport": "0.7.0",
|
"passport": "0.7.0",
|
||||||
|
|||||||
@@ -1,8 +1,9 @@
|
|||||||
{
|
{
|
||||||
"chat.room-id": "Room %1",
|
"chat.room-id": "Room %1",
|
||||||
"chat.chatting-with": "Chat with",
|
"chat.chatting-with": "Chat with",
|
||||||
"chat.placeholder": "Type chat message here, drag & drop images, press enter to send",
|
"chat.placeholder": "Type chat message here, drag & drop images",
|
||||||
"chat.placeholder.mobile": "Type chat message here",
|
"chat.placeholder.mobile": "Type chat message",
|
||||||
|
"chat.placeholder.message-room": "Message #%1",
|
||||||
"chat.scroll-up-alert": "Go to most recent message",
|
"chat.scroll-up-alert": "Go to most recent message",
|
||||||
"chat.usernames-and-x-others": "%1 & %2 others",
|
"chat.usernames-and-x-others": "%1 & %2 others",
|
||||||
"chat.chat-with-usernames": "Chat with %1",
|
"chat.chat-with-usernames": "Chat with %1",
|
||||||
|
|||||||
@@ -1,10 +1,9 @@
|
|||||||
'use strict';
|
'use strict';
|
||||||
|
|
||||||
define('chat', [
|
define('chat', [
|
||||||
'components', 'taskbar', 'translator', 'hooks', 'bootbox', 'alerts', 'api',
|
'components', 'taskbar', 'translator', 'hooks', 'bootbox', 'alerts', 'api', 'scrollStop',
|
||||||
], function (components, taskbar, translator, hooks, bootbox, alerts, api) {
|
], function (components, taskbar, translator, hooks, bootbox, alerts, api, scrollStop) {
|
||||||
const module = {};
|
const module = {};
|
||||||
let newMessage = false;
|
|
||||||
|
|
||||||
module.openChat = function (roomId, uid) {
|
module.openChat = function (roomId, uid) {
|
||||||
if (!app.user.uid) {
|
if (!app.user.uid) {
|
||||||
@@ -182,9 +181,11 @@ define('chat', [
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (module.modalExists(data.roomId)) {
|
if (module.modalExists(data.roomId)) {
|
||||||
|
const modal = module.getModal(data.roomId);
|
||||||
|
const newMessage = parseInt(modal.attr('new-message'), 10) === 1;
|
||||||
data.self = parseInt(app.user.uid, 10) === parseInt(data.fromUid, 10) ? 1 : 0;
|
data.self = parseInt(app.user.uid, 10) === parseInt(data.fromUid, 10) ? 1 : 0;
|
||||||
if (!newMessage) {
|
if (!newMessage) {
|
||||||
newMessage = data.self === 0;
|
modal.attr('new-message', data.self === 0 ? 1 : 0);
|
||||||
}
|
}
|
||||||
data.message.self = data.self;
|
data.message.self = data.self;
|
||||||
data.message.timestamp = Math.min(Date.now(), data.message.timestamp);
|
data.message.timestamp = Math.min(Date.now(), data.message.timestamp);
|
||||||
@@ -290,11 +291,61 @@ define('chat', [
|
|||||||
return $('#chat-modal-' + roomId).length !== 0;
|
return $('#chat-modal-' + roomId).length !== 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
module.initWidget = function (roomId, chatModal) {
|
||||||
|
require(['forum/chats', 'forum/chats/messages'], function (Chats, ChatsMessages) {
|
||||||
|
socket.emit('modules.chats.enter', roomId);
|
||||||
|
api.del(`/chats/${roomId}/state`, {});
|
||||||
|
|
||||||
|
chatModal.find('.timeago').timeago();
|
||||||
|
chatModal.find('[data-bs-toggle="tooltip"]').tooltip({
|
||||||
|
trigger: 'hover', container: '#content',
|
||||||
|
});
|
||||||
|
ChatsMessages.wrapImagesInLinks(chatModal.find('[component="chat/messages"] .chat-content'));
|
||||||
|
|
||||||
|
scrollStop.apply(chatModal.find('[component="chat/messages"] .chat-content'));
|
||||||
|
|
||||||
|
chatModal.on('mousemove keypress click', function () {
|
||||||
|
if (parseInt(chatModal.attr('new-message'), 10) === 1) {
|
||||||
|
console.log('marking chat read');
|
||||||
|
api.del(`/chats/${roomId}/state`, {});
|
||||||
|
chatModal.removeAttr('new-message');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
Chats.addActionHandlers(chatModal.find('[component="chat/message/window"]'), roomId);
|
||||||
|
Chats.addSendHandlers(roomId, chatModal.find('.chat-input'), chatModal.find('[data-action="send"]'));
|
||||||
|
|
||||||
|
Chats.createAutoComplete(roomId, chatModal.find('[component="chat/input"]'));
|
||||||
|
|
||||||
|
Chats.addScrollHandler(roomId, app.user.uid, chatModal.find('[component="chat/message/content"]'));
|
||||||
|
Chats.addScrollBottomHandler(roomId, chatModal.find('[component="chat/message/content"]'));
|
||||||
|
Chats.addParentHandler(chatModal.find('[component="chat/message/content"]'));
|
||||||
|
Chats.addCharactersLeftHandler(chatModal);
|
||||||
|
Chats.addTextareaResizeHandler(chatModal);
|
||||||
|
Chats.addTypingHandler(chatModal, roomId);
|
||||||
|
Chats.addIPHandler(chatModal);
|
||||||
|
Chats.addTooltipHandler(chatModal);
|
||||||
|
Chats.addUploadHandler({
|
||||||
|
dragDropAreaEl: chatModal.find('.modal-content'),
|
||||||
|
pasteEl: chatModal,
|
||||||
|
uploadFormEl: chatModal.find('[component="chat/upload"]'),
|
||||||
|
uploadBtnEl: chatModal.find('[component="chat/upload/button"]'),
|
||||||
|
inputEl: chatModal.find('[component="chat/input"]'),
|
||||||
|
});
|
||||||
|
|
||||||
|
ChatsMessages.addSocketListeners();
|
||||||
|
|
||||||
|
ChatsMessages.scrollToBottomAfterImageLoad(chatModal.find('.chat-content'));
|
||||||
|
|
||||||
|
hooks.fire('action:chat.loaded', chatModal);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
module.createModal = function (data, callback) {
|
module.createModal = function (data, callback) {
|
||||||
callback = callback || function () {};
|
callback = callback || function () {};
|
||||||
require([
|
require([
|
||||||
'scrollStop', 'forum/chats', 'forum/chats/messages', 'forum/chats/message-search',
|
'forum/chats', 'forum/chats/messages', 'forum/chats/message-search',
|
||||||
], function (scrollStop, Chats, ChatsMessages, messageSearch) {
|
], function (Chats, ChatsMessages, messageSearch) {
|
||||||
app.parseAndTranslate('chat', data, function (chatModal) {
|
app.parseAndTranslate('chat', data, function (chatModal) {
|
||||||
const roomId = data.roomId;
|
const roomId = data.roomId;
|
||||||
if (module.modalExists(roomId)) {
|
if (module.modalExists(roomId)) {
|
||||||
@@ -305,45 +356,19 @@ define('chat', [
|
|||||||
|
|
||||||
chatModal.attr('id', 'chat-modal-' + roomId);
|
chatModal.attr('id', 'chat-modal-' + roomId);
|
||||||
chatModal.attr('data-roomid', roomId);
|
chatModal.attr('data-roomid', roomId);
|
||||||
chatModal.attr('intervalId', 0);
|
|
||||||
chatModal.attr('data-uuid', uuid);
|
chatModal.attr('data-uuid', uuid);
|
||||||
chatModal.css('position', 'fixed');
|
chatModal.css('position', 'fixed');
|
||||||
chatModal.appendTo($('body'));
|
chatModal.appendTo($('body'));
|
||||||
chatModal.find('.timeago').timeago();
|
chatModal.find('.timeago').timeago();
|
||||||
chatModal.find('[data-bs-toggle="tooltip"]').tooltip({ trigger: 'hover', container: '#content' });
|
chatModal.find('[data-bs-toggle="tooltip"]').tooltip({ trigger: 'hover', container: '#content' });
|
||||||
ChatsMessages.wrapImagesInLinks(chatModal.find('[component="chat/messages"] .chat-content'));
|
ChatsMessages.wrapImagesInLinks(chatModal.find('[component="chat/messages"] .chat-content'));
|
||||||
module.center(chatModal);
|
|
||||||
|
|
||||||
app.loadJQueryUI(function () {
|
|
||||||
chatModal.find('.modal-content').resizable({
|
|
||||||
handles: 'n, e, s, w, se',
|
|
||||||
minHeight: 250,
|
|
||||||
minWidth: 400,
|
|
||||||
});
|
|
||||||
|
|
||||||
chatModal.find('.modal-content').on('resize', function (event, ui) {
|
|
||||||
if (ui.originalSize.height === ui.size.height) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
chatModal.find('.modal-body').css('height', module.calculateChatListHeight(chatModal));
|
|
||||||
});
|
|
||||||
|
|
||||||
chatModal.draggable({
|
|
||||||
start: function () {
|
|
||||||
taskbar.updateActive(uuid);
|
|
||||||
chatModal.css({ bottom: 'auto', right: 'auto' });
|
|
||||||
},
|
|
||||||
stop: function () {
|
|
||||||
module.focusInput(chatModal);
|
|
||||||
},
|
|
||||||
distance: 10,
|
|
||||||
handle: '.modal-header',
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
scrollStop.apply(chatModal.find('[component="chat/messages"] .chat-content'));
|
scrollStop.apply(chatModal.find('[component="chat/messages"] .chat-content'));
|
||||||
|
|
||||||
|
module.center(chatModal);
|
||||||
|
|
||||||
|
makeModalResizeableDraggable(chatModal, uuid);
|
||||||
|
|
||||||
chatModal.find('#chat-close-btn').on('click', function () {
|
chatModal.find('#chat-close-btn').on('click', function () {
|
||||||
module.close(uuid);
|
module.close(uuid);
|
||||||
});
|
});
|
||||||
@@ -380,9 +405,9 @@ define('chat', [
|
|||||||
});
|
});
|
||||||
|
|
||||||
chatModal.on('mousemove keypress click', function () {
|
chatModal.on('mousemove keypress click', function () {
|
||||||
if (newMessage) {
|
if (parseInt(chatModal.attr('new-message'), 10) === 1) {
|
||||||
api.del(`/chats/${roomId}/state`, {});
|
api.del(`/chats/${roomId}/state`, {});
|
||||||
newMessage = false;
|
chatModal.removeAttr('new-message');
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -433,6 +458,36 @@ define('chat', [
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
function makeModalResizeableDraggable(chatModal, uuid) {
|
||||||
|
app.loadJQueryUI(function () {
|
||||||
|
chatModal.find('.modal-content').resizable({
|
||||||
|
handles: 'n, e, s, w, se',
|
||||||
|
minHeight: 250,
|
||||||
|
minWidth: 400,
|
||||||
|
});
|
||||||
|
|
||||||
|
chatModal.find('.modal-content').on('resize', function (event, ui) {
|
||||||
|
if (ui.originalSize.height === ui.size.height) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
chatModal.find('.modal-body').css('height', module.calculateChatListHeight(chatModal));
|
||||||
|
});
|
||||||
|
|
||||||
|
chatModal.draggable({
|
||||||
|
start: function () {
|
||||||
|
taskbar.updateActive(uuid);
|
||||||
|
chatModal.css({ bottom: 'auto', right: 'auto' });
|
||||||
|
},
|
||||||
|
stop: function () {
|
||||||
|
module.focusInput(chatModal);
|
||||||
|
},
|
||||||
|
distance: 10,
|
||||||
|
handle: '.modal-header',
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
module.focusInput = function (chatModal) {
|
module.focusInput = function (chatModal) {
|
||||||
setTimeout(function () {
|
setTimeout(function () {
|
||||||
chatModal.find('[component="chat/input"]').focus();
|
chatModal.find('[component="chat/input"]').focus();
|
||||||
@@ -441,8 +496,6 @@ define('chat', [
|
|||||||
|
|
||||||
module.close = function (uuid) {
|
module.close = function (uuid) {
|
||||||
const chatModal = $('.chat-modal[data-uuid="' + uuid + '"]');
|
const chatModal = $('.chat-modal[data-uuid="' + uuid + '"]');
|
||||||
clearInterval(chatModal.attr('intervalId'));
|
|
||||||
chatModal.attr('intervalId', 0);
|
|
||||||
chatModal.remove();
|
chatModal.remove();
|
||||||
chatModal.data('modal', null);
|
chatModal.data('modal', null);
|
||||||
taskbar.discard('chat', uuid);
|
taskbar.discard('chat', uuid);
|
||||||
@@ -529,8 +582,6 @@ define('chat', [
|
|||||||
const chatModal = $('.chat-modal[data-uuid="' + uuid + '"]');
|
const chatModal = $('.chat-modal[data-uuid="' + uuid + '"]');
|
||||||
chatModal.addClass('hide');
|
chatModal.addClass('hide');
|
||||||
taskbar.minimize('chat', uuid);
|
taskbar.minimize('chat', uuid);
|
||||||
clearInterval(chatModal.attr('intervalId'));
|
|
||||||
chatModal.attr('intervalId', 0);
|
|
||||||
hooks.fire('action:chat.minimized', {
|
hooks.fire('action:chat.minimized', {
|
||||||
uuid: uuid,
|
uuid: uuid,
|
||||||
modal: chatModal,
|
modal: chatModal,
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
<div class="w-100 flex-grow-1 flex-nowrap position-relative d-flex rounded-2 border border-secondary p-1 align-items-end">
|
<div class="w-100 flex-grow-1 flex-nowrap position-relative d-flex rounded-2 border border-secondary p-1 align-items-end">
|
||||||
<button component="chat/upload/button" class="btn-ghost-sm px-2" type="button" title="[[global:upload]]" data-bs-toggle="tooltip"><i class="fa fa-fw fa-upload"></i></button>
|
<button component="chat/upload/button" class="btn-ghost-sm px-2" type="button" title="[[global:upload]]" data-bs-toggle="tooltip"><i class="fa fa-fw fa-upload"></i></button>
|
||||||
<div class="flex-grow-1 align-self-center">
|
<div class="flex-grow-1 align-self-center">
|
||||||
<textarea component="chat/input" placeholder="[[modules:chat.placeholder.mobile]]" class="bg-transparent text-body form-control chat-input mousetrap rounded-0 border-0 shadow-none px-1 py-0" style="min-height: 1.5rem;height:0;max-height:30vh;resize:none;"></textarea>
|
<textarea component="chat/input" placeholder="{{{ if roomName }}}[[modules:chat.placeholder.message-room, {roomName}]]{{{ else }}}[[modules:chat.placeholder.mobile]]{{{ end }}}" class="bg-transparent text-body form-control chat-input mousetrap rounded-0 border-0 shadow-none px-1 py-0" style="min-height: 1.5rem;height:0;max-height:30vh;resize:none;"></textarea>
|
||||||
</div>
|
</div>
|
||||||
<div class="d-flex gap-1">
|
<div class="d-flex gap-1">
|
||||||
{{{ each composerActions }}}
|
{{{ each composerActions }}}
|
||||||
|
|||||||
Reference in New Issue
Block a user