SeoKeywordsRankListCopy.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587
  1. <template>
  2. <div>
  3. <a-row class="r1" style="margin: 20px">
  4. <a-col :span="24">
  5. <span class="t1">{{ siteinfo.length > 1 ? '请选择站点:' : '套餐:' }}</span>
  6. <select-site v-if="siteinfo.length > 1" @com-methods="changeSite" selectWidth="300px" />
  7. <span style="margin-left: 16px" class="t1"
  8. >{{ selectSiteInfo.planName && selectSiteInfo.planType !== 'STATIONCONSTRUCTION' ? selectSiteInfo.planName : '暂无SEO推广套餐~' }}
  9. <a-popover placement="bottom" v-show="selectSiteInfo.planName && selectSiteInfo.planType !== 'STATIONCONSTRUCTION'">
  10. <template #content>
  11. <span> 指定词数:{{ last(comprehenInfo.specifyKeywordNum) }}&nbsp; </span>
  12. <span> 关键词数:{{ last(comprehenInfo.longTailKeywordNum) }}&nbsp; </span>
  13. <span> 外链数:{{ last(comprehenInfo.outerLinkNum) }}&nbsp; </span>
  14. <span> 文章数:{{ last(comprehenInfo.articleNum) }} </span>
  15. </template>
  16. <img
  17. src="https://cutomer-static-bucket.s3.cn-northwest-1.amazonaws.com.cn/public/material/220803883dfy/document/changjianwenti_1676606434975.png"
  18. alt="issue"
  19. width="25px"
  20. height="25px"
  21. />
  22. </a-popover>
  23. </span>
  24. </a-col>
  25. </a-row>
  26. <a-row class="r2" type="flex" justify="center" align="middle">
  27. <a-col :span="6">
  28. <p class="t1"><img src="../../../assets/seo/zhidingci.svg" />指定词</p>
  29. <p class="t2" v-if="comprehenInfo.specifyKeywordNum == null"> - </p>
  30. <p class="t2" v-else>
  31. <span>{{ comprehenInfo.specifyKeywordNum }}</span>
  32. </p>
  33. </a-col>
  34. <a-col :span="6">
  35. <p class="t1"><img src="../../../assets/seo/changweici.svg" />关键词</p>
  36. <p class="t2" v-if="comprehenInfo.longTailKeywordNum == null"> - </p>
  37. <p class="t2" v-else>
  38. <span>{{ comprehenInfo.longTailKeywordNum }}</span>
  39. </p>
  40. </a-col>
  41. <a-col :span="6">
  42. <p class="t1"><img src="../../../assets/seo/lianjieshu.svg" />外链数</p>
  43. <p class="t2" v-if="comprehenInfo.outerLinkNum == null"> - </p>
  44. <p class="t2">
  45. <span>{{ comprehenInfo.outerLinkNum }}</span>
  46. </p>
  47. </a-col>
  48. <a-col :span="6">
  49. <p class="t1"><img src="../../../assets/seo/wenzhangshu.svg" />文章数</p>
  50. <p class="t2" v-if="comprehenInfo.articleNum == null"> - </p>
  51. <p class="t2">
  52. <span>{{ comprehenInfo.articleNum }}</span>
  53. </p>
  54. </a-col>
  55. </a-row>
  56. <a-row class="r3" type="flex" :gutter="16">
  57. <a-col :span="12">
  58. <div class="wrap">
  59. <p class="t1">指定词排名</p>
  60. <div class="content">
  61. <div class="d1"><img src="../../../assets/seo/NO1.svg" />1-10位</div>
  62. <div class="d2"
  63. ><a @click="getTableInfoRank(3, 1)">{{ filtr_null(rankInfo.appointKeyword.firstNum) }}</a
  64. >个
  65. </div>
  66. </div>
  67. <div class="content">
  68. <div class="d1"><img src="../../../assets/seo/NO2.svg" />11-30位</div>
  69. <div class="d2"
  70. ><a @click="getTableInfoRank(7, 1)">{{ filtr_null(rankInfo.appointKeyword.secondNum) }}</a
  71. >个
  72. </div>
  73. </div>
  74. <div class="content">
  75. <div class="d1"><img src="../../../assets/seo/NO3.svg" />31-100位</div>
  76. <div class="d2"
  77. ><a @click="getTableInfoRank(8, 1)">{{ filtr_null(rankInfo.appointKeyword.thirdType) }}</a
  78. >个
  79. </div>
  80. </div>
  81. </div>
  82. </a-col>
  83. <a-col :span="12">
  84. <div class="wrap">
  85. <p class="t1">关键词排名</p>
  86. <div class="content">
  87. <div class="d1"><img src="../../../assets/seo/NO1.svg" />1-10位</div>
  88. <div class="d2">
  89. <a @click="longGetTableInfoRank(3)">{{ filtr_null(rankInfo.longTailKeyword.firstNum) }}</a
  90. >个
  91. </div>
  92. </div>
  93. <div class="content">
  94. <div class="d1"><img src="../../../assets/seo/NO2.svg" />11-30位</div>
  95. <div class="d2"
  96. ><a @click="longGetTableInfoRank(7)">{{ filtr_null(rankInfo.longTailKeyword.secondNum) }}</a
  97. >个
  98. </div>
  99. </div>
  100. <div class="content">
  101. <div class="d1"><img src="../../../assets/seo/NO3.svg" />31-100位</div>
  102. <div class="d2"
  103. ><a @click="longGetTableInfoRank(8)">{{ filtr_null(rankInfo.longTailKeyword.thirdType) }}</a
  104. >个
  105. </div>
  106. </div>
  107. </div>
  108. </a-col>
  109. </a-row>
  110. <!--引用表格-->
  111. <BasicTable @register="registerTable" :rowSelection="rowSelection">
  112. <!--插槽:table标题-->
  113. <template #tableTitle>
  114. <a-button type="primary" preIcon="ant-design:export-outlined" @click="onExportXls"> 导出</a-button>
  115. </template>
  116. <!--操作栏-->
  117. <template #action="{ record }">
  118. <TableAction :actions="getTableAction(record)" :dropDownActions="getDropDownAction(record)" />
  119. </template>
  120. <!--字段回显插槽-->
  121. <template #bodyCell="{ column, record, index, text }"> </template>
  122. </BasicTable>
  123. <!-- 表单区域 -->
  124. <SeoKpiStatisticsModal @register="registerModal" @success="handleSuccess" />
  125. </div>
  126. </template>
  127. <script lang="ts" name="serp-seoKpiStatistics" setup>
  128. import { ref, reactive, computed, unref, onMounted } from 'vue';
  129. import { BasicTable, useTable, TableAction } from '/src/components/Table';
  130. import { useModal } from '/src/components/Modal';
  131. import { useListPage } from '/src/hooks/system/useListPage';
  132. import { columns, searchFormSchema, superQuerySchema } from './SeoKeywordsRank.data';
  133. import { list, deleteOne, batchDelete, getImportUrl, getExportUrl, getAllSites } from './SeoKeywordsRank.api';
  134. import { downloadFile } from '/src/utils/common/renderUtils';
  135. import { useUserStore } from '/src/store/modules/user';
  136. import selectSite from '@/components/Adweb/selectSite.vue';
  137. import { getAction } from '@/api/manage/manage';
  138. const queryParam = reactive<any>({});
  139. const checkedKeys = ref<Array<string | number>>([]);
  140. const userStore = useUserStore();
  141. //注册model
  142. const [registerModal, { openModal }] = useModal();
  143. //注册table数据
  144. const { prefixCls, tableContext, onExportXls, onImportXls } = useListPage({
  145. tableProps: {
  146. title: 'seo_keywords_rank',
  147. api: list,
  148. columns,
  149. canResize: false,
  150. formConfig: {
  151. //labelWidth: 120,
  152. schemas: searchFormSchema,
  153. autoSubmitOnEnter: true,
  154. showAdvancedButton: true,
  155. fieldMapToNumber: [],
  156. fieldMapToTime: [],
  157. },
  158. actionColumn: {
  159. width: 120,
  160. fixed: 'right',
  161. },
  162. beforeFetch: (params) => {
  163. return Object.assign(params, queryParam);
  164. },
  165. },
  166. exportConfig: {
  167. name: 'seo_keywords_rank',
  168. url: getExportUrl,
  169. params: queryParam,
  170. },
  171. importConfig: {
  172. url: getImportUrl,
  173. success: handleSuccess,
  174. },
  175. });
  176. const [registerTable, { reload }, { rowSelection, selectedRowKeys }] = tableContext;
  177. // 高级查询配置
  178. const superQueryConfig = reactive(superQuerySchema);
  179. const siteinfo = ref([]);
  180. const selectSiteInfo = ref({ planId: '', planName: '', planType: '', subscriptionId: '' });
  181. const siteCode = ref('');
  182. let comprehenInfo = ref({ specifyKeywordNum: 0, longTailKeywordNum: 0, outerLinkNum: 0, articleNum: 0 });
  183. const keywords = ref('');
  184. const rankInfo = ref({
  185. appointKeyword: { firstNum: 0, secondNum: 0, thirdType: '' },
  186. longTailKeyword: { firstNum: 0, secondNum: 0, thirdType: '' },
  187. });
  188. const dataSource = ref([]);
  189. const ipagination = ref({ current: 1, total: 0 });
  190. const longDataSource = ref([]);
  191. const longIpagination = ref({ current: 1, total: 0 });
  192. const show = ref();
  193. const keywordType = ref();
  194. const longShow = ref();
  195. // const loading = ref(false);
  196. /**
  197. * 高级查询事件
  198. */
  199. function handleSuperQuery(params) {
  200. Object.keys(params).map((k) => {
  201. queryParam[k] = params[k];
  202. });
  203. reload();
  204. }
  205. /**
  206. * 新增事件
  207. */
  208. function handleAdd() {
  209. openModal(true, {
  210. isUpdate: false,
  211. showFooter: true,
  212. });
  213. }
  214. /**
  215. * 编辑事件
  216. */
  217. function handleEdit(record: Recordable) {
  218. openModal(true, {
  219. record,
  220. isUpdate: true,
  221. showFooter: true,
  222. });
  223. }
  224. /**
  225. * 详情
  226. */
  227. function handleDetail(record: Recordable) {
  228. openModal(true, {
  229. record,
  230. isUpdate: true,
  231. showFooter: false,
  232. });
  233. }
  234. /**
  235. * 删除事件
  236. */
  237. async function handleDelete(record) {
  238. await deleteOne({ id: record.id }, handleSuccess);
  239. }
  240. /**
  241. * 批量删除事件
  242. */
  243. async function batchHandleDelete() {
  244. await batchDelete({ ids: selectedRowKeys.value }, handleSuccess);
  245. }
  246. /**
  247. * 成功回调
  248. */
  249. function handleSuccess() {
  250. (selectedRowKeys.value = []) && reload();
  251. }
  252. /**
  253. * 操作栏
  254. */
  255. function getTableAction(record) {
  256. return [
  257. {
  258. label: '去查询',
  259. onClick: handleEdit.bind(null, record),
  260. },
  261. ];
  262. }
  263. //切换站点
  264. function changeSite(value, e) {
  265. getSiteInfo();
  266. selectSiteInfo.value = e.info;
  267. siteCode.value = value;
  268. queryParam.reachStandard = undefined;
  269. keywords.value = '';
  270. localStorage.setItem('siteCode', value);
  271. resetAllInfo();
  272. getAllInfo();
  273. }
  274. //先执行获取site信息,只有在获取到siteCode之后,才能执行以下方法
  275. function getSiteInfo() {
  276. getAllSites().then((res) => {
  277. siteinfo.value = res;
  278. if (res.length > 0) {
  279. let isInSite = false;
  280. for (let i in res) {
  281. if (localStorage.getItem('siteCode') !== null && res[i].code === localStorage.getItem('siteCode')) {
  282. isInSite = true;
  283. selectSiteInfo.value = res[i];
  284. }
  285. }
  286. if (localStorage.getItem('siteCode') !== null && isInSite) {
  287. siteCode.value = localStorage.getItem('siteCode');
  288. } else {
  289. siteCode.value = res[0].code;
  290. selectSiteInfo.value = res[0];
  291. localStorage.setItem('siteCode', res[0].code);
  292. }
  293. }
  294. resetAllInfo();
  295. getAllInfo();
  296. });
  297. }
  298. function resetAllInfo() {
  299. comprehenInfo.value = { specifyKeywordNum: 0, longTailKeywordNum: 0, outerLinkNum: 0, articleNum: 0 };
  300. rankInfo.value = { appointKeyword: {}, longTailKeyword: {} };
  301. dataSource.value = [];
  302. ipagination.value = { current: 1, total: 0 };
  303. longDataSource.value = [];
  304. longIpagination.value = { current: 1, total: 0 };
  305. }
  306. function getAllInfo() {
  307. let d = {
  308. siteCode: siteCode.value,
  309. subscriptionId: selectSiteInfo.value.subscriptionId ?? '',
  310. };
  311. getAction('/seo/seoKeywordsRank/comprehensiveInfo', d)
  312. .then((res) => {
  313. if (res.code == 200) {
  314. comprehenInfo.value = res.result;
  315. }
  316. })
  317. .finally(() => {
  318. getRankInfo();
  319. setTableQuery();
  320. // longSetTableQuery()
  321. });
  322. }
  323. function first(value) {
  324. if (value && value != null) {
  325. let a = value.split('/');
  326. return a[0];
  327. } else {
  328. return '- ';
  329. }
  330. }
  331. function last(value) {
  332. if (value && value != null) {
  333. let a = value.split('/');
  334. return a[1];
  335. } else {
  336. return ' -';
  337. }
  338. }
  339. function filtr_null(value) {
  340. if (value != null) {
  341. return value;
  342. } else {
  343. return '- ';
  344. }
  345. }
  346. //点击排名数量事件
  347. function getTableInfoRank(d, keywordType) {
  348. show.value = d;
  349. queryParam.reachStandard = undefined;
  350. if (keywordType) {
  351. keywordType.value = keywordType;
  352. }
  353. setTableQuery();
  354. }
  355. //由于页面的业务关联很多,故将所有的查询条件封装
  356. function setTableQuery() {
  357. queryParam.siteCode = siteCode.value;
  358. if (show.value == 3) {
  359. queryParam.rankStart = '1';
  360. queryParam.rankEnd = '10';
  361. } else if (show.value == 4) {
  362. queryParam.rankStart = '1';
  363. queryParam.rankEnd = '30';
  364. } else if (show.value == 5) {
  365. queryParam.rankStart = '1';
  366. queryParam.rankEnd = '50';
  367. } else if (show.value == 6) {
  368. queryParam.rankStart = '1';
  369. queryParam.rankEnd = '100';
  370. } else if (show.value == 7) {
  371. queryParam.rankStart = '11';
  372. queryParam.rankEnd = '30';
  373. } else if (show.value == 8) {
  374. queryParam.rankStart = '31';
  375. queryParam.rankEnd = '100';
  376. } else {
  377. queryParam.rankStart = undefined;
  378. queryParam.rankEnd = undefined;
  379. }
  380. queryParam.keywordType = keywordType.value;
  381. queryParam.keywords = keywords.value;
  382. // loadData(1)
  383. }
  384. // function loadData(arg) {
  385. // let a = []
  386. // if (arg === 1) {
  387. // ipagination.value.current = 1
  388. // }
  389. // var params = getQueryParams()//查询条件
  390. // params.subscriptionId = selectSiteInfo.value.subscriptionId;
  391. // params.keywordType = 1;
  392. // loading.value = true
  393. // getAction(url.list, params).then((res) => {
  394. // if (res.success) {
  395. // let alldata = res.result.records
  396. // if(alldata.length > 0){
  397. // for(let i in alldata){
  398. // let rankInfo = alldata[i].rankInfo;
  399. // a.push({
  400. // id:alldata[i].id,
  401. // keywords:alldata[i].keywords,
  402. // keywordType:alldata[i].keywordType,
  403. // keywordsLength:alldata[i].keywordsLength,
  404. // rankInfo:rankInfo,
  405. // ...rankInfo
  406. // })
  407. // }
  408. // dataSource = a;
  409. // getTableColums()
  410. // ipagination.total = res.result.total
  411. // }else{
  412. // dataSource = a;
  413. // ipagination.total = 0
  414. // }
  415. //
  416. // }
  417. // if (res.code === 510) {
  418. // $message.warning(res.message)
  419. // }
  420. // loading = false
  421. // })
  422. // },
  423. //点击排名数量事件
  424. function longGetTableInfoRank(d) {
  425. longShow.value = d;
  426. // longQueryParam.reachStandard = undefined
  427. // longSetTableQuery()
  428. }
  429. //获取排行榜数据
  430. function getRankInfo() {
  431. let d = {
  432. siteCode: siteCode.value,
  433. subscriptionId: selectSiteInfo.value.subscriptionId ? selectSiteInfo.value.subscriptionId : '',
  434. };
  435. getAction('/seo/seoKeywordsRank/getRankInfo', d).then((res) => {
  436. console.log(res.code == 200);
  437. if (res.code == 200) {
  438. rankInfo.value.appointKeyword = res.result.appointKeyword;
  439. rankInfo.value.longTailKeyword = res.result.longTailKeyword;
  440. console.log(rankInfo.value.a);
  441. }
  442. });
  443. }
  444. //动态处理表格列
  445. function getTableColums() {
  446. that.columns = [
  447. {
  448. title: '序号',
  449. dataIndex: 'rowIndex',
  450. key: 'rowIndex',
  451. width: 40,
  452. align: 'center',
  453. fixed: 'left',
  454. scopedSlots: { customRender: 'dataNo' },
  455. },
  456. {
  457. title: '关键词',
  458. dataIndex: 'keywords',
  459. key: 'keywords',
  460. fixed: 'left',
  461. scopedSlots: { customRender: 'keywords' },
  462. width: 350,
  463. },
  464. ];
  465. let obj = that.dataSource[0].rankInfo;
  466. let dateInfo = Object.keys(obj);
  467. dateInfo.sort().reverse();
  468. for (let i in dateInfo) {
  469. that.columns.push({
  470. title: dateInfo[i],
  471. dataIndex: dateInfo[i],
  472. key: dateInfo[i],
  473. scopedSlots: { customRender: 'tag' },
  474. align: 'center',
  475. });
  476. }
  477. that.columns.push({
  478. title: '操作',
  479. dataIndex: 'action',
  480. key: 'action',
  481. align: 'center',
  482. scopedSlots: { customRender: 'action' },
  483. fixed: 'right',
  484. width: 50,
  485. });
  486. }
  487. onMounted(() => {
  488. getSiteInfo();
  489. });
  490. </script>
  491. <style scoped>
  492. :deep(.ant-picker),
  493. :deep(.ant-input-number) {
  494. width: 100%;
  495. }
  496. .r2 {
  497. background: #fff;
  498. padding: 30px;
  499. margin: 10px;
  500. .ant-col {
  501. text-align: center;
  502. color: #000;
  503. img {
  504. width: 13px;
  505. margin: -3px 5px 0 0;
  506. }
  507. .t1 {
  508. font-size: 14px;
  509. }
  510. .t2 {
  511. font-size: 27px;
  512. margin-bottom: 0;
  513. font-weight: 500;
  514. }
  515. }
  516. }
  517. .r3 {
  518. padding-left: 10px;
  519. padding-right: 10px;
  520. .wrap {
  521. padding: 20px;
  522. background: #fff;
  523. margin-bottom: 10px;
  524. .t1 {
  525. color: #000;
  526. font-size: 14px;
  527. font-weight: 500;
  528. }
  529. .content {
  530. display: block;
  531. overflow: hidden;
  532. line-height: 1;
  533. border-bottom: 1px solid #ddd;
  534. padding: 20px 0;
  535. &:last-child {
  536. border-bottom: none;
  537. padding-bottom: 0;
  538. }
  539. .d1 {
  540. float: left;
  541. width: 50%;
  542. line-height: 25px;
  543. color: #000;
  544. img {
  545. width: 15px;
  546. margin: -4px 5px 0 0;
  547. }
  548. }
  549. .d2 {
  550. float: right;
  551. width: 50%;
  552. text-align: right;
  553. &.d3 {
  554. span {
  555. color: #000;
  556. }
  557. }
  558. a {
  559. font-size: 25px;
  560. font-weight: 500;
  561. }
  562. }
  563. }
  564. }
  565. }
  566. </style>