index.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402
  1. <route lang="json">
  2. { "style": { "navigationStyle": "custom" } }
  3. </route>
  4. <script setup lang="ts">
  5. import Card from '@/components/card.vue'
  6. import SectionHeading from '@/components/section-heading.vue'
  7. import {
  8. getMaterialDetail,
  9. getByDictType,
  10. getMaterialHomePage,
  11. downloadMaterials,
  12. } from '../../../core/libs/requests'
  13. import NavbarEvo from '@/components/navbar-evo.vue'
  14. import { DictType } from '../../../core/libs/models'
  15. import { close, phone } from '../../../core/libs/svgs'
  16. import { handleCall, openLocation, previewImage } from '../../../core/utils/common'
  17. import router from '@designer-hub/assets/src/assets/svgs/router'
  18. import { isVideoUrl } from 'wot-design-uni/components/common/util'
  19. import { useUserStore } from '@/store'
  20. import { AnalysisEventType, useAnalysis } from '@/composables/analysis'
  21. import BottomAppBar from '@/components/bottom-app-bar.vue'
  22. import { usePermissions } from '@/composables/permissions'
  23. const dealerPanelState = ref(false)
  24. const { clickByPermission } = usePermissions()
  25. const userStore = useUserStore()
  26. const { option } = useAnalysis(true, false)
  27. const id = ref()
  28. const downloading = ref(false)
  29. const currentBanner = ref(0)
  30. const handleSwiperChange = ({ detail: { current, source } }) => {
  31. currentBanner.value = current
  32. }
  33. const { data, run: setData } = useRequest(() => getMaterialDetail({ id: id.value }))
  34. const { data: materialHomePageData, run: setMaterialHomePageData } = useRequest(
  35. () => getMaterialHomePage(id.value),
  36. { initialData: {} },
  37. )
  38. const { data: materialBrandLevels, run: setMaterialBrandLevels } = useRequest(
  39. () => getByDictType(DictType.memberMaterialsBrandLevel),
  40. { initialData: [] },
  41. )
  42. const { data: materialOperationTypes, run: setMaterialOperationTypes } = useRequest(
  43. () => getByDictType(DictType.memberMaterialsOperationType),
  44. { initialData: [] },
  45. )
  46. const { data: materialBrandTypes, run: setMaterialBrandTypes } = useRequest(
  47. () => getByDictType(DictType.memberMaterialsBrandType),
  48. { initialData: [] },
  49. )
  50. const { data: materialsManageBrands, run: setMaterialsManageBrands } = useRequest(
  51. () => getByDictType(DictType.materialsManageBrand),
  52. { initialData: [] },
  53. )
  54. const copy = (url) => {
  55. uni.setClipboardData({
  56. data: url,
  57. success: () => {
  58. console.log('已复制')
  59. downloadMaterials({ materialsId: id.value })
  60. },
  61. })
  62. }
  63. const imageGroup = ref<any[]>([
  64. { height: '171px' },
  65. 1,
  66. { width: '107px' },
  67. [{ width: '93px' }, { width: '32px', marginLeft: '41px' }],
  68. ])
  69. const isLoaded = ref(false)
  70. const checkFodder = () => {
  71. if(materialHomePageData.value.fodderList[0].fodderUrl){
  72. dealerPanelState.value = true
  73. }else{
  74. uni.showToast({
  75. title:"暂无素材包",
  76. icon:"none"
  77. })
  78. }
  79. }
  80. const handleDownload = async () => {
  81. downloading.value = true
  82. const { filePath } = await uni.downloadFile({
  83. url: data.value.agreementFileUrl,
  84. filePath: `${uni.env.USER_DATA_PATH}/${data.value.materialsName}的素材包.zip`,
  85. header: {
  86. 'Content-Type': 'application/x-zip-compressed',
  87. },
  88. })
  89. downloading.value = false
  90. await downloadMaterials({ materialsId: id.value })
  91. const { deviceType } = await uni.getDeviceInfo()
  92. if (deviceType === 'pc') {
  93. uni.saveFileToDisk({
  94. filePath,
  95. success: (res) => {
  96. console.log('saveFileToDisk success', res)
  97. },
  98. fail: (err) => {
  99. console.log('saveFileToDisk fail', err)
  100. },
  101. })
  102. } else {
  103. uni.openDocument({
  104. filePath,
  105. success: (res) => {
  106. console.log('openDocument success', res)
  107. },
  108. fail: (err) => {
  109. console.log('openDocument fail', err)
  110. },
  111. })
  112. }
  113. }
  114. onLoad(async (query: { id: number }) => {
  115. console.log(query, '最近浏览品牌')
  116. if (query.id) {
  117. id.value = query.id
  118. await setData()
  119. option.value = {
  120. remark: `最近浏览品牌: ${data.value?.materialsName}`,
  121. }
  122. await setMaterialHomePageData()
  123. console.log(data.value)
  124. await Promise.all([
  125. setMaterialBrandLevels(),
  126. setMaterialBrandTypes(),
  127. setMaterialOperationTypes(),
  128. setMaterialsManageBrands(),
  129. ]).then(() => {
  130. isLoaded.value = true
  131. })
  132. }
  133. })
  134. onShareAppMessage(() => ({
  135. title: data.value?.materialsName,
  136. imageUrl: data.value?.bannerUrl,
  137. }))
  138. onShareTimeline(() => ({
  139. title: data.value?.materialsName,
  140. imageUrl: data.value?.bannerUrl,
  141. }))
  142. </script>
  143. <template>
  144. <view class="flex-grow flex flex-col" v-if="isLoaded">
  145. <div
  146. class="w-full aspect-[1.16/1] bg-[url(https://image.zhuchaohui.com/zhucaohui/9da274c73312f5ff828b60457bf4a469e631c001a4bf1c1f355338887b19f035.png)] bg-[length:100%]"
  147. :style="{ backgroundImage: `url(${materialHomePageData?.bannerUrl})` }"
  148. ></div>
  149. <NavbarEvo transparent></NavbarEvo>
  150. <div class="flex flex-col gap-6 relative top--14 px-3.5">
  151. <Card>
  152. <div class="flex flex-col gap-3">
  153. <div class="flex gap-5">
  154. <wd-img
  155. round
  156. width="78"
  157. height="78"
  158. custom-class="border border-[#f2f2f2] border-solid"
  159. :src="data?.logoUrl"
  160. ></wd-img>
  161. <div class="flex flex-col gap-1.5">
  162. <div class="flex gap-1 items-center">
  163. <div class="text-black/90 text-lg font-normal font-['PingFang_SC']">
  164. <!-- IMOLA瓷砖 -->
  165. {{ data?.materialsName }}
  166. </div>
  167. </div>
  168. <div class="text-black/40 text-xs font-normal font-['PingFang_SC'] leading-[10px]">
  169. {{
  170. materialBrandTypes.find(({ value }) => value === String(data?.brandType))?.label
  171. }}
  172. |
  173. {{
  174. materialOperationTypes
  175. .filter(({ value }) => data.manageType.includes(Number.parseInt(value)))
  176. .map((i) => i.label)
  177. .join(' / ')
  178. }}
  179. </div>
  180. <div>
  181. <span class="text-black/40 text-xs font-normal font-['PingFang_SC'] leading-[10px]">
  182. 经营品牌:
  183. </span>
  184. <span
  185. class="text-black/40 text-xs font-normal font-['PingFang_SC'] uppercase leading-[10px]"
  186. >
  187. <!-- imola / chedit -->
  188. {{
  189. materialsManageBrands
  190. .filter(({ value }) => data.manageBrand.includes(Number.parseInt(value)))
  191. .map((i) => i.label)
  192. .join(' / ')
  193. }}
  194. </span>
  195. </div>
  196. <div>
  197. <span class="text-black/40 text-xs font-normal font-['PingFang_SC'] leading-[10px]">
  198. 经营范围:
  199. </span>
  200. <span
  201. class="text-black/40 text-xs font-normal font-['PingFang_SC'] uppercase leading-[10px]"
  202. >
  203. <!-- imola / chedit -->
  204. {{ data?.businessScope }}
  205. </span>
  206. </div>
  207. </div>
  208. </div>
  209. <template
  210. v-for="({ shopAddr, shopName, shopContactPhone, ...shop }, i) in data?.shopList"
  211. :key="i"
  212. >
  213. <div class="bg-neutral-50 rounded-2.5 p-3.5 flex items-center">
  214. <div class="text-black/40 flex-1 border-r-2 border-r-[#f6f6f6] border-r-solid pr-4">
  215. <div class="text-black/90 text-lg font-normal font-['PingFang_SC'] leading-normal">
  216. {{ shopName }}
  217. </div>
  218. <div>
  219. <wd-icon name="location" size="15"></wd-icon>
  220. <span
  221. class="text-black/40 text-xs font-normal font-['PingFang_SC'] leading-normal"
  222. >
  223. {{ shopAddr }}
  224. </span>
  225. </div>
  226. </div>
  227. <wd-button
  228. type="text"
  229. size="small"
  230. custom-class=" bg-[#f2f2f2]! p-0! ml-4"
  231. @click="handleCall(shopContactPhone)"
  232. >
  233. <wd-img width="28" height="28" :src="phone"></wd-img>
  234. </wd-button>
  235. <wd-button
  236. type="text"
  237. size="small"
  238. custom-class=" bg-[#f2f2f2]! p-0! ml-4"
  239. @click="
  240. openLocation({
  241. latitude: shop.longitude,
  242. longitude: shop.latitude,
  243. name: shopName,
  244. address: shopAddr,
  245. })
  246. "
  247. >
  248. <wd-img width="28" height="28" :src="router"></wd-img>
  249. </wd-button>
  250. </div>
  251. </template>
  252. </div>
  253. </Card>
  254. <Card>
  255. <div class="flex flex-col gap-4">
  256. <div
  257. class="text-black/90 text-lg font-normal font-['PingFang_SC'] leading-[10.18px] pt-4"
  258. >
  259. 品牌介绍
  260. </div>
  261. <div
  262. v-if="(materialHomePageData.brandAdvantageUrl ?? '') !== ''"
  263. class="rounded-2xl overflow-hidden aspect-[1.72/1] whitespace-pre-wrap"
  264. >
  265. <template v-if="isVideoUrl(materialHomePageData.brandAdvantageUrl)">
  266. <video class="w-full h-194px" :src="materialHomePageData.brandAdvantageUrl"></video>
  267. </template>
  268. <template v-else>
  269. <wd-img
  270. width="100%"
  271. height="194"
  272. :src="materialHomePageData.brandAdvantageUrl"
  273. mode="aspectFill"
  274. custom-class=""
  275. ></wd-img>
  276. </template>
  277. </div>
  278. <!-- <div class="text-black/90 text-base font-normal font-['PingFang_SC'] leading-[10.18px]">-->
  279. <!-- {{-->
  280. <!-- materialsManageBrands.find(({ value }) => value === String(data?.manageBrand))?.label-->
  281. <!-- }}-->
  282. <!-- </div>-->
  283. <div
  284. class="text-justify text-black/60 text-sm font-normal font-['PingFang_SC'] leading-[25px] whitespace-pre-wrap"
  285. >
  286. {{ materialHomePageData.brandAdvantageDesc }}
  287. </div>
  288. </div>
  289. </Card>
  290. <template
  291. v-if="materialHomePageData.productDOList && materialHomePageData.productDOList?.length"
  292. >
  293. <SectionHeading title="产品展示"></SectionHeading>
  294. </template>
  295. <template
  296. v-if="materialHomePageData.productDOList && materialHomePageData.productDOList?.length"
  297. v-for="(it, index) in materialHomePageData.productDOList"
  298. :key="index"
  299. >
  300. <div class="aspect-[1.66/1] rounded-2xl overflow-hidden relative">
  301. <swiper class="h-[55.7vw]" @change="handleSwiperChange">
  302. <template v-for="(img, index) in it?.productImgUrl?.split(',')" :key="img">
  303. <swiper-item class="h-full">
  304. <wd-img
  305. width="100%"
  306. height="100%"
  307. class=""
  308. :src="img"
  309. mode="aspectFill"
  310. @click="previewImage(index, it?.productImgUrl?.split(','))"
  311. />
  312. </swiper-item>
  313. </template>
  314. </swiper>
  315. <div
  316. class="absolute bottom-0 w-full h-[66px] bg-gradient-to-t from-black to-[#5e5e5e00] rounded-bl-2xl rounded-br-2xl flex items-center px-4.5"
  317. >
  318. <div class="text-white text-base font-normal font-['PingFang_SC'] leading-[10.18px]">
  319. {{ it.productTitleName }}
  320. </div>
  321. </div>
  322. <div class="absolute flex gap-1 dots">
  323. <template v-for="(img, i) in it?.productImgUrl?.split(',')" :key="i">
  324. <div
  325. class="w-1 h-1 rounded-full"
  326. :class="`${currentBanner === i ? 'bg-white bg-active' : 'bg-white/40'}`"
  327. ></div>
  328. </template>
  329. </div>
  330. </div>
  331. </template>
  332. <BottomAppBar fixed>
  333. <wd-button
  334. custom-class="w-full! rounded-lg!"
  335. :disabled="(data?.agreementFileUrl ?? '') === ''"
  336. @click="clickByPermission('download', () => (checkFodder()))"
  337. >
  338. 下载所有素材包
  339. </wd-button>
  340. </BottomAppBar>
  341. </div>
  342. <wd-overlay :show="dealerPanelState" @click="dealerPanelState = false">
  343. <view class="wrapper flex flex-col justify-end h-full">
  344. <div class="w-full flex justify-end mb-4">
  345. <div class="mr-3.5">
  346. <wd-button type="text" custom-class="w-8! h-8! p-0!" size="small">
  347. <wd-img :src="close" width="28" height="28" custom-class="vertical-bottom"></wd-img>
  348. </wd-button>
  349. </div>
  350. </div>
  351. <div class="bg-neutral-100 rounded-tl-2xl rounded-tr-2xl p-3.5" @click.stop="">
  352. <div class="deal-title">请复制以下链接,打开默认浏览器,粘贴到浏览器进行下载</div>
  353. <div class="deal-link">{{ materialHomePageData?.fodderList[0].fodderUrl }}</div>
  354. <wd-button
  355. custom-class="custom-shadow"
  356. @click="copy(materialHomePageData?.fodderList[0].fodderUrl)"
  357. >
  358. 复制链接
  359. </wd-button>
  360. </div>
  361. </view>
  362. </wd-overlay>
  363. </view>
  364. <!-- 骨架 屏 -->
  365. <view v-else>
  366. <wd-skeleton animation="gradient" theme="paragraph" :row-col="imageGroup" />
  367. <wd-skeleton
  368. animation="gradient"
  369. theme="paragraph"
  370. :custom-style="{ marginTop: '20px' }"
  371. :row-col="imageGroup"
  372. />
  373. </view>
  374. </template>
  375. <style scoped>
  376. .dots {
  377. bottom: 30rpx;
  378. left: 50%;
  379. transform: translateX(-50%);
  380. }
  381. .bg-active {
  382. width: 40rpx;
  383. }
  384. .deal-title {
  385. padding: 60rpx 0;
  386. font-size: 32rpx;
  387. font-weight: bold;
  388. text-align: center;
  389. }
  390. .deal-link {
  391. padding: 20rpx 30rpx;
  392. margin-bottom: 60rpx;
  393. font-size: 28rpx;
  394. color: rgba(0, 0, 0, 0.45);
  395. word-wrap: break-word;
  396. background-color: rgba(200, 200, 200, 0w);
  397. border-radius: 20rpx;
  398. }
  399. :deep(.custom-shadow) {
  400. width: 100% !important;
  401. }
  402. </style>