mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-11-04 04:57:51 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			561 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			561 lines
		
	
	
		
			17 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
<script>
 | 
						|
import { ref, provide } from 'vue';
 | 
						|
// composable
 | 
						|
import { useConfig } from 'dashboard/composables/useConfig';
 | 
						|
import { useKeyboardEvents } from 'dashboard/composables/useKeyboardEvents';
 | 
						|
import { useAI } from 'dashboard/composables/useAI';
 | 
						|
 | 
						|
// components
 | 
						|
import ReplyBox from './ReplyBox.vue';
 | 
						|
import MessageList from 'next/message/MessageList.vue';
 | 
						|
import ConversationLabelSuggestion from './conversation/LabelSuggestion.vue';
 | 
						|
import Banner from 'dashboard/components/ui/Banner.vue';
 | 
						|
import Spinner from 'dashboard/components-next/spinner/Spinner.vue';
 | 
						|
 | 
						|
// stores and apis
 | 
						|
import { mapGetters } from 'vuex';
 | 
						|
 | 
						|
// mixins
 | 
						|
import inboxMixin, { INBOX_FEATURES } from 'shared/mixins/inboxMixin';
 | 
						|
 | 
						|
// utils
 | 
						|
import { emitter } from 'shared/helpers/mitt';
 | 
						|
import { getTypingUsersText } from '../../../helper/commons';
 | 
						|
import { calculateScrollTop } from './helpers/scrollTopCalculationHelper';
 | 
						|
import { LocalStorage } from 'shared/helpers/localStorage';
 | 
						|
import {
 | 
						|
  filterDuplicateSourceMessages,
 | 
						|
  getReadMessages,
 | 
						|
  getUnreadMessages,
 | 
						|
} from 'dashboard/helper/conversationHelper';
 | 
						|
 | 
						|
// constants
 | 
						|
import { BUS_EVENTS } from 'shared/constants/busEvents';
 | 
						|
import { REPLY_POLICY } from 'shared/constants/links';
 | 
						|
import wootConstants from 'dashboard/constants/globals';
 | 
						|
import { LOCAL_STORAGE_KEYS } from 'dashboard/constants/localStorage';
 | 
						|
import { INBOX_TYPES } from 'dashboard/helper/inbox';
 | 
						|
 | 
						|
