AdwebSiteList.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445
  1. <template>
  2. <div class="p-2">
  3. <!--查询区域-->
  4. <div class="jeecg-basic-table-form-container">
  5. <a-form ref="formRef" @keyup.enter.native="searchQuery" :model="queryParam" :label-col="labelCol" :wrapper-col="wrapperCol">
  6. <a-row :gutter="24">
  7. <a-col :xxl="6" :xl="7">
  8. <a-form-item name="name" label="站点名称">
  9. <j-input placeholder="请输入站点名称" v-model:value="queryParam.name" allow-clear />
  10. </a-form-item>
  11. </a-col>
  12. <a-col :xxl="6" :xl="7">
  13. <a-form-item name="domain" label="站点域名">
  14. <j-input placeholder="请输入站点域名" v-model:value="queryParam.domain" allow-clear />
  15. </a-form-item>
  16. </a-col>
  17. <a-col :xxl="6" :xl="8">
  18. <a-form-item label="发布状态">
  19. <j-search-select placeholder="请选择发布状态" v-model:value="queryParam.status" dict="site_status" />
  20. </a-form-item>
  21. </a-col>
  22. <template v-if="toggleSearchStatus">
  23. <a-col :xxl="6" :xl="7">
  24. <a-form-item label="渠道商">
  25. <a-select placeholder="请选择渠道商" v-model:value="queryParam.channelProviderId" showSearch allowClear :filterOption="filterOption">
  26. <a-select-option v-for="data in channelProviderList" :key="data.username" :value="data.workNo">
  27. {{ data.username }}
  28. </a-select-option>
  29. </a-select>
  30. </a-form-item>
  31. </a-col>
  32. <!-- <a-col :xxl="12" :xl="10">-->
  33. <!-- <a-form-item label="创建时间">-->
  34. <!-- <j-range-date placeholder="请选择开始日期" class="query-group-cust" v-model:value="queryParam.ctime" />-->
  35. <!-- </a-form-item>-->
  36. <!-- </a-col>-->
  37. <!-- <a-col :xl="10" :lg="11" :md="12" :sm="24">-->
  38. <!-- <a-form-item label="到期时间">-->
  39. <!-- <j-range-date placeholder="请选择开始日期" class="query-group-cust" v-model:value="queryParam.etime" />-->
  40. <!-- </a-form-item>-->
  41. <!-- </a-col>-->
  42. </template>
  43. <a-col :xl="6" :lg="7" :md="8" :sm="24">
  44. <span style="float: left; overflow: hidden" class="table-page-search-submitButtons">
  45. <a-col :lg="6">
  46. <a-button type="primary" preIcon="ant-design:search-outlined" @click="searchQuery">查询</a-button>
  47. <a-button type="primary" preIcon="ant-design:reload-outlined" @click="searchReset" style="margin-left: 8px">重置</a-button>
  48. <a @click="toggleSearchStatus = !toggleSearchStatus" style="margin-left: 8px">
  49. {{ toggleSearchStatus ? '收起' : '展开' }}
  50. <Icon :icon="toggleSearchStatus ? 'ant-design:up-outlined' : 'ant-design:down-outlined'" />
  51. </a>
  52. </a-col>
  53. </span>
  54. </a-col>
  55. </a-row>
  56. </a-form>
  57. </div>
  58. <!--引用表格-->
  59. <BasicTable @register="registerTable" :rowSelection="rowSelection">
  60. <!--插槽:table标题-->
  61. <template #tableTitle>
  62. <a-button type="primary" @click="handleAdd" preIcon="ant-design:plus-outlined"> 新增</a-button>
  63. <a-button type="primary" preIcon="ant-design:export-outlined" @click="onExportXls"> 导出</a-button>
  64. <j-upload-button type="primary" preIcon="ant-design:import-outlined" @click="onImportXls">导入</j-upload-button>
  65. <a-dropdown v-if="selectedRowKeys.length > 0">
  66. <template #overlay>
  67. <a-menu>
  68. <a-menu-item key="1" @click="batchHandleDelete">
  69. <Icon icon="ant-design:delete-outlined" />
  70. 删除
  71. </a-menu-item>
  72. </a-menu>
  73. </template>
  74. <a-button
  75. >批量操作
  76. <Icon icon="mdi:chevron-down" />
  77. </a-button>
  78. </a-dropdown>
  79. <!-- 高级查询 -->
  80. <super-query :config="superQueryConfig" @search="handleSuperQuery" />
  81. </template>
  82. <!--操作栏-->
  83. <template #action="{ record }">
  84. <TableAction :actions="getTableAction(record)" :dropDownActions="getDropDownAction(record)" />
  85. </template>
  86. <template #bodyCell="{ column, record, index, text }">
  87. <div v-if="column.dataIndex == 'name'">
  88. <a :href="record.domain" target="_blank">{{ text }}</a>
  89. </div>
  90. <div v-if="column.dataIndex == 'domain'">
  91. <a :href="record.domain" target="_blank">{{ text }}</a>
  92. </div>
  93. <div v-if="column.dataIndex == 'subscribePlan'">
  94. <template v-if="text === null || text === ''"> 无 </template>
  95. <template v-else>
  96. <a-popover>
  97. <template #content>
  98. <a-descriptions bordered :column="{ xxl: 1, xl: 1, lg: 1, md: 1, sm: 1, xs: 1 }" title="SEO套餐详情" size="default">
  99. <a-descriptions-item label="套餐名">
  100. {{ text.planName }}
  101. </a-descriptions-item>
  102. <a-descriptions-item label="关键词"> {{ text.keywordCount }}个 </a-descriptions-item>
  103. <a-descriptions-item label="服务时间">
  104. <template v-if="text.serviceTime != 0"> {{ text.serviceTime }}个月 </template>
  105. <template v-else> 长期 </template>
  106. </a-descriptions-item>
  107. <a-descriptions-item label="价格"> {{ text.price }}元 </a-descriptions-item>
  108. </a-descriptions>
  109. </template>
  110. <a-tag color="blue">
  111. {{ text.planName }}
  112. </a-tag>
  113. </a-popover>
  114. </template>
  115. </div>
  116. <div v-if="column.dataIndex == 'status_dictText'">
  117. <template v-if="record.status == 1">
  118. <a-tag color="blue">
  119. {{ text }}
  120. </a-tag>
  121. </template>
  122. <template v-else-if="record.status == 2">
  123. <a-tag color="green">
  124. {{ text }}
  125. </a-tag>
  126. </template>
  127. <template v-else>
  128. <a-tag color="red">
  129. {{ text }}
  130. </a-tag>
  131. </template>
  132. </div>
  133. <div v-if="column.dataIndex == 'siteCurrentStep'">
  134. <span v-if="text === null || text === undefined || text === ''"> - </span>
  135. <template v-else>
  136. <a @click="handleProcess(record, 'site')"> <span class="circle" style="background-color: #7194f8"></span>{{ text }} </a>
  137. </template>
  138. </div>
  139. <div v-if="column.dataIndex == 'completeProcess'">
  140. <a-progress :stroke-color="{ '0%': '#108ee9', '100%': '#87d068' }" :percent="record.completeProcess" />
  141. </div>
  142. </template>
  143. </BasicTable>
  144. <!-- 表单区域 -->
  145. <AdwebSiteModal ref="registerModal" @success="handleSuccess" />
  146. <site-set-enquiry ref="siteSetEnquiryRef" @success="reload" />
  147. <!-- 自动化嵌入GTM 代码 -->
  148. <gtm-add ref="gtmRef" />
  149. <!--SEO流程-->
  150. <seo-process ref="seoProcessRef" :visible="seoProcessVisible" :title="processTitle" @close="closeProcess" @reload="reload" />
  151. </div>
  152. </template>
  153. <script lang="ts" name="adweb-adwebSite" setup>
  154. import { computed, onBeforeMount, reactive, ref } from 'vue';
  155. import { getAction } from '/@/api/manage/manage';
  156. import { BasicTable, TableAction } from '/src/components/Table';
  157. import { useListPage } from '/src/hooks/system/useListPage';
  158. import { columns, superQuerySchema } from './AdwebSite.data';
  159. import { batchDelete, deleteOne, getExportUrl, getImportUrl, list, gtmListUrl, saveOrUpdate } from './AdwebSite.api';
  160. import AdwebSiteModal from './components/AdwebSiteModal.vue';
  161. import { useUserStore } from '/src/store/modules/user';
  162. import SeoProcess from '@/views/adweb/site/components/SeoProcess.vue';
  163. import JInput from '@/components/Form/src/jeecg/components/JInput.vue';
  164. import JSearchSelect from '@/components/Form/src/jeecg/components/JSearchSelect.vue';
  165. import GtmAdd from '@/views/adweb/site/components/GtmAdd.vue';
  166. import { RoleEnum } from '@/enums/roleEnum';
  167. import { useMessage } from '@/hooks/web/useMessage';
  168. import SiteSetEnquiry from '@/views/adweb/site/components/SiteSetEnquiry.vue';
  169. const formRef = ref();
  170. const queryParam = reactive<any>({});
  171. const toggleSearchStatus = ref<boolean>(false);
  172. const registerModal = ref();
  173. const userStore = useUserStore();
  174. const processTitle = ref('');
  175. const seoProcessVisible = ref(false);
  176. const seoProcessRef = ref();
  177. const gtmRef = ref();
  178. const siteSetEnquiryRef = ref();
  179. const isAdmin = computed(() => userStore.getRoleList.includes(RoleEnum.ADMIN));
  180. const { createMessage } = useMessage();
  181. let channelProviderList = reactive([{ username: '', workNo: '' }]);
  182. //注册table数据
  183. const { prefixCls, tableContext, onExportXls, onImportXls } = useListPage({
  184. tableProps: {
  185. title: 'adweb站点配置表单',
  186. api: list,
  187. columns,
  188. canResize: false,
  189. useSearchForm: true,
  190. showTableSetting: true,
  191. actionColumn: {
  192. width: 120,
  193. fixed: 'right',
  194. },
  195. beforeFetch: (params) => {
  196. return Object.assign(params, queryParam);
  197. },
  198. },
  199. exportConfig: {
  200. name: 'adweb站点配置表单',
  201. url: getExportUrl,
  202. params: queryParam,
  203. },
  204. importConfig: {
  205. url: getImportUrl,
  206. success: handleSuccess,
  207. },
  208. });
  209. const [registerTable, { reload, collapseAll, updateTableDataRecord, findTableDataRecord, getDataSource }, { rowSelection, selectedRowKeys }] =
  210. tableContext;
  211. const labelCol = reactive({
  212. xs: 24,
  213. sm: 4,
  214. xl: 6,
  215. xxl: 4,
  216. });
  217. const wrapperCol = reactive({
  218. xs: 24,
  219. sm: 20,
  220. });
  221. // 高级查询配置
  222. const superQueryConfig = reactive(superQuerySchema);
  223. //渠道商搜索
  224. const filterOption = (input: string, option: any) => {
  225. console.log(option, 'optionoption');
  226. return option.key.toLowerCase().indexOf(input.toLowerCase()) >= 0;
  227. };
  228. onBeforeMount(() => {
  229. getChannelProvider();
  230. });
  231. function getChannelProvider() {
  232. getAction('/adweb/system/getChannelProvider', null).then((res) => {
  233. channelProviderList = res.result;
  234. });
  235. }
  236. /**
  237. * 高级查询事件
  238. */
  239. function handleSuperQuery(params) {
  240. Object.keys(params).map((k) => {
  241. queryParam[k] = params[k];
  242. });
  243. searchQuery();
  244. }
  245. /**
  246. * 新增事件
  247. */
  248. function handleAdd() {
  249. registerModal.value.disableSubmit = false;
  250. registerModal.value.add();
  251. }
  252. /**
  253. * 编辑事件
  254. */
  255. function handleEdit(record: Recordable) {
  256. registerModal.value.disableSubmit = false;
  257. registerModal.value.edit(record);
  258. }
  259. /**
  260. * 详情
  261. */
  262. function handleDetail(record: Recordable) {
  263. registerModal.value.disableSubmit = true;
  264. registerModal.value.edit(record);
  265. }
  266. /**
  267. * 删除事件
  268. */
  269. async function handleDelete(record) {
  270. await deleteOne({ id: record.id }, handleSuccess);
  271. }
  272. async function siteDeleteStatus(record) {
  273. record.status = 0;
  274. await saveOrUpdate(record, true)
  275. .then((res) => {
  276. if (res.success) {
  277. createMessage.success(res.message);
  278. } else {
  279. createMessage.warning(res.message);
  280. }
  281. reload();
  282. })
  283. .finally(() => {});
  284. }
  285. /**
  286. * 批量删除事件
  287. */
  288. async function batchHandleDelete() {
  289. await batchDelete({ ids: selectedRowKeys.value }, handleSuccess);
  290. }
  291. /**
  292. * 成功回调
  293. */
  294. function handleSuccess() {
  295. (selectedRowKeys.value = []) && reload();
  296. }
  297. /**
  298. * 操作栏
  299. */
  300. function getTableAction(record) {
  301. return [
  302. {
  303. label: '编辑网站',
  304. onClick: handleEdit.bind(null, record),
  305. },
  306. ];
  307. }
  308. /**
  309. * 下拉操作栏
  310. */
  311. function getDropDownAction(record) {
  312. let dropDown = [
  313. {
  314. label: '详情',
  315. onClick: handleDetail.bind(null, record),
  316. },
  317. {
  318. label: '询盘设置',
  319. onClick: setEnquiry.bind(null, record),
  320. },
  321. ];
  322. // 站点不是删除状态
  323. if (record.status != 0) {
  324. dropDown.push({
  325. label: '站点删除',
  326. onClick: siteDeleteStatus.bind(null, record),
  327. });
  328. }
  329. // 站点已发布,并且是正式域名时可以添加GTM 代码
  330. if (record.isDomain === 1 && record.status === 1) {
  331. dropDown.push({
  332. label: '跟踪代码',
  333. onClick: addGtm.bind(null, record),
  334. });
  335. }
  336. if (isAdmin.value) {
  337. dropDown.push({
  338. label: '资源删除',
  339. popConfirm: {
  340. title: '该站点将从系统中删除,站点资源将会保留10天,是否确认删除',
  341. confirm: handleDelete.bind(null, record),
  342. placement: 'topLeft',
  343. },
  344. });
  345. }
  346. return dropDown;
  347. }
  348. // 添加gtm
  349. function addGtm(r) {
  350. gtmRef.value.init(r);
  351. }
  352. // 询盘设置
  353. function setEnquiry(r) {
  354. siteSetEnquiryRef.value.init(r);
  355. }
  356. /**
  357. * 查询
  358. */
  359. function searchQuery() {
  360. reload();
  361. }
  362. /**
  363. * 重置
  364. */
  365. function searchReset() {
  366. formRef.value.resetFields();
  367. selectedRowKeys.value = [];
  368. //刷新数据
  369. reload();
  370. }
  371. function handleProcess(record, type) {
  372. seoProcessVisible.value = true;
  373. processTitle.value = record.name;
  374. seoProcessRef.value.init(record, type);
  375. seoProcessRef.value.isCustomer = true;
  376. }
  377. function closeProcess() {
  378. seoProcessVisible.value = false;
  379. }
  380. </script>
  381. <style lang="less" scoped>
  382. .jeecg-basic-table-form-container {
  383. padding: 0;
  384. .table-page-search-submitButtons {
  385. display: block;
  386. margin-bottom: 24px;
  387. white-space: nowrap;
  388. }
  389. .query-group-cust {
  390. min-width: 100px !important;
  391. }
  392. .query-group-split-cust {
  393. width: 30px;
  394. display: inline-block;
  395. text-align: center;
  396. }
  397. .ant-form-item:not(.ant-form-item-with-help) {
  398. margin-bottom: 16px;
  399. height: 32px;
  400. }
  401. :deep(.ant-picker),
  402. :deep(.ant-input-number) {
  403. width: 100%;
  404. }
  405. }
  406. </style>