462 lines
12 KiB
Vue
462 lines
12 KiB
Vue
<script lang="ts" setup>
|
|
import { computed, ref, onMounted, watch, reactive, onBeforeMount, getCurrentInstance } from 'vue'
|
|
import { onBeforeRouteUpdate } from 'vue-router'
|
|
import { useDialogueStore, useTalkStore } from '@/store'
|
|
import { NDropdown, NIcon, NInput, NPopover, NTabs, NTab, NCard } from 'naive-ui'
|
|
import { Search, Plus } from '@icon-park/vue-next'
|
|
import TalkItem from './TalkItem.vue'
|
|
import Skeleton from './Skeleton.vue'
|
|
import { ServeClearTalkUnreadNum } from '@/api/chat'
|
|
import GroupLaunch from '@/components/group/GroupLaunch.vue'
|
|
import { getCacheIndexName } from '@/utils/talk'
|
|
import { ISession } from '@/types/chat'
|
|
import { useSessionMenu } from '@/hooks'
|
|
import customModal from '@/components/common/customModal.vue'
|
|
import xSearchForm from '@/components/x-naive-ui/x-search-form/index.vue'
|
|
import flTree from '@/components/flnlayout/tree/flnindex.vue'
|
|
import { processError, processSuccess } from '@/utils/helper/message.js'
|
|
|
|
const currentInstance = getCurrentInstance()
|
|
const { $request } = currentInstance?.appContext.config.globalProperties
|
|
|
|
const {
|
|
dropdown,
|
|
onContextMenuTalkHandle,
|
|
onContextMenu: onContextMenuTalk,
|
|
onCloseContextMenu,
|
|
onToTopTalk
|
|
} = useSessionMenu()
|
|
|
|
const dialogueStore = useDialogueStore()
|
|
const talkStore = useTalkStore()
|
|
const isShowGroup = ref(false)
|
|
const searchKeyword = ref('')
|
|
const topItems = computed((): ISession[] => talkStore.topItems)
|
|
const unreadNum = computed(() => talkStore.talkUnreadNum)
|
|
|
|
const state = reactive({
|
|
isShowAddressBookModal: false, // 是否显示通讯录模态框
|
|
customModalStyle: {
|
|
width: '1288px',
|
|
height: '846px',
|
|
backgroundColor: '#F9F9FD'
|
|
}, //自定义模态框样式
|
|
searchConfig: [
|
|
{
|
|
label: '姓名',
|
|
key: 'name',
|
|
type: 'input',
|
|
valueType: 'string'
|
|
}
|
|
], // 搜索配置
|
|
|
|
treeData: [],
|
|
expandedKeys: [],
|
|
clickKey: '',
|
|
treeRefreshCount: 0,
|
|
treeSelectData: {}
|
|
})
|
|
|
|
const items = computed((): ISession[] => {
|
|
if (searchKeyword.value.length === 0) {
|
|
return talkStore.talkItems
|
|
}
|
|
|
|
return talkStore.talkItems.filter((item: ISession) => {
|
|
let keyword = item.remark || item.name
|
|
|
|
return keyword.toLowerCase().indexOf(searchKeyword.value.toLowerCase()) != -1
|
|
})
|
|
})
|
|
|
|
watch(
|
|
() => talkStore,
|
|
(newValue, oldValue) => {
|
|
console.log(newValue)
|
|
},
|
|
{ deep: true, immediate: true }
|
|
)
|
|
|
|
// 列表加载状态
|
|
const loadStatus = computed(() => talkStore.loadStatus)
|
|
|
|
// 当前会话索引
|
|
const indexName = computed(() => dialogueStore.index_name)
|
|
|
|
// 切换会话
|
|
const onTabTalk = (item: ISession, follow = false) => {
|
|
if (item.index_name === indexName.value) return
|
|
|
|
searchKeyword.value = ''
|
|
|
|
// 更新编辑信息
|
|
dialogueStore.setDialogue(item)
|
|
|
|
// 清空消息未读数
|
|
if (item.unread_num > 0) {
|
|
ServeClearTalkUnreadNum({
|
|
talk_type: item.talk_type,
|
|
receiver_id: item.receiver_id
|
|
}).then(() => {
|
|
talkStore.updateItem({
|
|
index_name: item.index_name,
|
|
unread_num: 0
|
|
})
|
|
})
|
|
}
|
|
|
|
// 设置滚动条跟随
|
|
if (follow) {
|
|
const el = document.getElementById('talk-session-list')
|
|
if (el) {
|
|
let index = talkStore.findTalkIndex(item.index_name)
|
|
|
|
el.scrollTo({
|
|
top: index * 66 + index * 5,
|
|
behavior: 'smooth'
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
const onReload = () => {
|
|
talkStore.loadTalkList()
|
|
}
|
|
|
|
// 初始化加载
|
|
const onInitialize = () => {
|
|
let index_name = getCacheIndexName()
|
|
|
|
index_name && onTabTalk(talkStore.findItem(index_name), true)
|
|
}
|
|
|
|
// 路由更新事件
|
|
onBeforeRouteUpdate(onInitialize)
|
|
|
|
onBeforeMount(() => {
|
|
getTreeData()
|
|
})
|
|
|
|
onMounted(() => {
|
|
onInitialize()
|
|
})
|
|
|
|
// 点击显示通讯录模态框
|
|
const showAddressBookModal = () => {
|
|
state.isShowAddressBookModal = true
|
|
}
|
|
const handleTreeClick = ({ selectedKey, tree }) => {
|
|
state.clickKey = tree.key
|
|
state.treeSelectData = tree
|
|
}
|
|
const calcTreeData = (data) => {
|
|
for (let item of data) {
|
|
item.key = item.ID
|
|
item.label = item.name
|
|
item.title = item.name
|
|
if (item.sons) {
|
|
item.children = item.sons
|
|
calcTreeData(item.children)
|
|
}
|
|
delete item.ID
|
|
delete item.name
|
|
delete item.sons
|
|
}
|
|
}
|
|
// 获取组织树数据
|
|
const getTreeData = () => {
|
|
let url = '/department/v2/tree/filter'
|
|
let params = {}
|
|
$request.HTTP.components.postDataByParams(url, params).then(
|
|
(res) => {
|
|
if (res.status === 0 && Array.isArray(res.data.nodes)) {
|
|
let data = res.data.nodes
|
|
calcTreeData(data)
|
|
state.treeData = data
|
|
// // 获取最近点击的部门
|
|
// let localSelect = Local.get("posimanage_treeSelectData");
|
|
// if (localSelect && JSON.stringify(localSelect) !== "{}") {
|
|
// state.treeSelectData = localSelect;
|
|
// state.expandedKeys = localSelect.pathIds;
|
|
// state.clickKey = localSelect.key;
|
|
// } else {
|
|
// if (JSON.stringify(state.treeSelectData) === "{}") {
|
|
// state.treeSelectData = data[0];
|
|
// state.clickKey = data[0].key;
|
|
// }
|
|
// if (
|
|
// state.clickKey === data[0].key &&
|
|
// !state.expandedKeys.includes(data[0].key)
|
|
// ) {
|
|
// state.expandedKeys.push(data[0].key);
|
|
// }
|
|
// if (!state.expandedKeys.includes(state.clickKey)) {
|
|
// state.expandedKeys.push(state.clickKey);
|
|
// }
|
|
// }
|
|
state.treeRefreshCount++
|
|
// state.tableConfig.refreshCount++;
|
|
} else {
|
|
processError(res.msg || '获取失败!')
|
|
}
|
|
},
|
|
() => {
|
|
processError('获取失败!')
|
|
},
|
|
() => {
|
|
processError('获取失败!')
|
|
}
|
|
)
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<!-- 右键菜单 -->
|
|
<n-dropdown
|
|
class="dropdown-menus"
|
|
:show="dropdown.show"
|
|
:x="dropdown.x"
|
|
:y="dropdown.y"
|
|
:options="dropdown.options"
|
|
@select="onContextMenuTalkHandle"
|
|
@clickoutside="onCloseContextMenu"
|
|
/>
|
|
|
|
<section class="el-container is-vertical height100">
|
|
<!-- 工具栏目 -->
|
|
<header class="el-header header-tools">
|
|
<n-input
|
|
placeholder="搜索好友 / 群聊"
|
|
v-model:value.trim="searchKeyword"
|
|
round
|
|
clearable
|
|
style="width: 78%;"
|
|
>
|
|
<template #prefix>
|
|
<n-icon :component="Search" />
|
|
</template>
|
|
</n-input>
|
|
|
|
<n-button circle @click="isShowGroup = true">
|
|
<template #icon>
|
|
<n-icon :component="Plus" />
|
|
</template>
|
|
</n-button>
|
|
<img
|
|
style="width: 19px; height: 20px;"
|
|
src="@/assets/image/chatList/addressBook.png"
|
|
alt=""
|
|
@click="showAddressBookModal"
|
|
/>
|
|
</header>
|
|
|
|
<!-- 置顶栏目 -->
|
|
<header class="el-header header-top" v-show="loadStatus == 3 && topItems.length > 0">
|
|
<n-popover v-for="item in topItems" :key="item.index_name" placement="bottom" trigger="hover">
|
|
<template #trigger>
|
|
<div
|
|
class="top-item pointer"
|
|
@click="onTabTalk(item, true)"
|
|
:class="{
|
|
active: item.index_name == indexName
|
|
}"
|
|
>
|
|
<im-avatar :src="item.avatar" :size="34" :username="item.name" />
|
|
|
|
<span class="icon-mark robot" v-show="item.is_robot == 1"> 助 </span>
|
|
|
|
<span class="icon-mark group" v-show="item.talk_type == 2 && item.is_robot == 0">
|
|
群
|
|
</span>
|
|
|
|
<span class="text">{{ item.remark || item.name }}</span>
|
|
</div>
|
|
</template>
|
|
<span> {{ item.remark || item.name }} </span>
|
|
</n-popover>
|
|
</header>
|
|
|
|
<!-- 标题栏目 -->
|
|
<header
|
|
v-show="loadStatus == 3 && talkStore.talkItems.length > 0"
|
|
class="el-header header-badge"
|
|
:class="{ shadow: false }"
|
|
>
|
|
<p>会话记录({{ talkStore.talkItems.length }})</p>
|
|
<p>
|
|
<span class="badge unread" v-show="unreadNum">{{ unreadNum }}未读</span>
|
|
</p>
|
|
</header>
|
|
|
|
<main id="talk-session-list" class="el-main me-scrollbar me-scrollbar-thumb">
|
|
<template v-if="loadStatus == 2"><Skeleton /></template>
|
|
<template v-else>
|
|
<TalkItem
|
|
v-for="item in items"
|
|
:key="item.index_name"
|
|
:data="item"
|
|
:avatar="item.avatar"
|
|
:username="item.remark || item.name"
|
|
:active="item.index_name == indexName"
|
|
@tab-talk="onTabTalk"
|
|
@top-talk="onToTopTalk"
|
|
@contextmenu.prevent="onContextMenuTalk($event, item)"
|
|
/>
|
|
</template>
|
|
</main>
|
|
</section>
|
|
|
|
<GroupLaunch v-if="isShowGroup" @close="isShowGroup = false" @on-submit="onReload" />
|
|
|
|
<customModal
|
|
v-model:show="state.isShowAddressBookModal"
|
|
title="通讯录"
|
|
:style="state.customModalStyle"
|
|
:customCloseBtn="true"
|
|
:closable="false"
|
|
>
|
|
<template #content>
|
|
<div class="custom-modal-content">
|
|
<n-card>
|
|
<n-tabs type="line">
|
|
<n-tab name="employeeAddressBook">员工通讯录</n-tab>
|
|
<n-tab name="groupChatList">群聊列表</n-tab>
|
|
</n-tabs>
|
|
<xSearchForm :search-config="state.searchConfig"></xSearchForm>
|
|
<div class="addressBook-content">
|
|
<div class="addressBook-tree">
|
|
<fl-tree
|
|
:data="state.treeData"
|
|
:expandedKeys="state.expandedKeys"
|
|
:refreshCount="state.treeRefreshCount"
|
|
:clickKey="state.clickKey"
|
|
@triggerTreeClick="handleTreeClick"
|
|
></fl-tree>
|
|
</div>
|
|
</div>
|
|
</n-card>
|
|
</div>
|
|
</template>
|
|
</customModal>
|
|
</template>
|
|
|
|
<style lang="less" scoped>
|
|
.header-tools {
|
|
height: 60px;
|
|
flex-shrink: 0;
|
|
display: flex;
|
|
flex-direction: row;
|
|
align-items: center;
|
|
justify-content: space-around;
|
|
padding: 0 8px;
|
|
}
|
|
|
|
.header-badge {
|
|
height: 38px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
padding-left: 10px;
|
|
|
|
&.shadow {
|
|
box-shadow: 0 2px 6px 0 rgb(31 35 41 / 5%);
|
|
}
|
|
|
|
.unread {
|
|
background-color: #ff4d4f;
|
|
color: white;
|
|
cursor: pointer;
|
|
}
|
|
}
|
|
|
|
.header-top {
|
|
padding: 5px 8px;
|
|
padding-right: 0;
|
|
padding-right: 8px;
|
|
-webkit-justify-content: space-between;
|
|
-ms-flex-pack: justify;
|
|
justify-content: space-between;
|
|
grid-gap: 0 14px;
|
|
grid-template-columns: repeat(auto-fill, 32px);
|
|
display: grid;
|
|
box-sizing: border-box;
|
|
|
|
.top-item {
|
|
flex-basis: 46px;
|
|
flex-shrink: 0;
|
|
height: 56px;
|
|
margin: 3px 2px 3px 2px;
|
|
display: flex;
|
|
flex-direction: column;
|
|
justify-content: space-between;
|
|
align-items: center;
|
|
position: relative;
|
|
|
|
.icon-mark {
|
|
position: absolute;
|
|
height: 25px;
|
|
width: 25px;
|
|
font-size: 14px;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: center;
|
|
right: -12px;
|
|
bottom: 15px;
|
|
transform: scale(0.6);
|
|
border-radius: 50%;
|
|
|
|
&.group {
|
|
color: #3370ff;
|
|
background-color: #e1eaff;
|
|
}
|
|
|
|
&.robot {
|
|
color: #dc9b04 !important;
|
|
background-color: #faf1d1 !important;
|
|
}
|
|
}
|
|
&.active {
|
|
.text {
|
|
color: rgb(80 138 254);
|
|
}
|
|
}
|
|
|
|
.text {
|
|
display: inline-block;
|
|
height: 20px;
|
|
font-size: 12px;
|
|
transform: scale(0.9);
|
|
text-align: center;
|
|
line-height: 20px;
|
|
word-break: break-all;
|
|
overflow: hidden;
|
|
}
|
|
}
|
|
}
|
|
|
|
html[theme-mode='dark'] {
|
|
.header-badge {
|
|
&.shadow {
|
|
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.3);
|
|
}
|
|
}
|
|
}
|
|
|
|
.custom-modal-content {
|
|
box-sizing: border-box;
|
|
width: 100%;
|
|
padding: 0 12px;
|
|
.addressBook-content {
|
|
.addressBook-tree {
|
|
width: 328px;
|
|
height: 524px;
|
|
overflow: auto;
|
|
border: 1px solid #efeff5;
|
|
border-radius: 4px;
|
|
padding: 12px 20px;
|
|
box-sizing: border-box;
|
|
}
|
|
}
|
|
}
|
|
</style>
|