index.vue 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342
  1. <script setup>
  2. import { workbenchApi } from '@/api/voice/workbench';
  3. import RecordCardItem from './components/RecordCardItem';
  4. import CallView from '@/components/CallView';
  5. const queryParams = ref({
  6. pageNum: 1,
  7. pageSize: 10,
  8. category: 0,
  9. phone: ''
  10. });
  11. const { proxy } = getCurrentInstance();
  12. const tabCallRecordList = ref([]);
  13. const tabCurrentActive = ref(null);
  14. const callDetails = ref({});
  15. const isTransitionVoiceStatus = ref(false);
  16. const total = ref(0);
  17. const loading = ref(false);
  18. const tabEnum = ['通话呼入', '通话呼出'];
  19. const disabled = computed(() => loading.value || total.value == tabCallRecordList.value.length);
  20. const getTimeOfDayGreeting = () => {
  21. const date = new Date();
  22. const hour = date.getHours();
  23. if (hour >= 0 && hour < 12) {
  24. return "上午好";
  25. } else if (hour >= 12 && hour < 18) {
  26. return "下午好";
  27. } else {
  28. return "晚上好";
  29. }
  30. }
  31. // 切换tabs
  32. const handleChangeTab = (index) => {
  33. if (isTransitionVoiceStatus.value) {
  34. return proxy.$modal.msgWarning("当前语音正在转换中,请稍后");
  35. }
  36. if ( queryParams.value.category != index ) {
  37. queryParams.value.pageNum = 1;
  38. queryParams.value.category = index;
  39. queryParams.value.phone = '';
  40. tabCurrentActive.value = null;
  41. tabCallRecordList.value = [];
  42. initTabsData();
  43. getTimeOfDayGreeting();
  44. }
  45. }
  46. // 选中通话记录
  47. const hanldeTabItem = async (id) => {
  48. if ( !isTransitionVoiceStatus.value ) {
  49. const typeEnum = { 0: '白名单', 1: 'AI客服', 2: '传统服务' };
  50. const { data } = await workbenchApi.getCallRecordDetails(id);
  51. callDetails.value = {
  52. ...data,
  53. typeText: data.category == 1 ? '传统服务' : typeEnum[data.type],
  54. };
  55. tabCurrentActive.value = id;
  56. } else {
  57. proxy.$modal.msgWarning("当前语音正在转换中,请稍后");
  58. }
  59. }
  60. // 搜索
  61. const onSearch = ( type ) => {
  62. if (isTransitionVoiceStatus.value) {
  63. return proxy.$modal.msgWarning("当前语音正在转换中,请稍后");
  64. }
  65. if ( type === 'refresh' ) {
  66. queryParams.value.phone = '';
  67. }
  68. queryParams.value.pageNum = 1;
  69. tabCallRecordList.value = [];
  70. tabCurrentActive.value = null;
  71. initTabsData();
  72. }
  73. // 语音转化完成
  74. const handleVoiceParsed = ({ parsedVoiceContent }) => {
  75. callDetails.value.parsedVoiceContent = parsedVoiceContent;
  76. }
  77. const initTabsData = async () => {
  78. loading.value = true;
  79. const { rows, total: t } = await workbenchApi.getCallRecordList(queryParams.value);
  80. tabCallRecordList.value = [...tabCallRecordList.value, ...rows];
  81. total.value = t;
  82. loading.value = false;
  83. }
  84. onMounted(async () => {
  85. initTabsData();
  86. });
  87. const loadMoreData = () => {
  88. queryParams.value.pageNum += 1;
  89. initTabsData();
  90. }
  91. </script>
  92. <template>
  93. <div class="workbench-viewport space-x-[16px]">
  94. <div class="record-section">
  95. <ul class="tabs-nav space-x-[48px]">
  96. <li v-for="item, index in tabEnum" :class="['tabs-nav-item', { active: queryParams.category === index }]"
  97. :key="item" @click="handleChangeTab(index)">{{ item }}</li>
  98. </ul>
  99. <div class="tabs-content">
  100. <div class="search-inp-wrapper">
  101. <div class="search-inp">
  102. <input type="text" class="inp" placeholder="请输入电话号码" v-model.trim="queryParams.phone">
  103. <div class="btn" @click="onSearch">搜索</div>
  104. </div>
  105. <el-tooltip
  106. effect="dark"
  107. content="刷新"
  108. placement="top"
  109. >
  110. <el-icon style="cursor: pointer;" @click="onSearch('refresh')"><Refresh /></el-icon>
  111. </el-tooltip>
  112. </div>
  113. <div class="search-result-wrapper">
  114. <el-scrollbar height="100%">
  115. <div
  116. class="search-result-inner space-y-[8px]"
  117. v-infinite-scroll="loadMoreData"
  118. :infinite-scroll-disabled="disabled"
  119. v-show="tabCallRecordList.length"
  120. >
  121. <RecordCardItem
  122. v-for="item, index in tabCallRecordList"
  123. :data="item"
  124. :index="index"
  125. :active="tabCurrentActive === item.id"
  126. :key="item.id"
  127. @on-click="hanldeTabItem(item.id)"
  128. ></RecordCardItem>
  129. <div class="flex justify-center text-[#999] text-[12px]">
  130. <p class="pb-[6px]" v-if="loading">Loading...</p>
  131. </div>
  132. </div>
  133. <div class="flex items-center justify-center pt-[100px]" v-show="!tabCallRecordList.length">
  134. <span class="text-[#999] text-[14px]">暂无数据</span>
  135. </div>
  136. </el-scrollbar>
  137. </div>
  138. </div>
  139. </div>
  140. <div class="details-section">
  141. <div class="empty-wrapper" v-show="!callDetails.id || tabCurrentActive === null">
  142. <img src="@/assets/images/workbench/img-empty.png" alt="">
  143. <p class="empty-text">
  144. <span>Hi, {{ getTimeOfDayGreeting() }}~</span>
  145. <span>欢迎登录智能语音客服</span>
  146. </p>
  147. </div>
  148. <div class="details-wrapper" v-show="callDetails.id && tabCurrentActive !== null">
  149. <h4 class="title">通话详情</h4>
  150. <el-scrollbar class="details-scrollbar">
  151. <CallView :data="callDetails" noInit @on-end="handleVoiceParsed" v-model="isTransitionVoiceStatus"></CallView>
  152. </el-scrollbar>
  153. </div>
  154. </div>
  155. </div>
  156. </template>
  157. <style lang="scss" scoped>
  158. $primaryColor: #165DFF;
  159. .workbench-viewport {
  160. display: flex;
  161. height: 100%;
  162. background: #eceff6;
  163. .record-section {
  164. flex-shrink: 0;
  165. width: 292px;
  166. height: 100%;
  167. border-radius: 8px;
  168. background: linear-gradient(180deg, #FFF 0%, #FFF 100%);
  169. .tabs-nav {
  170. display: flex;
  171. align-items: center;
  172. justify-content: center;
  173. height: 46px;
  174. padding-top: 15px;
  175. border-bottom: 1px solid #E5E6EB;
  176. font-size: 14px;
  177. line-height: 20px;
  178. color: #4E5969;
  179. .tabs-nav-item {
  180. position: relative;
  181. cursor: pointer;
  182. &.active {
  183. color: $primaryColor;
  184. font-weight: bold;
  185. &::after {
  186. position: absolute;
  187. left: 0;
  188. bottom: -6px;
  189. content: ' ';
  190. display: block;
  191. width: 100%;
  192. height: 2px;
  193. background: $primaryColor;
  194. }
  195. }
  196. }
  197. }
  198. .tabs-content {
  199. height: calc(100% - 46px);
  200. .search-inp-wrapper {
  201. display: flex;
  202. align-items: center;
  203. justify-content: space-between;
  204. padding: 12px 22px;
  205. .search-inp {
  206. display: flex;
  207. align-items: center;
  208. height: 34px;
  209. padding: 2px;
  210. border-radius: 8px;
  211. background: #F2F4F7;
  212. .inp {
  213. width: 100%;
  214. padding: 0 10px;
  215. background: transparent;
  216. outline: none;
  217. font-size: 13px;
  218. color: #1D2129;
  219. }
  220. .btn {
  221. flex-shrink: 0;
  222. width: 52px;
  223. height: 30px;
  224. border-radius: 8px;
  225. background: #165DFF;
  226. color: #FFF;
  227. font-size: 13px;
  228. line-height: 30px;
  229. text-align: center;
  230. cursor: pointer;
  231. }
  232. }
  233. }
  234. .search-result-wrapper {
  235. height: calc(100% - 58px);
  236. .search-result-inner {
  237. padding: 0 22px;
  238. }
  239. }
  240. }
  241. }
  242. .details-section {
  243. width: 100%;
  244. min-width: 700px;
  245. height: 100%;
  246. border-radius: 8px;
  247. overflow: hidden;
  248. padding: 20px;
  249. background: #fff;
  250. .details-scrollbar {
  251. height: calc(100% - 50px);
  252. }
  253. .empty-wrapper {
  254. display: flex;
  255. align-items: center;
  256. justify-content: center;
  257. flex-flow: column;
  258. width: 100%;
  259. height: 100%;
  260. .empty-text {
  261. span {
  262. display: block;
  263. text-align: center;
  264. font-weight: bold;
  265. font-size: 24px;
  266. line-height: 32px;
  267. &:nth-child(1) {
  268. color: #165DFF;
  269. }
  270. &:nth-child(2) {
  271. color: #1D2129;
  272. }
  273. }
  274. }
  275. }
  276. .details-wrapper {
  277. height: 100%;
  278. .title {
  279. margin-bottom: 24px;
  280. color: #1D2129;
  281. font-size: 18px;
  282. font-weight: bold;
  283. line-height: 26px;
  284. }
  285. }
  286. }
  287. }
  288. .dialog-footer {
  289. display: flex;
  290. justify-content: center;
  291. }
  292. :deep(.el-textarea__inner) {
  293. background: #f2f4f7;
  294. }
  295. </style>