export default {
 | 
						|
  components: {
 | 
						|
    MessageList,
 | 
						|
    ReplyBox,
 | 
						|
    Banner,
 | 
						|
    ConversationLabelSuggestion,
 | 
						|
    Spinner,
 | 
						|
  },
 | 
						|
  mixins: [inboxMixin],
 | 
						|
  setup() {
 | 
						|
    const isPopOutReplyBox = ref(false);
 | 
						|
    const conversationPanelRef = ref(null);
 | 
						|
    const { isEnterprise } = useConfig();
 | 
						|
 | 
						|
    const keyboardEvents = {
 | 
						|
      Escape: {
 | 
						|
        action: () => {
 | 
						|
          isPopOutReplyBox.value = false;
 | 
						|
        },
 | 
						|
      },
 | 
						|
    };
 | 
						|
 | 
						|
    useKeyboardEvents(keyboardEvents);
 | 
						|
 | 
						|
    const {
 | 
						|
      isAIIntegrationEnabled,
 | 
						|
      isLabelSuggestionFeatureEnabled,
 | 
						|
      fetchIntegrationsIfRequired,
 | 
						|
      fetchLabelSuggestions,
 | 
						|
    } = useAI();
 | 
						|
 | 
						|
    provide('contextMenuElementTarget', conversationPanelRef);
 | 
						|
 | 
						|
    return {
 | 
						|
      isEnterprise,
 | 
						|
      isPopOutReplyBox,
 | 
						|
      isAIIntegrationEnabled,
 | 
						|
      isLabelSuggestionFeatureEnabled,
 | 
						|
      fetchIntegrationsIfRequired,
 | 
						|
      fetchLabelSuggestions,
 | 
						|
      conversationPanelRef,
 | 
						|
    };
 | 
						|
  },
 | 
						|
  data() {
 | 
						|
    return {
 | 
						|
      isLoadingPrevious: true,
 | 
						|
      heightBeforeLoad: null,
 | 
						|
      conversationPanel: null,
 | 
						|
      hasUserScrolled: false,
 | 
						|
      isProgrammaticScroll: false,
 | 
						|
      messageSentSinceOpened: false,
 | 
						|
      labelSuggestions: [],
 | 
						|
    };
 | 
						|
  },
 | 
						|
 | 
						|
  computed: {
 | 
						|
    ...mapGetters({
 | 
						|
      currentChat: 'getSelectedChat',
 | 
						|
      currentUserId: 'getCurrentUserID',
 | 
						|
      listLoadingStatus: 'getAllMessagesLoaded',
 | 
						|
      currentAccountId: 'getCurrentAccountId',
 | 
						|
    }),
 | 
						|
    isOpen() {
 | 
						|
      return this.currentChat?.status === wootConstants.STATUS_TYPE.OPEN;
 | 
						|
    },
 | 
						|
    shouldShowLabelSuggestions() {
 | 
						|
      return (
 | 
						|
        this.isOpen &&
 | 
						|
        this.isEnterprise &&
 | 
						|
        this.isAIIntegrationEnabled &&
 | 
						|
        !this.messageSentSinceOpened
 | 
						|
      );
 | 
						|
    },
 | 
						|
    inboxId() {
 | 
						|
      return this.currentChat.inbox_id;
 | 
						|
    },
 | 
						|
    inbox() {
 | 
						|
      return this.$store.getters['inboxes/getInbox'](this.inboxId);
 | 
						|
    },
 | 
						|
    typingUsersList() {
 | 
						|
      const userList = this.$store.getters[
 | 
						|
        'conversationTypingStatus/getUserList'
 | 
						|
      ](this.currentChat.id);
 | 
						|
      return userList;
 | 
						|
    },
 | 
						|
    isAnyoneTyping() {
 | 
						|
      const userList = this.typingUsersList;
 | 
						|
      return userList.length !== 0;
 | 
						|
    },
 | 
						|
    typingUserNames() {
 | 
						|
      const userList = this.typingUsersList;
 | 
						|
      if (this.isAnyoneTyping) {
 | 
						|
        const [i18nKey, params] = getTypingUsersText(userList);
 | 
						|
        return this.$t(i18nKey, params);
 | 
						|
      }
 | 
						|
 | 
						|
      return '';
 | 
						|
    },
 | 
						|
    getMessages() {
 | 
						|
      const messages = this.currentChat.messages || [];
 | 
						|
      if (this.isAWhatsAppChannel) {
 | 
						|
        return filterDuplicateSourceMessages(messages);
 | 
						|
      }
 | 
						|
      return messages;
 | 
						|
    },
 | 
						|
    readMessages() {
 | 
						|
      return getReadMessages(
 | 
						|
        this.getMessages,
 | 
						|
        this.currentChat.agent_last_seen_at
 | 
						|
      );
 | 
						|
    },
 | 
						|
    unReadMessages() {
 | 
						|
      return getUnreadMessages(
 | 
						|
        this.getMessages,
 | 
						|
        this.currentChat.agent_last_seen_at
 | 
						|
      );
 | 
						|
    },
 | 
						|
    shouldShowSpinner() {
 | 
						|
      return (
 | 
						|
        (this.currentChat && this.currentChat.dataFetched === undefined) ||
 | 
						|
        (!this.listLoadingStatus && this.isLoadingPrevious)
 | 
						|
      );
 | 
						|
    },
 | 
						|
    // Check there is a instagram inbox exists with the same instagram_id
 | 
						|
    hasDuplicateInstagramInbox() {
 | 
						|
      const instagramId = this.inbox.instagram_id;
 | 
						|
      const { additional_attributes: additionalAttributes = {} } = this.inbox;
 | 
						|
      const instagramInbox =
 | 
						|
        this.$store.getters['inboxes/getInstagramInboxByInstagramId'](
 | 
						|
          instagramId
 | 
						|
        );
 | 
						|
 | 
						|
      return (
 | 
						|
        this.inbox.channel_type === INBOX_TYPES.FB &&
 | 
						|
        additionalAttributes.type === 'instagram_direct_message' &&
 | 
						|
        instagramInbox
 | 
						|
      );
 | 
						|
    },
 | 
						|
 | 
						|
    replyWindowBannerMessage() {
 | 
						|
      if (this.isAWhatsAppChannel) {
 | 
						|
        return this.$t('CONVERSATION.TWILIO_WHATSAPP_CAN_REPLY');
 | 
						|
      }
 | 
						|
      if (this.isAPIInbox) {
 | 
						|
        const { additional_attributes: additionalAttributes = {} } = this.inbox;
 | 
						|
        if (additionalAttributes) {
 | 
						|
          const {
 | 
						|
            agent_reply_time_window_message: agentReplyTimeWindowMessage,
 | 
						|
            agent_reply_time_window: agentReplyTimeWindow,
 | 
						|
          } = additionalAttributes;
 | 
						|
          return (
 | 
						|
            agentReplyTimeWindowMessage ||
 | 
						|
            this.$t('CONVERSATION.API_HOURS_WINDOW', {
 | 
						|
              hours: agentReplyTimeWindow,
 | 
						|
            })
 | 
						|
          );
 | 
						|
        }
 | 
						|
        return '';
 | 
						|
      }
 | 
						|
      return this.$t('CONVERSATION.CANNOT_REPLY');
 | 
						|
    },
 | 
						|
    replyWindowLink() {
 | 
						|
      if (this.isAFacebookInbox || this.isAnInstagramChannel) {
 | 
						|
        return REPLY_POLICY.FACEBOOK;
 | 
						|
      }
 | 
						|
      if (this.isAWhatsAppCloudChannel) {
 | 
						|
        return REPLY_POLICY.WHATSAPP_CLOUD;
 | 
						|
      }
 | 
						|
      if (!this.isAPIInbox) {
 | 
						|
        return REPLY_POLICY.TWILIO_WHATSAPP;
 | 
						|
      }
 | 
						|
      return '';
 | 
						|
    },
 | 
						|
    replyWindowLinkText() {
 | 
						|
      if (
 | 
						|
        this.isAWhatsAppChannel ||
 | 
						|
        this.isAFacebookInbox ||
 | 
						|
        this.isAnInstagramChannel
 | 
						|
      ) {
 | 
						|
        return this.$t('CONVERSATION.24_HOURS_WINDOW');
 | 
						|
      }
 | 
						|
      if (!this.isAPIInbox) {
 | 
						|
        return this.$t('CONVERSATION.TWILIO_WHATSAPP_24_HOURS_WINDOW');
 | 
						|
      }
 | 
						|
      return '';
 | 
						|
    },
 | 
						|
    unreadMessageCount() {
 | 
						|
      return this.currentChat.unread_count || 0;
 | 
						|
    },
 | 
						|
    unreadMessageLabel() {
 | 
						|
      const count =
 | 
						|
        this.unreadMessageCount > 9 ? '9+' : this.unreadMessageCount;
 | 
						|
      const label =
 | 
						|
        this.unreadMessageCount > 1
 | 
						|
          ? 'CONVERSATION.UNREAD_MESSAGES'
 | 
						|
          : 'CONVERSATION.UNREAD_MESSAGE';
 | 
						|
      return `${count} ${this.$t(label)}`;
 | 
						|
    },
 | 
						|
    inboxSupportsReplyTo() {
 | 
						|
      const incoming = this.inboxHasFeature(INBOX_FEATURES.REPLY_TO);
 | 
						|
      const outgoing =
 | 
						|
        this.inboxHasFeature(INBOX_FEATURES.REPLY_TO_OUTGOING) &&
 | 
						|
        !this.is360DialogWhatsAppChannel;
 | 
						|
 | 
						|
      return { incoming, outgoing };
 | 
						|
    },
 | 
						|
  },
 | 
						|
 | 
						|
  watch: {
 | 
						|
    currentChat(newChat, oldChat) {
 | 
						|
      if (newChat.id === oldChat.id) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      this.fetchAllAttachmentsFromCurrentChat();
 | 
						|
      this.fetchSuggestions();
 | 
						|
      this.messageSentSinceOpened = false;
 | 
						|
    },
 | 
						|
  },
 | 
						|
 | 
						|
  created() {
 | 
						|
    emitter.on(BUS_EVENTS.SCROLL_TO_MESSAGE, this.onScrollToMessage);
 | 
						|
    // when a new message comes in, we refetch the label suggestions
 | 
						|
    emitter.on(BUS_EVENTS.FETCH_LABEL_SUGGESTIONS, this.fetchSuggestions);
 | 
						|
    // when a message is sent we set the flag to true this hides the label suggestions,
 | 
						|
    // until the chat is changed and the flag is reset in the watch for currentChat
 | 
						|
    emitter.on(BUS_EVENTS.MESSAGE_SENT, () => {
 | 
						|
      this.messageSentSinceOpened = true;
 | 
						|
    });
 | 
						|
  },
 | 
						|
 | 
						|
  mounted() {
 | 
						|
    this.addScrollListener();
 | 
						|
    this.fetchAllAttachmentsFromCurrentChat();
 | 
						|
    this.fetchSuggestions();
 | 
						|
  },
 | 
						|
 | 
						|
  unmounted() {
 | 
						|
    this.removeBusListeners();
 | 
						|
    this.removeScrollListener();
 | 
						|
  },
 | 
						|
 | 
						|
  methods: {
 | 
						|
    async fetchSuggestions() {
 | 
						|
      // start empty, this ensures that the label suggestions are not shown
 | 
						|
      this.labelSuggestions = [];
 | 
						|
 | 
						|
      if (this.isLabelSuggestionDismissed()) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      if (!this.isEnterprise) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      // method available in mixin, need to ensure that integrations are present
 | 
						|
      await this.fetchIntegrationsIfRequired();
 | 
						|
 | 
						|
      if (!this.isLabelSuggestionFeatureEnabled) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      this.labelSuggestions = await this.fetchLabelSuggestions({
 | 
						|
        conversationId: this.currentChat.id,
 | 
						|
      });
 | 
						|
 | 
						|
      // once the labels are fetched, we need to scroll to bottom
 | 
						|
      // but we need to wait for the DOM to be updated
 | 
						|
      // so we use the nextTick method
 | 
						|
      this.$nextTick(() => {
 | 
						|
        // this param is added to route, telling the UI to navigate to the message
 | 
						|
        // it is triggered by the SCROLL_TO_MESSAGE method
 | 
						|
        // see setActiveChat on ConversationView.vue for more info
 | 
						|
        const { messageId } = this.$route.query;
 | 
						|
 | 
						|
        // only trigger the scroll to bottom if the user has not scrolled
 | 
						|
        // and there's no active messageId that is selected in view
 | 
						|
        if (!messageId && !this.hasUserScrolled) {
 | 
						|
          this.scrollToBottom();
 | 
						|
        }
 | 
						|
      });
 | 
						|
    },
 | 
						|
    isLabelSuggestionDismissed() {
 | 
						|
      return LocalStorage.getFlag(
 | 
						|
        LOCAL_STORAGE_KEYS.DISMISSED_LABEL_SUGGESTIONS,
 | 
						|
        this.currentAccountId,
 | 
						|
        this.currentChat.id
 | 
						|
      );
 | 
						|
    },
 | 
						|
    fetchAllAttachmentsFromCurrentChat() {
 | 
						|
      this.$store.dispatch('fetchAllAttachments', this.currentChat.id);
 | 
						|
    },
 | 
						|
    removeBusListeners() {
 | 
						|
      emitter.off(BUS_EVENTS.SCROLL_TO_MESSAGE, this.onScrollToMessage);
 | 
						|
    },
 | 
						|
    onScrollToMessage({ messageId = '' } = {}) {
 | 
						|
      this.$nextTick(() => {
 | 
						|
        const messageElement = document.getElementById('message' + messageId);
 | 
						|
        if (messageElement) {
 | 
						|
          this.isProgrammaticScroll = true;
 | 
						|
          messageElement.scrollIntoView({ behavior: 'smooth' });
 | 
						|
          this.fetchPreviousMessages();
 | 
						|
        } else {
 | 
						|
          this.scrollToBottom();
 | 
						|
        }
 | 
						|
      });
 | 
						|
      this.makeMessagesRead();
 | 
						|
    },
 | 
						|
    addScrollListener() {
 | 
						|
      this.conversationPanel = this.$el.querySelector('.conversation-panel');
 | 
						|
      this.setScrollParams();
 | 
						|
      this.conversationPanel.addEventListener('scroll', this.handleScroll);
 | 
						|
      this.$nextTick(() => this.scrollToBottom());
 | 
						|
      this.isLoadingPrevious = false;
 | 
						|
    },
 | 
						|
    removeScrollListener() {
 | 
						|
      this.conversationPanel.removeEventListener('scroll', this.handleScroll);
 | 
						|
    },
 | 
						|
    scrollToBottom() {
 | 
						|
      this.isProgrammaticScroll = true;
 | 
						|
      let relevantMessages = [];
 | 
						|
 | 
						|
      // label suggestions are not part of the messages list
 | 
						|
      // so we need to handle them separately
 | 
						|
      let labelSuggestions =
 | 
						|
        this.conversationPanel.querySelector('.label-suggestion');
 | 
						|
 | 
						|
      // if there are unread messages, scroll to the first unread message
 | 
						|
      if (this.unreadMessageCount > 0) {
 | 
						|
        // capturing only the unread messages
 | 
						|
        relevantMessages =
 | 
						|
          this.conversationPanel.querySelectorAll('.message--unread');
 | 
						|
      } else if (labelSuggestions) {
 | 
						|
        // when scrolling to the bottom, the label suggestions is below the last message
 | 
						|
        // so we scroll there if there are no unread messages
 | 
						|
        // Unread messages always take the highest priority
 | 
						|
        relevantMessages = [labelSuggestions];
 | 
						|
      } else {
 | 
						|
        // if there are no unread messages or label suggestion, scroll to the last message
 | 
						|
        // capturing last message from the messages list
 | 
						|
        relevantMessages = Array.from(
 | 
						|
          this.conversationPanel.querySelectorAll('.message--read')
 | 
						|
        ).slice(-1);
 | 
						|
      }
 | 
						|
 | 
						|
      this.conversationPanel.scrollTop = calculateScrollTop(
 | 
						|
        this.conversationPanel.scrollHeight,
 | 
						|
        this.$el.scrollHeight,
 | 
						|
        relevantMessages
 | 
						|
      );
 | 
						|
    },
 | 
						|
    setScrollParams() {
 | 
						|
      this.heightBeforeLoad = this.conversationPanel.scrollHeight;
 | 
						|
      this.scrollTopBeforeLoad = this.conversationPanel.scrollTop;
 | 
						|
    },
 | 
						|
 | 
						|
    async fetchPreviousMessages(scrollTop = 0) {
 | 
						|
      this.setScrollParams();
 | 
						|
      const shouldLoadMoreMessages =
 | 
						|
        this.currentChat.dataFetched === true &&
 | 
						|
        !this.listLoadingStatus &&
 | 
						|
        !this.isLoadingPrevious;
 | 
						|
 | 
						|
      if (
 | 
						|
        scrollTop < 100 &&
 | 
						|
        !this.isLoadingPrevious &&
 | 
						|
        shouldLoadMoreMessages
 | 
						|
      ) {
 | 
						|
        this.isLoadingPrevious = true;
 | 
						|
        try {
 | 
						|
          await this.$store.dispatch('fetchPreviousMessages', {
 | 
						|
            conversationId: this.currentChat.id,
 | 
						|
            before: this.currentChat.messages[0].id,
 | 
						|
          });
 | 
						|
          const heightDifference =
 | 
						|
            this.conversationPanel.scrollHeight - this.heightBeforeLoad;
 | 
						|
          this.conversationPanel.scrollTop =
 | 
						|
            this.scrollTopBeforeLoad + heightDifference;
 | 
						|
          this.setScrollParams();
 | 
						|
        } catch (error) {
 | 
						|
          // Ignore Error
 | 
						|
        } finally {
 | 
						|
          this.isLoadingPrevious = false;
 | 
						|
        }
 | 
						|
      }
 | 
						|
    },
 | 
						|
 | 
						|
    handleScroll(e) {
 | 
						|
      if (this.isProgrammaticScroll) {
 | 
						|
        // Reset the flag
 | 
						|
        this.isProgrammaticScroll = false;
 | 
						|
        this.hasUserScrolled = false;
 | 
						|
      } else {
 | 
						|
        this.hasUserScrolled = true;
 | 
						|
      }
 | 
						|
      emitter.emit(BUS_EVENTS.ON_MESSAGE_LIST_SCROLL);
 | 
						|
      this.fetchPreviousMessages(e.target.scrollTop);
 | 
						|
    },
 | 
						|
 | 
						|
    makeMessagesRead() {
 | 
						|
      this.$store.dispatch('markMessagesRead', { id: this.currentChat.id });
 | 
						|
    },
 | 
						|
  },
 | 
						|
};
 | 
						|
