AdwebThemeList.vue 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345
  1. <template>
  2. <a-row>
  3. <a-col class="theme-wrap">
  4. <a-col class="tag-wrap" />
  5. <a-list
  6. :grid="{ gutter: 16, column: 4 }"
  7. :data-source="dataSource"
  8. :loading="loading"
  9. class="j-table-force-nowrap template_list"
  10. :style="{ height: '100%' }"
  11. >
  12. <template #loadMore>
  13. <div v-if="!loading && !moreLoading" :style="{ textAlign: 'center', marginTop: '12px', height: '32px', lineHeight: '32px' }">
  14. <a-button @click="onLoadMore">loading more</a-button>
  15. </div>
  16. </template>
  17. <template #renderItem="{ item, index }">
  18. <a-list-item>
  19. <a-card :title="item.name" class="text-center">
  20. <div class="item" @mouseover="hoverThis">
  21. <div class="Pic">
  22. <a class="img">
  23. <img v-if="checkUrl(item.images)" :key="item.code" :alt="item.name" :src="domainUrl + '/sys/common/static/' + item.images" />
  24. <img v-else alt="item.name" :src="domainUrl + '/sys/common/static/' + item.images" />
  25. </a>
  26. <div class="Content">
  27. <a-button class="fl" block @click="visit(item, index)">模板预览</a-button>
  28. <a-button type="primary" block class="fr" @click="createSite(item)">创建站点</a-button>
  29. </div>
  30. </div>
  31. <div class="clear"></div>
  32. </div>
  33. </a-card>
  34. </a-list-item>
  35. </template>
  36. </a-list>
  37. <adweb-create-site ref="modelRef" />
  38. <a-modal v-model:open="visible" width="80%" @cancel="visitImageCancel" :footer="null" class="self_buy_modal">
  39. <template #title>
  40. <div>
  41. <div>
  42. <p class="ant-modal-title">
  43. 模板预览
  44. <template
  45. v-if="
  46. (userRole.indexOf('admin') > -1 ||
  47. userRole.indexOf('adweb_admin') > -1 ||
  48. userRole.indexOf('adweb_seo_manager') > -1 ||
  49. userRole.indexOf('adweb_site_manager') > -1) &&
  50. modelText != null &&
  51. modelText != ''
  52. "
  53. >
  54. <a :href="modelText" target="_blank" style="font-size: 14px; text-decoration: underline"><span>购买链接</span></a>
  55. </template>
  56. </p>
  57. </div>
  58. </div>
  59. </template>
  60. <img alt="example" style="width: 100%" :src="theImage" />
  61. </a-modal>
  62. </a-col>
  63. </a-row>
  64. </template>
  65. <script lang="ts" setup name="AdwebThemeCList">
  66. import '/@/assets/less/TableExpand.less';
  67. import '/@/assets/less/common.less';
  68. import { getAction } from '/@/api/manage/manage';
  69. import $ from 'jquery';
  70. import { nextTick, onMounted, reactive, ref } from 'vue';
  71. import { useMessage } from '@/hooks/web/useMessage';
  72. import AdwebCreateSite from '@/views/adweb/theme/components/AdwebCreateSite.vue';
  73. const url = reactive({
  74. list: '/adweb/adwebTheme/list',
  75. delete: '/adweb/adwebThemeC/delete',
  76. deleteBatch: '/adweb/adwebThemeC/deleteBatch',
  77. createSiteUrl: 'siteManage/createSite',
  78. addSiteLanguageUrl: 'siteManage/addSiteLanguage',
  79. getTemplateTagsUrl: '/adweb/adwebThemeC/getTemplateTags',
  80. wpSiteCreateUrl: 'wpWebSite/addWebsite',
  81. getAdwebVipUrl: 'adweb/adwebSiteManage/getAdwebVip',
  82. });
  83. const domainUrl = ref(window._CONFIG.domianURL);
  84. const userRole = ref('');
  85. const modelText = ref('');
  86. const theImage = ref('');
  87. let dataSource = reactive([
  88. {
  89. createBy: '',
  90. createTime: '',
  91. descirbe: '',
  92. domain: '',
  93. id: '',
  94. images: '',
  95. name: '',
  96. putaway: '',
  97. runStatus: null,
  98. templateColorId: '',
  99. templatePrice: 0,
  100. templateTags: '',
  101. updateBy: null,
  102. updateTime: null,
  103. visible: true,
  104. },
  105. ]);
  106. const loading = ref(true);
  107. const moreLoading = ref(false);
  108. const modelRef = ref();
  109. const visible = ref(false);
  110. const pageNo = ref(1);
  111. const { createMessage } = useMessage();
  112. onMounted(() => {
  113. getTemplateList();
  114. });
  115. function getTemplateList() {
  116. loading.value = true;
  117. let listUrl = `${url.list}` + '?pageNo=' + pageNo.value + '&pageSize=8&templateTags=' + 22;
  118. getAction(listUrl, null).then(function (res) {
  119. if (res.code == 200) {
  120. loading.value = false;
  121. // 第一页数据直接赋值数组,后续合并数组
  122. if (res.result.current === 1) {
  123. dataSource = res.result.records.map((item) => ({
  124. ...item,
  125. visible: false,
  126. }));
  127. } else {
  128. dataSource = dataSource.concat(
  129. res.result.records.map((item) => ({
  130. ...item,
  131. visible: false,
  132. }))
  133. );
  134. }
  135. if (res.result.current >= res.result.pages) {
  136. moreLoading.value = true;
  137. }
  138. } else {
  139. loading.value = false;
  140. createMessage.error('模板获取失败!');
  141. }
  142. nextTick(() => {
  143. // Resetting window's offsetTop so as to display react-virtualized demo underfloor.
  144. // In real scene, you can using public method of react-virtualized:
  145. // https://stackoverflow.com/questions/46700726/how-to-use-public-method-updateposition-of-react-virtualized
  146. window.dispatchEvent(new Event('resize'));
  147. });
  148. });
  149. }
  150. function checkUrl(val) {
  151. if (val != null) {
  152. if (val.substr(0, 7).toLowerCase() == 'http://' || val.substr(0, 8).toLowerCase() == 'https://') {
  153. return false;
  154. } else {
  155. return true;
  156. }
  157. } else {
  158. return false;
  159. }
  160. }
  161. function hoverThis() {
  162. $('.item').hover(
  163. function () {
  164. if ($(this).height() >= $(this).find('.Pic img').height()) {
  165. return;
  166. } else {
  167. $(this)
  168. .find('.Pic img')
  169. .stop()
  170. .animate({ top: $(this).height() - $(this).find('.Pic img').height() }, 5000);
  171. }
  172. },
  173. function () {
  174. $(this).find('.Pic img').stop().animate({ top: 0 }, 'fast');
  175. }
  176. );
  177. }
  178. //预览
  179. function visit(data, index) {
  180. if (data.domain != null && data.domain != '' && data.putaway === 1) {
  181. return window.open(data.domain, '_blank');
  182. } else {
  183. visitImage(data);
  184. }
  185. }
  186. function visitImage(data) {
  187. visible.value = true;
  188. modelText.value = data.buyLinks;
  189. theImage.value = domainUrl.value + '/sys/common/static/' + data.images;
  190. }
  191. function visitImageCancel() {
  192. visible.value = false;
  193. modelText.value = '';
  194. theImage.value = '';
  195. }
  196. function onLoadMore() {
  197. loading.value = true;
  198. pageNo.value += 1;
  199. getTemplateList();
  200. }
  201. function createSite(data) {
  202. modelRef.value.initCreateSiteModel(data);
  203. }
  204. </script>
  205. <style scoped lang="less">
  206. .mt-20 {
  207. margin-top: 20px;
  208. }
  209. .fl {
  210. float: left;
  211. }
  212. .fr {
  213. float: right;
  214. }
  215. .template-image {
  216. width: 100%;
  217. height: 300px;
  218. }
  219. .text-center {
  220. text-align: center;
  221. }
  222. .template_list /deep/ .ant-card-body {
  223. padding: 0;
  224. }
  225. .template_list .item {
  226. width: 100%;
  227. display: block;
  228. height: 467px;
  229. position: relative;
  230. transition: all 0.4s ease-out;
  231. }
  232. .template_list .item .Pic {
  233. width: 100%;
  234. height: 100%;
  235. position: relative;
  236. overflow: hidden;
  237. }
  238. .template_list .item .Pic img {
  239. width: 100%;
  240. max-height: unset;
  241. position: absolute;
  242. top: 0;
  243. left: 0;
  244. }
  245. .template_list .item .Pic .Content {
  246. display: flex;
  247. justify-content: center;
  248. align-items: flex-end;
  249. opacity: 0;
  250. position: absolute;
  251. width: 100%;
  252. height: 100%;
  253. background: rgba(0, 0, 0, 0.4);
  254. top: 0;
  255. left: 0;
  256. font-size: 0;
  257. transition: all 0.4s ease-out;
  258. padding-bottom: 20px;
  259. cursor: pointer;
  260. }
  261. .template_list .item .Pic .Content button {
  262. margin: 0 10px;
  263. }
  264. .template_list .item:hover {
  265. box-shadow: 3px 3px 12px 2px #aaaaaad4;
  266. }
  267. .template_list .item:hover .Pic .Content {
  268. opacity: 1;
  269. }
  270. .tag-wrap .ant-btn.ant-btn-link.act {
  271. color: @primary-color!important;
  272. background-color: #fff !important;
  273. font-weight: 600;
  274. border-bottom: 1px solid #fff;
  275. }
  276. .tag-wrap {
  277. margin-bottom: 12px;
  278. }
  279. .tag-wrap .ant-btn-link {
  280. color: #fff !important;
  281. width: 10%;
  282. margin: 5px 0.5%;
  283. }
  284. .template_list {
  285. overflow-y: scroll;
  286. overflow-x: hidden;
  287. }
  288. .ant-btn-link {
  289. margin-right: 10px;
  290. }
  291. img[lazy='error'],
  292. img[lazy='loading'] {
  293. top: calc(50% - 140px) !important;
  294. position: relative !important;
  295. margin: 0 auto;
  296. width: 240px !important;
  297. }
  298. .self_buy_modal {
  299. /deep/ .ant-modal-body {
  300. padding: 0;
  301. }
  302. }
  303. </style>