Files
lingji-work-fe/src/views/components/dataEngine/keyWord.vue

627 lines
19 KiB
Vue
Raw Normal View History

2025-06-30 18:37:27 +08:00
<!-- eslint-disable vue/no-duplicate-attributes -->
2025-06-16 14:42:26 +08:00
<template>
<view>
2025-06-17 11:18:39 +08:00
<topHeader ref="topHeaderRef" @search="search"></topHeader>
2025-06-16 14:42:26 +08:00
<!-- 关键词热度榜 -->
2025-06-30 18:37:27 +08:00
<a-space direction="vertical" style="background-color: #fff; width: 100%; padding: 0 20px" class="mb-24px">
<div class="title-row">
<span class="title mr-4px">关键词热度榜</span>
<a-tooltip>
<template #content>基于该行业用户内容中提及频率较高的关键词按热度进行排序反映近期关注焦点</template>
<icon-question-circle size="16" class="color-#737478" />
</a-tooltip>
</div>
2025-06-22 16:01:48 +08:00
<a-table
:columns="columns"
:data="dataList"
:filter-icon-align-left="alignLeft"
:scroll="true"
:pagination="false"
2025-06-30 18:37:27 +08:00
@change="handleChange"
2025-06-22 16:01:48 +08:00
>
<template #heatLevel>
<a-space>
<span>热度指数</span>
2025-06-30 18:37:27 +08:00
<a-tooltip>
<template #content>综合话题出现频次互动数据如点赞收藏评论加权计算的热度得分</template>
<icon-question-circle size="14" class="color-#737478" />
</a-tooltip>
2025-06-22 16:01:48 +08:00
</a-space>
</template>
<template #trendTitle>
<a-space>
<span>变化幅度</span>
2025-06-30 18:37:27 +08:00
<a-tooltip>
<template #content>仅基于关键词出现频次</template>
<icon-question-circle size="14" class="color-#737478" />
</a-tooltip>
2025-06-22 16:01:48 +08:00
</a-space>
</template>
<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 #hot="{ record }">
<img v-for="i in record.hot" :key="i" :src="starImages[i - 1]" style="width: 16px; height: 16px" />
</template>
<template #sentiment="{ record }">
2025-06-30 18:37:27 +08:00
<img v-if="record.felling == '2'" src="@/assets/img/hottranslation/good.png" class="w-24px h-24px" />
<img v-else-if="record.felling == '1'" src="@/assets/img/hottranslation/normal.png" class="w-24px h-24px" />
<img v-else-if="record.felling == '0'" src="@/assets/img/hottranslation/poor.png" class="w-24px h-24px" />
2025-06-22 16:01:48 +08:00
</template>
<template #tred="{ record }">
2025-06-30 18:37:27 +08:00
<div class="flex items-center" :class="record.trend > 0 ? 'color-#F64B31' : 'color-#25C883'">
<icon-arrow-up v-if="record.trend > 0" size="16" />
<icon-arrow-down v-else size="16" />
{{ `${record.trend * 100}%` }}
</div>
2025-06-16 14:42:26 +08:00
</template>
</a-table>
</a-space>
<!-- 行业情绪 -->
2025-06-30 18:37:27 +08:00
<a-space direction="vertical" style="background-color: #fff; width: 100%; padding: 0 20px" class="mb-24px">
<div class="title-row">
<span class="title mr-4px">行业情绪</span>
<a-tooltip>
<template #content
>对该行业下用户内容进行情绪分析按情绪类别统计占比提取占比最高者作为行业情绪代表</template
>
<icon-question-circle size="16" class="color-#737478" />
</a-tooltip>
</div>
2025-06-16 14:42:26 +08:00
<a-space align="center">
<a-space>
2025-06-22 16:01:48 +08:00
<a-space style="width: 320px">
2025-06-16 14:42:26 +08:00
<div id="container" style="height: 180px; width: 180px"></div>
2025-06-30 18:37:27 +08:00
<a-space v-if="fellingRate.length > 0" direction="vertical" style="font-size: 14px">
2025-06-16 14:42:26 +08:00
<a-space>
<span style="width: 8px; height: 8px; background-color: #25c883; border-radius: 50%"></span>
<span>正面情绪 </span>
2025-06-22 16:01:48 +08:00
<span style="width: 40px">{{ getFormatter(fellingRate[0] * 100) }}</span>
2025-06-16 14:42:26 +08:00
</a-space>
<a-space>
<span style="width: 8px; height: 8px; background-color: #f64b31; border-radius: 50%"></span>
<span>负面情绪 </span>
2025-06-22 16:01:48 +08:00
<span style="width: 40px">{{ getFormatter(fellingRate[1] * 100) }}</span>
2025-06-16 14:42:26 +08:00
</a-space>
</a-space>
</a-space>
2025-06-22 16:01:48 +08:00
<a-table
:columns="columns2"
:data="rowData"
:span-method="spanMethod"
:filter-icon-align-left="alignLeft"
:scroll="true"
:pagination="false"
2025-06-30 18:37:27 +08:00
@change="handleChange"
2025-06-22 16:01:48 +08:00
>
<template #felling="{ 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"
/>
2025-06-16 14:42:26 +08:00
</template>
</a-table>
</a-space>
</a-space>
</a-space>
<!-- 新兴关键词 -->
2025-06-30 18:37:27 +08:00
<a-space direction="vertical" style="background-color: #fff; width: 100%; padding: 0 20px" class="mb-24px">
<div class="title-row">
<span class="title mr-4px">新兴关键词</span>
<a-tooltip>
<template #content
>指当前周期中首次出现或相较上一周期词频显著增长的关键词反映近期出现的新关注点</template
>
<icon-question-circle size="16" class="color-#737478" />
</a-tooltip>
</div>
2025-06-22 16:01:48 +08:00
<a-table
:columns="columns3"
:data="keywordList"
:filter-icon-align-left="alignLeft"
:scroll="true"
:pagination="false"
2025-06-30 18:37:27 +08:00
@change="handleChange"
2025-06-22 16:01:48 +08:00
>
<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 #felling="{ 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 #first_appeared_at="{ record }">
<div>{{ formatTimestamp(record.first_appeared_at) }}</div>
</template>
<template #hot="{ record }">
<img v-for="i in record.hot" :key="i" :src="starImages[i - 1]" style="width: 16px; height: 16px" />
</template>
<template #hotTitle="{ record }">
<a-space>
<span>当前热度指数</span>
<a-popover position="tr">
<a-button type="primary" class="pop-btn2">
<template #icon>
<icon-question-circle />
2025-06-16 14:42:26 +08:00
</template>
2025-06-22 16:01:48 +08:00
</a-button>
<template #content>
<p style="margin: 0">综合关键词出现频次互动表现如点赞收藏评论加权计算的热度得分</p>
</template>
</a-popover>
</a-space>
</template>
<template #trendTitle="{ record }">
<a-space>
<span>变化幅度</span>
<a-popover position="tr">
<a-button type="primary" class="pop-btn2">
<template #icon>
<icon-question-circle />
2025-06-17 11:18:39 +08:00
</template>
2025-06-22 16:01:48 +08:00
</a-button>
<template #content>
<p style="margin: 0">仅基于关键词出现频次</p>
</template>
</a-popover>
</a-space>
</template>
<template #tred="{ record }">
2025-06-30 18:37:27 +08:00
<div class="flex items-center" :class="record.trend > 0 ? 'color-#F64B31' : 'color-#25C883'">
<icon-arrow-up v-if="record.trend > 0" size="16" />
<icon-arrow-down v-else size="16" />
{{ `${record.trend * 100}%` }}
</div>
2025-06-22 16:01:48 +08:00
</template>
<template #optional="{ record }">
<a-button type="outline" @click="gotoDetail(record)">详情</a-button>
2025-06-16 14:42:26 +08:00
</template>
</a-table>
</a-space>
2025-06-17 11:18:39 +08:00
<!-- modal -->
2025-06-30 18:37:27 +08:00
<a-modal :visible="visible" unmountOnClose @ok="handleOk" @cancel="handleCancel">
2025-06-17 11:18:39 +08:00
<template #title>
<span style="text-align: left; width: 100%">新兴关键词</span>
</template>
<div>
<a-space direction="vertical">
<a-space>
<span style="margin-right: 16px">关键词</span>
<span>{{ topicInfo.name }}</span>
</a-space>
<a-space>
<span style="margin-right: 16px">最大规模出现</span>
<span>{{ formatTimestamp(topicInfo.first_appeared_at) }}</span>
</a-space>
<a-space>
<span style="margin-right: 16px">变化幅度</span>
<div>
<a-statistic
v-if="topicInfo?.trend > 0"
style="font-size: 14px"
:value="topicInfo.trend * 100"
:value-style="{ color: '#F64B31' }"
>
<template #prefix>
<IconArrowRise />
</template>
<template #suffix>%</template>
</a-statistic>
<a-statistic
v-else
style="font-size: 14px"
:value="topicInfo?.trend * 100 || 0"
:value-style="{ color: '#25C883' }"
>
<template #prefix>
<IconArrowFall />
</template>
<template #suffix>%</template>
</a-statistic>
</div>
</a-space>
<a-space>
<span style="margin-right: 16px">热度指数</span>
<img v-for="i in topicInfo.hot" :key="i" :src="starImages[i - 1]" style="width: 16px; height: 16px" />
</a-space>
<a-space direction="top">
<span style="margin-right: 16px; width: 60px; font-size: 12px">原始来源 </span>
<a-space direction="vertical" style="margin-left: 15px">
<a-space v-for="item in topicInfo.industry_new_keyword_sources" :key="item">
<a-link style="background-color: initial" :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>
2025-06-16 14:42:26 +08:00
</view>
</template>
<script setup>
import topHeader from './topHeader.vue';
2025-06-17 11:18:39 +08:00
import {
fetchKeywordTrendsList,
fetchIndustryEmotions,
fetchNewKeywordList,
fetchNewKeywordDetail,
} from '@/api/all/index';
2025-06-16 14:42:26 +08:00
import { ref, onMounted, onBeforeUnmount, watchEffect, computed } from 'vue';
import * as echarts from 'echarts';
import star1 from '@/assets/img/hottranslation/star-fill1.png';
import star2 from '@/assets/img/hottranslation/star-fill2.png';
import star3 from '@/assets/img/hottranslation/star-fill3.png';
import star4 from '@/assets/img/hottranslation/star-fill4.png';
import star5 from '@/assets/img/hottranslation/star-fill5.png';
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 starImages = [star1, star2, star3, star4, star5];
const topImages = [top1, top2, top3];
const chartRef = (ref < HTMLElement) | (null > null);
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 rowData = ref([]);
const keywordList = ref([]);
const fellingRate = ref([]);
2025-06-17 11:18:39 +08:00
const visible = ref(false);
const topicInfo = ref({});
2025-06-22 16:01:48 +08:00
const columns = [
{
title: '排名',
dataIndex: 'rank',
slotName: 'rank',
2025-06-30 18:37:27 +08:00
width: 100,
2025-06-22 16:01:48 +08:00
},
{
title: '关键词名称',
dataIndex: 'name',
2025-06-30 18:37:27 +08:00
width: 300,
2025-06-22 16:01:48 +08:00
},
{
titleSlotName: 'heatLevel',
title: '热度指数',
dataIndex: 'hot',
sortable: {
sortDirections: ['ascend', 'descend'],
},
slotName: 'hot',
2025-06-30 18:37:27 +08:00
width: 220,
2025-06-22 16:01:48 +08:00
},
{
titleSlotName: 'trendTitle',
title: '变化幅度',
sortable: {
sortDirections: ['ascend', 'descend'],
},
dataIndex: 'tred',
slotName: 'tred',
2025-06-30 18:37:27 +08:00
width: 220,
2025-06-22 16:01:48 +08:00
},
{
title: '情感倾向',
dataIndex: 'sentiment',
slotName: 'sentiment',
2025-06-30 18:37:27 +08:00
width: 220,
2025-06-22 16:01:48 +08:00
},
];
const columns2 = [
{
title: '情绪分布',
dataIndex: 'felling',
slotName: 'felling',
width: 120,
minWidth: 120,
},
{
title: '主要观点',
dataIndex: 'content',
},
];
const spanMethod = ({ record, columnIndex }) => {
// console.log(record.felling);
// if (record.felling === 2) {
// return {
// rowspan: 3,
// };
// }
};
const columns3 = [
{
title: '排名',
dataIndex: 'rank',
slotName: 'rank',
width: 180,
minWidth: 180,
},
{
title: '新兴关键词名称',
dataIndex: 'name',
},
{
title: '首次大规模出现',
dataIndex: 'first_appeared_at',
slotName: 'first_appeared_at',
},
{
titleSlotName: 'hotTitle',
title: '当前热度指数',
dataIndex: 'hot',
slotName: 'hot',
sortable: {
sortDirections: ['ascend', 'descend'],
},
},
{
titleSlotName: 'trendTitle',
title: '变化幅度',
sortable: {
sortDirections: ['ascend', 'descend'],
},
dataIndex: 'tred',
slotName: 'tred',
width: 180,
minWidth: 180,
},
{
title: '操作',
slotName: 'optional',
width: 120,
minWidth: 120,
},
];
const getFormatter = (value) => {
const formattedValue = Number.isInteger(value) ? value.toString() : value.toFixed(2);
return formattedValue + '%';
};
2025-06-16 14:42:26 +08:00
const getIndustryEmotions = async () => {
const params = {
industry_id: selectedIndustry.value,
time_dimension: selectedTimePeriod.value,
};
if (selectedIndustry.value == undefined) {
return;
}
if (selectedSubCategory.value == undefined) {
return;
}
2025-06-17 11:18:39 +08:00
if (selectedSubCategory.value != 0) {
params['industry_id'] = selectedSubCategory.value;
}
2025-06-16 14:42:26 +08:00
const res = await fetchIndustryEmotions(params);
2025-06-22 16:01:48 +08:00
if (res.code == 200) {
let data = res['data'];
fellingRate.value = [];
fellingRate.value.push(data['good_felling_rate']);
fellingRate.value.push(data['bad_felling_rate']);
2025-06-16 14:42:26 +08:00
2025-06-22 16:01:48 +08:00
drawChart();
rowData.value = data['industry_emotion_view_points'];
let items = groupedData();
}
2025-06-16 14:42:26 +08:00
};
2025-06-17 11:18:39 +08:00
// 详情
const gotoDetail = async (record) => {
const res = await fetchNewKeywordDetail(record.id);
2025-06-22 16:01:48 +08:00
if (res.code === 200) {
visible.value = true;
topicInfo.value = res.data;
}
2025-06-17 11:18:39 +08:00
};
2025-06-16 14:42:26 +08:00
const groupedData = () => {
const groups = {
negative: { name: '负面', items: [], color: '#F64B31' },
neutral: { name: '中性', items: [], color: '#FFAA16' },
positive: { name: '正面', items: [], color: '#25C883' },
};
rowData.value.forEach((item) => {
if (item.felling === 0) groups.negative.items.push(item);
else if (item.felling === 1) groups.neutral.items.push(item);
else if (item.felling === 2) groups.positive.items.push(item);
});
return groups;
};
2025-06-17 11:18:39 +08:00
// 弹窗的取消
const handleCancel = () => {
visible.value = false;
};
2025-06-16 14:42:26 +08:00
2025-06-17 11:18:39 +08:00
// 弹窗的确定
const handleOk = () => {
visible.value = false;
};
2025-06-16 14:42:26 +08:00
const getKeywordTrendsList = async () => {
const params = {
industry_id: selectedIndustry.value,
time_dimension: selectedTimePeriod.value,
};
if (selectedIndustry.value == undefined) {
return;
}
if (selectedSubCategory.value == undefined) {
return;
}
2025-06-17 11:18:39 +08:00
if (selectedSubCategory.value != 0) {
params['industry_id'] = selectedSubCategory.value;
}
2025-06-16 14:42:26 +08:00
const res = await fetchKeywordTrendsList(params);
2025-06-22 16:01:48 +08:00
if (res.code === 200) {
dataList.value = res.data;
}
2025-06-16 14:42:26 +08:00
};
const formatTimestamp = (timestamp) => {
if (!timestamp) return '未记录';
try {
return dayjs.unix(timestamp).format('YYYY-MM-DD HH:mm');
} catch (e) {
return '格式错误';
}
};
const getNewKeywordList = 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 && selectedSubCategory.value != undefined) {
2025-06-17 11:18:39 +08:00
params['industry_id'] = selectedSubCategory.value;
}
2025-06-16 14:42:26 +08:00
const res = await fetchNewKeywordList(params);
if (res.code == 200) {
// 这里需要根据API返回的数据结构处理成tagRows需要的格式
keywordList.value = res.data;
}
2025-06-16 14:42:26 +08:00
};
const drawChart = () => {
2025-06-30 18:37:27 +08:00
let dom = document.getElementById('container');
let myChart = echarts.init(dom, null, {
2025-06-16 14:42:26 +08:00
renderer: 'canvas',
useDirtyRect: false,
});
2025-06-30 18:37:27 +08:00
let option;
2025-06-16 14:42:26 +08:00
option = {
color: ['#25C883', '#F64B31'],
series: [
{
type: 'pie',
avoidLabelOverlap: false,
data: fellingRate.value,
labelLine: {
show: false, // 不显示引导线
},
radius: ['40%', '55%'],
},
],
};
if (option && typeof option === 'object') {
myChart.setOption(option);
}
};
2025-06-17 11:18:39 +08:00
const search = () => {
getKeywordTrendsList();
getIndustryEmotions();
getNewKeywordList();
};
2025-06-16 14:42:26 +08:00
// 监听筛选条件变化
2025-06-17 11:18:39 +08:00
watch([selectedTimePeriod, selectedSubCategory], () => {
2025-06-16 14:42:26 +08:00
getKeywordTrendsList();
getIndustryEmotions();
getNewKeywordList();
});
2025-06-17 11:18:39 +08:00
watch([selectedIndustry], () => {
selectedSubCategory.value = 0;
2025-06-16 14:42:26 +08:00
getKeywordTrendsList();
getIndustryEmotions();
getNewKeywordList();
});
2025-06-17 11:18:39 +08:00
onMounted(() => {
getKeywordTrendsList();
getIndustryEmotions();
getNewKeywordList();
});
2025-06-16 14:42:26 +08:00
</script>
2025-06-30 18:37:27 +08:00
<style scoped lang="scss">
2025-06-16 14:42:26 +08:00
/* 自定义样式 */
: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;
2025-06-22 16:01:48 +08:00
margin-left: -8px;
}
.pop-btn2 {
background: transparent !important;
border-color: transparent !important;
color: #737478 !important;
margin-left: -8px;
2025-06-16 14:42:26 +08:00
}
2025-06-30 18:37:27 +08:00
.title-row {
display: flex;
height: 64px;
padding: 10px 0 2px 0;
align-items: center;
.title {
color: var(--Text-1, #211f24);
font-family: 'Alibaba PuHuiTi';
font-size: 16px;
font-style: normal;
font-weight: 400;
line-height: 24px; /* 150% */
}
}
2025-06-16 14:42:26 +08:00
</style>