</script>
 | 
						|
 | 
						|
<template>
 | 
						|
  <div class="flex flex-col justify-between flex-grow h-full min-w-0 m-0">
 | 
						|
    <Banner
 | 
						|
      v-if="!currentChat.can_reply"
 | 
						|
      color-scheme="alert"
 | 
						|
      class="mx-2 mt-2 overflow-hidden rounded-lg"
 | 
						|
      :banner-message="replyWindowBannerMessage"
 | 
						|
      :href-link="replyWindowLink"
 | 
						|
      :href-link-text="replyWindowLinkText"
 | 
						|
    />
 | 
						|
    <Banner
 | 
						|
      v-else-if="hasDuplicateInstagramInbox"
 | 
						|
      color-scheme="alert"
 | 
						|
      class="mx-2 mt-2 overflow-hidden rounded-lg"
 | 
						|
      :banner-message="$t('CONVERSATION.OLD_INSTAGRAM_INBOX_REPLY_BANNER')"
 | 
						|
    />
 | 
						|
    <MessageList
 | 
						|
      ref="conversationPanelRef"
 | 
						|
      class="conversation-panel flex-shrink flex-grow basis-px flex flex-col overflow-y-auto relative h-full m-0 pb-4"
 | 
						|
      :current-user-id="currentUserId"
 | 
						|
      :first-unread-id="unReadMessages[0]?.id"
 | 
						|
      :is-an-email-channel="isAnEmailChannel"
 | 
						|
      :inbox-supports-reply-to="inboxSupportsReplyTo"
 | 
						|
      :messages="getMessages"
 | 
						|
    >
 | 
						|
      <template #beforeAll>
 | 
						|
        <transition name="slide-up">
 | 
						|
          <!-- eslint-disable-next-line vue/require-toggle-inside-transition -->
 | 
						|
          <li
 | 
						|
            class="min-h-[4rem] flex flex-shrink-0 flex-grow-0 items-center flex-auto justify-center max-w-full mt-0 mr-0 mb-1 ml-0 relative first:mt-auto last:mb-0"
 | 
						|
          >
 | 
						|
            <Spinner v-if="shouldShowSpinner" class="text-n-brand" />
 | 
						|
          </li>
 | 
						|
        </transition>
 | 
						|
      </template>
 | 
						|
      <template #unreadBadge>
 | 
						|
        <li
 | 
						|
          v-show="unreadMessageCount != 0"
 | 
						|
          class="list-none flex justify-center items-center"
 | 
						|
        >
 | 
						|
          <span
 | 
						|
            class="shadow-lg rounded-full bg-n-brand text-white text-xs font-medium my-2.5 mx-auto px-2.5 py-1.5"
 | 
						|
          >
 | 
						|
            {{ unreadMessageLabel }}
 | 
						|
          </span>
 | 
						|
        </li>
 | 
						|
      </template>
 | 
						|
      <template #after>
 | 
						|
        <ConversationLabelSuggestion
 | 
						|
          v-if="shouldShowLabelSuggestions"
 | 
						|
          :suggested-labels="labelSuggestions"
 | 
						|
          :chat-labels="currentChat.labels"
 | 
						|
          :conversation-id="currentChat.id"
 | 
						|
        />
 | 
						|
      </template>
 | 
						|
    </MessageList>
 | 
						|
    <div
 | 
						|
      class="flex relative flex-col"
 | 
						|
      :class="{
 | 
						|
        'modal-mask': isPopOutReplyBox,
 | 
						|
        'bg-n-background': !isPopOutReplyBox,
 | 
						|
      }"
 | 
						|
    >
 | 
						|
      <div
 | 
						|
        v-if="isAnyoneTyping"
 | 
						|
        class="absolute flex items-center w-full h-0 -top-7"
 | 
						|
      >
 | 
						|
        <div
 | 
						|
          class="flex py-2 pr-4 pl-5 shadow-md rounded-full bg-white dark:bg-n-solid-3 text-n-slate-11 text-xs font-semibold my-2.5 mx-auto"
 | 
						|
        >
 | 
						|
          {{ typingUserNames }}
 | 
						|
          <img
 | 
						|
            class="w-6 ltr:ml-2 rtl:mr-2"
 | 
						|
            src="assets/images/typing.gif"
 | 
						|
            alt="Someone is typing"
 | 
						|
          />
 | 
						|
        </div>
 | 
						|
      </div>
 | 
						|
      <ReplyBox
 | 
						|
        :pop-out-reply-box="isPopOutReplyBox"
 | 
						|
        @update:pop-out-reply-box="isPopOutReplyBox = $event"
 | 
						|
      />
 | 
						|
    </div>
 | 
						|
  </div>
 | 
						|
</template>
 | 
						|
 | 
						|
<style scoped lang="scss">
 | 
						|
.modal-mask {
 | 
						|
  @apply fixed;
 | 
						|
 | 
						|
  &::v-deep {
 | 
						|
    .ProseMirror-woot-style {
 | 
						|
      @apply max-h-[25rem];
 | 
						|
    }
 | 
						|
 | 
						|
    .reply-box {
 | 
						|
      @apply border border-n-weak max-w-[75rem] w-[70%];
 | 
						|
 | 
						|
      &.is-private {
 | 
						|
        @apply dark:border-n-amber-3/30 border-n-amber-12/5;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    .reply-box .reply-box__top {
 | 
						|
      @apply relative min-h-[27.5rem];
 | 
						|
    }
 | 
						|
 | 
						|
    .reply-box__top .input {
 | 
						|
      @apply min-h-[27.5rem];
 | 
						|
    }
 | 
						|
 | 
						|
    .emoji-dialog {
 | 
						|
      @apply absolute ltr:left-auto rtl:right-auto bottom-1;
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
</style>
 |