mirror of
https://github.com/lingble/chatwoot.git
synced 2025-11-01 19:48:08 +00:00
# Pull Request Template ## Description This PR includes the following enhancements to the conversation card context menu: 1. **Added "Open in New Tab" and "Copy Conversation Link" options.** * "Open in New Tab" allows users to quickly open a conversation in a separate browser tab. * "Copy Conversation Link" copies the conversation URL to the clipboard for easy sharing. 2. **Enabled the context menu in Previous Conversations card** with support for these two options. Fixes https://linear.app/chatwoot/issue/CW-4722/cannot-open-previous-conversations-in-a-new-tab ## Type of change - [x] Bug fix (non-breaking change which fixes an issue) ## How Has This Been Tested? ### Loom video https://www.loom.com/share/37b45d23c6804db292568d093b645ac0?sid=c3105971-f938-41bd-9f52-0f00d419d1b3 ## Checklist: - [x] My code follows the style guidelines of this project - [x] I have performed a self-review of my code - [x] I have commented on my code, particularly in hard-to-understand areas - [ ] I have made corresponding changes to the documentation - [x] My changes generate no new warnings - [ ] I have added tests that prove my fix is effective or that my feature works - [x] New and existing unit tests pass locally with my changes - [ ] Any dependent changes have been merged and published in downstream modules --------- Co-authored-by: Pranav <pranav@chatwoot.com>
439 lines
12 KiB
Vue
439 lines
12 KiB
Vue
<script>
|
|
import { mapGetters } from 'vuex';
|
|
import { getLastMessage } from 'dashboard/helper/conversationHelper';
|
|
import Thumbnail from '../Thumbnail.vue';
|
|
import MessagePreview from './MessagePreview.vue';
|
|
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 TimeAgo from 'dashboard/components/ui/TimeAgo.vue';
|
|
import CardLabels from './conversationCardComponents/CardLabels.vue';
|
|
import PriorityMark from './PriorityMark.vue';
|
|
import SLACardLabel from './components/SLACardLabel.vue';
|
|
import ContextMenu from 'dashboard/components/ui/ContextMenu.vue';
|
|
|
|
export default {
|
|
components: {
|
|
CardLabels,
|
|
InboxName,
|
|
Thumbnail,
|
|
ConversationContextMenu,
|
|
TimeAgo,
|
|
MessagePreview,
|
|
PriorityMark,
|
|
SLACardLabel,
|
|
ContextMenu,
|
|
},
|
|
mixins: [inboxMixin],
|
|
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,
|
|
},
|
|
allowedContextMenuOptions: {
|
|
type: Array,
|
|
default: () => [],
|
|
},
|
|
},
|
|
emits: [
|
|
'contextMenuToggle',
|
|
'assignAgent',
|
|
'assignLabel',
|
|
'assignTeam',
|
|
'markAsUnread',
|
|
'markAsRead',
|
|
'assignPriority',
|
|
'updateConversationStatus',
|
|
'deleteConversation',
|
|
],
|
|
data() {
|
|
return {
|
|
hovered: false,
|
|
showContextMenu: false,
|
|
contextMenu: {
|
|
x: null,
|
|
y: null,
|
|
},
|
|
};
|
|
},
|
|
computed: {
|
|
...mapGetters({
|
|
currentChat: 'getSelectedChat',
|
|
inboxesList: 'inboxes/getInboxes',
|
|
activeInbox: 'getSelectedInbox',
|
|
accountId: 'getCurrentAccountId',
|
|
}),
|
|
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 getLastMessage(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;
|
|
},
|
|
conversationPath() {
|
|
const { activeInbox, chat } = this;
|
|
return frontendURL(
|
|
conversationUrl({
|
|
accountId: this.accountId,
|
|
activeInbox,
|
|
id: chat.id,
|
|
label: this.activeLabel,
|
|
teamId: this.teamId,
|
|
foldersId: this.foldersId,
|
|
conversationType: this.conversationType,
|
|
})
|
|
);
|
|
},
|
|
},
|
|
methods: {
|
|
onCardClick(e) {
|
|
const path = this.conversationPath;
|
|
if (!path) return;
|
|
|
|
// Handle Ctrl/Cmd + Click for new tab
|
|
if (e.metaKey || e.ctrlKey) {
|
|
e.preventDefault();
|
|
window.open(
|
|
`${window.chatwootConfig.hostURL}${path}`,
|
|
'_blank',
|
|
'noopener,noreferrer'
|
|
);
|
|
return;
|
|
}
|
|
|
|
// Skip if already active
|
|
if (this.isActiveChat) return;
|
|
|
|
router.push({ path });
|
|
},
|
|
onThumbnailHover() {
|
|
this.hovered = !this.hideThumbnail;
|
|
},
|
|
onThumbnailLeave() {
|
|
this.hovered = false;
|
|
},
|
|
onSelectConversation(checked) {
|
|
const action = checked ? 'selectConversation' : 'deSelectConversation';
|
|
this.$emit(action, this.chat.id, this.inbox.id);
|
|
},
|
|
openContextMenu(e) {
|
|
if (!this.enableContextMenu) return;
|
|
e.preventDefault();
|
|
this.$emit('contextMenuToggle', true);
|
|
this.contextMenu.x = e.pageX || e.clientX;
|
|
this.contextMenu.y = e.pageY || e.clientY;
|
|
this.showContextMenu = true;
|
|
},
|
|
closeContextMenu() {
|
|
this.$emit('contextMenuToggle', false);
|
|
this.showContextMenu = false;
|
|
this.contextMenu.x = null;
|
|
this.contextMenu.y = null;
|
|
},
|
|
onUpdateConversation(status, snoozedUntil) {
|
|
this.closeContextMenu();
|
|
this.$emit(
|
|
'updateConversationStatus',
|
|
this.chat.id,
|
|
status,
|
|
snoozedUntil
|
|
);
|
|
},
|
|
async onAssignAgent(agent) {
|
|
this.$emit('assignAgent', agent, [this.chat.id]);
|
|
this.closeContextMenu();
|
|
},
|
|
async onAssignLabel(label) {
|
|
this.$emit('assignLabel', [label.title], [this.chat.id]);
|
|
this.closeContextMenu();
|
|
},
|
|
async onAssignTeam(team) {
|
|
this.$emit('assignTeam', team, this.chat.id);
|
|
this.closeContextMenu();
|
|
},
|
|
async markAsUnread() {
|
|
this.$emit('markAsUnread', this.chat.id);
|
|
this.closeContextMenu();
|
|
},
|
|
async markAsRead() {
|
|
this.$emit('markAsRead', this.chat.id);
|
|
this.closeContextMenu();
|
|
},
|
|
async assignPriority(priority) {
|
|
this.$emit('assignPriority', priority, this.chat.id);
|
|
this.closeContextMenu();
|
|
},
|
|
async deleteConversation() {
|
|
this.$emit('deleteConversation', this.chat.id);
|
|
this.closeContextMenu();
|
|
},
|
|
},
|
|
};
|
|
</script>
|
|
|
|
<template>
|
|
<div
|
|
class="relative flex items-start flex-grow-0 flex-shrink-0 w-auto max-w-full px-3 py-0 border-t-0 border-b-0 border-l-2 border-r-0 border-transparent border-solid cursor-pointer conversation hover:bg-n-alpha-1 dark:hover:bg-n-alpha-3 group"
|
|
:class="{
|
|
'active animate-card-select bg-n-alpha-1 dark:bg-n-alpha-3 border-n-weak':
|
|
isActiveChat,
|
|
'unread-chat': hasUnread,
|
|
'has-inbox-name': showInboxName,
|
|
'conversation-selected': selected,
|
|
}"
|
|
@click="onCardClick"
|
|
@contextmenu="openContextMenu($event)"
|
|
>
|
|
<div
|
|
class="relative"
|
|
@mouseenter="onThumbnailHover"
|
|
@mouseleave="onThumbnailLeave"
|
|
>
|
|
<label
|
|
v-if="hovered || selected"
|
|
class="checkbox-wrapper absolute inset-0 z-20 backdrop-blur-[2px]"
|
|
@click.stop
|
|
>
|
|
<input
|
|
:value="selected"
|
|
:checked="selected"
|
|
class="checkbox"
|
|
type="checkbox"
|
|
@change="onSelectConversation($event.target.checked)"
|
|
/>
|
|
</label>
|
|
<Thumbnail
|
|
v-if="!hideThumbnail"
|
|
:src="currentContact.thumbnail"
|
|
:badge="inboxBadge"
|
|
:username="currentContact.name"
|
|
:status="currentContact.availability_status"
|
|
size="32px"
|
|
/>
|
|
</div>
|
|
<div
|
|
class="px-0 py-3 border-b group-hover:border-transparent flex-1 border-n-slate-3 w-[calc(100%-40px)]"
|
|
>
|
|
<div class="flex justify-between conversation-card--meta">
|
|
<InboxName 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-n-slate-11 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-n-slate-11" />
|
|
{{ assignee.name }}
|
|
</span>
|
|
<PriorityMark :priority="chat.priority" />
|
|
</div>
|
|
</div>
|
|
<h4
|
|
class="conversation--user text-sm my-0 mx-2 capitalize pt-0.5 text-ellipsis overflow-hidden whitespace-nowrap w-[calc(100%-70px)] text-n-slate-12"
|
|
:class="hasUnread ? 'font-semibold' : 'font-medium'"
|
|
>
|
|
{{ currentContact.name }}
|
|
</h4>
|
|
<MessagePreview
|
|
v-if="lastMessageInChat"
|
|
:message="lastMessageInChat"
|
|
class="conversation--message my-0 mx-2 leading-6 h-6 max-w-[96%] w-[16.875rem] text-sm"
|
|
:class="hasUnread ? 'font-medium text-n-slate-12' : 'text-n-slate-11'"
|
|
/>
|
|
<p
|
|
v-else
|
|
class="conversation--message text-n-slate-11 text-sm my-0 mx-2 leading-6 h-6 max-w-[96%] w-[16.875rem] overflow-hidden text-ellipsis whitespace-nowrap"
|
|
:class="hasUnread ? 'font-medium text-n-slate-12' : 'text-n-slate-11'"
|
|
>
|
|
<fluent-icon
|
|
size="16"
|
|
class="-mt-0.5 align-middle inline-block text-n-slate-10"
|
|
icon="info"
|
|
/>
|
|
<span>
|
|
{{ $t(`CHAT_LIST.NO_MESSAGES`) }}
|
|
</span>
|
|
</p>
|
|
<div class="absolute flex flex-col mt-4 ltr:right-4 rtl:left-4 top-4">
|
|
<span class="ml-auto font-normal leading-4 text-xxs">
|
|
<TimeAgo
|
|
: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 ltr:ml-auto rtl:mr-auto mt-1 min-w-[1rem] px-1 py-0 text-center text-white bg-n-teal-9"
|
|
>
|
|
{{ unreadCount > 9 ? '9+' : unreadCount }}
|
|
</span>
|
|
</div>
|
|
<CardLabels :conversation-labels="chat.labels" class="mt-0.5 mx-2 mb-0">
|
|
<template v-if="hasSlaPolicyId" #before>
|
|
<SLACardLabel :chat="chat" class="ltr:mr-1 rtl:ml-1" />
|
|
</template>
|
|
</CardLabels>
|
|
</div>
|
|
<ContextMenu
|
|
v-if="showContextMenu"
|
|
:x="contextMenu.x"
|
|
:y="contextMenu.y"
|
|
@close="closeContextMenu"
|
|
>
|
|
<ConversationContextMenu
|
|
:status="chat.status"
|
|
:inbox-id="inbox.id"
|
|
:priority="chat.priority"
|
|
:chat-id="chat.id"
|
|
:has-unread-messages="hasUnread"
|
|
:conversation-url="conversationPath"
|
|
:allowed-options="allowedContextMenuOptions"
|
|
@update-conversation="onUpdateConversation"
|
|
@assign-agent="onAssignAgent"
|
|
@assign-label="onAssignLabel"
|
|
@assign-team="onAssignTeam"
|
|
@mark-as-unread="markAsUnread"
|
|
@mark-as-read="markAsRead"
|
|
@assign-priority="assignPriority"
|
|
@delete-conversation="deleteConversation"
|
|
@close="closeContextMenu"
|
|
/>
|
|
</ContextMenu>
|
|
</div>
|
|
</template>
|
|
|
|
<style lang="scss" scoped>
|
|
.conversation {
|
|
&.unread-chat {
|
|
.unread {
|
|
@apply block;
|
|
}
|
|
}
|
|
|
|
&.compact {
|
|
@apply pl-0;
|
|
|
|
.conversation-card--meta {
|
|
@apply ltr:pr-4 rtl:pl-4;
|
|
}
|
|
|
|
.conversation--details {
|
|
@apply rounded-sm ml-0 pl-5 pr-2;
|
|
}
|
|
}
|
|
|
|
&::v-deep .user-thumbnail-box {
|
|
@apply mt-4;
|
|
}
|
|
|
|
&.conversation-selected {
|
|
@apply bg-n-slate-2 dark:bg-n-slate-3;
|
|
}
|
|
|
|
&.has-inbox-name {
|
|
&::v-deep .user-thumbnail-box {
|
|
@apply mt-8;
|
|
}
|
|
|
|
.checkbox-wrapper {
|
|
@apply mt-8;
|
|
}
|
|
|
|
.conversation--meta {
|
|
@apply mt-4;
|
|
}
|
|
}
|
|
|
|
.checkbox-wrapper {
|
|
@apply flex items-center justify-center rounded-full cursor-pointer mt-4;
|
|
|
|
input[type='checkbox'] {
|
|
@apply m-0 cursor-pointer;
|
|
}
|
|
}
|
|
}
|
|
</style>
|