467 lines
12 KiB
Vue
467 lines
12 KiB
Vue
<template>
|
|
<div>
|
|
<div
|
|
v-show="showSmallCallPopup"
|
|
class="ml-2 flex cursor-pointer select-none items-center justify-between gap-1 rounded-full bg-surface-gray-7 px-2 py-[7px] text-base text-ink-gray-2"
|
|
@click="toggleCallPopup"
|
|
>
|
|
<div
|
|
class="flex justify-center items-center size-5 rounded-full bg-surface-gray-6 shrink-0 mr-1"
|
|
>
|
|
<Avatar
|
|
v-if="contact?.image"
|
|
:image="contact.image"
|
|
:label="contact.full_name"
|
|
class="!size-5"
|
|
/>
|
|
<AvatarIcon v-else class="size-3" />
|
|
</div>
|
|
<span>{{ contact?.full_name ?? phoneNumber }}</span>
|
|
<span>·</span>
|
|
<div v-if="callStatus == 'In progress'">
|
|
{{ counterUp?.updatedTime }}
|
|
</div>
|
|
<div
|
|
v-else-if="callStatus == 'Call ended' || callStatus == 'No answer'"
|
|
class="blink"
|
|
:class="{
|
|
'text-red-700':
|
|
callStatus == 'Call ended' || callStatus == 'No answer',
|
|
}"
|
|
>
|
|
<span>{{ __(callStatus) }}</span>
|
|
<span v-if="callStatus == 'Call ended'">
|
|
<span> · </span>
|
|
<span>{{ callDuration }}</span>
|
|
</span>
|
|
</div>
|
|
<div v-else>{{ __(callStatus) }}</div>
|
|
</div>
|
|
<div
|
|
v-show="showCallPopup"
|
|
class="fixed z-20 w-[280px] min-h-44 flex gap-2 flex-col rounded-lg bg-surface-gray-7 p-4 pt-2.5 text-ink-gray-2 shadow-2xl"
|
|
:style="style"
|
|
@click.stop
|
|
>
|
|
<div
|
|
ref="callPopupHeader"
|
|
class="header flex items-center justify-between gap-1 text-base cursor-move select-none"
|
|
>
|
|
<div class="flex gap-2 items-center truncate">
|
|
<div v-if="showNote || showTask" class="flex items-center gap-3">
|
|
<Avatar
|
|
v-if="contact?.image"
|
|
:image="contact.image"
|
|
:label="contact.full_name"
|
|
class="!size-7"
|
|
/>
|
|
<div
|
|
v-else
|
|
class="flex justify-center items-center size-7 rounded-full bg-surface-gray-6"
|
|
>
|
|
<AvatarIcon class="size-3" />
|
|
</div>
|
|
<div class="flex flex-col gap-1 text-base leading-4">
|
|
<div class="font-medium">
|
|
{{ contact?.full_name ?? phoneNumber }}
|
|
</div>
|
|
<div class="text-ink-gray-6">
|
|
<div v-if="callStatus == 'In progress'">
|
|
<span>{{ phoneNumber }}</span>
|
|
<span> · </span>
|
|
<span>{{ counterUp?.updatedTime }}</span>
|
|
</div>
|
|
<div
|
|
v-else-if="
|
|
callStatus == 'Call ended' || callStatus == 'No answer'
|
|
"
|
|
class="blink"
|
|
:class="{
|
|
'text-red-700':
|
|
callStatus == 'Call ended' || callStatus == 'No answer',
|
|
}"
|
|
>
|
|
<span>{{ __(callStatus) }}</span>
|
|
<span v-if="callStatus == 'Call ended'">
|
|
<span> · </span>
|
|
<span>{{ callDuration }}</span>
|
|
</span>
|
|
</div>
|
|
<div v-else>{{ __(callStatus) }}</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div v-else>
|
|
<div v-if="callStatus == 'In progress'">
|
|
{{ counterUp?.updatedTime }}
|
|
</div>
|
|
<div
|
|
v-else-if="
|
|
callStatus == 'Call ended' || callStatus == 'No answer'
|
|
"
|
|
class="blink"
|
|
:class="{
|
|
'text-red-700':
|
|
callStatus == 'Call ended' || callStatus == 'No answer',
|
|
}"
|
|
>
|
|
<span>{{ __(callStatus) }}</span>
|
|
<span v-if="callStatus == 'Call ended'">
|
|
<span> · </span>
|
|
<span>{{ callDuration }}</span>
|
|
</span>
|
|
</div>
|
|
<div v-else>{{ __(callStatus) }}</div>
|
|
</div>
|
|
</div>
|
|
|
|
<Button
|
|
@click="toggleCallPopup"
|
|
class="bg-surface-gray-7 text-ink-white hover:bg-surface-gray-6 shrink-0"
|
|
size="md"
|
|
>
|
|
<template #icon>
|
|
<MinimizeIcon class="h-4 w-4 cursor-pointer" />
|
|
</template>
|
|
</Button>
|
|
</div>
|
|
<div class="body flex-1">
|
|
<div v-if="showNote">
|
|
<TextEditor
|
|
variant="ghost"
|
|
ref="content"
|
|
editor-class="prose-sm h-[290px] text-ink-white overflow-auto mt-1"
|
|
:bubbleMenu="true"
|
|
:content="note"
|
|
@change="(val) => (note = val)"
|
|
:placeholder="__('Take a note...')"
|
|
/>
|
|
</div>
|
|
<TaskPanel ref="taskRef" v-else-if="showTask" :task="task" />
|
|
<div v-else class="flex items-center gap-3">
|
|
<Avatar
|
|
v-if="contact?.image"
|
|
:image="contact.image"
|
|
:label="contact.full_name"
|
|
class="!size-8"
|
|
/>
|
|
<div
|
|
v-else
|
|
class="flex justify-center items-center size-8 rounded-full bg-surface-gray-6"
|
|
>
|
|
<AvatarIcon class="size-4" />
|
|
</div>
|
|
<div v-if="contact?.full_name" class="flex flex-col gap-1">
|
|
<div class="text-lg font-medium leading-5">
|
|
{{ contact.full_name }}
|
|
</div>
|
|
<div class="text-base text-ink-gray-6 leading-4">
|
|
{{ phoneNumber }}
|
|
</div>
|
|
</div>
|
|
<div v-else class="text-lg font-medium leading-5">
|
|
{{ phoneNumber }}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="footer flex justify-between gap-2">
|
|
<div class="flex gap-2">
|
|
<Button
|
|
class="bg-surface-gray-6 text-ink-white hover:bg-surface-gray-5"
|
|
size="md"
|
|
@click="showNoteWindow"
|
|
>
|
|
<template #icon>
|
|
<NoteIcon class="w-4 h-4" />
|
|
</template>
|
|
</Button>
|
|
<Button
|
|
class="bg-surface-gray-6 text-ink-white hover:bg-surface-gray-5"
|
|
size="md"
|
|
@click="showTaskWindow"
|
|
>
|
|
<template #icon>
|
|
<TaskIcon class="w-4 h-4" />
|
|
</template>
|
|
</Button>
|
|
</div>
|
|
<div class="flex gap-2">
|
|
<Button
|
|
v-if="callStatus == 'Call ended' || callStatus == 'No answer'"
|
|
@click="closeCallPopup"
|
|
class="bg-surface-gray-6 text-ink-white hover:bg-surface-gray-5"
|
|
:label="__('Close')"
|
|
size="md"
|
|
/>
|
|
<Button
|
|
v-if="(note && note != '<p></p>') || task.title"
|
|
@click="save"
|
|
class="bg-surface-white !text-ink-gray-9 hover:!bg-surface-gray-3"
|
|
variant="solid"
|
|
:label="__('Save')"
|
|
size="md"
|
|
/>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<CountUpTimer ref="counterUp" />
|
|
</div>
|
|
</template>
|
|
<script setup>
|
|
import AvatarIcon from '@/components/Icons/AvatarIcon.vue'
|
|
import MinimizeIcon from '@/components/Icons/MinimizeIcon.vue'
|
|
import NoteIcon from '@/components/Icons/NoteIcon.vue'
|
|
import TaskIcon from '@/components/Icons/TaskIcon.vue'
|
|
import TaskPanel from './TaskPanel.vue'
|
|
import CountUpTimer from '@/components/CountUpTimer.vue'
|
|
import { TextEditor, Avatar, Button, call } from 'frappe-ui'
|
|
import { globalStore } from '@/stores/global'
|
|
import { contactsStore } from '@/stores/contacts'
|
|
import { useDraggable, useWindowSize } from '@vueuse/core'
|
|
import { ref, computed, onBeforeUnmount } from 'vue'
|
|
|
|
const { getContact, getLeadContact } = contactsStore()
|
|
const { $socket } = globalStore()
|
|
|
|
const callPopupHeader = ref(null)
|
|
const showCallPopup = ref(false)
|
|
const showSmallCallPopup = ref(false)
|
|
|
|
function toggleCallPopup() {
|
|
showCallPopup.value = !showCallPopup.value
|
|
if (showSmallCallPopup.value == undefined) {
|
|
showSmallCallPopup = !showSmallCallPopup
|
|
} else {
|
|
showSmallCallPopup.value = !showSmallCallPopup.value
|
|
}
|
|
}
|
|
|
|
const { width, height } = useWindowSize()
|
|
|
|
let { style } = useDraggable(callPopupHeader, {
|
|
initialValue: { x: width.value - 350, y: height.value - 250 },
|
|
preventDefault: true,
|
|
})
|
|
|
|
const callStatus = ref('')
|
|
const phoneNumber = ref('')
|
|
const callData = ref(null)
|
|
const counterUp = ref(null)
|
|
|
|
const contact = computed(() => {
|
|
if (!phoneNumber.value) {
|
|
return {
|
|
full_name: '',
|
|
image: '',
|
|
}
|
|
}
|
|
let _contact = getContact(phoneNumber.value)
|
|
if (!_contact) {
|
|
_contact = getLeadContact(phoneNumber.value)
|
|
}
|
|
return _contact
|
|
})
|
|
|
|
const note = ref('')
|
|
|
|
const showNote = ref(false)
|
|
|
|
function showNoteWindow() {
|
|
showNote.value = !showNote.value
|
|
if (!showTask.value) {
|
|
updateWindowHeight(showNote.value)
|
|
}
|
|
if (showNote.value) {
|
|
showTask.value = false
|
|
}
|
|
}
|
|
|
|
function createNote() {
|
|
call('crm.integrations.api.create_and_add_note_to_call_log', {
|
|
call_sid: callData.value.CallSid,
|
|
content: note.value,
|
|
})
|
|
note.value = ''
|
|
}
|
|
|
|
const task = ref({
|
|
title: '',
|
|
description: '',
|
|
assigned_to: '',
|
|
due_date: '',
|
|
status: 'Backlog',
|
|
priority: 'Low',
|
|
})
|
|
|
|
const showTask = ref(false)
|
|
|
|
function showTaskWindow() {
|
|
showTask.value = !showTask.value
|
|
if (!showNote.value) {
|
|
updateWindowHeight(showTask.value)
|
|
}
|
|
if (showTask.value) {
|
|
showNote.value = false
|
|
}
|
|
}
|
|
|
|
function createTask() {
|
|
call('crm.integrations.api.create_and_add_task_to_call_log', {
|
|
call_sid: callData.value.CallSid,
|
|
task: task.value,
|
|
})
|
|
task.value = {
|
|
title: '',
|
|
description: '',
|
|
assigned_to: '',
|
|
due_date: '',
|
|
status: 'Backlog',
|
|
priority: 'Low',
|
|
}
|
|
}
|
|
|
|
function updateWindowHeight(condition) {
|
|
let callPopup = callPopupHeader.value.parentElement
|
|
let top = parseInt(callPopup.style.top)
|
|
let updatedTop = 0
|
|
|
|
updatedTop = condition ? top - 224 : top + 224
|
|
|
|
if (updatedTop < 0) {
|
|
updatedTop = 10
|
|
}
|
|
|
|
callPopup.style.top = updatedTop + 'px'
|
|
}
|
|
|
|
function makeOutgoingCall(number) {
|
|
phoneNumber.value = number
|
|
callStatus.value = 'Calling...'
|
|
showCallPopup.value = true
|
|
showSmallCallPopup.value = false
|
|
|
|
call('crm.integrations.exotel.handler.make_a_call', {
|
|
to_number: phoneNumber.value,
|
|
})
|
|
}
|
|
|
|
function setup() {
|
|
$socket.on('exotel_call', (data) => {
|
|
callData.value = data
|
|
console.log(data)
|
|
|
|
callStatus.value = updateStatus(data)
|
|
|
|
if (!showCallPopup.value && !showSmallCallPopup.value) {
|
|
showCallPopup.value = true
|
|
}
|
|
})
|
|
}
|
|
|
|
onBeforeUnmount(() => {
|
|
$socket.off('exotel_call')
|
|
})
|
|
|
|
function closeCallPopup() {
|
|
showCallPopup.value = false
|
|
showSmallCallPopup.value = false
|
|
note.value = ''
|
|
task.value = {
|
|
title: '',
|
|
description: '',
|
|
assigned_to: '',
|
|
due_date: '',
|
|
status: 'Backlog',
|
|
priority: 'Low',
|
|
}
|
|
}
|
|
|
|
function save() {
|
|
if (note.value) createNote()
|
|
if (task.value.title) createTask()
|
|
}
|
|
|
|
const callDuration = ref('00:00')
|
|
|
|
function updateStatus(data) {
|
|
// outgoing call
|
|
if (
|
|
data.EventType == 'answered' &&
|
|
data.Direction == 'outbound-api' &&
|
|
data.Status == 'in-progress' &&
|
|
data['Legs[0][Status]'] == 'in-progress' &&
|
|
data['Legs[1][Status]'] == ''
|
|
) {
|
|
return 'Ringing...'
|
|
} else if (
|
|
data.EventType == 'answered' &&
|
|
data.Direction == 'outbound-api' &&
|
|
data.Status == 'in-progress' &&
|
|
data['Legs[1][Status]'] == 'in-progress'
|
|
) {
|
|
counterUp.value.start()
|
|
return 'In progress'
|
|
} else if (
|
|
data.EventType == 'terminal' &&
|
|
data.Direction == 'outbound-api' &&
|
|
data.Status == 'no-answer' &&
|
|
data['Legs[1][Status]'] == 'no-answer'
|
|
) {
|
|
counterUp.value.stop()
|
|
return 'No answer'
|
|
} else if (
|
|
data.EventType == 'terminal' &&
|
|
data.Direction == 'outbound-api' &&
|
|
data.Status == 'completed'
|
|
) {
|
|
counterUp.value.stop()
|
|
callDuration.value = getTime(
|
|
data['Legs[0][OnCallDuration]'] || data.DialCallDuration,
|
|
)
|
|
return 'Call ended'
|
|
}
|
|
|
|
// incoming call
|
|
if (
|
|
data.EventType == 'Dial' &&
|
|
data.Direction == 'incoming' &&
|
|
data.Status == 'busy'
|
|
) {
|
|
phoneNumber.value = data.From || data.CallFrom
|
|
return 'Incoming call'
|
|
} else if (
|
|
data.Direction == 'incoming' &&
|
|
(data.EventType == 'Terminal' || data.CallType == 'completed') &&
|
|
(data.Status == 'free' || data.DialCallStatus == 'completed')
|
|
) {
|
|
callDuration.value = counterUp.value.getTime(
|
|
data['Legs[0][OnCallDuration]'] || data.DialCallDuration,
|
|
)
|
|
return 'Call ended'
|
|
}
|
|
}
|
|
|
|
defineExpose({ makeOutgoingCall, setup })
|
|
</script>
|
|
<style scoped>
|
|
@keyframes blink {
|
|
0% {
|
|
opacity: 1;
|
|
}
|
|
50% {
|
|
opacity: 0;
|
|
}
|
|
100% {
|
|
opacity: 1;
|
|
}
|
|
}
|
|
|
|
.blink {
|
|
animation: blink 1s ease-in-out 6;
|
|
}
|
|
|
|
:deep(.ProseMirror) {
|
|
caret-color: var(--ink-white);
|
|
}
|
|
</style>
|