mirror of
				https://github.com/lingble/chatwoot.git
				synced 2025-11-03 20:48:07 +00:00 
			
		
		
		
	* fix: Resolve and go next keyboard shortcuts doesn't work * refactor: use buildHotKeys instead of hasPressedCommandPlusAltAndEKey * feat: install tinykeys * refactor: use tinykeys * test: update buildKeyEvents * fix: remove stray import * feat: handle action list globally * feat: allow configuring `allowOnFocusedInput` * chore: Navigate chat list item * chore: Navigate dashboard * feat: Navigate editor top panel * feat: Toggle file upload * chore: More keyboard shortcuts * chore: Update mention selection mixin * chore: Phone input * chore: Clean up * chore: Clean up * chore: Dropdown and editor * chore: Enter key to send and clean up * chore: Rename mixin * chore: Review fixes * chore: Removed unused shortcut from modal * fix: Specs --------- Co-authored-by: iamsivin <iamsivin@gmail.com> Co-authored-by: Sivin Varghese <64252451+iamsivin@users.noreply.github.com>
		
			
				
	
	
		
			400 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
			
		
		
	
	
			400 lines
		
	
	
		
			11 KiB
		
	
	
	
		
			Vue
		
	
	
	
	
	
<template>
 | 
						|
  <div
 | 
						|
    class="relative flex items-start flex-grow-0 flex-shrink-0 w-auto max-w-full px-4 py-0 border-t-0 border-b-0 border-l-2 border-r-0 border-transparent border-solid cursor-pointer conversation hover:bg-slate-25 dark:hover:bg-slate-800 group"
 | 
						|
    :class="{
 | 
						|
      'active animate-card-select bg-slate-25 dark:bg-slate-800 border-woot-500':
 | 
						|
        isActiveChat,
 | 
						|
      'unread-chat': hasUnread,
 | 
						|
      'has-inbox-name': showInboxName,
 | 
						|
      'conversation-selected': selected,
 | 
						|
    }"
 | 
						|
    @mouseenter="onCardHover"
 | 
						|
    @mouseleave="onCardLeave"
 | 
						|
    @click="onCardClick"
 | 
						|
    @contextmenu="openContextMenu($event)"
 | 
						|
  >
 | 
						|
    <label v-if="hovered || selected" class="checkbox-wrapper" @click.stop>
 | 
						|
      <input
 | 
						|
        :value="selected"
 | 
						|
        :checked="selected"
 | 
						|
        class="checkbox"
 | 
						|
        type="checkbox"
 | 
						|
        @change="onSelectConversation($event.target.checked)"
 | 
						|
      />
 | 
						|
    </label>
 | 
						|
    <thumbnail
 | 
						|
      v-if="bulkActionCheck"
 | 
						|
      :src="currentContact.thumbnail"
 | 
						|
      :badge="inboxBadge"
 | 
						|
      :username="currentContact.name"
 | 
						|
      :status="currentContact.availability_status"
 | 
						|
      size="40px"
 | 
						|
    />
 | 
						|
    <div
 | 
						|
      class="px-0 py-3 border-b group-hover:border-transparent flex-1 border-slate-50 dark:border-slate-800/75 w-[calc(100%-40px)]"
 | 
						|
    >
 | 
						|
      <div class="flex justify-between">
 | 
						|
        <inbox-name v-if="showInboxName" :inbox="inbox" />
 | 
						|
        <div class="flex gap-2 ml-2 rtl:mr-2 rtl:ml-0">
 | 
						|
          <span
 | 
						|
            v-if="showAssignee && assignee.name"
 | 
						|
            class="text-slate-500 dark:text-slate-400 text-xs font-medium leading-3 py-0.5 px-0 inline-flex text-ellipsis overflow-hidden whitespace-nowrap"
 | 
						|
          >
 | 
						|
            <fluent-icon
 | 
						|
              icon="person"
 | 
						|
              size="12"
 | 
						|
              class="text-slate-500 dark:text-slate-400"
 | 
						|
            />
 | 
						|
            {{ assignee.name }}
 | 
						|
          </span>
 | 
						|
          <priority-mark :priority="chat.priority" />
 | 
						|
        </div>
 | 
						|
      </div>
 | 
						|
      <h4
 | 
						|
        class="conversation--user text-sm my-0 mx-2 capitalize pt-0.5 text-ellipsis font-medium overflow-hidden whitespace-nowrap w-[calc(100%-70px)] text-slate-900 dark:text-slate-100"
 | 
						|
      >
 | 
						|
        {{ currentContact.name }}
 | 
						|
      </h4>
 | 
						|
      <message-preview
 | 
						|
        v-if="lastMessageInChat"
 | 
						|
        :message="lastMessageInChat"
 | 
						|
        class="conversation--message my-0 mx-2 leading-6 h-6 max-w-[96%] w-[16.875rem] text-sm text-slate-700 dark:text-slate-200"
 | 
						|
      />
 | 
						|
      <p
 | 
						|
        v-else
 | 
						|
        class="conversation--message text-slate-700 dark:text-slate-200 text-sm my-0 mx-2 leading-6 h-6 max-w-[96%] w-[16.875rem] overflow-hidden text-ellipsis whitespace-nowrap"
 | 
						|
      >
 | 
						|
        <fluent-icon
 | 
						|
          size="16"
 | 
						|
          class="-mt-0.5 align-middle inline-block text-slate-600 dark:text-slate-300"
 | 
						|
          icon="info"
 | 
						|
        />
 | 
						|
        <span>
 | 
						|
          {{ $t(`CHAT_LIST.NO_MESSAGES`) }}
 | 
						|
        </span>
 | 
						|
      </p>
 | 
						|
      <div class="absolute flex flex-col conversation--meta right-4 top-4">
 | 
						|
        <span class="ml-auto font-normal leading-4 text-black-600 text-xxs">
 | 
						|
          <time-ago
 | 
						|
            :last-activity-timestamp="chat.timestamp"
 | 
						|
            :created-at-timestamp="chat.created_at"
 | 
						|
          />
 | 
						|
        </span>
 | 
						|
        <span
 | 
						|
          class="unread shadow-lg rounded-full hidden text-xxs font-semibold h-4 leading-4 ml-auto mt-1 min-w-[1rem] px-1 py-0 text-center text-white bg-green-400"
 | 
						|
        >
 | 
						|
          {{ unreadCount > 9 ? '9+' : unreadCount }}
 | 
						|
        </span>
 | 
						|
      </div>
 | 
						|
      <card-labels :conversation-id="chat.id" class="mt-0.5 mx-2 mb-0">
 | 
						|
        <template v-if="hasSlaPolicyId" #before>
 | 
						|
          <SLA-card-label :chat="chat" class="ltr:mr-1 rtl:ml-1" />
 | 
						|
        </template>
 | 
						|
      </card-labels>
 | 
						|
    </div>
 | 
						|
    <woot-context-menu
 | 
						|
      v-if="showContextMenu"
 | 
						|
      ref="menu"
 | 
						|
      :x="contextMenu.x"
 | 
						|
      :y="contextMenu.y"
 | 
						|
      @close="closeContextMenu"
 | 
						|
    >
 | 
						|
      <conversation-context-menu
 | 
						|
        :status="chat.status"
 | 
						|
        :inbox-id="inbox.id"
 | 
						|
        :priority="chat.priority"
 | 
						|
        :has-unread-messages="hasUnread"
 | 
						|
        @update-conversation="onUpdateConversation"
 | 
						|
        @assign-agent="onAssignAgent"
 | 
						|
        @assign-label="onAssignLabel"
 | 
						|
        @assign-team="onAssignTeam"
 | 
						|
        @mark-as-unread="markAsUnread"
 | 
						|
        @assign-priority="assignPriority"
 | 
						|
      />
 | 
						|
    </woot-context-menu>
 | 
						|
  </div>
 | 
						|
