瀏覽代碼

Merge branch 'refs/heads/wang' into test

WanGxC 11 月之前
父節點
當前提交
44d3e11e38

+ 0 - 19
src/views/keyword/topSearchTermTable/api.ts

@@ -1,19 +0,0 @@
-import { request } from '/@/utils/service';
-
-const apiPrefix = '/api/searchterm/';
-
-export function getTopSearchTermTable(query: any) {
-  return request({
-    url: apiPrefix + 'topsearchtermTable/',
-    method: 'GET',
-    params: query,
-  });
-}
-
-// export function postCreateSearchTerm(body: any) {
-//   return request({
-//     url: apiPrefix + 'topsearchtermroot/',
-//     method: 'POST',
-//     data: body,
-//   });
-// }

+ 0 - 0
src/views/keyword/rootWordManage/api.ts → src/views/searchTerm/rootWordManage/api.ts


+ 0 - 0
src/views/keyword/rootWordManage/components/root-word-manage-table.vue → src/views/searchTerm/rootWordManage/components/root-word-manage-table.vue


+ 0 - 0
src/views/keyword/rootWordManage/index.vue → src/views/searchTerm/rootWordManage/index.vue


+ 11 - 0
src/views/searchTerm/topSearchTermRank/api.ts

@@ -0,0 +1,11 @@
+import { request } from '/@/utils/service';
+
+const apiPrefix = '/api/searchterm/';
+
+export function getTopSearchTermTable(query: any) {
+  return request({
+    url: apiPrefix + 'topsearchtermRank/',
+    method: 'GET',
+    params: query,
+  });
+}

+ 97 - 0
src/views/searchTerm/topSearchTermRank/column-chart.vue

@@ -0,0 +1,97 @@
+<script setup lang="ts">
+/**
+ * @Name: column-chart.vue
+ * @Description:
+ * @Author: Cheney
+ */
+
+import { PropType, ref, onMounted, onUnmounted, watch } from 'vue';
+import * as echarts from 'echarts';
+
+const props = defineProps({
+  rowData: {
+    type: Array as PropType<{ [key: string]: number }[]>,
+    required: true,
+  },
+});
+
+const chartRef = ref<HTMLElement | null>(null);
+let chart: echarts.ECharts | null = null;
+let resizeObserver: ResizeObserver | null = null
+
+onMounted(() => {
+  initChart();
+});
+
+onUnmounted(() => {
+  // 清理 ResizeObserver
+  if (resizeObserver) {
+    resizeObserver.disconnect()
+  }
+  if (chart) {
+    chart.dispose()
+    chart = null
+  }
+})
+
+watch(
+    () => props.rowData,
+    () => {
+      updateChart();
+    },
+    { immediate: true, deep: true }
+);
+
+function updateChart() {
+  if (!chart || !props.rowData || props.rowData.length === 0) return;
+
+  const data = props.rowData[0];
+  const dates = Object.keys(data);
+  const values = Object.values(data);
+
+  const option: echarts.EChartsOption = {
+    xAxis: {
+      type: 'category',
+      data: dates,
+      axisLabel: {
+        rotate: 45,
+        interval: 0,
+      },
+    },
+    yAxis: {
+      type: 'value',
+    },
+    series: [
+      {
+        data: values,
+        type: 'line',
+      },
+    ],
+    tooltip: {
+      trigger: 'axis',
+    },
+  };
+  chart.setOption(option);
+}
+
+function initChart() {
+  if (chartRef.value) {
+    chart = echarts.init(chartRef.value);
+    updateChart();
+
+    // 创建 ResizeObserver
+    resizeObserver = new ResizeObserver(() => {
+      chart?.resize()
+    })
+
+    // 观察 chartRef 元素
+    resizeObserver.observe(chartRef.value)
+  }
+}
+</script>
+
+<template>
+  <div ref="chartRef" style="width: 100%; height: 300px"></div>
+</template>
+
+<style scoped></style>

