This commit adds variousMessageBus.last_ids to serializer payloads for chat channels and the chat view (for chat live pane) so we can use those IDs when subscribing to MessageBus channels from chat. This allows us to ensure that any messages created between the server being hit and the UI loaded and subscribing end up being delivered to the client, rather than just silently dropped. This commit also fixes an issue where we were subscribing to the new-messages and new-mentions MessageBus channels multiple times when following/unfollowing a channel multiple times.
186 lines
5.4 KiB
JavaScript
186 lines
5.4 KiB
JavaScript
import { withPluginApi } from "discourse/lib/plugin-api";
|
|
import I18n from "I18n";
|
|
import { bind } from "discourse-common/utils/decorators";
|
|
import { getOwner } from "discourse-common/lib/get-owner";
|
|
import { MENTION_KEYWORDS } from "discourse/plugins/chat/discourse/components/chat-message";
|
|
import { clearChatComposerButtons } from "discourse/plugins/chat/discourse/lib/chat-composer-buttons";
|
|
|
|
let _lastForcedRefreshAt;
|
|
const MIN_REFRESH_DURATION_MS = 180000; // 3 minutes
|
|
|
|
export default {
|
|
name: "chat-setup",
|
|
initialize(container) {
|
|
this.chatService = container.lookup("service:chat");
|
|
|
|
if (!this.chatService.userCanChat) {
|
|
return;
|
|
}
|
|
|
|
this.siteSettings = container.lookup("service:site-settings");
|
|
this.appEvents = container.lookup("service:appEvents");
|
|
this.appEvents.on("discourse:focus-changed", this, "_handleFocusChanged");
|
|
|
|
withPluginApi("0.12.1", (api) => {
|
|
api.registerChatComposerButton({
|
|
id: "chat-upload-btn",
|
|
icon: "far-image",
|
|
label: "chat.upload",
|
|
position: "dropdown",
|
|
action: "uploadClicked",
|
|
dependentKeys: ["canAttachUploads"],
|
|
displayed() {
|
|
return this.canAttachUploads;
|
|
},
|
|
});
|
|
|
|
if (this.siteSettings.discourse_local_dates_enabled) {
|
|
api.registerChatComposerButton({
|
|
label: "discourse_local_dates.title",
|
|
id: "local-dates",
|
|
class: "chat-local-dates-btn",
|
|
icon: "calendar-alt",
|
|
position: "dropdown",
|
|
action() {
|
|
this.insertDiscourseLocalDate();
|
|
},
|
|
});
|
|
}
|
|
|
|
api.registerChatComposerButton({
|
|
label: "chat.emoji",
|
|
id: "emoji",
|
|
class: "chat-emoji-btn",
|
|
icon: "discourse-emojis",
|
|
position: "dropdown",
|
|
action() {
|
|
const chatEmojiPickerManager = container.lookup(
|
|
"service:chat-emoji-picker-manager"
|
|
);
|
|
chatEmojiPickerManager.startFromComposer(this.didSelectEmoji);
|
|
},
|
|
});
|
|
|
|
// we want to decorate the chat quote dates regardless
|
|
// of whether the current user has chat enabled
|
|
api.decorateCookedElement(
|
|
(elem) => {
|
|
const currentUser = getOwner(this).lookup("service:current-user");
|
|
const currentUserTimezone =
|
|
currentUser?.resolvedTimezone(currentUser);
|
|
const chatTranscriptElements =
|
|
elem.querySelectorAll(".chat-transcript");
|
|
|
|
chatTranscriptElements.forEach((el) => {
|
|
const dateTimeRaw = el.dataset["datetime"];
|
|
const dateTimeEl = el.querySelector(
|
|
".chat-transcript-datetime a, .chat-transcript-datetime span"
|
|
);
|
|
|
|
if (currentUserTimezone) {
|
|
dateTimeEl.innerText = moment
|
|
.tz(dateTimeRaw, currentUserTimezone)
|
|
.format(I18n.t("dates.long_no_year"));
|
|
} else {
|
|
dateTimeEl.innerText = moment(dateTimeRaw).format(
|
|
I18n.t("dates.long_no_year")
|
|
);
|
|
}
|
|
});
|
|
},
|
|
{ id: "chat-transcript-datetime" }
|
|
);
|
|
|
|
if (!this.chatService.userCanChat) {
|
|
return;
|
|
}
|
|
|
|
document.body.classList.add("chat-enabled");
|
|
|
|
const currentUser = api.getCurrentUser();
|
|
if (currentUser?.chat_channels) {
|
|
this.chatService.setupWithPreloadedChannels(currentUser.chat_channels);
|
|
} else {
|
|
this.chatService.setupWithoutPreloadedChannels();
|
|
}
|
|
|
|
const chatNotificationManager = container.lookup(
|
|
"service:chat-notification-manager"
|
|
);
|
|
chatNotificationManager.start();
|
|
|
|
if (!this._registeredDocumentTitleCountCallback) {
|
|
api.addDocumentTitleCounter(this.documentTitleCountCallback);
|
|
this._registeredDocumentTitleCountCallback = true;
|
|
}
|
|
|
|
api.addCardClickListenerSelector(".chat-drawer-outlet");
|
|
|
|
api.dispatchWidgetAppEvent(
|
|
"site-header",
|
|
"header-chat-link",
|
|
"chat:rerender-header"
|
|
);
|
|
|
|
api.dispatchWidgetAppEvent(
|
|
"sidebar-header",
|
|
"header-chat-link",
|
|
"chat:rerender-header"
|
|
);
|
|
|
|
api.addToHeaderIcons("header-chat-link");
|
|
|
|
api.decorateChatMessage(function (chatMessage) {
|
|
if (!this.currentUser) {
|
|
return;
|
|
}
|
|
|
|
const highlightable = [
|
|
`@${this.currentUser.username}`,
|
|
...MENTION_KEYWORDS.map((k) => `@${k}`),
|
|
];
|
|
|
|
chatMessage.querySelectorAll(".mention").forEach((node) => {
|
|
const mention = node.textContent.trim();
|
|
if (highlightable.includes(mention)) {
|
|
node.classList.add("highlighted", "valid-mention");
|
|
}
|
|
});
|
|
});
|
|
});
|
|
},
|
|
|
|
@bind
|
|
documentTitleCountCallback() {
|
|
return this.chatService.getDocumentTitleCount();
|
|
},
|
|
|
|
teardown() {
|
|
if (!this.chatService.userCanChat) {
|
|
return;
|
|
}
|
|
|
|
this.appEvents.off("discourse:focus-changed", this, "_handleFocusChanged");
|
|
_lastForcedRefreshAt = null;
|
|
clearChatComposerButtons();
|
|
},
|
|
|
|
@bind
|
|
_handleFocusChanged(hasFocus) {
|
|
if (!hasFocus) {
|
|
_lastForcedRefreshAt = Date.now();
|
|
return;
|
|
}
|
|
|
|
_lastForcedRefreshAt = _lastForcedRefreshAt || Date.now();
|
|
|
|
const duration = Date.now() - _lastForcedRefreshAt;
|
|
if (duration <= MIN_REFRESH_DURATION_MS) {
|
|
return;
|
|
}
|
|
|
|
_lastForcedRefreshAt = Date.now();
|
|
this.chatService.refreshTrackingState();
|
|
},
|
|
};
|