</template>
 | 
						|
<script>
 | 
						|
import { mapGetters } from 'vuex';
 | 
						|
import Thumbnail from '../Thumbnail.vue';
 | 
						|
import MessagePreview from './MessagePreview.vue';
 | 
						|
import conversationMixin from '../../../mixins/conversations';
 | 
						|
import timeMixin from '../../../mixins/time';
 | 
						|
import router from '../../../routes';
 | 
						|
import { frontendURL, conversationUrl } from '../../../helper/URLHelper';
 | 
						|
import InboxName from '../InboxName.vue';
 | 
						|
import inboxMixin from 'shared/mixins/inboxMixin';
 | 
						|
import ConversationContextMenu from './contextMenu/Index.vue';
 | 
						|
import alertMixin from 'shared/mixins/alertMixin';
 | 
						|
import TimeAgo from 'dashboard/components/ui/TimeAgo.vue';
 | 
						|
import CardLabels from './conversationCardComponents/CardLabels.vue';
 | 
						|
import PriorityMark from './PriorityMark.vue';
 | 
						|
import SLACardLabel from './components/SLACardLabel.vue';
 | 
						|
 | 
						|
export default {
 | 
						|
  components: {
 | 
						|
    CardLabels,
 | 
						|
    InboxName,
 | 
						|
    Thumbnail,
 | 
						|
    ConversationContextMenu,
 | 
						|
    TimeAgo,
 | 
						|
    MessagePreview,
 | 
						|
    PriorityMark,
 | 
						|
    SLACardLabel,
 | 
						|
  },
 | 
						|
 | 
						|
  mixins: [inboxMixin, timeMixin, conversationMixin, alertMixin],
 | 
						|
  props: {
 | 
						|
    activeLabel: {
 | 
						|
      type: String,
 | 
						|
      default: '',
 | 
						|
    },
 | 
						|
    chat: {
 | 
						|
      type: Object,
 | 
						|
      default: () => {},
 | 
						|
    },
 | 
						|
    hideInboxName: {
 | 
						|
      type: Boolean,
 | 
						|
      default: false,
 | 
						|
    },
 | 
						|
    hideThumbnail: {
 | 
						|
      type: Boolean,
 | 
						|
      default: false,
 | 
						|
    },
 | 
						|
    teamId: {
 | 
						|
      type: [String, Number],
 | 
						|
      default: 0,
 | 
						|
    },
 | 
						|
    foldersId: {
 | 
						|
      type: [String, Number],
 | 
						|
      default: 0,
 | 
						|
    },
 | 
						|
    showAssignee: {
 | 
						|
      type: Boolean,
 | 
						|
      default: false,
 | 
						|
    },
 | 
						|
    conversationType: {
 | 
						|
      type: String,
 | 
						|
      default: '',
 | 
						|
    },
 | 
						|
    selected: {
 | 
						|
      type: Boolean,
 | 
						|
      default: false,
 | 
						|
    },
 | 
						|
    enableContextMenu: {
 | 
						|
      type: Boolean,
 | 
						|
      default: false,
 | 
						|
    },
 | 
						|
  },
 | 
						|
  data() {
 | 
						|
    return {
 | 
						|
      hovered: false,
 | 
						|
      showContextMenu: false,
 | 
						|
      contextMenu: {
 | 
						|
        x: null,
 | 
						|
        y: null,
 | 
						|
      },
 | 
						|
    };
 | 
						|
  },
 | 
						|
  computed: {
 | 
						|
    ...mapGetters({
 | 
						|
      currentChat: 'getSelectedChat',
 | 
						|
      inboxesList: 'inboxes/getInboxes',
 | 
						|
      activeInbox: 'getSelectedInbox',
 | 
						|
      currentUser: 'getCurrentUser',
 | 
						|
      accountId: 'getCurrentAccountId',
 | 
						|
    }),
 | 
						|
    bulkActionCheck() {
 | 
						|
      return !this.hideThumbnail && !this.hovered && !this.selected;
 | 
						|
    },
 | 
						|
    chatMetadata() {
 | 
						|
      return this.chat.meta || {};
 | 
						|
    },
 | 
						|
 | 
						|
    assignee() {
 | 
						|
      return this.chatMetadata.assignee || {};
 | 
						|
    },
 | 
						|
 | 
						|
    currentContact() {
 | 
						|
      return this.$store.getters['contacts/getContact'](
 | 
						|
        this.chatMetadata.sender.id
 | 
						|
      );
 | 
						|
    },
 | 
						|
 | 
						|
    isActiveChat() {
 | 
						|
      return this.currentChat.id === this.chat.id;
 | 
						|
    },
 | 
						|
 | 
						|
    unreadCount() {
 | 
						|
      return this.chat.unread_count;
 | 
						|
    },
 | 
						|
 | 
						|
    hasUnread() {
 | 
						|
      return this.unreadCount > 0;
 | 
						|
    },
 | 
						|
 | 
						|
    isInboxNameVisible() {
 | 
						|
      return !this.activeInbox;
 | 
						|
    },
 | 
						|
 | 
						|
    lastMessageInChat() {
 | 
						|
      return this.lastMessage(this.chat);
 | 
						|
    },
 | 
						|
 | 
						|
    inbox() {
 | 
						|
      const { inbox_id: inboxId } = this.chat;
 | 
						|
      const stateInbox = this.$store.getters['inboxes/getInbox'](inboxId);
 | 
						|
      return stateInbox;
 | 
						|
    },
 | 
						|
 | 
						|
    showInboxName() {
 | 
						|
      return (
 | 
						|
        !this.hideInboxName &&
 | 
						|
        this.isInboxNameVisible &&
 | 
						|
        this.inboxesList.length > 1
 | 
						|
      );
 | 
						|
    },
 | 
						|
    inboxName() {
 | 
						|
      const stateInbox = this.inbox;
 | 
						|
      return stateInbox.name || '';
 | 
						|
    },
 | 
						|
    hasSlaPolicyId() {
 | 
						|
      return this.chat?.sla_policy_id;
 | 
						|
    },
 | 
						|
  },
 | 
						|
  methods: {
 | 
						|
    onCardClick(e) {
 | 
						|
      const { activeInbox, chat } = this;
 | 
						|
      const path = frontendURL(
 | 
						|
        conversationUrl({
 | 
						|
          accountId: this.accountId,
 | 
						|
          activeInbox,
 | 
						|
          id: chat.id,
 | 
						|
          label: this.activeLabel,
 | 
						|
          teamId: this.teamId,
 | 
						|
          foldersId: this.foldersId,
 | 
						|
          conversationType: this.conversationType,
 | 
						|
        })
 | 
						|
      );
 | 
						|
 | 
						|
      if (e.metaKey || e.ctrlKey) {
 | 
						|
        window.open(
 | 
						|
          window.chatwootConfig.hostURL + path,
 | 
						|
          '_blank',
 | 
						|
          'noopener noreferrer nofollow'
 | 
						|
        );
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      if (this.isActiveChat) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
 | 
						|
      router.push({ path });
 | 
						|
    },
 | 
						|
    onCardHover() {
 | 
						|
      this.hovered = !this.hideThumbnail;
 | 
						|
    },
 | 
						|
    onCardLeave() {
 | 
						|
      this.hovered = false;
 | 
						|
    },
 | 
						|
    onSelectConversation(checked) {
 | 
						|
      const action = checked ? 'select-conversation' : 'de-select-conversation';
 | 
						|
      this.$emit(action, this.chat.id, this.inbox.id);
 | 
						|
    },
 | 
						|
    openContextMenu(e) {
 | 
						|
      if (!this.enableContextMenu) return;
 | 
						|
      e.preventDefault();
 | 
						|
      this.$emit('context-menu-toggle', true);
 | 
						|
      this.contextMenu.x = e.pageX || e.clientX;
 | 
						|
      this.contextMenu.y = e.pageY || e.clientY;
 | 
						|
      this.showContextMenu = true;
 | 
						|
    },
 | 
						|
    closeContextMenu() {
 | 
						|
      this.$emit('context-menu-toggle', false);
 | 
						|
      this.showContextMenu = false;
 | 
						|
      this.contextMenu.x = null;
 | 
						|
      this.contextMenu.y = null;
 | 
						|
    },
 | 
						|
    onUpdateConversation(status, snoozedUntil) {
 | 
						|
      this.closeContextMenu();
 | 
						|
      this.$emit(
 | 
						|
        'update-conversation-status',
 | 
						|
        this.chat.id,
 | 
						|
        status,
 | 
						|
        snoozedUntil
 | 
						|
      );
 | 
						|
    },
 | 
						|
    async onAssignAgent(agent) {
 | 
						|
      this.$emit('assign-agent', agent, [this.chat.id]);
 | 
						|
      this.closeContextMenu();
 | 
						|
    },
 | 
						|
    async onAssignLabel(label) {
 | 
						|
      this.$emit('assign-label', [label.title], [this.chat.id]);
 | 
						|
      this.closeContextMenu();
 | 
						|
    },
 | 
						|
    async onAssignTeam(team) {
 | 
						|
      this.$emit('assign-team', team, this.chat.id);
 | 
						|
      this.closeContextMenu();
 | 
						|
    },
 | 
						|
    async markAsUnread() {
 | 
						|
      this.$emit('mark-as-unread', this.chat.id);
 | 
						|
      this.closeContextMenu();
 | 
						|
    },
 | 
						|
    async assignPriority(priority) {
 | 
						|
      this.$emit('assign-priority', priority, this.chat.id);
 | 
						|
      this.closeContextMenu();
 | 
						|
    },
 | 
						|
  },
 | 
						|
};
 | 
						|
</script>
 | 
						|
<style lang="scss" scoped>
 | 
						|
.conversation {
 | 
						|
  &.unread-chat {
 | 
						|
    .unread {
 | 
						|
      @apply block;
 | 
						|
    }
 | 
						|
    .conversation--message {
 | 
						|
      @apply font-semibold;
 | 
						|
    }
 | 
						|
    .conversation--user {
 | 
						|
      @apply font-semibold;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  &.compact {
 | 
						|
    @apply pl-0;
 | 
						|
    .conversation--details {
 | 
						|
      @apply rounded-sm ml-0 pl-5 pr-2;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  &::v-deep .user-thumbnail-box {
 | 
						|
    @apply mt-4;
 | 
						|
  }
 | 
						|
 | 
						|
  &.conversation-selected {
 | 
						|
    @apply bg-slate-25 dark:bg-slate-800;
 | 
						|
  }
 | 
						|
 | 
						|
  &.has-inbox-name {
 | 
						|
    &::v-deep .user-thumbnail-box {
 | 
						|
      @apply mt-8;
 | 
						|
    }
 | 
						|
    .checkbox-wrapper {
 | 
						|
      @apply mt-8;
 | 
						|
    }
 | 
						|
    .conversation--meta {
 | 
						|
      @apply mt-4;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  .checkbox-wrapper {
 | 
						|
    @apply h-10 w-10 flex items-center justify-center rounded-full cursor-pointer mt-4 hover:bg-woot-100 dark:hover:bg-woot-800;
 | 
						|
 | 
						|
    input[type='checkbox'] {
 | 
						|
      @apply m-0 cursor-pointer;
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
</style>
 |