+ 52 - 138
src/views/keyword/topSearchTermTable/index.vue → src/views/searchTerm/topSearchTermRank/index.vue

@@ -1,43 +1,29 @@
 <script setup lang="ts">
 /**
  * @Name: index.vue
- * @Description: 关键词-TopSearchTerm Table
+ * @Description:
  * @Author: Cheney
  */
 
-import { onMounted, ref, watch } from 'vue';
+import { Key, PictureRounded, Refresh, Search, TopRight } from '@element-plus/icons-vue';
+import { nextTick, onMounted, ref, watch } from 'vue';
+import dayjs from 'dayjs';
+import { useRouter } from 'vue-router';
+import { marketplaceIdEnum } from '/@/utils/marketplaceIdEnum';
 import { usePagination } from '/@/utils/usePagination';
 import { getTopSearchTermTable } from './api';
-import { marketplaceIdEnum } from '/@/utils/marketplaceIdEnum';
-import {
-  Download,
-  Goods,
-  Key,
-  Medal,
-  Memo,
-  Pointer,
-  Rank,
-  Reading,
-  RefreshRight,
-  Search,
-  Switch,
-  TopRight,
-} from '@element-plus/icons-vue';
-import { useRouter } from 'vue-router';
 import { ElMessage } from 'element-plus';
-import dayjs from 'dayjs';
+import ColumnChart from '/src/views/searchTerm/topSearchTermRank/column-chart.vue';
 
 const router = useRouter();
 
-const { tableData, total, currentPage, pageSize, handlePageChange } = usePagination(fetchTableData);
-
 const date = ref([dayjs().subtract(7, 'day').format('YYYY-MM-DD'), dayjs().subtract(1, 'day').format('YYYY-MM-DD')]);
-const searchTermInp = ref('');
-const asinInp = ref('');
+const { tableData, total, currentPage, pageSize, handlePageChange } = usePagination(fetchTableData);
 const marketplaceSelect = ref(marketplaceIdEnum[0].value); // 当前只有美国区 默认第一个为美国
 const marketplaceOptions = marketplaceIdEnum;
 const reportTypeSelect = ref('weekly');
