267 lines
8.1 KiB
Vue
267 lines
8.1 KiB
Vue
<template>
|
||
<view>
|
||
<topHeader ref="topHeaderRef" @search="search"></topHeader>
|
||
|
||
<!-- 用户痛点观察 -->
|
||
<a-space direction="vertical" style="background-color: #fff; width: 100%; padding: 24px; margin: 24px 0">
|
||
<a-space align="center">
|
||
<span>用户痛点观察 </span>
|
||
<a-popover position="tl">
|
||
<a-button type="primary" class="pop-btn">
|
||
<template #icon>
|
||
<icon-question-circle />
|
||
</template>
|
||
</a-button>
|
||
<template #content>
|
||
<p style="margin: 0">基于用户内容中的情绪分析与表达模式,提取反复出现的负面倾向主题,反映典型使用痛点。</p>
|
||
</template>
|
||
</a-popover>
|
||
</a-space>
|
||
<a-table
|
||
:columns="columns"
|
||
:data="dataList"
|
||
:filter-icon-align-left="alignLeft"
|
||
@change="handleChange"
|
||
:scroll="true"
|
||
:pagination="false"
|
||
>
|
||
<template #rank="{ record }">
|
||
<img v-if="record.rank == 1" :src="topImages[0]" style="width: 25px; height: 17px" />
|
||
<img v-else-if="record.rank == 2" :src="topImages[1]" style="width: 25px; height: 17px" />
|
||
<img v-else-if="record.rank == 3" :src="topImages[2]" style="width: 25px; height: 17px" />
|
||
<span v-else>{{ record.rank }}</span>
|
||
</template>
|
||
<template #keywords="{ record }">
|
||
<a-tag v-for="item in record.keywords" :key="item" style="margin-right: 5px">{{ item }}</a-tag>
|
||
</template>
|
||
<template #frequency="{ record }">
|
||
<a-tag v-if="record.frequency == 0" style="margin-right: 5px; background-color: #ebf7f2; color: #1bae71"
|
||
>低频</a-tag
|
||
>
|
||
<a-tag v-else-if="record.frequency == 1" style="margin-right: 5px; background-color: #fff5de; color: #cc8b00"
|
||
>中频</a-tag
|
||
>
|
||
<a-tag v-else-if="record.frequency == 2" style="margin-right: 5px; background-color: #ffe7e4; color: #c53c27"
|
||
>高频</a-tag
|
||
>
|
||
</template>
|
||
<template #sentiment="{ record }">
|
||
<img
|
||
v-if="record.felling == '2'"
|
||
src="@/assets/img/hottranslation/good.png"
|
||
style="width: 16px; height: 16px"
|
||
/>
|
||
<img
|
||
v-else-if="record.felling == '1'"
|
||
src="@/assets/img/hottranslation/normal.png"
|
||
style="width: 16px; height: 16px"
|
||
/>
|
||
<img
|
||
v-else-if="record.felling == '0'"
|
||
src="@/assets/img/hottranslation/poor.png"
|
||
style="width: 16px; height: 16px"
|
||
/>
|
||
</template>
|
||
|
||
<template #optional="{ record }">
|
||
<a-button type="outline" @click="gotoDetail(record)">详情</a-button>
|
||
</template>
|
||
</a-table>
|
||
</a-space>
|
||
|
||
<a-modal :visible="visible" @ok="handleOk" @cancel="handleCancel" unmountOnClose>
|
||
<template #title>
|
||
<span style="text-align: left; width: 100%">用户痛点观察</span>
|
||
</template>
|
||
<div>
|
||
<a-space direction="vertical" style="font-size: 12px">
|
||
<a-space>
|
||
<span style="width: 80px">痛点</span>
|
||
<span>{{ topicInfo.name }}</span>
|
||
</a-space>
|
||
<a-space>
|
||
<span style="width: 80px; font-size: 12px">关键词</span>
|
||
<a-tag v-for="item in topicInfo.keywords" :key="item" style="margin-right: 5px">{{ item }}</a-tag>
|
||
</a-space>
|
||
<a-space>
|
||
<span style="width: 80px; font-size: 12px">频次</span>
|
||
<a-tag v-if="topicInfo.frequency == 0" style="margin-right: 5px; background-color: #ebf7f2; color: #1bae71"
|
||
>低频</a-tag
|
||
>
|
||
<a-tag
|
||
v-else-if="topicInfo.frequency == 1"
|
||
style="margin-right: 5px; background-color: #fff5de; color: #cc8b00"
|
||
>中频</a-tag
|
||
>
|
||
<a-tag
|
||
v-else-if="topicInfo.frequency == 2"
|
||
style="margin-right: 5px; background-color: #ffe7e4; color: #c53c27"
|
||
>高频</a-tag
|
||
>
|
||
</a-space>
|
||
<a-space>
|
||
<span style="width: 80px; font-size: 12px">代表性发言</span>
|
||
<span>{{ topicInfo.content }}</span>
|
||
</a-space>
|
||
<a-space direction="top">
|
||
<span style="width: 80px; font-size: 12px; display: inline-block">原始来源</span>
|
||
<a-space direction="vertical">
|
||
<a-space v-for="item in topicInfo.user_pain_point_sources" :key="item">
|
||
<a-link
|
||
style="
|
||
background-color: initial;
|
||
display: inline-block;
|
||
width: 280px;
|
||
text-overflow: ellipsis;
|
||
"
|
||
:href="item.link"
|
||
target="_blank"
|
||
>{{ item.title }}</a-link
|
||
>
|
||
<img src="@/assets/img/hottranslation/xhs.png" style="width: 16px; height: 16px" />
|
||
</a-space>
|
||
</a-space>
|
||
</a-space>
|
||
</a-space>
|
||
</div>
|
||
</a-modal>
|
||
</view>
|
||
</template>
|
||
|
||
<script setup>
|
||
import topHeader from './topHeader.vue';
|
||
import { fetchUserPainPointsDetail, fetchUserPainPointsList } from '@/api/all/index';
|
||
import { ref, onMounted, computed } from 'vue';
|
||
import top1 from '@/assets/img/captcha/top1.svg';
|
||
import top2 from '@/assets/img/captcha/top2.svg';
|
||
import top3 from '@/assets/img/captcha/top3.svg';
|
||
const topImages = [top1, top2, top3];
|
||
const visible = ref(false);
|
||
const topicInfo = ref({});
|
||
const topHeaderRef = ref();
|
||
// 从topHeader获取统一的状态
|
||
const selectedIndustry = computed(() => topHeaderRef.value?.selectedIndustry);
|
||
const selectedSubCategory = computed(() => topHeaderRef.value?.selectedSubCategory);
|
||
const selectedTimePeriod = computed(() => topHeaderRef.value?.selectedTimePeriod);
|
||
const dataList = ref([]);
|
||
const columns = [
|
||
{
|
||
title: '排名',
|
||
dataIndex: 'rank',
|
||
slotName: 'rank',
|
||
width: 60,
|
||
minWidth: 60,
|
||
},
|
||
{
|
||
title: '痛点名称',
|
||
dataIndex: 'name',
|
||
width: 300,
|
||
minWidth: 300,
|
||
},
|
||
{
|
||
title: '关键词',
|
||
dataIndex: 'keywords',
|
||
slotName: 'keywords',
|
||
width: 250,
|
||
minWidth: 250,
|
||
},
|
||
{
|
||
width: 180,
|
||
minWidth: 180,
|
||
title: '频次',
|
||
dataIndex: 'frequency',
|
||
sortable: {
|
||
sortDirections: ['ascend', 'descend'],
|
||
},
|
||
slotName: 'frequency',
|
||
},
|
||
{
|
||
title: '代表性发言',
|
||
dataIndex: 'content',
|
||
slotName: 'content',
|
||
width: 320,
|
||
minWidth: 320,
|
||
},
|
||
{
|
||
title: '操作',
|
||
slotName: 'optional',
|
||
width: 120,
|
||
minWidth: 120,
|
||
},
|
||
];
|
||
// 详情
|
||
const gotoDetail = async (record) => {
|
||
const res = await fetchUserPainPointsDetail(record.id);
|
||
if (res.code == 200) {
|
||
topicInfo.value = res.data;
|
||
visible.value = true;
|
||
}
|
||
};
|
||
|
||
const getUserPainPointsList = async () => {
|
||
const params = {
|
||
industry_id: selectedIndustry.value,
|
||
time_dimension: selectedTimePeriod.value,
|
||
};
|
||
if (selectedIndustry.value == undefined) {
|
||
return;
|
||
}
|
||
if (selectedSubCategory.value == undefined) {
|
||
return;
|
||
}
|
||
if (selectedSubCategory.value != 0) {
|
||
params['industry_id'] = selectedSubCategory.value;
|
||
}
|
||
const res = await fetchUserPainPointsList(params);
|
||
if (res.code === 200) {
|
||
dataList.value = res.data;
|
||
}
|
||
};
|
||
// 弹窗的取消
|
||
const handleCancel = () => {
|
||
visible.value = false;
|
||
};
|
||
|
||
// 弹窗的确定
|
||
const handleOk = () => {
|
||
visible.value = false;
|
||
};
|
||
// 监听筛选条件变化
|
||
watch([selectedIndustry, selectedTimePeriod, selectedSubCategory], () => {
|
||
getUserPainPointsList();
|
||
});
|
||
|
||
onMounted(() => {
|
||
getUserPainPointsList();
|
||
});
|
||
const search = () => {
|
||
getUserPainPointsList();
|
||
};
|
||
</script>
|
||
|
||
<style scoped>
|
||
/* 自定义样式 */
|
||
:deep(.arco-table-th) {
|
||
background-color: var(--color-fill-2);
|
||
}
|
||
|
||
:deep(.arco-table-tr):hover {
|
||
background-color: var(--color-fill-1);
|
||
}
|
||
:deep(.arco-statistic-content .arco-statistic-value-integer) {
|
||
font-size: 14px;
|
||
}
|
||
|
||
.pop-btn {
|
||
background: #fff !important;
|
||
border-color: #fff !important;
|
||
color: #737478 !important;
|
||
margin-left: -5px;
|
||
}
|
||
|
||
:deep(.arco-btn-outline) {
|
||
color: #6d4cfe !important;
|
||
border-color: #6d4cfe !important;
|
||
}
|
||
</style>
|