import './message-actions.js'; import './message-body.js'; import 'shared/components/dropdown.js'; import 'shared/modals/message-versions.js'; import 'shared/modals/user-details.js'; import 'shared/registry'; import 'plugins/muc-views/modals/occupant.js'; import tpl_file_progress from './templates/file-progress.js'; import log from '@converse/headless/log'; import tpl_info_message from './templates/info-message.js'; import tpl_mep_message from 'plugins/muc-views/templates/mep-message.js'; import tpl_message from './templates/message.js'; import tpl_message_text from './templates/message-text.js'; import tpl_retraction from './templates/retraction.js'; import tpl_spinner from 'templates/spinner.js'; import { CustomElement } from 'shared/components/element.js'; import { __ } from 'i18n'; import { _converse, api, converse } from '@converse/headless/core'; import { getAppSettings } from '@converse/headless/shared/settings/utils.js'; import { getHats } from './utils.js'; const { Strophe, dayjs } = converse.env; export default class Message extends CustomElement { static get properties () { return { jid: { type: String }, mid: { type: String } } } async initialize () { await this.setModels(); if (!this.model) { // Happen during tests due to a race condition log.error('Could not find module for converse-chat-message'); return; } const settings = getAppSettings(); this.listenTo(settings, 'change:render_media', () => { // Reset individual show/hide state of media this.model.save('hide_url_previews', undefined) this.requestUpdate(); }); this.listenTo(this.chatbox, 'change:first_unread_id', () => this.requestUpdate()); this.listenTo(this.model, 'change', () => this.requestUpdate()); this.model.vcard && this.listenTo(this.model.vcard, 'change', () => this.requestUpdate()); if (this.model.get('type') === 'groupchat') { if (this.model.occupant) { this.listenTo(this.model.occupant, 'change', () => this.requestUpdate()); } else { this.listenTo(this.model, 'occupantAdded', () => { this.requestUpdate(); this.listenTo(this.model.occupant, 'change', () => this.requestUpdate()) }); } } } async setModels () { this.chatbox = await api.chatboxes.get(this.jid); await this.chatbox.initialized; await this.chatbox.messages.fetched; this.model = this.chatbox.messages.get(this.mid); this.model && this.requestUpdate(); } render () { if (!this.model) { return ''; } else if (this.show_spinner) { return tpl_spinner(); } else if (this.model.get('file') && this.model.get('upload') !== _converse.SUCCESS) { return this.renderFileProgress(); } else if (['mep'].includes(this.model.get('type'))) { return this.renderMEPMessage(); } else if (['error', 'info'].includes(this.model.get('type'))) { return this.renderInfoMessage(); } else { return this.renderChatMessage(); } } getProps () { return Object.assign( this.model.toJSON(), this.getDerivedMessageProps() ); } renderRetraction () { return tpl_retraction(this); } renderMessageText () { return tpl_message_text(this); } renderMEPMessage () { return tpl_mep_message(this); } renderInfoMessage () { return tpl_info_message(this); } renderFileProgress () { if (!this.model.file) { // Can happen when file upload failed and page was reloaded return ''; } return tpl_file_progress(this); } renderChatMessage () { return tpl_message(this, this.getProps()); } shouldShowAvatar () { return api.settings.get('show_message_avatar') && !this.model.isMeCommand() && ['chat', 'groupchat'].includes(this.model.get('type')); } onUnfurlAnimationEnd () { if (this.model.get('url_preview_transition') === 'fade-out') { this.model.save({ 'hide_url_previews': true, 'url_preview_transition': 'fade-in' }); } } async onRetryClicked () { this.show_spinner = true; this.requestUpdate(); await api.trigger(this.model.get('retry_event_id'), {'synchronous': true}); this.model.destroy(); this.parentElement.removeChild(this); } isRetracted () { return this.model.get('retracted') || this.model.get('moderated') === 'retracted'; } hasMentions () { const is_groupchat = this.model.get('type') === 'groupchat'; return is_groupchat && this.model.get('sender') === 'them' && this.chatbox.isUserMentioned(this.model); } getOccupantAffiliation () { return this.model.occupant?.get('affiliation'); } getOccupantRole () { return this.model.occupant?.get('role'); } getExtraMessageClasses () { const extra_classes = [ this.model.isFollowup() ? 'chat-msg--followup' : null, this.model.get('is_delayed') ? 'delayed' : null, this.model.isMeCommand() ? 'chat-msg--action' : null, this.isRetracted() ? 'chat-msg--retracted' : null, this.model.get('type'), this.shouldShowAvatar() ? 'chat-msg--with-avatar' : null, ].map(c => c); if (this.model.get('type') === 'groupchat') { extra_classes.push(this.getOccupantRole() ?? ''); extra_classes.push(this.getOccupantAffiliation() ?? ''); if (this.model.get('sender') === 'them' && this.hasMentions()) { extra_classes.push('mentioned'); } } this.model.get('correcting') && extra_classes.push('correcting'); return extra_classes.filter(c => c).join(" "); } getDerivedMessageProps () { const format = api.settings.get('time_format'); return { 'pretty_time': dayjs(this.model.get('edited') || this.model.get('time')).format(format), 'has_mentions': this.hasMentions(), 'hats': getHats(this.model), 'is_first_unread': this.chatbox.get('first_unread_id') === this.model.get('id'), 'is_me_message': this.model.isMeCommand(), 'is_retracted': this.isRetracted(), 'username': this.model.getDisplayName(), 'should_show_avatar': this.shouldShowAvatar(), } } getRetractionText () { if (['groupchat', 'mep'].includes(this.model.get('type')) && this.model.get('moderated_by')) { const retracted_by_mod = this.model.get('moderated_by'); const chatbox = this.model.collection.chatbox; if (!this.model.mod) { this.model.mod = chatbox.occupants.findOccupant({'jid': retracted_by_mod}) || chatbox.occupants.findOccupant({'nick': Strophe.getResourceFromJid(retracted_by_mod)}); } const modname = this.model.mod ? this.model.mod.getDisplayName() : 'A moderator'; return __('%1$s has removed this message', modname); } else { return __('%1$s has removed this message', this.model.getDisplayName()); } } showUserModal (ev) { if (this.model.get('sender') === 'me') { api.modal.show('converse-profile-modal', {model: this.model}, ev); } else if (this.model.get('type') === 'groupchat') { ev.preventDefault(); api.modal.show('converse-muc-occupant-modal', { 'model': this.model.occupant, 'message': this.model }, ev); } else { ev.preventDefault(); const chatbox = this.model.collection.chatbox; api.modal.show('converse-user-details-modal', { model: chatbox }, ev); } } showMessageVersionsModal (ev) { ev.preventDefault(); api.modal.show('converse-message-versions-modal', {'model': this.model}, ev); } toggleSpoilerMessage (ev) { ev?.preventDefault(); this.model.save({'is_spoiler_visible': !this.model.get('is_spoiler_visible')}); } } api.elements.define('converse-chat-message', Message);