-
+const searchTermInp = ref('');
+const asinInp = ref('');
 const tableLoading = ref(false);
 
 onMounted(() => {
@@ -70,10 +56,18 @@ async function fetchTableData() {
     date_start: date.value[0],
     date_end: date.value[1],
   };
-  const response = await getTopSearchTermTable(query);
-  total.value = response.total;
-  tableData.value = response.data;
-  tableLoading.value = false;
+  try {
+    const response = await getTopSearchTermTable(query);
+    total.value = response.total;
+    tableData.value = response.data;
+  } catch (error) {
+    console.error('==Error==:', error);
+  } finally {
+    tableLoading.value = false;
+    await nextTick();
+    // 触发窗口 resize 事件
+    window.dispatchEvent(new Event('resize'));
+  }
 }
 
 async function handleSelectChange() {
@@ -85,16 +79,14 @@ async function handleQueryChange() {
     if (searchTermInp.value.length == 0) {
       return;
     } else {
-      ElMessage.warning({ message: '关键词只能输入数字和英文字母', plain: true });
+      ElMessage.warning({ message: '搜索词只能输入数字和英文字母', plain: true });
       return;
     }
   }
-
   if (asinInp.value.length > 0 && !validateAsinInput(asinInp.value)) {
     ElMessage.warning({ message: '不符合匹配规范', plain: true });
     return;
   }
-
   await fetchTableData();
 }
 
@@ -117,53 +109,28 @@ function validateAsinInput(input: string) {
 }
 
 function handleJump() {
-  // console.log('All defined routes:', router.getRoutes());
-  router.push({ path: '/keyword/rootWordManage' });
-}
-
-function getTagStyle(clickShareRank: number): Record<string, string> {
-  switch (clickShareRank) {
-    case 1:
-      return { backgroundColor: '#fbbf24', color: '#fff', border: '1px solid #fbbf24' }; // 金色
-    case 2:
-      return { backgroundColor: '#C0C0C0', color: '#fff', border: '1px solid #C0C0C0' }; // 银色
-    case 3:
-      return { backgroundColor: '#CD7F32', color: '#fff', border: '1px solid #CD7F32' }; // 铜色
-    default:
-      return { backgroundColor: '#e0e0e0', color: '#000', border: '1px solid #e0e0e0' }; // 默认颜色
-  }
+  router.push({ path: '/searchTerm/rootWordManage' });
 }
 </script>
 
 <template>
-  <div class="mt-3 mx-1.5" style="background-color: #f7f7f7">
-    <div class="flex justify-between mt-1.5 mx-2">
-      <div class="font-bold text-lg">
+  <div class="mx-3">
+    <el-divider>
+      <div class="font-bold text-xl">
         <el-icon style="top: 3px">
-          <Memo />
+          <DataLine />
         </el-icon>
-        Top Search Term - Table
-      </div>
-      <div>
-        <el-button type="primary" plain @click="handleJump" :icon="TopRight">关键词管理</el-button>
-        <el-button type="success" plain round :icon="Download">下载表格</el-button>
+        Top Search Term - Rank
       </div>
-    </div>
-  </div>
-  <div class="mx-3" style="margin-top: -8px">
-    <el-divider>
-      <el-icon>
-        <star-filled />
-      </el-icon>
     </el-divider>
   </div>
   <el-card class="mx-3" v-loading="tableLoading" style="border: none">
     <!-- table筛选栏 -->
     <div class="flex justify-between">
-      <div class="flex gap-6 flex-wrap">
+      <div class="flex gap-5 flex-wrap">
         <div>
           <span class="font-medium mr-0.5">市场 </span>
-          <el-select v-model="marketplaceSelect" @change="handleSelectChange" style="width: 130px">
+          <el-select v-model="marketplaceSelect" @change="handleSelectChange" style="width: 90px">
             <el-option
               v-for="item in marketplaceOptions"
               :disabled="item.disabled"
@@ -174,20 +141,21 @@ function getTagStyle(clickShareRank: number): Record<string, string> {
         </div>
         <div>
           <span class="font-medium mr-0.5">报告类型 </span>
-          <el-select v-model="reportTypeSelect" @change="handleSelectChange" style="width: 100px">
+          <el-select v-model="reportTypeSelect" @change="handleSelectChange" style="width: 90px">
             <el-option label="周度" value="weekly" />
             <el-option label="月度" value="monthly" />
           </el-select>
         </div>
         <div>
-          <span class="font-medium mr-0.5">关键词 </span>
+          <span class="font-medium mr-0.5">搜索词 </span>
           <el-input
             v-model="searchTermInp"
             @keyup.enter="handleQueryChange"
             :prefix-icon="Search"
             placeholder="输入后回车查询"
             clearable
-            style="width: 300px"></el-input>
+            @clear="handleSelectChange"
+            style="width: 240px" />
         </div>
         <div>
           <span class="font-medium mr-0.5">ASIN </span>
@@ -197,7 +165,8 @@ function getTagStyle(clickShareRank: number): Record<string, string> {
             :prefix-icon="Search"
             placeholder="输入后回车查询"
             clearable
-            style="width: 180px"></el-input>
+            @clear="handleSelectChange"
+            style="width: 180px" />
         </div>
         <div>
           <span class="font-medium mr-0.5">报告日期 </span>
@@ -208,95 +177,40 @@ function getTagStyle(clickShareRank: number): Record<string, string> {
             :popper-options="{ placement: 'bottom-end' }"
             :clearable="false"
             :disabled-date="(time: Date) => time > new Date()"
-            range-separator="至"
-            start-placeholder="开始日期"
-            end-placeholder="结束日期" />
+            range-separator="至" />
         </div>
       </div>
-      <el-button @click="refreshTable" :icon="RefreshRight" circle></el-button>
+      <div class="flex">
+        <el-button type="primary" plain @click="handleJump" :icon="TopRight">搜索词管理</el-button>
+        <el-button @click="refreshTable" :icon="Refresh" circle></el-button>
+      </div>
     </div>
-
     <!-- table -->
     <el-card shadow="never" class="mt-5">
       <div style="height: 795px; overflow: auto">
-        <el-table :data="tableData" stripe style="width: 100%">
-          <el-table-column fixed prop="searchTerm" label="关键词" width="260">
+        <el-table :data="tableData" height="795" stripe style="width: 100%">
+          <el-table-column fixed prop="searchTerm" label="搜索词" width="260">
             <template #header>
               <el-icon style="top: 2px; margin-right: 3px">
                 <Key />
               </el-icon>
-              <span>关键词</span>
+              <span>搜索词</span>
             </template>
             <template #default="{ row }">
-              <el-link :underline="false" href="https://www.bilibili.com/" target="_blank" style="color: #0b3289">{{
-                row.searchTerm
-              }}</el-link>
+              <el-link :underline="false" href="https://www.bilibili.com/" target="_blank" style="color: #0b3289">
+                {{ row.searchTerm }}
+              </el-link>
             </template>
           </el-table-column>
-          <el-table-column prop="searchFrequencyRank" label="关键词搜索排名" align="center" width="150">
+          <el-table-column prop="rank" label="搜索词搜索排名" align="center">
             <template #header>
               <el-icon style="top: 2px; margin-right: 4px">
-                <Rank />
-              </el-icon>
-              <span>关键词搜索排名</span>
-            </template>
-            <template #default="{ row }">
-              <span class="font-medium">{{ row.searchFrequencyRank }}</span>
-            </template>
-          </el-table-column>
-          <el-table-column prop="clickedAsin" align="center" label="Asin">
-            <template #header>
-              <el-icon style="top: 2px; margin-right: 5px">
-                <Goods />
-              </el-icon>
-              <span>Asin</span>
-            </template>
-            <template #default="{ row }">
-              <span class="font-medium">{{ row.clickedAsin }}</span>
-            </template>
-          </el-table-column>
-          <el-table-column prop="clickedItemName" label="标题">
-            <template #header>
-              <el-icon style="top: 2px; margin-right: 5px">
-                <Reading />
-              </el-icon>
-              <span>标题</span>
-            </template>
-          </el-table-column>
-          <el-table-column prop="clickShareRank" label="点击分享率排名" align="center" width="150">
-            <template #header>
-              <el-icon style="top: 2px; margin-right: 4px">
-                <Medal />
-              </el-icon>
-              <span>点击分享率排名</span>
-            </template>
-            <template #default="{ row }">
-              <!--<span class="font-semibold">{{ row.clickShareRank }}</span>-->
-              <el-tag :style="getTagStyle(row.clickShareRank)">
-                {{ row.clickShareRank }}
-              </el-tag>
-            </template>
-          </el-table-column>
-          <el-table-column prop="clickShare" align="center" label="点击分享率">
-            <template #header>
-              <el-icon style="top: 2px; margin-right: 4px">
-                <Pointer />
-              </el-icon>
-              <span>点击分享率</span>
-            </template>
-            <template #default="{ row }">
-              <span class="font-semibold">{{ row.clickShare }}</span>
-            </template>
-          </el-table-column>
-          <el-table-column prop="conversionShare" align="center" label="转化分享率">
-            <template #header>
-              <el-icon style="top: 2px; margin-right: 5px">
-                <Switch />
+                <PictureRounded />
               </el-icon>
-              <span>转化分享率</span>
+              <span>Tendency</span>
             </template>
             <template #default="{ row }">
-              <span class="font-semibold">{{ row.conversionShare }}</span>
+              <ColumnChart :rowData="row.rank" />
             </template>
           </el-table-column>
         </el-table>

+ 23 - 0
src/views/searchTerm/topSearchTermTable/api.ts

@@ -0,0 +1,23 @@
+import { request } from '/@/utils/service';
+
+const apiPrefix = '/api/searchterm/';
+
+export function getTopSearchTermTable(query: any) {
+  return request({
+    url: apiPrefix + 'topsearchtermTable/',
+    method: 'GET',
+    params: query,
+  });
+}
+
+export function postDownload(body: any) {
+  return request({
+    url: apiPrefix + 'topsearchtermTable/',
+    method: 'POST',
+    params: body,
+    responseType: 'blob',
+    headers: {
+      'Accept': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' // 指定接受 Excel 文件
+    }
+  });
+}

+ 428 - 0
src/views/searchTerm/topSearchTermTable/index.vue

@@ -0,0 +1,428 @@
+<script setup lang="ts">
+/**
+ * @Name: index.vue
+ * @Description: 搜索词-TopSearchTerm Table
+ * @Author: Cheney
+ */
+
+import { nextTick, onBeforeMount, onMounted, ref, watch } from 'vue';
+import { usePagination } from '/@/utils/usePagination';
+import { getTopSearchTermTable, postDownload } from './api';
+import { marketplaceIdEnum } from '/@/utils/marketplaceIdEnum';
+import { Download, Goods, Key, Medal, Pointer, Rank, Refresh, Search, Switch, TopRight } from '@element-plus/icons-vue';
+import { useRouter } from 'vue-router';
+import { ElMessage } from 'element-plus';
+import dayjs from 'dayjs';
+import enLocale from 'element-plus/es/locale/lang/en';
+
+const router = useRouter();
+
+const { tableData, total, currentPage, pageSize, handlePageChange } = usePagination(fetchTableData);
+
+const marketplaceSelect = ref(marketplaceIdEnum[0].value); // 当前只有美国区 默认第一个为美国
+const marketplaceOptions = marketplaceIdEnum;
+const reportTypeSelect = ref('weekly');
+const searchTermInp = ref('');
+const asinInp = ref('');
+const tableLoading = ref(false);
+const downloadLoading = ref(false);
+const date = ref(calculateLastWeek()); // 设置默认日期为上周的周日到周六
+const dateDimension = ref(date.value[0]);
+
+
+onBeforeMount(() => {
+  fetchTableData();
+});
+
+watch(dateDimension, () => {
+  calculateDate();
+  // console.log('==Date==', date.value[0], date.value[1]);
+  fetchTableData();
+});
+
+function calculateDate() {
+  if (reportTypeSelect.value === 'weekly') {
+    date.value[0] = dateDimension.value;
+    date.value[1] = calculateEndDate(dateDimension.value);
+  } else if (reportTypeSelect.value === 'monthly') {
+    const selectedMonth = dayjs(dateDimension.value);
+    date.value[0] = selectedMonth.startOf('month').format('YYYY-MM-DD');
+    date.value[1] = selectedMonth.endOf('month').format('YYYY-MM-DD');
+  }
+}
+
+/**
+ * 计算上周的周日到周六的日期范围
+ */
+function calculateLastWeek() {
+  const today = dayjs();
+  const lastSaturday = today.subtract(today.day() + 1, 'day'); // 上周六
+  const lastSunday = lastSaturday.subtract(6, 'day'); // 上周日
+  return [lastSunday.format('YYYY-MM-DD'), lastSaturday.format('YYYY-MM-DD')];
+}
+
+function calculateEndDate(startDate: string) {
+  return dayjs(startDate).add(6, 'day').format('YYYY-MM-DD');
+}
+
+async function refreshTable() {
+  currentPage.value = 1;
+  pageSize.value = 10;
+  asinInp.value = '';
+  searchTermInp.value = '';
+  reportTypeSelect.value = 'weekly';
+  marketplaceSelect.value = marketplaceIdEnum[0].value;
+  await fetchTableData();
+}
+
+async function fetchTableData() {
+  tableLoading.value = true;
+  const query = {
+    page: currentPage.value,
+    limit: pageSize.value,
+    asin: asinInp.value,
+    search_term: searchTermInp.value,
+    report_type: reportTypeSelect.value,
+    marketplace_Ids: marketplaceSelect.value,
+    date_start: date.value[0],
+    date_end: date.value[1],
+  };
+  try {
+    const response = await getTopSearchTermTable(query);
+    total.value = response.total;
+    tableData.value = response.data;
+  } catch (error) {
+    console.error('==Error==:', error);
+  } finally {
+    tableLoading.value = false;
+    await nextTick();
+    // 触发窗口 resize 事件
+    window.dispatchEvent(new Event('resize'));
+  }
+}
+
+async function handleSelectChange() {
+  calculateDate();
+  await fetchTableData();
+}
+
+async function handleQueryChange() {
+  if (!validateSearchTermInput(searchTermInp.value)) {
+    if (searchTermInp.value.length == 0) {
+      return;
+    } else {
+      ElMessage.warning({ message: '搜索词只能输入数字和英文字母', plain: true });
+      return;
+    }
+  }
+  if (asinInp.value.length > 0 && !validateAsinInput(asinInp.value)) {
+    ElMessage.warning({ message: '不符合匹配规范', plain: true });
+    return;
+  }
+  await fetchTableData();
+}
+
+/**
+ * 校验SearchTerm输入是否合法
+ * @param input 输入的字符串
+ */
+function validateSearchTermInput(input: string) {
+  const regex = /^[a-zA-Z0-9\s]*$/;
+  return regex.test(input);
+}
+
+/**
+ * 校验Asin输入是否合法
+ * @param input 输入的字符串
+ */
+function validateAsinInput(input: string) {
+  const regex = /^[Bb]0[A-Za-z0-9\s]*$/i;
+  return regex.test(input);
+}
+
+function handleJump() {
+  // console.log('All defined routes:', router.getRoutes());
+  router.push({ path: '/searchTerm/rootWordManage' });
+}
+
+async function handleDownload() {
+  downloadLoading.value = true;
+  try {
+    const body = {
+      asin: asinInp.value,
+      date_start: date.value[0],
+      date_end: date.value[1],
+      search_term: searchTermInp.value,
+      marketplace_Ids: marketplaceSelect.value,
+      report_type: reportTypeSelect.value,
+    };
+
+    const response = await postDownload(body);
+
+    const blob = new Blob([response.data], { type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' });
+
+    // 创建一个临时 URL
+    const url = window.URL.createObjectURL(blob);
+
+    // 创建一个临时的 <a> 元素并触发下载
+    const link = document.createElement('a');
+    link.href = url;
+
+    // 设置文件名
+    const currentTime = dayjs().format('YYYY-MM-DD_HH_mm_ss');
+    const filename = `TopSearchTerm_${currentTime}.xlsx`;
+
+    link.setAttribute('download', filename);
+
+    // 添加到 body, 触发点击, 然后移除
+    document.body.appendChild(link);
+    link.click();
+    document.body.removeChild(link);
+
+    // 释放 URL 对象
+    window.URL.revokeObjectURL(url);
+
+    ElMessage.success('文件下载成功');
+  } catch (error) {
+    console.error('==Error==:', error);
+    ElMessage.error('文件下载失败,请重试');
+  } finally {
+    downloadLoading.value = false;
+  }
+}
+
+function getTagStyle(clickShareRank: number): Record<string, string> {
+  switch (clickShareRank) {
+    case 1:
+      return { backgroundColor: '#fbbf24', color: '#fff', border: '1px solid #fbbf24' }; // 金色
+    case 2:
+      return { backgroundColor: '#C0C0C0', color: '#fff', border: '1px solid #C0C0C0' }; // 银色
+    case 3:
+      return { backgroundColor: '#CD7F32', color: '#fff', border: '1px solid #CD7F32' }; // 铜色
+    default:
+      return { backgroundColor: '#e0e0e0', color: '#000', border: '1px solid #e0e0e0' };
+  }
+}
+</script>
+
+<template>
+  <div class="mx-3">
+    <el-divider>
+      <div class="font-bold text-xl">
+        <el-icon style="top: 3px">
+          <DataAnalysis />
+        </el-icon>
+        Top Search Term - Table
+      </div>
+    </el-divider>
+  </div>
+  <el-card class="mx-3" v-loading="tableLoading" style="border: none">
+    <!-- table筛选栏 -->
+    <div class="flex justify-between">
+      <div class="flex gap-5 flex-wrap">
+        <div>
+          <span class="font-medium mr-0.5">市场 </span>
+          <el-select v-model="marketplaceSelect" @change="handleSelectChange" style="width: 90px">
+            <el-option
+              v-for="item in marketplaceOptions"
+              :disabled="item.disabled"
+              :key="item.value"
+              :value="item.value"
+              :label="item.label" />
+          </el-select>
+        </div>
+        <div>
+          <span class="font-medium mr-0.5">报告类型 </span>
+          <el-select v-model="reportTypeSelect" @change="handleSelectChange" style="width: 90px">
+            <el-option label="周度" value="weekly" />
+            <el-option label="月度" value="monthly" />
+          </el-select>
+        </div>
+        <div>
+          <span class="font-medium mr-0.5">搜索词 </span>
+          <el-input
+            v-model="searchTermInp"
+            @keyup.enter="handleQueryChange"
+            :prefix-icon="Search"
+            placeholder="输入后回车查询"
+            clearable
+            @clear="handleSelectChange"
+            style="width: 240px" />
+        </div>
+        <div>
+          <span class="font-medium mr-0.5">ASIN </span>
+          <el-input
+            v-model="asinInp"
+            @keyup.enter="handleQueryChange"
+            :prefix-icon="Search"
+            placeholder="输入后回车查询"
+            clearable
+            @clear="handleSelectChange"
+            style="width: 180px" />
+        </div>
+        <div>
+          <span class="font-medium mr-0.5">报告日期 </span>
+
+          <el-config-provider :locale="enLocale">
+            <el-date-picker
+              v-if="reportTypeSelect === 'weekly'"
+              v-model="dateDimension"
+              type="week"
+              value-format="YYYY-MM-DD"
+              :format="`${date[0]} To ${date[1]}`"
+              :popper-options="{ placement: 'bottom-end' }"
+              :disabled-date="(time: Date) => time > new Date()"
+              :clearable="false" />
+            <el-date-picker
+              v-else
+              v-model="dateDimension"
+              type="month"
+              value-format="YYYY-MM"
+              :format="`${date[0]} To ${date[1]}`"
+              :popper-options="{ placement: 'bottom-end' }"
+              :disabled-date="(time: Date) => time > new Date()"
+              :clearable="false">
+              <template #default>
+                123
+              </template>
+            </el-date-picker>
+          </el-config-provider>
+        </div>
+      </div>
+      <div class="flex">
+        <el-button type="primary" plain @click="handleJump" :icon="TopRight">搜索词管理</el-button>
+        <el-button
+          type="success"
+          plain
+          @click="handleDownload"
+          :icon="Download"
+          round
+          :loading="downloadLoading"
+          :disabled="!tableData.length"
+          >下载表格
+        </el-button>
+        <el-button @click="refreshTable" :icon="Refresh" circle></el-button>
+      </div>
+    </div>
+    <!-- table -->
+    <el-card shadow="never" class="mt-5">
+      <div style="height: 795px; overflow: auto">
+        <el-table :data="tableData" height="795" stripe style="width: 100%">
+          <el-table-column fixed prop="searchTerm" label="搜索词" width="260">
+            <template #header>
+              <el-icon style="top: 2px; margin-right: 3px">
+                <Key />
+              </el-icon>
+              <span>搜索词</span>
+            </template>
+            <template #default="{ row }">
+              <el-link :underline="false" href="https://www.bilibili.com/" target="_blank" style="color: #0b3289"
+                >{{ row.searchTerm }}
+              </el-link>
+            </template>
+          </el-table-column>
+          <el-table-column prop="searchFrequencyRank" label="搜索词搜索排名" align="center" width="150">
+            <template #header>
+              <el-icon style="top: 2px; margin-right: 4px">
+                <Rank />
+              </el-icon>
+              <span>搜索词搜索排名</span>
+            </template>
+            <template #default="{ row }">
+              <span class="font-medium">{{ row.searchFrequencyRank }}</span>
+            </template>
+          </el-table-column>
+          <el-table-column prop="clickedAsin" label="Asin" align="center">
+            <template #header>
+              <el-icon style="top: 2px; margin-right: 5px">
+                <Goods />
+              </el-icon>
+              <span>Asin</span>
+            </template>
+            <template #default="{ row }">
+              <div class="font-medium" style="color: black">{{ row.clickedAsin }}</div>
+              <div class="text-sm text-left">
+                <el-tooltip class="box-item" effect="dark" :content="row.clickedItemName" placement="top-start">
+                  <div class="tooltip-text">
+                    <span class="font-medium mr-1">Title:</span>
+                    {{ row.clickedItemName }}
+                  </div>
+                </el-tooltip>
+              </div>
+            </template>
+          </el-table-column>
+          <!--<el-table-column prop="clickedItemName" label="标题">-->
+          <!--  <template #header>-->
+          <!--    <el-icon style="top: 2px; margin-right: 5px">-->
+          <!--      <Reading />-->
+          <!--    </el-icon>-->
+          <!--    <span>标题</span>-->
+          <!--  </template>-->
+          <!--</el-table-column>-->
+          <el-table-column prop="clickShareRank" label="点击分享率排名" align="center" width="150">
+            <template #header>
+              <el-icon style="top: 2px; margin-right: 4px">
+                <Medal />
+              </el-icon>
+              <span>点击分享率排名</span>
+            </template>
+            <template #default="{ row }">
+              <!--<span class="font-semibold">{{ row.clickShareRank }}</span>-->
+              <el-tag :style="getTagStyle(row.clickShareRank)">
+                {{ row.clickShareRank }}
+              </el-tag>
+            </template>
+          </el-table-column>
+          <el-table-column prop="clickShare" align="center" label="点击分享率">
+            <template #header>
+              <el-icon style="top: 2px; margin-right: 4px">
+                <Pointer />
+              </el-icon>
+              <span>点击分享率</span>
+            </template>
+            <template #default="{ row }">
+              <span class="font-semibold">{{ row.clickShare }}</span>
+            </template>
+          </el-table-column>
+          <el-table-column prop="conversionShare" align="center" label="转化分享率">
+            <template #header>
+              <el-icon style="top: 2px; margin-right: 5px">
+                <Switch />
+              </el-icon>
+              <span>转化分享率</span>
+            </template>
+            <template #default="{ row }">
+              <span class="font-semibold">{{ row.conversionShare }}</span>
+            </template>
+          </el-table-column>
+        </el-table>
+      </div>
+      <div class="mt-3.5 flex justify-end">
+        <el-pagination
+          v-model:current-page="currentPage"
+          v-model:page-size="pageSize"
+          :page-sizes="[10, 20, 30, 50, 100, 200]"
+          layout="sizes, prev, pager, next"
+          :total="total"
+          @change="handlePageChange" />
+      </div>
+    </el-card>
+  </el-card>
+</template>
+
+<style scoped>
+:deep(.el-divider__text.is-center.el-divider__text) {
+  background-color: #f8f8f8;
+}
+
+.tooltip-text {
+  display: -webkit-box;
+  -webkit-box-orient: vertical;
+  -webkit-line-clamp: 2;
+  overflow: hidden;
+  text-overflow: ellipsis;
+  white-space: normal;
+  line-height: 1.2em;
+  max-height: 2.4em;
+}
+</style>