list.vue 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809
  1. <template>
  2. <view class="app-container">
  3. <view class="fixed-container">
  4. <view class="top-container">
  5. <scroll-view
  6. :scroll-x="true"
  7. :scroll-left="scrollLeft"
  8. scroll-with-animation
  9. :show-scrollbar="false"
  10. class="tabs-view"
  11. ref="tabs-view">
  12. <view class="tab">
  13. <block v-for="(item, index) in tabList" :key='index'>
  14. <view class="item" :ref="`tabs-item-${index}`" :class="[`tabs-item-${index}`, item.categoryId == tabCurrent ? 'current':'']" @tap="changeTab(item.categoryId)">{{item.name}}</view>
  15. </block>
  16. </view>
  17. </scroll-view>
  18. </view>
  19. <view class="search-container">
  20. <view class="search">
  21. <image src="/static/icon/search.png" class=""></image>
  22. <input type="text" confirm-type="search" placeholder="搜索商品名称" v-model="keyword" @confirm="searchSubmit">
  23. </view>
  24. </view>
  25. </view>
  26. <view class="banner-container" v-if="bannerUrl">
  27. <image :src="bannerUrl" mode="widthFix"></image>
  28. </view>
  29. <view class="list-container">
  30. <block v-for="(item, index) in dataList" :key='index'>
  31. <div class="item" @tap="toGoodsDetail(item.goodsId)">
  32. <image :src="item.goodsImgSrc" mode="aspectFill"></image>
  33. <view class="right">
  34. <view class="title ellipsis-2">{{item.goodsName}}</view>
  35. <view class="des ellipsis-2">{{item.describeText ? item.describeText : ''}}</view>
  36. <view class="stock-sales">
  37. <view class="stock">
  38. <text>剩余{{item.stock}}件</text>
  39. <view class="progress-box">
  40. <!-- 库存 / 总数 * 100 = 剩余百分比 -->
  41. <progress :percent="item.stock / (item.stock+100) * 100" activeColor="#FF3F42" active stroke-width="6" />
  42. </view>
  43. </view>
  44. <view class="sales">销量:{{item.saleNum || '0'}}</view>
  45. </view>
  46. <view class="bottom">
  47. <view class="price">
  48. <view class="price-1">¥{{item.groupPrice | numToFixed}}</view>
  49. <view class="price-2">¥{{item.orgGoodsPrice | numToFixed}}</view>
  50. </view>
  51. <view class="btn">去拼团</view>
  52. </view>
  53. </view>
  54. </div>
  55. </block>
  56. </view>
  57. <no-data v-if="!dataList.length" :showText="'暂无数据'"></no-data>
  58. <loading-text v-if="dataList.length" :loading="loading" :noMore="noMore" ></loading-text>
  59. <view class="share-container" :style="{top: top+'px'}" v-if="dataList.length">
  60. <button @tap="clickShare">
  61. <image src="@/static/icon/share.png"></image>
  62. <text>分享</text>
  63. </button>
  64. </view>
  65. <view class="global-mask" v-show="isShareDialog" @tap="isShareDialog = false"></view>
  66. <view class="sharelist-container" v-show="isShareDialog">
  67. <button class="item" open-type="share" @tap="isShareDialog = false">
  68. <image src="@/static/icon/wechat.png"></image>
  69. <text>分享给微信好友</text>
  70. </button>
  71. <view class="item" @tap="markImage">
  72. <image src="@/static/icon/image.png"></image>
  73. <text>生成图片分享</text>
  74. </view>
  75. </view>
  76. <view class="global-mask" v-show="isShowCanvas" @tap="closeCanvas"></view>
  77. <view class="canvas-container" v-show="isShowCanvas">
  78. <view class="content">
  79. <canvas style="width: 340px; height: 340px;" canvas-id="myCanvas" id="myCanvas"></canvas>
  80. </view>
  81. <view class="button"><text @tap="saveImage">保存图片</text></view>
  82. </view>
  83. <drag-button :isDock="true" :customBar="true" ref="dragButton"></drag-button>
  84. </view>
  85. </template>
  86. <script>
  87. import {mapState} from 'vuex';
  88. import dragButton from '@/components/drag-button.vue';
  89. export default {
  90. components:{
  91. dragButton
  92. },
  93. data() {
  94. return {
  95. configInfo: uni.getStorageSync('configInfo'),
  96. dataList: [], // 优惠券列表
  97. pageNum: 1,
  98. pageSize: 8,
  99. noMore: false,
  100. loading: false,
  101. isShareDialog: false, // 是否显示分享弹窗
  102. isShowCanvas: false, // 是否显示海报弹窗
  103. isFinishCanvas: false, // 是否已完成海报
  104. codeUrl: '',
  105. bgUrl: '',
  106. headUrl: '',
  107. top: 300,
  108. tabList: [], // 分类列表
  109. tabCurrent: '', // 分类当前值
  110. keyword: '',
  111. scrollLeft: 50,
  112. scrollViewWidth: 0,
  113. tabsRect: {
  114. left: 0
  115. },
  116. bannerUrl: '',
  117. }
  118. },
  119. watch: {
  120. tabList() {
  121. this.$nextTick(() => {
  122. this.resize()
  123. })
  124. },
  125. },
  126. computed:{
  127. ...mapState(['userInfo', 'isLogin', 'userId'])
  128. },
  129. onShow() {
  130. this.$refs.dragButton.init();
  131. this.top = uni.getStorageSync('top') < 200 ? uni.getStorageSync('top') : uni.getStorageSync('top') - 124;
  132. },
  133. onLoad() {
  134. this.getTabList();
  135. this.getBanner();
  136. },
  137. // 下拉刷新
  138. onPullDownRefresh() {
  139. this.pageNum = 1;
  140. this.getList();
  141. },
  142. // 上拉加载
  143. onReachBottom() {
  144. this.getList(1);
  145. },
  146. onShareAppMessage(options) {
  147. if (options && options.from == 'button') {
  148. // 来自页面内的转发按钮
  149. } else {
  150. // 点击微信右上角的分享按钮
  151. }
  152. return {
  153. title: `${this.userInfo.nickName}向你推荐了${this.configInfo.minAppName}的团购活动`,
  154. // imageUrl: this.detail.imgUrl,
  155. path: '/pages/index/index?serviceId=' + this.userId + '&otherType=groupbuyList',
  156. query: {
  157. // id: this.goodsId,
  158. },
  159. success: function(res) {
  160. if(res.errMsg == 'shareAppMessage:ok'){
  161. this.$successToast('分享完成');
  162. }
  163. }
  164. }
  165. },
  166. methods: {
  167. getList(loadMore) {
  168. if(this.noMore && loadMore)return;
  169. this.noMore = false
  170. if(!loadMore){
  171. this.pageNum = 1;
  172. }else{
  173. this.loading = true;
  174. }
  175. this.$axios({
  176. url: '/goods/promotion/group/list',
  177. method: 'get',
  178. params: {
  179. pageNo: this.pageNum,
  180. pageSize: this.pageSize,
  181. userId: this.userId,
  182. goodsCategoryId: this.tabCurrent,
  183. keyword: this.keyword,
  184. },
  185. isLoading: !loadMore
  186. }).then(res => {
  187. let _list = res.data.records;
  188. let pageTotal = res.data.pages;
  189. if(this.pageNum >= pageTotal){
  190. this.noMore = true;
  191. }
  192. if (_list.length) {
  193. this.pageNum += 1
  194. }
  195. if (loadMore) {
  196. this.dataList = this.dataList.concat(_list);
  197. this.loading = false;
  198. } else {
  199. this.dataList = _list;
  200. }
  201. uni.stopPullDownRefresh();
  202. })
  203. },
  204. // 获取一级菜单
  205. getTabList() {
  206. this.$axios({
  207. url: '/goods/category/list',
  208. method: 'get',
  209. params: {}
  210. }).then(res => {
  211. res.data.unshift({name: "全部", categoryId: ""})
  212. this.tabList = res.data;
  213. this.tabCurrent = res.data.length > 0 ? res.data[0].categoryId : 0;
  214. this.getList();
  215. })
  216. },
  217. // 切换一级菜单
  218. changeTab(current) {
  219. this.pageNum = 1;
  220. this.tabCurrent = current;
  221. this.resize();
  222. this.getList();
  223. },
  224. searchSubmit() {
  225. this.pageNum = 1;
  226. this.getList();
  227. },
  228. setScrollLeft() {
  229. // 当前活动tab的布局信息,有tab菜单的width和left(为元素左边界到父元素左边界的距离)等信息
  230. const index = this.findElem(this.tabList, 'categoryId', this.tabCurrent);
  231. const tabRect = this.tabList[index]
  232. // 累加得到当前item到左边的距离
  233. const offsetLeft = this.tabList
  234. .slice(0, index)
  235. .reduce((total, curr) => {
  236. return total + curr.rect.width
  237. }, 0)
  238. // 此处为屏幕宽度
  239. const res = uni.getSystemInfoSync();
  240. const windowWidth = res.windowWidth;
  241. // 将活动的tabs-item移动到屏幕正中间,实际上是对scroll-view的移动
  242. let scrollLeft = offsetLeft - (this.tabsRect.width - tabRect.rect.width) / 2 - (windowWidth - this.tabsRect.right) / 2 + this.tabsRect.left / 2
  243. // 这里做一个限制,限制scrollLeft的最大值为整个scroll-view宽度减去tabs组件的宽度
  244. scrollLeft = Math.min(scrollLeft, this.scrollViewWidth - this.tabsRect.width)
  245. this.scrollLeft = Math.max(0, scrollLeft)
  246. },
  247. // 获取所有标签的尺寸
  248. resize() {
  249. // 如果不存在list,则不处理
  250. if(this.tabList.length === 0) {
  251. return
  252. }
  253. Promise.all([this.getTabsRect(), this.getAllItemRect()]).then(([tabsRect, itemRect = []]) => {
  254. this.tabsRect = tabsRect
  255. this.scrollViewWidth = 0
  256. itemRect.map((item, index) => {
  257. // 计算scroll-view的宽度,这里
  258. this.scrollViewWidth += item.width
  259. // 另外计算每一个item的中心点X轴坐标
  260. this.tabList[index].rect = item
  261. })
  262. // 获取了tabs的尺寸之后,设置滑块的位置
  263. this.setScrollLeft()
  264. })
  265. },
  266. // 获取导航菜单的尺寸
  267. getTabsRect() {
  268. return new Promise(resolve => {
  269. this.queryRect('tabs-view').then(size => resolve(size))
  270. })
  271. },
  272. // 获取所有标签的尺寸
  273. getAllItemRect() {
  274. return new Promise(resolve => {
  275. const promiseAllArr = this.tabList.map((item, index) => this.queryRect(
  276. `tabs-item-${index}`, true))
  277. Promise.all(promiseAllArr).then(sizes => resolve(sizes))
  278. })
  279. },
  280. // 获取各个标签的尺寸
  281. queryRect(el, item) {
  282. const query = uni.createSelectorQuery().in(this);
  283. return new Promise(resolve => {
  284. query.select(`.${el}`).boundingClientRect(data => {
  285. resolve(data)
  286. }).exec();
  287. })
  288. },
  289. findElem(array, attr, val) {
  290. for (var i = 0; i < array.length; i++) {
  291. if (array[i][attr] == val) {
  292. return i; //返回当前索引值
  293. }
  294. }
  295. return -1;
  296. },
  297. // 进入商品详情
  298. toGoodsDetail(id) {
  299. if(!id) {
  300. return false;
  301. }
  302. uni.navigateTo({
  303. url: '/packageGoods/pages/detail?id=' + id
  304. })
  305. },
  306. // 获取海报图
  307. getBanner() {
  308. this.$axios({
  309. url: '/goods/promotion/share/qrcode',
  310. method: 'get',
  311. params: {
  312. userId: this.userId,
  313. }
  314. }).then(res => {
  315. this.bannerUrl = res.data.posterImgUrl;
  316. })
  317. },
  318. // 点击分享
  319. clickShare() {
  320. if(!this.isLogin) {
  321. return uni.navigateTo({
  322. url: '/pages/login/index'
  323. })
  324. }
  325. this.getCode();
  326. this.isShareDialog = true;
  327. },
  328. // 获取二维码
  329. getCode() {
  330. let that = this;
  331. this.$axios({
  332. url: '/goods/promotion/share/qrcode',
  333. method: 'get',
  334. params: {
  335. userId: this.userId,
  336. }
  337. }).then(res => {
  338. if(res.data) {
  339. // this.codeUrl = res.data.qrcode;
  340. // this.bgUrl = res.data.promotionImgUrl;
  341. if(!this.isFinishCanvas) {
  342. uni.downloadFile({
  343. url: res.data.promotionImgUrl,
  344. success: function (fileRes) {
  345. that.bgUrl = fileRes.tempFilePath;
  346. }
  347. })
  348. uni.downloadFile({
  349. url: res.data.qrcode,
  350. success: function (fileRes) {
  351. that.codeUrl = fileRes.tempFilePath;
  352. }
  353. })
  354. uni.downloadFile({
  355. url: that.userInfo.avatar.indexOf('http') >= 0 ? that.userInfo.avatar : (that.$imageUrl + that.userInfo.avatar),
  356. success: function (fileRes) {
  357. that.headUrl = fileRes.tempFilePath;
  358. }
  359. })
  360. }
  361. }
  362. })
  363. },
  364. // 生成图片
  365. markImage() {
  366. if(!this.bgUrl || !this.codeUrl || !this.headUrl) {
  367. this.getCode();
  368. return this.$toast('生成失败,请重新操作');
  369. }
  370. let that = this;
  371. this.isShareDialog = false;
  372. this.isShowCanvas = true;
  373. if(this.isFinishCanvas) {
  374. return false;
  375. }
  376. uni.showLoading({
  377. title: '海报生成中'
  378. });
  379. let codeUrl = this.codeUrl;
  380. let bgUrl = this.bgUrl;
  381. let headUrl = this.headUrl;
  382. // this.userInfo.nickName = '阿里巴巴的的';
  383. let name = this.userInfo.nickName.length > 4 ? this.userInfo.nickName.slice(0, 4) + '...' : this.userInfo.nickName;
  384. var ctx = uni.createCanvasContext('myCanvas');
  385. // 背景图片
  386. ctx.drawImage(bgUrl, 0, 0, 340, 340)
  387. // 二维码
  388. this.circleImgOne(ctx, codeUrl, 260, 260, uni.upx2px(70));
  389. // ctx.drawImage(codeUrl, 260, 260, 70, 70)
  390. // 矩形
  391. // ctx.setFillStyle('rgba(0,0,0,0.4)')
  392. // ctx.fillRect(185, 10, 145, 30)
  393. // 圆角矩形
  394. this.fillRoundRect(ctx, 175, 10, 155, 30, 15, 'rgba(0,0,0,0.4)');
  395. // 头像
  396. // this.circleImgOne(ctx, headUrl, 200, 10, 15);
  397. ctx.drawImage(headUrl, 187, 12, 26, 26)
  398. // 文字
  399. ctx.setFontSize(12)
  400. ctx.setFillStyle('#FFFFFF')
  401. ctx.fillText(name+'向你推荐', 217, 30)
  402. ctx.draw()
  403. uni.hideLoading();
  404. this.isFinishCanvas = true;
  405. },
  406. // ctx=Canvas实例, img=图片地址, x=x轴坐标, y=y轴坐标, r=圆形半径
  407. circleImgOne(ctx, img, x, y, r) {
  408. // 如果在绘制图片之后还有需要绘制别的元素,需启动 save() 、restore() 方法,否则 clip() 方法会导致之后元素都不可见
  409. // save():保存当前 Canvas 画布状态
  410. // restore():恢复到保存时的状态
  411. ctx.save();
  412. let d = r * 2;
  413. let cx = x + r;
  414. let cy = y + r;
  415. ctx.arc(cx, cy, r, 0, 2 * Math.PI);
  416. ctx.strokeStyle = '#FFFFFF'; // 设置绘制圆形边框的颜色
  417. ctx.stroke(); // 绘制出圆形,默认为黑色,可通过 ctx.strokeStyle = '#FFFFFF', 设置想要的颜色
  418. ctx.clip();
  419. ctx.drawImage(img, x, y, d, d);
  420. ctx.restore();
  421. },
  422. /**该方法用来绘制一个有填充色的圆角矩形
  423. *@param cxt:canvas的上下文环境
  424. *@param x:左上角x轴坐标
  425. *@param y:左上角y轴坐标
  426. *@param width:矩形的宽度
  427. *@param height:矩形的高度
  428. *@param radius:圆的半径
  429. *@param fillColor:填充颜色
  430. **/
  431. fillRoundRect(cxt, x, y, width, height, radius, fillColor) {
  432. //圆的直径必然要小于矩形的宽高
  433. if (2 * radius > width || 2 * radius > height) { return false; }
  434. cxt.save();
  435. cxt.translate(x, y);
  436. //绘制圆角矩形的各个边
  437. this.drawRoundRectPath(cxt, width, height, radius);
  438. cxt.fillStyle = fillColor;
  439. cxt.fill();
  440. cxt.restore();
  441. },
  442. drawRoundRectPath(cxt, width, height, radius) {
  443. cxt.beginPath(0);
  444. //从右下角顺时针绘制,弧度从0到1/2PI
  445. cxt.arc(width - radius, height - radius, radius, 0, Math.PI / 2);
  446. //矩形下边线
  447. cxt.lineTo(radius, height);
  448. //左下角圆弧,弧度从1/2PI到PI
  449. cxt.arc(radius, height - radius, radius, Math.PI / 2, Math.PI);
  450. //矩形左边线
  451. cxt.lineTo(0, radius);
  452. //左上角圆弧,弧度从PI到3/2PI
  453. cxt.arc(radius, radius, radius, Math.PI, Math.PI * 3 / 2);
  454. //上边线
  455. cxt.lineTo(width - radius, 0);
  456. //右上角圆弧
  457. cxt.arc(width - radius, radius, radius, Math.PI * 3 / 2, Math.PI * 2);
  458. //右边线
  459. cxt.lineTo(width, height - radius);
  460. cxt.closePath();
  461. },
  462. // 保存图片
  463. saveImage() {
  464. let that = this;
  465. uni.canvasToTempFilePath({
  466. x: 0,
  467. y: 0,
  468. width: 340,
  469. height: 340,
  470. canvasId: 'myCanvas',
  471. success: function(res) {
  472. uni.saveImageToPhotosAlbum({
  473. filePath: res.tempFilePath,
  474. success: function () {
  475. that.$successToast('保存成功');
  476. }
  477. });
  478. }
  479. })
  480. },
  481. // 关闭canvas
  482. closeCanvas() {
  483. this.isShowCanvas = false;
  484. uni.hideLoading();
  485. },
  486. }
  487. }
  488. </script>
  489. <style lang="scss">
  490. .app-container {
  491. background: #F4F2F2;
  492. box-sizing: border-box;
  493. padding-top: 184rpx;
  494. }
  495. .fixed-container {
  496. position: fixed;
  497. top: 0;
  498. left: 0;
  499. width: 100%;
  500. }
  501. .share-container {
  502. position: fixed;
  503. right: 20rpx;
  504. // bottom: 30vh;
  505. button {
  506. width: 108rpx;
  507. height: 108rpx;
  508. padding: 0;
  509. display: flex;
  510. flex-direction: column;
  511. justify-content: center;
  512. align-items: center;
  513. border-radius: 50%;
  514. // box-shadow: 0 1px 4px rgba(0, 21, 41, .8);
  515. background: #ffffff;
  516. border: 4rpx solid #eaeaea;
  517. image {
  518. width: 28rpx;
  519. height: 28rpx;
  520. display: block;
  521. flex-shrink: 0;
  522. margin-bottom: 6rpx;
  523. }
  524. text {
  525. font-size: 22rpx;
  526. line-height: 24rpx;
  527. color: #666666;
  528. margin-top: 8rpx;
  529. }
  530. }
  531. }
  532. .top-container {
  533. background: #FFFFFF;
  534. .tab {
  535. display: flex;
  536. margin-left: -10rpx;
  537. .item {
  538. display: flex;
  539. flex-direction: column;
  540. align-items: center;
  541. flex-shrink: 0;
  542. font-size: 28rpx;
  543. color: #666666;
  544. height: 80rpx;
  545. line-height: 80rpx;
  546. position: relative;
  547. padding: 0 30rpx;
  548. &.current {
  549. color: #FF3F42;
  550. font-weight: 600;
  551. &::after {
  552. content: '';
  553. display: block;
  554. width: 50rpx;
  555. height: 6rpx;
  556. background: #FF3F42;
  557. position: absolute;
  558. bottom: 0;
  559. left: 50%;
  560. margin-left: -25rpx;
  561. }
  562. }
  563. }
  564. }
  565. }
  566. .search-container {
  567. background: #FFFFFF;
  568. padding: 20rpx;
  569. .search {
  570. height: 64rpx;
  571. display: flex;
  572. align-items: center;
  573. border-radius: 64rpx;
  574. padding: 0 20rpx;
  575. border: 1px solid #eaeaea;
  576. image {
  577. width: 28rpx;
  578. height: 28rpx;
  579. }
  580. input {
  581. width: 100%;
  582. padding-left: 15rpx;
  583. }
  584. }
  585. }
  586. .banner-container {
  587. padding: 20rpx 20rpx 0;
  588. image {
  589. display: block;
  590. width: 100%;
  591. border-radius: 10rpx;
  592. }
  593. }
  594. .list-container {
  595. padding: 20rpx;
  596. .item {
  597. background: #FFFFFF;
  598. border-radius: 10rpx;
  599. display: flex;
  600. padding: 20rpx;
  601. margin-bottom: 20rpx;
  602. image {
  603. display: block;
  604. width: 180rpx;
  605. height: 180rpx;
  606. flex-shrink: 0;
  607. }
  608. .right {
  609. width: 490rpx;
  610. box-sizing: border-box;
  611. padding-left: 20rpx;
  612. .title {
  613. font-size: 30rpx;
  614. color: #333333;
  615. line-height: 36rpx;
  616. font-weight: 600;
  617. }
  618. .des {
  619. font-size: 24rpx;
  620. line-height: 30rpx;
  621. color: #999999;
  622. margin-top: 6rpx;
  623. }
  624. .stock-sales {
  625. display: flex;
  626. justify-content: space-between;
  627. align-items: center;
  628. margin-top: 10rpx;
  629. font-size: 24rpx;
  630. color: #666666;
  631. .stock {
  632. display: flex;
  633. align-items: center;
  634. text {
  635. font-size: 24rpx;
  636. color: #666666;
  637. }
  638. .progress-box {
  639. width: 140rpx;
  640. border-radius: 6px;
  641. overflow: hidden;
  642. margin-left: 10rpx;
  643. }
  644. }
  645. }
  646. .bottom {
  647. display: flex;
  648. justify-content: space-between;
  649. align-items: center;
  650. margin-top: 10rpx;
  651. .price {
  652. display: flex;
  653. flex-direction: column;
  654. }
  655. .price-1 {
  656. font-size: 32rpx;
  657. color: #FF3F42;
  658. line-height: 36rpx;
  659. }
  660. .price-2 {
  661. font-size: 26rpx;
  662. color: #666666;
  663. line-height: 30rpx;
  664. text-decoration: line-through;
  665. }
  666. .btn {
  667. width: 110rpx;
  668. height: 44rpx;
  669. background: #FF3F42;
  670. border-radius: 5rpx;
  671. font-size: 28rpx;
  672. color: #FFFFFF;
  673. text-align: center;
  674. line-height: 44rpx;
  675. }
  676. .btn2 {
  677. width: 110rpx;
  678. height: 44rpx;
  679. background: #AAAAAA;
  680. border-radius: 5rpx;
  681. font-size: 28rpx;
  682. color: #FFFFFF;
  683. text-align: center;
  684. line-height: 44rpx;
  685. }
  686. }
  687. }
  688. }
  689. }
  690. .sharelist-container {
  691. position: fixed;
  692. bottom: 0;
  693. left: 0;
  694. z-index: 999;
  695. width: 100%;
  696. box-sizing: border-box;
  697. background: #FFFFFF;
  698. padding: 30rpx 0;
  699. display: flex;
  700. button {
  701. background: none;
  702. border-radius:0;
  703. &::after {
  704. border: none;
  705. }
  706. }
  707. .item {
  708. display: flex;
  709. width: 50%;
  710. flex-direction: column;
  711. align-items: center;
  712. image {
  713. width: 100rpx;
  714. height: 100rpx;
  715. display: block;
  716. margin-bottom: 20rpx;
  717. }
  718. text {
  719. font-size: 28rpx;
  720. line-height: 32rpx;
  721. color: #333333;
  722. }
  723. }
  724. }
  725. .canvas-container {
  726. position: fixed;
  727. left: 50%;
  728. top: 50%;
  729. z-index: 999;
  730. margin-left: -170px;
  731. margin-top: -170px;
  732. .button {
  733. display: flex;
  734. justify-content: center;
  735. text {
  736. display: block;
  737. width: 280rpx;
  738. height: 70rpx;
  739. border-radius: 70rpx;
  740. background: linear-gradient(-90deg,#ff3f42 0%, #fe781f 100%);
  741. font-size: 28rpx;
  742. color: #FFFFFF;
  743. text-align: center;
  744. line-height: 70rpx;
  745. margin-top: 20rpx;
  746. }
  747. }
  748. }
  749. </style>