502 lines
14 KiB
Vue
502 lines
14 KiB
Vue
|
<template>
|
|||
|
<view ref="specialSharePage" class="special-share-page">
|
|||
|
<!-- 特殊场景背景图片 -->
|
|||
|
<image
|
|||
|
class="share-bg-image"
|
|||
|
:src="specialBackgroundImage"
|
|||
|
mode="scaleToFill"
|
|||
|
crossorigin="anonymous"
|
|||
|
@load="onBackgroundImageLoad"
|
|||
|
@error="onBackgroundImageError"
|
|||
|
v-if="specialBackgroundImage"
|
|||
|
></image>
|
|||
|
|
|||
|
<!-- 加载状态 -->
|
|||
|
<view
|
|||
|
class="loading-container"
|
|||
|
v-if="!specialBackgroundImage || isLoadingBackground"
|
|||
|
>
|
|||
|
<view class="loader"></view>
|
|||
|
<text class="loading-text">{{
|
|||
|
isLoadingBackground ? '正在获取背景图片...' : '正在加载分享背景...'
|
|||
|
}}</text>
|
|||
|
</view>
|
|||
|
|
|||
|
<view
|
|||
|
class="share-wrapper"
|
|||
|
v-if="specialBackgroundImage && !isLoadingBackground"
|
|||
|
>
|
|||
|
<view class="portal-frame" :class="{ 'is-loaded': isLoaded }">
|
|||
|
<view class="qr-code-outer special-qr-style">
|
|||
|
<image
|
|||
|
class="qr-code"
|
|||
|
:src="qrCodeImage"
|
|||
|
mode="aspectFit"
|
|||
|
v-if="qrCodeImage"
|
|||
|
></image>
|
|||
|
<view v-else class="qr-code-placeholder">
|
|||
|
<view class="loader"></view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
|
|||
|
<text
|
|||
|
class="member-code-text special-member-code-style"
|
|||
|
style="
|
|||
|
font-size: 30rpx;
|
|||
|
color: #005bac;
|
|||
|
font-weight: bold;
|
|||
|
margin-top: 20rpx;
|
|||
|
"
|
|||
|
>{{ desensitization(userInfo.memberCode) }}</text
|
|||
|
>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</view>
|
|||
|
</template>
|
|||
|
|
|||
|
<script>
|
|||
|
import { snapdom } from '@zumer/snapdom'
|
|||
|
import html2canvas from 'html2canvas'
|
|||
|
import { getSharedImg } from '@/config/login'
|
|||
|
export default {
|
|||
|
name: 'SpecialSharePage',
|
|||
|
props: {
|
|||
|
qrCodeImage: {
|
|||
|
type: String,
|
|||
|
default: '',
|
|||
|
},
|
|||
|
userInfo: {
|
|||
|
type: Object,
|
|||
|
default: () => ({}),
|
|||
|
},
|
|||
|
isWechat: {
|
|||
|
type: Boolean,
|
|||
|
default: false,
|
|||
|
},
|
|||
|
isLoaded: {
|
|||
|
type: Boolean,
|
|||
|
default: false,
|
|||
|
},
|
|||
|
// 移除specialBackgroundImage prop,改为接口获取
|
|||
|
// 调试用:跳过图片验证
|
|||
|
skipImageVerification: {
|
|||
|
type: Boolean,
|
|||
|
default: false,
|
|||
|
},
|
|||
|
},
|
|||
|
data() {
|
|||
|
return {
|
|||
|
backgroundImageLoaded: false,
|
|||
|
imageLoadRetryCount: 0,
|
|||
|
maxRetryCount: 3,
|
|||
|
specialBackgroundImage: '', // 存储接口返回的背景图
|
|||
|
isLoadingBackground: false, // 是否正在加载背景图
|
|||
|
}
|
|||
|
},
|
|||
|
created() {
|
|||
|
this.getBackgroundImage()
|
|||
|
},
|
|||
|
methods: {
|
|||
|
desensitization(str) {
|
|||
|
if (!str) return ''
|
|||
|
if (str.length <= 8) return str.slice(0, 4) + '****'
|
|||
|
const len = str.length - 6
|
|||
|
const placeholder = '*'.repeat(len)
|
|||
|
return str.slice(0, 4) + placeholder + str.slice(-2)
|
|||
|
},
|
|||
|
|
|||
|
// 背景图片加载成功
|
|||
|
onBackgroundImageLoad() {
|
|||
|
this.backgroundImageLoaded = true
|
|||
|
console.log('特殊场景背景图片加载成功')
|
|||
|
},
|
|||
|
|
|||
|
// 背景图片加载失败
|
|||
|
onBackgroundImageError(e) {
|
|||
|
console.error('特殊场景背景图片加载失败:', e)
|
|||
|
this.backgroundImageLoaded = false
|
|||
|
this.imageLoadRetryCount++
|
|||
|
|
|||
|
// 如果重试次数未达到上限,可以通知父组件重试
|
|||
|
if (this.imageLoadRetryCount < this.maxRetryCount) {
|
|||
|
console.log(
|
|||
|
`图片加载失败,重试中 (${this.imageLoadRetryCount}/${this.maxRetryCount})`
|
|||
|
)
|
|||
|
this.$emit('background-image-retry', this.imageLoadRetryCount)
|
|||
|
} else {
|
|||
|
this.$emit('background-image-error', e)
|
|||
|
}
|
|||
|
},
|
|||
|
|
|||
|
// 获取特殊场景背景图片
|
|||
|
async getBackgroundImage() {
|
|||
|
if (this.skipImageVerification) {
|
|||
|
this.specialBackgroundImage = '' // 跳过验证时清空背景图
|
|||
|
this.backgroundImageLoaded = true
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
if (this.specialBackgroundImage) {
|
|||
|
// 如果背景图已加载,则直接使用
|
|||
|
this.backgroundImageLoaded = true
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
this.isLoadingBackground = true
|
|||
|
try {
|
|||
|
const result = await getSharedImg()
|
|||
|
if (result && result.code === 200 && result.data) {
|
|||
|
this.specialBackgroundImage =
|
|||
|
'data:image/png;base64,' + result.data.base64
|
|||
|
this.backgroundImageLoaded = true
|
|||
|
} else {
|
|||
|
this.backgroundImageLoaded = false
|
|||
|
this.imageLoadRetryCount++
|
|||
|
}
|
|||
|
} catch (error) {
|
|||
|
this.backgroundImageLoaded = false
|
|||
|
this.imageLoadRetryCount++
|
|||
|
if (this.imageLoadRetryCount < this.maxRetryCount) {
|
|||
|
this.$emit('background-image-retry', this.imageLoadRetryCount)
|
|||
|
} else {
|
|||
|
this.$emit('background-image-error', error)
|
|||
|
}
|
|||
|
} finally {
|
|||
|
this.isLoadingBackground = false
|
|||
|
}
|
|||
|
return this.specialBackgroundImage
|
|||
|
},
|
|||
|
|
|||
|
async generateShareImage() {
|
|||
|
// await this.getBackgroundImage()
|
|||
|
try {
|
|||
|
this.$nextTick(() => {
|
|||
|
const info = uni.getSystemInfoSync()
|
|||
|
console.log(info.platform, '.....info')
|
|||
|
if (info.platform === 'ios') {
|
|||
|
console.log('iOS detected, using html2canvas for capturing.')
|
|||
|
this.capturePageWithHtml2Canvas()
|
|||
|
} else {
|
|||
|
console.log('Non-iOS detected, using snapdom for capturing.')
|
|||
|
this.capturePageWithSnapDom()
|
|||
|
}
|
|||
|
})
|
|||
|
} catch (error) {
|
|||
|
uni.hideLoading()
|
|||
|
uni.showToast({ title: '图片生成失败,请稍后重试', icon: 'none' })
|
|||
|
}
|
|||
|
},
|
|||
|
|
|||
|
// 新增:使用html2canvas截取整个页面 (For iOS)
|
|||
|
async capturePageWithHtml2Canvas() {
|
|||
|
return new Promise(async (resolve, reject) => {
|
|||
|
// 确保所有图片都已加载完成
|
|||
|
const waitForImages = async () => {
|
|||
|
// 检查是否正在加载背景图片
|
|||
|
if (this.isLoadingBackground) {
|
|||
|
console.log('等待背景图片接口调用完成...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
// 检查背景图片是否加载完成
|
|||
|
if (!this.backgroundImageLoaded) {
|
|||
|
console.log('等待背景图片加载完成...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
// 检查二维码是否存在
|
|||
|
if (!this.qrCodeImage) {
|
|||
|
console.log('等待二维码生成...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
// 检查特殊背景图片参数是否存在
|
|||
|
if (!this.specialBackgroundImage) {
|
|||
|
console.log('等待特殊背景图片参数...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
console.log('所有图片准备就绪,开始使用html2canvas截图...')
|
|||
|
|
|||
|
// 等待确保渲染完成
|
|||
|
setTimeout(() => {
|
|||
|
const element = this.$el
|
|||
|
if (!element) {
|
|||
|
uni.hideLoading()
|
|||
|
reject(new Error('找不到组件容器'))
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
html2canvas(element, {
|
|||
|
useCORS: true,
|
|||
|
allowTaint: true,
|
|||
|
backgroundColor: null,
|
|||
|
scale: 2, // 提高清晰度
|
|||
|
dpi: 300,
|
|||
|
width: element.offsetWidth,
|
|||
|
height: element.offsetHeight,
|
|||
|
})
|
|||
|
.then(canvas => {
|
|||
|
const dataUrl = canvas.toDataURL('image/jpeg', 1.0)
|
|||
|
uni.hideLoading()
|
|||
|
// this.handleShareGenerated(dataUrl)
|
|||
|
this.$emit('share-generated', dataUrl)
|
|||
|
resolve()
|
|||
|
})
|
|||
|
.catch(err => {
|
|||
|
uni.hideLoading()
|
|||
|
console.error('html2canvas截取失败:', err)
|
|||
|
uni.showToast({
|
|||
|
title: '图片生成失败,请稍后重试',
|
|||
|
icon: 'none',
|
|||
|
})
|
|||
|
reject(err)
|
|||
|
})
|
|||
|
}, 1000)
|
|||
|
}
|
|||
|
|
|||
|
// 开始等待图片加载
|
|||
|
waitForImages()
|
|||
|
})
|
|||
|
},
|
|||
|
|
|||
|
// 使用snapDOM截取整个页面
|
|||
|
async capturePageWithSnapDom() {
|
|||
|
return new Promise(async (resolve, reject) => {
|
|||
|
// 确保所有图片都已加载完成
|
|||
|
const waitForImages = async () => {
|
|||
|
// 检查是否正在加载背景图片
|
|||
|
if (this.isLoadingBackground) {
|
|||
|
console.log('等待背景图片接口调用完成...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
// 检查背景图片是否加载完成
|
|||
|
if (!this.backgroundImageLoaded) {
|
|||
|
console.log('等待背景图片加载完成...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
// 检查二维码是否存在
|
|||
|
if (!this.qrCodeImage) {
|
|||
|
console.log('等待二维码生成...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
// 检查特殊背景图片参数是否存在
|
|||
|
if (!this.specialBackgroundImage) {
|
|||
|
console.log('等待特殊背景图片参数...')
|
|||
|
setTimeout(waitForImages, 100)
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
console.log('所有图片准备就绪,开始截图...')
|
|||
|
|
|||
|
// 等待确保渲染完成
|
|||
|
setTimeout(async () => {
|
|||
|
const element = this.$el
|
|||
|
if (!element) {
|
|||
|
reject(new Error('找不到组件容器'))
|
|||
|
return
|
|||
|
}
|
|||
|
|
|||
|
try {
|
|||
|
// 获取元素的所有可能尺寸
|
|||
|
const rect = element.getBoundingClientRect()
|
|||
|
const offsetWidth = element.offsetWidth
|
|||
|
const offsetHeight = element.offsetHeight
|
|||
|
const scrollWidth = element.scrollWidth
|
|||
|
const scrollHeight = element.scrollHeight
|
|||
|
const clientWidth = element.clientWidth
|
|||
|
const clientHeight = element.clientHeight
|
|||
|
|
|||
|
// 计算实际需要的尺寸(取最大值确保完整)
|
|||
|
const elementWidth = Math.max(
|
|||
|
offsetWidth,
|
|||
|
scrollWidth,
|
|||
|
clientWidth,
|
|||
|
rect.width
|
|||
|
)
|
|||
|
const elementHeight = Math.max(
|
|||
|
offsetHeight,
|
|||
|
scrollHeight,
|
|||
|
clientHeight,
|
|||
|
rect.height
|
|||
|
)
|
|||
|
|
|||
|
setTimeout(async () => {
|
|||
|
try {
|
|||
|
const result = await snapdom(element, {
|
|||
|
width: elementWidth,
|
|||
|
height: elementHeight,
|
|||
|
quality: 1,
|
|||
|
compress: false,
|
|||
|
|
|||
|
// useProxy: true,
|
|||
|
})
|
|||
|
const canvas = await result.toCanvas()
|
|||
|
|
|||
|
const dataUrl = canvas.toDataURL('image/jpeg')
|
|||
|
|
|||
|
uni.hideLoading()
|
|||
|
// this.handleShareGenerated(dataUrl)
|
|||
|
this.$emit('share-generated', dataUrl)
|
|||
|
resolve()
|
|||
|
} catch (err) {
|
|||
|
uni.hideLoading()
|
|||
|
console.error('snapDOM截取失败:', err)
|
|||
|
uni.showToast({
|
|||
|
title: '图片生成失败,请稍后重试',
|
|||
|
icon: 'none',
|
|||
|
})
|
|||
|
reject(err)
|
|||
|
}
|
|||
|
}, 500)
|
|||
|
} catch (err) {
|
|||
|
uni.hideLoading()
|
|||
|
console.error('snapDOM初始化失败:', err)
|
|||
|
uni.showToast({
|
|||
|
title: '图片生成失败,请稍后重试',
|
|||
|
icon: 'none',
|
|||
|
})
|
|||
|
reject(err)
|
|||
|
}
|
|||
|
}, 1000)
|
|||
|
}
|
|||
|
|
|||
|
// 开始等待图片加载
|
|||
|
waitForImages()
|
|||
|
})
|
|||
|
},
|
|||
|
},
|
|||
|
}
|
|||
|
</script>
|
|||
|
|
|||
|
<style lang="scss" scoped>
|
|||
|
.special-share-page {
|
|||
|
position: relative;
|
|||
|
width: 100%;
|
|||
|
height: 100%;
|
|||
|
display: flex;
|
|||
|
flex-direction: column;
|
|||
|
align-items: center;
|
|||
|
}
|
|||
|
|
|||
|
/* 背景图片样式 */
|
|||
|
.share-bg-image {
|
|||
|
position: absolute;
|
|||
|
top: 0;
|
|||
|
left: 0;
|
|||
|
width: 100%;
|
|||
|
height: 100%;
|
|||
|
z-index: 1;
|
|||
|
}
|
|||
|
|
|||
|
/* 加载状态 */
|
|||
|
.loading-container {
|
|||
|
position: absolute;
|
|||
|
top: 50%;
|
|||
|
left: 50%;
|
|||
|
transform: translate(-50%, -50%);
|
|||
|
display: flex;
|
|||
|
flex-direction: column;
|
|||
|
align-items: center;
|
|||
|
z-index: 2;
|
|||
|
}
|
|||
|
|
|||
|
.loading-text {
|
|||
|
margin-top: 20rpx;
|
|||
|
font-size: 28rpx;
|
|||
|
color: #666;
|
|||
|
}
|
|||
|
|
|||
|
.share-wrapper {
|
|||
|
position: absolute;
|
|||
|
z-index: 2;
|
|||
|
bottom: 22%;
|
|||
|
display: flex;
|
|||
|
flex-direction: column;
|
|||
|
align-items: center;
|
|||
|
}
|
|||
|
|
|||
|
.portal-frame {
|
|||
|
padding: 32rpx;
|
|||
|
width: 520rpx;
|
|||
|
border-radius: 40rpx;
|
|||
|
display: flex;
|
|||
|
box-sizing: border-box;
|
|||
|
flex-direction: column;
|
|||
|
align-items: center;
|
|||
|
margin: 0 auto;
|
|||
|
opacity: 0;
|
|||
|
transform: translateY(20rpx);
|
|||
|
transition: all 0.3s ease;
|
|||
|
}
|
|||
|
|
|||
|
.portal-frame.is-loaded {
|
|||
|
opacity: 1;
|
|||
|
transform: translateY(0);
|
|||
|
}
|
|||
|
|
|||
|
/* 特殊场景二维码样式区域 - 用户自定义样式位置 */
|
|||
|
.qr-code-outer {
|
|||
|
width: 400rpx;
|
|||
|
height: 400rpx;
|
|||
|
border-radius: 20rpx;
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
justify-content: center;
|
|||
|
padding: 12rpx;
|
|||
|
box-sizing: border-box;
|
|||
|
}
|
|||
|
|
|||
|
.qr-code,
|
|||
|
.qr-code-placeholder {
|
|||
|
width: 100%;
|
|||
|
height: 100%;
|
|||
|
}
|
|||
|
|
|||
|
.qr-code {
|
|||
|
border-radius: 16rpx;
|
|||
|
}
|
|||
|
|
|||
|
.qr-code-placeholder {
|
|||
|
display: flex;
|
|||
|
align-items: center;
|
|||
|
justify-content: center;
|
|||
|
}
|
|||
|
|
|||
|
.loader {
|
|||
|
width: 100rpx;
|
|||
|
height: 100rpx;
|
|||
|
border: 8rpx solid rgba(0, 0, 0, 0.1);
|
|||
|
border-left-color: #0072ff;
|
|||
|
border-radius: 50%;
|
|||
|
animation: spin 1s linear infinite;
|
|||
|
}
|
|||
|
|
|||
|
@keyframes spin {
|
|||
|
to {
|
|||
|
transform: rotate(360deg);
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
/* 特殊场景会员编号样式区域 - 用户自定义样式位置 */
|
|||
|
.member-code-text {
|
|||
|
/* 用户可以在这里自定义会员编号的样式 */
|
|||
|
}
|
|||
|
|
|||
|
/* 特殊场景下的会员编号样式 */
|
|||
|
.special-member-code-style {
|
|||
|
/* 用户可以在这里自定义特殊场景下的会员编号样式 */
|
|||
|
/* 例如:不同的位置、颜色、字体等 */
|
|||
|
}
|
|||
|
</style>
|