完成视频聊天页面,优化修改花呗逾期页面
|
|
@ -1,5 +1,5 @@
|
||||||
<template>
|
<template>
|
||||||
<view style="width: 100%;" :style="{ height: `${data.statusBarHeight + 44}px` }"></view>
|
<view style="width: 100%;" :style="{ height: `calc(${data.statusBarHeight}px + 88rpx)` }"></view>
|
||||||
<view class="nav-bar-container" :style="{ backgroundColor: bgColor, zIndex: zIndex }">
|
<view class="nav-bar-container" :style="{ backgroundColor: bgColor, zIndex: zIndex }">
|
||||||
<view class="status-placeholder" :style="{ height: `${data.statusBarHeight}px` }"></view>
|
<view class="status-placeholder" :style="{ height: `${data.statusBarHeight}px` }"></view>
|
||||||
<view style="width: 100%;height: 88rpx;" @click="openPopup">
|
<view style="width: 100%;height: 88rpx;" @click="openPopup">
|
||||||
|
|
@ -57,21 +57,35 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</popup>
|
</popup>
|
||||||
|
|
||||||
|
|
||||||
|
<view class="tipLayer" :style="{ top: `${45 + data.statusBarHeight}px` }" v-if="isTipLayer && showTipLayer">
|
||||||
|
<view class="tipLayer-content">
|
||||||
|
<view class="title">
|
||||||
|
<slot name="tipLayer">点击此处<text>[{{ tipLayerText }}]</text></slot>
|
||||||
|
</view>
|
||||||
|
<image class="close" src="/static/image/common/tipLayer-close.png" mode="" @click="closeTipLayer">
|
||||||
|
</image>
|
||||||
|
<image class="triangleImg" src="/static/image/common/tipLayer-eye.png"></image>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup>
|
<script setup>
|
||||||
import popup from '../popup/popup.vue'
|
import popup from '../popup/popup.vue'
|
||||||
import {
|
import {
|
||||||
onMounted,
|
onMounted,
|
||||||
reactive,
|
reactive,
|
||||||
ref
|
ref,
|
||||||
} from 'vue'
|
toRefs
|
||||||
|
} from 'vue'
|
||||||
|
|
||||||
const topPopup = ref()
|
const topPopup = ref()
|
||||||
|
|
||||||
// 定义组件属性
|
// 定义组件属性
|
||||||
const props = defineProps({
|
const props = defineProps({
|
||||||
bgColor: {
|
bgColor: {
|
||||||
type: String,
|
type: String,
|
||||||
default: '#fff'
|
default: '#fff'
|
||||||
|
|
@ -107,52 +121,83 @@ const props = defineProps({
|
||||||
noBack: {
|
noBack: {
|
||||||
type: Boolean,
|
type: Boolean,
|
||||||
default: false
|
default: false
|
||||||
|
},
|
||||||
|
tipLayerText: {
|
||||||
|
type: String,
|
||||||
|
default: ''
|
||||||
|
},
|
||||||
|
isTipLayer: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false
|
||||||
|
},
|
||||||
|
tipLayerType: {
|
||||||
|
type: String,
|
||||||
|
default: ''
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
// 定义事件
|
// 定义事件
|
||||||
const emit = defineEmits(['back', 'right-click', 'button-click'])
|
const emit = defineEmits(['back', 'right-click', 'button-click', 'refresh'])
|
||||||
|
|
||||||
const data = reactive({
|
const data = reactive({
|
||||||
statusBarHeight: 0
|
statusBarHeight: 0,
|
||||||
})
|
showTipLayer: true,
|
||||||
|
})
|
||||||
|
|
||||||
onMounted(() => {
|
let {
|
||||||
|
showTipLayer
|
||||||
|
} = toRefs(data)
|
||||||
|
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
// 同步获取系统信息
|
// 同步获取系统信息
|
||||||
const systemInfo = uni.getSystemInfoSync();
|
const systemInfo = uni.getSystemInfoSync();
|
||||||
data.statusBarHeight = systemInfo.statusBarHeight || 0;
|
data.statusBarHeight = systemInfo.statusBarHeight || 0;
|
||||||
})
|
|
||||||
|
|
||||||
const openPopup = () => {
|
if (props.isTipLayer) {
|
||||||
|
if (uni.getStorageSync(props.tipLayerType) == props.tipLayerType) {
|
||||||
|
showTipLayer.value = false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
const closeTipLayer = () => {
|
||||||
|
showTipLayer.value = false
|
||||||
|
uni.setStorageSync(props.tipLayerType, props.tipLayerType)
|
||||||
|
emit("refresh")
|
||||||
|
}
|
||||||
|
|
||||||
|
const openPopup = () => {
|
||||||
if (props.buttonGroup.length > 0) {
|
if (props.buttonGroup.length > 0) {
|
||||||
topPopup.value.open()
|
topPopup.value.open()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 返回按钮点击事件
|
// 返回按钮点击事件
|
||||||
const onBack = () => {
|
const onBack = () => {
|
||||||
emit('back')
|
emit('back')
|
||||||
// 默认返回上一页
|
// 默认返回上一页
|
||||||
if (props.noBack) return
|
if (props.noBack) return
|
||||||
uni.navigateBack()
|
uni.navigateBack()
|
||||||
}
|
}
|
||||||
|
|
||||||
// 右侧按钮点击事件
|
// 右侧按钮点击事件
|
||||||
const onRightClick = () => {
|
const onRightClick = () => {
|
||||||
emit('right-click')
|
emit('right-click')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
const buttonClick = (button) => {
|
const buttonClick = (button) => {
|
||||||
topPopup.value.close()
|
topPopup.value.close()
|
||||||
emit('button-click', button)
|
emit('button-click', button)
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style scoped>
|
<style scoped>
|
||||||
@import "/common/main.css";
|
@import "/common/main.css";
|
||||||
|
|
||||||
.nav-bar-container {
|
.nav-bar-container {
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
position: fixed !important;
|
position: fixed !important;
|
||||||
|
|
@ -160,42 +205,42 @@ const buttonClick = (button) => {
|
||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
z-index: 999;
|
z-index: 999;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-bar {
|
.nav-bar {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: space-between;
|
justify-content: space-between;
|
||||||
position: relative;
|
position: relative;
|
||||||
z-index: 1;
|
z-index: 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
.status-placeholder {
|
.status-placeholder {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
::v-deep .uni-navbar__content {
|
::v-deep .uni-navbar__content {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-bar-left {
|
.nav-bar-left {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: flex-start;
|
justify-content: flex-start;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-icon-back {
|
.nav-icon-back {
|
||||||
width: 24px;
|
width: 24px;
|
||||||
height: 24px;
|
height: 24px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-icon-more {
|
.nav-icon-more {
|
||||||
width: 26px;
|
width: 26px;
|
||||||
height: 26px;
|
height: 26px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-bar-title {
|
.nav-bar-title {
|
||||||
flex: 1;
|
flex: 1;
|
||||||
margin: auto;
|
margin: auto;
|
||||||
display: flex;
|
display: flex;
|
||||||
|
|
@ -203,15 +248,15 @@ const buttonClick = (button) => {
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
font-size: 17px;
|
font-size: 17px;
|
||||||
font-weight: 500;
|
font-weight: 500;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-bar-right {
|
.nav-bar-right {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: flex-end;
|
justify-content: flex-end;
|
||||||
}
|
}
|
||||||
|
|
||||||
.right-button {
|
.right-button {
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
border-radius: 16px;
|
border-radius: 16px;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
|
|
@ -220,19 +265,85 @@ const buttonClick = (button) => {
|
||||||
height: 30px;
|
height: 30px;
|
||||||
min-width: 60px;
|
min-width: 60px;
|
||||||
background: linear-gradient(90deg, #187AFF 0%, #3295FC 100%);
|
background: linear-gradient(90deg, #187AFF 0%, #3295FC 100%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.button-box {
|
.button-box {
|
||||||
width: calc(50% - 8rpx);
|
width: calc(50% - 8rpx);
|
||||||
text-align: center;
|
text-align: center;
|
||||||
margin-top: 16rpx;
|
margin-top: 16rpx;
|
||||||
}
|
}
|
||||||
|
|
||||||
.button {
|
.button {
|
||||||
border: 1px solid #E4E4E4;
|
border: 1px solid #E4E4E4;
|
||||||
border-radius: 8px;
|
border-radius: 8px;
|
||||||
height: 42px;
|
height: 42px;
|
||||||
line-height: 42px;
|
line-height: 42px;
|
||||||
font-size: 28rpx;
|
font-size: 28rpx;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.tipLayer {
|
||||||
|
box-sizing: border-box;
|
||||||
|
min-width: 200px !important;
|
||||||
|
height: 48px;
|
||||||
|
background: #B8EDFE;
|
||||||
|
border-radius: 8px 8px 8px 8px;
|
||||||
|
position: fixed;
|
||||||
|
/* top: 115px; */
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
z-index: 999;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.tipLayer-content {
|
||||||
|
position: relative;
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.title {
|
||||||
|
font-weight: 450;
|
||||||
|
font-size: 14px;
|
||||||
|
color: #268FFF;
|
||||||
|
line-height: 48px;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
text {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: bold;
|
||||||
|
color: #006ADD;
|
||||||
|
}
|
||||||
|
|
||||||
|
::v-deep text {
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: bold;
|
||||||
|
color: #006ADD;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.triangleImg {
|
||||||
|
width: 111px;
|
||||||
|
height: 52px;
|
||||||
|
pointer-events: none;
|
||||||
|
position: absolute;
|
||||||
|
top: -23px;
|
||||||
|
left: calc(50% - 111px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.triangle {
|
||||||
|
position: absolute;
|
||||||
|
top: -57px;
|
||||||
|
left: calc(50% - 40px);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close {
|
||||||
|
position: absolute;
|
||||||
|
top: -5px;
|
||||||
|
right: -5px;
|
||||||
|
width: 18px;
|
||||||
|
height: 18px;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
2
main.js
|
|
@ -27,7 +27,7 @@ export function createApp() {
|
||||||
const systemInfo = uni.getStorageSync('systemInfo') || {}
|
const systemInfo = uni.getStorageSync('systemInfo') || {}
|
||||||
app.config.globalProperties.$system = systemInfo.platform == 'ios' ? 'iOS' : 'Android'
|
app.config.globalProperties.$system = systemInfo.platform == 'ios' ? 'iOS' : 'Android'
|
||||||
app.config.globalProperties.$systemInfo = systemInfo
|
app.config.globalProperties.$systemInfo = systemInfo
|
||||||
uni.setStorageSync('version', '1.0.0.sp4')
|
uni.setStorageSync('version', '1.0.0.sp7')
|
||||||
app.config.globalProperties.$version = uni.getStorageSync('version')
|
app.config.globalProperties.$version = uni.getStorageSync('version')
|
||||||
|
|
||||||
app.use(globalMethods);
|
app.use(globalMethods);
|
||||||
|
|
|
||||||
102
manifest.json
|
|
@ -1,37 +1,37 @@
|
||||||
{
|
{
|
||||||
"name": "alipay-emulator",
|
"name" : "alipay-emulator",
|
||||||
"appid": "__UNI__D535736",
|
"appid" : "__UNI__D535736",
|
||||||
"description": "",
|
"description" : "",
|
||||||
"versionName": "1.0.0",
|
"versionName" : "1.0.0",
|
||||||
"versionCode": "100",
|
"versionCode" : 100,
|
||||||
"transformPx": false,
|
"transformPx" : false,
|
||||||
/* 5+App特有相关 */
|
/* 5+App特有相关 */
|
||||||
"app-plus": {
|
"app-plus" : {
|
||||||
"darkmode": false,
|
"darkmode" : false,
|
||||||
"usingComponents": true,
|
"usingComponents" : true,
|
||||||
"nvueStyleCompiler": "uni-app",
|
"nvueStyleCompiler" : "uni-app",
|
||||||
"compilerVersion": 3,
|
"compilerVersion" : 3,
|
||||||
"splashscreen": {
|
"splashscreen" : {
|
||||||
"alwaysShowBeforeRender": true,
|
"alwaysShowBeforeRender" : true,
|
||||||
"waiting": true,
|
"waiting" : true,
|
||||||
"autoclose": true,
|
"autoclose" : true,
|
||||||
"delay": 0
|
"delay" : 0
|
||||||
},
|
},
|
||||||
"optimization": {
|
"optimization" : {
|
||||||
"subPackages": true
|
"subPackages" : true
|
||||||
},
|
},
|
||||||
"runmode": "liberate", // 开启分包优化后,必须配置资源释放模式
|
"runmode" : "liberate", // 开启分包优化后,必须配置资源释放模式
|
||||||
|
|
||||||
/* 模块配置 */
|
/* 模块配置 */
|
||||||
"modules": {
|
"modules" : {
|
||||||
"Camera": {},
|
"Camera" : {},
|
||||||
"Payment": {}
|
"Payment" : {}
|
||||||
},
|
},
|
||||||
/* 应用发布信息 */
|
/* 应用发布信息 */
|
||||||
"distribute": {
|
"distribute" : {
|
||||||
/* android打包配置 */
|
/* android打包配置 */
|
||||||
"android": {
|
"android" : {
|
||||||
"permissions": [
|
"permissions" : [
|
||||||
"<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>",
|
"<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>",
|
||||||
"<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>",
|
"<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>",
|
||||||
"<uses-permission android:name=\"android.permission.VIBRATE\"/>",
|
"<uses-permission android:name=\"android.permission.VIBRATE\"/>",
|
||||||
|
|
@ -50,46 +50,46 @@
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
/* ios打包配置 */
|
/* ios打包配置 */
|
||||||
"ios": {
|
"ios" : {
|
||||||
"dSYMs": false
|
"dSYMs" : false
|
||||||
},
|
},
|
||||||
/* SDK配置 */
|
/* SDK配置 */
|
||||||
"sdkConfigs": {
|
"sdkConfigs" : {
|
||||||
"payment": {
|
"payment" : {
|
||||||
"weixin": {
|
"weixin" : {
|
||||||
"__platform__": ["ios", "android"],
|
"__platform__" : [ "ios", "android" ],
|
||||||
"appid": "123456",
|
"appid" : "123456",
|
||||||
"UniversalLinks": "https://hhhhh.com/apple-app-site-association/"
|
"UniversalLinks" : "https://hhhhh.com/apple-app-site-association/"
|
||||||
},
|
},
|
||||||
"alipay": {
|
"alipay" : {
|
||||||
"__platform__": ["ios", "android"]
|
"__platform__" : [ "ios", "android" ]
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"nvueLaunchMode": ""
|
"nvueLaunchMode" : ""
|
||||||
},
|
},
|
||||||
/* 快应用特有相关 */
|
/* 快应用特有相关 */
|
||||||
"quickapp": {},
|
"quickapp" : {},
|
||||||
/* 小程序特有相关 */
|
/* 小程序特有相关 */
|
||||||
"mp-weixin": {
|
"mp-weixin" : {
|
||||||
"appid": "",
|
"appid" : "",
|
||||||
"setting": {
|
"setting" : {
|
||||||
"urlCheck": false
|
"urlCheck" : false
|
||||||
},
|
},
|
||||||
"usingComponents": true
|
"usingComponents" : true
|
||||||
},
|
},
|
||||||
"mp-alipay": {
|
"mp-alipay" : {
|
||||||
"usingComponents": true
|
"usingComponents" : true
|
||||||
},
|
},
|
||||||
"mp-baidu": {
|
"mp-baidu" : {
|
||||||
"usingComponents": true
|
"usingComponents" : true
|
||||||
},
|
},
|
||||||
"mp-toutiao": {
|
"mp-toutiao" : {
|
||||||
"usingComponents": true
|
"usingComponents" : true
|
||||||
},
|
},
|
||||||
"uniStatistics": {
|
"uniStatistics" : {
|
||||||
"enable": false
|
"enable" : false
|
||||||
},
|
},
|
||||||
"vueVersion": "3"
|
"vueVersion" : "3"
|
||||||
}
|
}
|
||||||
19
pages.json
|
|
@ -95,6 +95,25 @@
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
"root": "pages/other",
|
||||||
|
"pages": [{
|
||||||
|
"path": "/video-group-chat/video-group-chat",
|
||||||
|
"style": {
|
||||||
|
"navigationBarTitleText": "视频群聊",
|
||||||
|
"navigationStyle": "custom",
|
||||||
|
"navigationBarTextStyle": "white"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"path": "splash/splash",
|
||||||
|
"style": {
|
||||||
|
"navigationBarTitleText": "工资单",
|
||||||
|
"navigationStyle": "custom"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
{
|
{
|
||||||
"root": "pages/common",
|
"root": "pages/common",
|
||||||
"pages": [{
|
"pages": [{
|
||||||
|
|
|
||||||
|
|
@ -8,8 +8,8 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="page-container">
|
<view class="page-container">
|
||||||
<view class="main-container">
|
<view class="main-container">
|
||||||
<NavBar v-if="!selectedImage" title="花呗" :bgColor="data.navBar.bgColor" :buttonGroup="buttonGroup"
|
<NavBar v-if="!selectedImage" title="花呗" :bgColor="data.navBar.bgColor" tipLayerType="huabei-tip" isTipLayer
|
||||||
@button-click="clickTitlePopupButton">
|
tipLayerText="修改花呗信息" :buttonGroup="buttonGroup" @button-click="clickTitlePopupButton">
|
||||||
<!-- 使用作用域插槽自定义按钮渲染,特别是switch的checked绑定 -->
|
<!-- 使用作用域插槽自定义按钮渲染,特别是switch的checked绑定 -->
|
||||||
<template #button="{ button }">
|
<template #button="{ button }">
|
||||||
<view class="button flex-align-center flex-justify-center">
|
<view class="button flex-align-center flex-justify-center">
|
||||||
|
|
@ -32,7 +32,7 @@
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<view class="right">
|
<view class="right flex-align-center">
|
||||||
<image class="icon" src="/static/image/ant-credit-pay/service.png"></image>
|
<image class="icon" src="/static/image/ant-credit-pay/service.png"></image>
|
||||||
<image class="icon" src="/static/image/ant-credit-pay/setting.png"></image>
|
<image class="icon" src="/static/image/ant-credit-pay/setting.png"></image>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -45,7 +45,7 @@
|
||||||
<view v-else class="current-month">{{ huabeiInfo.mouth }}月账单累计中(元)</view>
|
<view v-else class="current-month">{{ huabeiInfo.mouth }}月账单累计中(元)</view>
|
||||||
<view class="money-box flex-align-center">
|
<view class="money-box flex-align-center">
|
||||||
<text class="money alipay-font">{{ numberUtil.formatMoneyWithThousand(huabeiInfo.money) }}</text>
|
<text class="money alipay-font">{{ numberUtil.formatMoneyWithThousand(huabeiInfo.money) }}</text>
|
||||||
<uni-icons type="right" size="16" color="#B9D6FF"></uni-icons>
|
<uni-icons type="right" size="18" color="#B9D6FF"></uni-icons>
|
||||||
</view>
|
</view>
|
||||||
<!-- 样式一 按钮样式 -->
|
<!-- 样式一 按钮样式 -->
|
||||||
<view v-if="huabeiInfo.styleType == 1 || !huabeiInfo.styleType" class="style-1 button-group">
|
<view v-if="huabeiInfo.styleType == 1 || !huabeiInfo.styleType" class="style-1 button-group">
|
||||||
|
|
@ -69,7 +69,7 @@
|
||||||
<view class="bubble-box">
|
<view class="bubble-box">
|
||||||
<view class="arrow"></view>
|
<view class="arrow"></view>
|
||||||
<text class="text flex-align-center">{{ huabeiInfo.descText }}
|
<text class="text flex-align-center">{{ huabeiInfo.descText }}
|
||||||
<uni-icons type="right" size="16" color="#B9D6FF"></uni-icons>
|
<uni-icons type="right" size="18" color="#B9D6FF"></uni-icons>
|
||||||
</text>
|
</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -249,7 +249,8 @@ const data = reactive({
|
||||||
installmentBadgeText: '4折起',
|
installmentBadgeText: '4折起',
|
||||||
image: "",
|
image: "",
|
||||||
isOverdue: false,
|
isOverdue: false,
|
||||||
daysPastDue: 1
|
daysPastDue: 1,
|
||||||
|
isOverdueDeactivate: false,
|
||||||
},
|
},
|
||||||
selectedImage: '',
|
selectedImage: '',
|
||||||
showMask: false
|
showMask: false
|
||||||
|
|
|
||||||
|
|
@ -7,14 +7,15 @@
|
||||||
<view class="header-bg"></view>
|
<view class="header-bg"></view>
|
||||||
|
|
||||||
<!-- 导航栏 - 降低层级以显示水印 -->
|
<!-- 导航栏 - 降低层级以显示水印 -->
|
||||||
<NavBar title="花呗 | 信用购" :bgColor="'#717E91'" textColor="#fff" :isFixed="true" :zIndex="10"
|
<NavBar title="花呗 | 信用购" :bgColor="data.navBar.bgColor" :textColor="data.navBar.textColor" :isFixed="true"
|
||||||
:buttonGroup="buttonGroup" @button-click="clickTitlePopupButton">
|
:zIndex="10" tipLayerType="huabei-tip" isTipLayer tipLayerText="修改花呗信息" :buttonGroup="buttonGroup"
|
||||||
|
@button-click="clickTitlePopupButton">
|
||||||
<!-- 使用作用域插槽自定义按钮渲染,特别是switch的checked绑定 -->
|
<!-- 使用作用域插槽自定义按钮渲染,特别是switch的checked绑定 -->
|
||||||
<template #button="{ button }">
|
<template #button="{ button }">
|
||||||
<view class="button flex-align-center flex-justify-center">
|
<view class="button flex-align-center flex-justify-center">
|
||||||
{{ button.name }}
|
{{ button.name }}
|
||||||
<view @tap.stop>
|
<view @tap.stop>
|
||||||
<switch v-if="button.isSwitch" :checked="data.huabeiInfo.isOverdue" @change="button.click"
|
<switch v-if="button.isSwitch" :checked="data.huabeiInfo[button.key]" @change="button.click"
|
||||||
style="transform: scale(0.7);"></switch>
|
style="transform: scale(0.7);"></switch>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -22,33 +23,61 @@
|
||||||
<view class="nav-bar flex-between w100" :class="{ 'ios-nav-bar': $system == 'iOS' }">
|
<view class="nav-bar flex-between w100" :class="{ 'ios-nav-bar': $system == 'iOS' }">
|
||||||
<view class="flex-align-center flex-1">
|
<view class="flex-align-center flex-1">
|
||||||
<view class="left flex-align-center" @click.stop="goBack">
|
<view class="left flex-align-center" @click.stop="goBack">
|
||||||
<image class=" back-icon" src="/static/image/nav-bar/back-white.png" mode="">
|
<image class=" back-icon"
|
||||||
|
:src="`/static/image/nav-bar/back-${data.isTop ? 'white' : 'black'}.png`" mode="">
|
||||||
</image>
|
</image>
|
||||||
</view>
|
</view>
|
||||||
<view class="title flex-align-center">
|
<view class="title flex-align-center" :style="{ color: data.navBar.textColor }">
|
||||||
花呗
|
花呗
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<view class="right">
|
<view class="right flex-align-center">
|
||||||
<image class="icon" src="/static/image/ant-credit-pay/service.png"></image>
|
<image class="icon"
|
||||||
<image class="icon" src="/static/image/ant-credit-pay/setting.png"></image>
|
:src="`/static/image/ant-credit-pay/${data.isTop ? 'service' : 'service-black'}.png`">
|
||||||
|
</image>
|
||||||
|
<image class="icon"
|
||||||
|
:src="`/static/image/ant-credit-pay/${data.isTop ? 'setting' : 'setting-black'}.png`">
|
||||||
|
</image>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</NavBar>
|
</NavBar>
|
||||||
|
|
||||||
<view class="header-content-wrapper">
|
<view class="header-content-wrapper">
|
||||||
<view class="current-month flex-align-center flex-justify-center">逾期金额(元) | <text
|
<view v-if="!huabeiInfo.isOverdueDeactivate"
|
||||||
style="color: #FFF3A8;margin-left: 14rpx;">
|
class="current-month flex-align-center flex-justify-center">
|
||||||
|
逾期金额(元) |
|
||||||
|
<text style="color: #FFF3A8;margin-left: 14rpx;">
|
||||||
已逾期{{ huabeiInfo.daysPastDue }}天,逾期影响信用</text>
|
已逾期{{ huabeiInfo.daysPastDue }}天,逾期影响信用</text>
|
||||||
</view>
|
</view>
|
||||||
|
<view v-else class="current-month">{{ huabeiInfo.mouth }}月应还(元)</view>
|
||||||
<view class="money-box flex-align-center">
|
<view class="money-box flex-align-center">
|
||||||
<text class="money alipay-font">{{ numberUtil.formatMoneyWithThousand(huabeiInfo.money) }}</text>
|
<text class="money alipay-font">{{ numberUtil.formatMoneyWithThousand(huabeiInfo.money) }}</text>
|
||||||
<uni-icons type="right" size="16" color="#D2DBE3"></uni-icons>
|
<image style="width: 32rpx;height: 32rpx;" class="icon"
|
||||||
|
src="/static/image/ant-credit-pay/overdue-payment/right-icon.png"></image>
|
||||||
</view>
|
</view>
|
||||||
<view class="style-1 button-group">
|
<!-- 样式一 按钮样式 -->
|
||||||
|
<view v-if="huabeiInfo.styleType == 1 || !huabeiInfo.styleType" class="style-1 button-group">
|
||||||
<view class="button-item second-button" :class="{ 'ios-button': $system == 'iOS' }">立即还款</view>
|
<view class="button-item second-button" :class="{ 'ios-button': $system == 'iOS' }">立即还款</view>
|
||||||
</view>
|
</view>
|
||||||
|
<!-- 样式二 纯气泡样式 -->
|
||||||
|
<view v-if="huabeiInfo.styleType == 2" class="style-2 bubble-container">
|
||||||
|
<view class="bubble-box">
|
||||||
|
<view class="arrow"></view>
|
||||||
|
<text class="text">{{ huabeiInfo.descText }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<!-- 样式三 气泡带箭头样式 -->
|
||||||
|
<view v-if="huabeiInfo.styleType == 3" class="style-3 bubble-container">
|
||||||
|
<view class="bubble-box">
|
||||||
|
<view class="arrow"></view>
|
||||||
|
<view class="text flex-align-center">{{ huabeiInfo.descText }}
|
||||||
|
<!-- <uni-icons type="right" size="16" color="#D2DBE3"></uni-icons> -->
|
||||||
|
<image style="width: 32rpx;height: 32rpx;" class="icon"
|
||||||
|
src="/static/image/ant-credit-pay/overdue-payment/right-icon.png"></image>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
<view class="total-info-box flex-align-center">
|
<view class="total-info-box flex-align-center">
|
||||||
<view class="info-item">
|
<view class="info-item">
|
||||||
<view class="label">总计账单</view>
|
<view class="label">总计账单</view>
|
||||||
|
|
@ -69,7 +98,10 @@
|
||||||
<view class="overdue-info">
|
<view class="overdue-info">
|
||||||
<image class="icon" src="/static/image/ant-credit-pay/overdue-payment/warring-icon.png"></image>
|
<image class="icon" src="/static/image/ant-credit-pay/overdue-payment/warring-icon.png"></image>
|
||||||
<view class="err-text">抱歉,您暂时无法使用该服务!服务机构将不定期评估您的使用资格,请保持良好信用行为并耐心等待通知</view>
|
<view class="err-text">抱歉,您暂时无法使用该服务!服务机构将不定期评估您的使用资格,请保持良好信用行为并耐心等待通知</view>
|
||||||
<view class="info-text">你已逾期{{ huabeiInfo.daysPastDue }}天,还款后花呗可恢复正常</view>
|
<view v-if="!huabeiInfo.isOverdueDeactivate" class="info-text">
|
||||||
|
你已逾期{{ huabeiInfo.daysPastDue }}天,还款后花呗可恢复正常
|
||||||
|
</view>
|
||||||
|
<view v-else class="info-text-bold">可通过【额度快充】服务恢复花呗正常使用</view>
|
||||||
<view class="button">立即还款并恢复花呗</view>
|
<view class="button">立即还款并恢复花呗</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -83,17 +115,25 @@
|
||||||
<view class="left flex-align-center">
|
<view class="left flex-align-center">
|
||||||
<image class="img " src="/static/image/ant-credit-pay/overdue-payment/consume-dot.png">
|
<image class="img " src="/static/image/ant-credit-pay/overdue-payment/consume-dot.png">
|
||||||
</image>
|
</image>
|
||||||
<view class="text-box">消费成功{{ Number(huabeiInfo.consumptionAmount).toFixed(2) }}元</view>
|
<view class="text-box">{{ huabeiInfo.consumptionText }}</view>
|
||||||
</view>
|
</view>
|
||||||
<view class="right flex-align-center">
|
<view class="right flex-align-center">
|
||||||
<view class="date">{{ huabeiInfo.consumptionDate }}</view>
|
<view class="date">{{ huabeiInfo.consumptionDate }}</view>
|
||||||
<image class="icon" src="/static/image/ant-credit-pay/overdue-payment/right-icon.png"></image>
|
<image class="icon" src="/static/image/ant-credit-pay/overdue-payment/right-icon.png"></image>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
|
<view class="bottom-image-box">
|
||||||
|
<image class="img" src="/static/image/ant-credit-pay/overdue-payment/bottom-image-1.png" mode="widthFix">
|
||||||
|
</image>
|
||||||
|
<image class="img" src="/static/image/ant-credit-pay/overdue-payment/bottom-image-2.png" mode="widthFix">
|
||||||
|
</image>
|
||||||
|
</view>
|
||||||
<!-- 编辑弹窗 -->
|
<!-- 编辑弹窗 -->
|
||||||
<uni-popup ref="popup" type="center" :mask-click="false">
|
<uni-popup ref="popup" type="center" :mask-click="false">
|
||||||
<view class="popup-content">
|
<view class="popup-content">
|
||||||
<view class="popup-title">编辑花呗数据</view>
|
<view class="popup-title">编辑花呗数据</view>
|
||||||
|
|
||||||
<view class="form-item">
|
<view class="form-item">
|
||||||
<text class="label">还款月份</text>
|
<text class="label">还款月份</text>
|
||||||
<picker :range="monthRange" :value="editHuabeiInfo.mouth - 1" @change="onMonthChange"
|
<picker :range="monthRange" :value="editHuabeiInfo.mouth - 1" @change="onMonthChange"
|
||||||
|
|
@ -117,11 +157,15 @@
|
||||||
<text class="label">总计额度</text>
|
<text class="label">总计额度</text>
|
||||||
<input class="input" type="digit" v-model="editHuabeiInfo.totalAmount" placeholder="请输入总计额度" />
|
<input class="input" type="digit" v-model="editHuabeiInfo.totalAmount" placeholder="请输入总计额度" />
|
||||||
</view>
|
</view>
|
||||||
<!-- <text>消费信息</text> -->
|
<view v-if="huabeiInfo.styleType != 1" class="form-item">
|
||||||
|
<text class="label">气泡文本</text>
|
||||||
|
<input class="input" type="text" v-model="editHuabeiInfo.descText" placeholder="请输入描述文本" />
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- <text class="form-title">消费信息</text> -->
|
||||||
<view class="form-item">
|
<view class="form-item">
|
||||||
<text class="label">消费金额</text>
|
<text class="label">消费文案</text>
|
||||||
<input class="input" type="digit" v-model="editHuabeiInfo.consumptionAmount"
|
<input class="input" type="text" v-model="editHuabeiInfo.consumptionText" placeholder="请输入消费文案" />
|
||||||
placeholder="请输入消费金额" />
|
|
||||||
</view>
|
</view>
|
||||||
<view class="form-item">
|
<view class="form-item">
|
||||||
<text class="label">消费日期</text>
|
<text class="label">消费日期</text>
|
||||||
|
|
@ -138,6 +182,24 @@
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
</uni-popup>
|
</uni-popup>
|
||||||
|
|
||||||
|
<!-- 样式选择弹窗 -->
|
||||||
|
<uni-popup ref="stylePopup" type="center">
|
||||||
|
<view class="popup-content">
|
||||||
|
<view class="popup-title">选择展示样式</view>
|
||||||
|
<view class="style-list">
|
||||||
|
<view class="style-item" v-for="(item, index) in styleList" :key="index"
|
||||||
|
@click="confirmStyleDialog(item.value)">
|
||||||
|
<text>{{ item.label }}</text>
|
||||||
|
<uni-icons v-if="huabeiInfo.styleType == item.value" type="checkmarkempty" size="20"
|
||||||
|
color="#1777FF"></uni-icons>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="popup-btns">
|
||||||
|
<view class="btn cancel" @click="closeStyleDialog">取消</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</uni-popup>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<!-- 水印 -->
|
<!-- 水印 -->
|
||||||
|
|
@ -164,20 +226,49 @@ import {
|
||||||
} from 'vue';
|
} from 'vue';
|
||||||
import {
|
import {
|
||||||
onLoad,
|
onLoad,
|
||||||
onShow
|
onShow,
|
||||||
|
onPageScroll
|
||||||
} from '@dcloudio/uni-app';
|
} from '@dcloudio/uni-app';
|
||||||
|
|
||||||
const instance = getCurrentInstance();
|
const instance = getCurrentInstance();
|
||||||
const { proxy } = instance;
|
const { proxy } = instance;
|
||||||
|
|
||||||
|
const styleList = [{
|
||||||
|
label: '样式 1 (默认)',
|
||||||
|
value: 1
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '样式 2 (纯气泡)',
|
||||||
|
value: 2
|
||||||
|
},
|
||||||
|
{
|
||||||
|
label: '样式 3 (带箭头气泡)',
|
||||||
|
value: 3
|
||||||
|
}
|
||||||
|
]
|
||||||
|
|
||||||
const buttonGroup = [{
|
const buttonGroup = [{
|
||||||
name: "编辑花呗数据",
|
name: "编辑花呗数据",
|
||||||
click: () => {
|
click: () => {
|
||||||
openDialog()
|
openDialog()
|
||||||
}
|
}
|
||||||
|
}, {
|
||||||
|
name: "切换展示样式",
|
||||||
|
click: () => {
|
||||||
|
openStyleDialog()
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
name: "逾期后停用",
|
||||||
|
isSwitch: true,
|
||||||
|
key: 'isOverdueDeactivate',
|
||||||
|
click: () => {
|
||||||
|
data.huabeiInfo.isOverdueDeactivate = !data.huabeiInfo.isOverdueDeactivate
|
||||||
|
uni.setStorageSync(data.huabeiInfoStorageKey, data.huabeiInfo)
|
||||||
|
}
|
||||||
}, {
|
}, {
|
||||||
name: "花呗逾期",
|
name: "花呗逾期",
|
||||||
isSwitch: true,
|
isSwitch: true,
|
||||||
|
key: 'isOverdue',
|
||||||
click: () => {
|
click: () => {
|
||||||
data.huabeiInfo.isOverdue = !data.huabeiInfo.isOverdue
|
data.huabeiInfo.isOverdue = !data.huabeiInfo.isOverdue
|
||||||
uni.setStorageSync(data.huabeiInfoStorageKey, data.huabeiInfo)
|
uni.setStorageSync(data.huabeiInfoStorageKey, data.huabeiInfo)
|
||||||
|
|
@ -218,26 +309,30 @@ const clickTitlePopupButton = (button) => {
|
||||||
button.click()
|
button.click()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
const data = reactive({
|
const data = reactive({
|
||||||
navBar: {
|
navBar: {
|
||||||
bgColor: "#717E91",
|
bgColor: "#717E91",
|
||||||
textColor: "#fff"
|
textColor: "#fff"
|
||||||
},
|
},
|
||||||
huabeiInfoStorageKey: 'huabei_info_storage',
|
huabeiInfoStorageKey: 'huabei_info_storage',
|
||||||
|
isTop: true,
|
||||||
huabeiInfo: {
|
huabeiInfo: {
|
||||||
mouth: 1,
|
mouth: 1,
|
||||||
money: 100,
|
money: 100,
|
||||||
dueDate: 15,
|
dueDate: 15,
|
||||||
totalAmount: 200000,
|
totalAmount: 200000,
|
||||||
descText: "当前账单进度已超出预期,花超了",
|
descText: "当前账单进度已超出预期,花超了",
|
||||||
|
isOverdueDeactivate: false,
|
||||||
isInstallment: false,
|
isInstallment: false,
|
||||||
styleType: 1,
|
styleType: 1,
|
||||||
installmentBadgeText: '4折起',
|
installmentBadgeText: '4折起',
|
||||||
image: "",
|
image: "",
|
||||||
isOverdue: false,
|
isOverdue: false,
|
||||||
daysPastDue: 1,
|
daysPastDue: 1,
|
||||||
consumptionAmount: 66.5,
|
consumptionDate: '2025/09/22',
|
||||||
consumptionDate: '2025/09/22'
|
consumptionText: '消费成功66.23元'
|
||||||
},
|
},
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
@ -248,6 +343,24 @@ let {
|
||||||
const editHuabeiInfo = ref({})
|
const editHuabeiInfo = ref({})
|
||||||
|
|
||||||
const popup = ref(null)
|
const popup = ref(null)
|
||||||
|
const stylePopup = ref(null)
|
||||||
|
// 打开样式选择弹窗
|
||||||
|
const openStyleDialog = () => {
|
||||||
|
stylePopup.value.open()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 关闭样式选择弹窗
|
||||||
|
const closeStyleDialog = () => {
|
||||||
|
stylePopup.value.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 确认样式选择
|
||||||
|
const confirmStyleDialog = (type) => {
|
||||||
|
data.huabeiInfo.styleType = type
|
||||||
|
// 保存到缓存
|
||||||
|
uni.setStorageSync(data.huabeiInfoStorageKey, data.huabeiInfo)
|
||||||
|
stylePopup.value.close()
|
||||||
|
}
|
||||||
|
|
||||||
const monthRange = Array.from({
|
const monthRange = Array.from({
|
||||||
length: 12
|
length: 12
|
||||||
|
|
@ -281,6 +394,25 @@ onLoad((option) => {
|
||||||
...savedInfo
|
...savedInfo
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
})
|
||||||
|
|
||||||
|
// 监听页面滚动
|
||||||
|
onPageScroll((e) => {
|
||||||
|
// 当滚动超过40px时,将导航栏背景设置为#F6F6F6
|
||||||
|
if (e.scrollTop > 40) {
|
||||||
|
data.navBar.bgColor = '#F6F6F6'
|
||||||
|
data.navBar.textColor = '#1a1a1a'
|
||||||
|
data.isTop = false
|
||||||
|
plus.navigator.setStatusBarStyle("dark");
|
||||||
|
} else {
|
||||||
|
// 回到顶部时恢复原色
|
||||||
|
data.navBar.bgColor = '#717E91'
|
||||||
|
data.navBar.textColor = '#fff'
|
||||||
|
data.isTop = true
|
||||||
|
plus.navigator.setStatusBarStyle("light");
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
onShow(() => {
|
onShow(() => {
|
||||||
|
|
@ -455,6 +587,41 @@ const goBack = () => {
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bubble-container {
|
||||||
|
margin-top: 10rpx;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
.bubble-box {
|
||||||
|
position: relative;
|
||||||
|
border: 1px solid #D2DBE3;
|
||||||
|
border-radius: 30rpx;
|
||||||
|
padding: 12rpx 24rpx;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
.text {
|
||||||
|
color: #ffffff;
|
||||||
|
font-size: 26rpx;
|
||||||
|
line-height: 1.2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.arrow {
|
||||||
|
position: absolute;
|
||||||
|
top: -5px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%) rotate(45deg);
|
||||||
|
width: 8px;
|
||||||
|
height: 8px;
|
||||||
|
background-color: #717E91;
|
||||||
|
border-left: 1px solid #D2DBE3;
|
||||||
|
border-top: 1px solid #D2DBE3;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
.total-info-box {
|
.total-info-box {
|
||||||
margin-top: 48rpx;
|
margin-top: 48rpx;
|
||||||
|
|
||||||
|
|
@ -490,7 +657,7 @@ const goBack = () => {
|
||||||
|
|
||||||
.overdue-info {
|
.overdue-info {
|
||||||
width: 654rpx;
|
width: 654rpx;
|
||||||
height: 628rpx;
|
// height: 628rpx;
|
||||||
background: linear-gradient(180deg, #FFF5F4 0%, #FFFBF7 100%);
|
background: linear-gradient(180deg, #FFF5F4 0%, #FFFBF7 100%);
|
||||||
border-radius: 20rpx 20rpx 20rpx 20rpx;
|
border-radius: 20rpx 20rpx 20rpx 20rpx;
|
||||||
padding: 100rpx 24rpx 60rpx;
|
padding: 100rpx 24rpx 60rpx;
|
||||||
|
|
@ -517,6 +684,13 @@ const goBack = () => {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.info-text-bold {
|
||||||
|
color: #333333;
|
||||||
|
font-size: 36rpx;
|
||||||
|
font-weight: bold;
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
.button {
|
.button {
|
||||||
background-color: #1777FF;
|
background-color: #1777FF;
|
||||||
width: 100%;
|
width: 100%;
|
||||||
|
|
@ -609,6 +783,18 @@ const goBack = () => {
|
||||||
width: 24px;
|
width: 24px;
|
||||||
height: 24px;
|
height: 24px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bottom-image-box {
|
||||||
|
padding: 0 24rpx;
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding-bottom: 16rpx;
|
||||||
|
|
||||||
|
.img {
|
||||||
|
flex-shrink: 0;
|
||||||
|
width: calc(50% - 12rpx);
|
||||||
|
}
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
|
|
@ -676,4 +862,20 @@ const goBack = () => {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.style-list {
|
||||||
|
.style-item {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 30rpx 0;
|
||||||
|
border-bottom: 1px solid #f5f5f5;
|
||||||
|
font-size: 30rpx;
|
||||||
|
color: #333;
|
||||||
|
|
||||||
|
&:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
@ -8,8 +8,8 @@
|
||||||
</view>
|
</view>
|
||||||
<view class="container" :style="{ height: data.windowHeight + 'px' }">
|
<view class="container" :style="{ height: data.windowHeight + 'px' }">
|
||||||
<view class="bg-container"></view>
|
<view class="bg-container"></view>
|
||||||
<NavBar class="nav-bar" isRightIcon title="" :bgColor="data.navBar.bgColor" :buttonGroup="buttonGroup"
|
<NavBar class="nav-bar" isRightIcon title="" tipLayerType="balance-tip" isTipLayer tipLayerText="修改余额"
|
||||||
@button-click="clickTitlePopupButton">
|
:bgColor="data.navBar.bgColor" :buttonGroup="buttonGroup" @button-click="clickTitlePopupButton">
|
||||||
<template v-slot:left>
|
<template v-slot:left>
|
||||||
<view class="nav-bar-left" @click="util.goBack()">
|
<view class="nav-bar-left" @click="util.goBack()">
|
||||||
<image class="nav-icon" src="/static/image/nav-bar/back-white.png" mode=""></image>
|
<image class="nav-icon" src="/static/image/nav-bar/back-white.png" mode=""></image>
|
||||||
|
|
|
||||||
|
|
@ -7,8 +7,9 @@
|
||||||
</liu-drag-button>
|
</liu-drag-button>
|
||||||
</view>
|
</view>
|
||||||
<view style="overflow: hidden;overflow-y: scroll; height: 100vh;">
|
<view style="overflow: hidden;overflow-y: scroll; height: 100vh;">
|
||||||
<navBar :title="data.navBar.title" :bgColor="data.navBar.bgColor" :buttonGroup="data.navBar.buttonGroup"
|
<navBar :title="data.navBar.title" :bgColor="data.navBar.bgColor" tipLayerType="bill-detail-tip" isTipLayer
|
||||||
@button-click="util.clickTitlePopupButton"></navBar>
|
tipLayerText="编辑账单信息" :buttonGroup="data.navBar.buttonGroup" @button-click="util.clickTitlePopupButton">
|
||||||
|
</navBar>
|
||||||
<!-- 账单信息容器 -->
|
<!-- 账单信息容器 -->
|
||||||
<view class=" add-bill-container">
|
<view class=" add-bill-container">
|
||||||
<!-- 头像 -->
|
<!-- 头像 -->
|
||||||
|
|
|
||||||
|
|
@ -8,7 +8,7 @@
|
||||||
</view>
|
</view>
|
||||||
<view style="overflow: hidden;height: 100vh;;">
|
<view style="overflow: hidden;height: 100vh;;">
|
||||||
<navBar isRightIcon :bgColor="data.navBar.bgColor" :buttonGroup="data.navBar.buttonGroup"
|
<navBar isRightIcon :bgColor="data.navBar.bgColor" :buttonGroup="data.navBar.buttonGroup"
|
||||||
@button-click="clickTitlePopupButton">
|
@button-click="clickTitlePopupButton" tipLayerType="bill-list-tip" isTipLayer tipLayerText="新增账单信息">
|
||||||
<template v-slot:center>
|
<template v-slot:center>
|
||||||
<view class="nav-bar-search flex-align-center flex-1">
|
<view class="nav-bar-search flex-align-center flex-1">
|
||||||
<image class="search-icon" src="/static/image/bill/bill-list/search-black.png" mode=""></image>
|
<image class="search-icon" src="/static/image/bill/bill-list/search-black.png" mode=""></image>
|
||||||
|
|
|
||||||
|
|
@ -39,6 +39,10 @@
|
||||||
<!-- 资产信息 -->
|
<!-- 资产信息 -->
|
||||||
<view class="section-title">资产信息</view>
|
<view class="section-title">资产信息</view>
|
||||||
<view class="card">
|
<view class="card">
|
||||||
|
<view class="form-item">
|
||||||
|
<text class="label">家庭保障份数</text>
|
||||||
|
<input class="input" type="number" v-model="financeInfo.familyProtection" />
|
||||||
|
</view>
|
||||||
<view class="form-item">
|
<view class="form-item">
|
||||||
<text class="label">保险信息</text>
|
<text class="label">保险信息</text>
|
||||||
<input class="input" v-model="financeInfo.myAssets.baoxian.labelSub" />
|
<input class="input" v-model="financeInfo.myAssets.baoxian.labelSub" />
|
||||||
|
|
@ -68,7 +72,9 @@ const defualtData = {
|
||||||
cardHolderName: 'XiaoMing',
|
cardHolderName: 'XiaoMing',
|
||||||
vipLevel: "v3",
|
vipLevel: "v3",
|
||||||
navigationMenuStyle: "style-1",
|
navigationMenuStyle: "style-1",
|
||||||
bgImage: "",
|
bgImage: "/static/image/finance-management/bg-style/style-1.png",
|
||||||
|
familyProtection: "3",
|
||||||
|
searchText: "小而美的基金",
|
||||||
fortune: {
|
fortune: {
|
||||||
yuebao: { value: 3.53, labelSub: "100000" },
|
yuebao: { value: 3.53, labelSub: "100000" },
|
||||||
dingqi: { value: 50, labelSub: "100000" },
|
dingqi: { value: 50, labelSub: "100000" },
|
||||||
|
|
@ -124,7 +130,10 @@ const onTotalAmountInput = (item) => {
|
||||||
}
|
}
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
const stored = uni.getStorageSync('financeInfo')
|
const stored = {
|
||||||
|
...defualtData,
|
||||||
|
...uni.getStorageSync('financeInfo')
|
||||||
|
}
|
||||||
if (stored) {
|
if (stored) {
|
||||||
// Deep merge to ensure structure integrity
|
// Deep merge to ensure structure integrity
|
||||||
Object.assign(data.financeInfo, stored)
|
Object.assign(data.financeInfo, stored)
|
||||||
|
|
|
||||||
|
|
@ -1,4 +1,4 @@
|
||||||
<template>
|
<template>
|
||||||
<!-- 水印 -->
|
<!-- 水印 -->
|
||||||
<view v-if="$isVip()">
|
<view v-if="$isVip()">
|
||||||
<watermark :dark="data.dark" />
|
<watermark :dark="data.dark" />
|
||||||
|
|
@ -12,15 +12,15 @@
|
||||||
<view class="main-info-container" :class="{ 'open': isOpen }"
|
<view class="main-info-container" :class="{ 'open': isOpen }"
|
||||||
:style="{ 'padding-bottom': `${cardArrowHeight}px` }">
|
:style="{ 'padding-bottom': `${cardArrowHeight}px` }">
|
||||||
<!-- 导航栏 placeholder -->
|
<!-- 导航栏 placeholder -->
|
||||||
<NavBar v-if="!selectedImage" bgColor="transparent" :buttonGroup="buttonGroup"
|
<NavBar v-if="!selectedImage" bgColor="transparent" tipLayerType="finance-management-tip" isTipLayer
|
||||||
@button-click="clickTitlePopupButton">
|
tipLayerText="修改理财信息" :buttonGroup="buttonGroup" @button-click="clickTitlePopupButton">
|
||||||
<view class="nav-bar flex-align-center h100">
|
<view class="nav-bar flex-align-center h100">
|
||||||
<view class="left flex-align-center">
|
<view class="left flex-align-center">
|
||||||
<image src="/static/image/finance-management/search/search-left.png"></image>
|
<image src="/static/image/finance-management/search/search-left.png"></image>
|
||||||
</view>
|
</view>
|
||||||
<view class="nav-bar-search flex-align-center flex-1">
|
<view class="nav-bar-search flex-align-center flex-1">
|
||||||
<image class="search-icon" src="/static/image/bill/bill-list/search-black.png" mode=""></image>
|
<image class="search-icon" src="/static/image/bill/bill-list/search-black.png" mode=""></image>
|
||||||
<input type="text" class="search-input flex-1" placeholder="小而美的基金" />
|
<input type="text" class="search-input flex-1" :placeholder="financeInfo.searchText" />
|
||||||
<view class="line h100"></view>
|
<view class="line h100"></view>
|
||||||
<view class="search-button">搜索</view>
|
<view class="search-button">搜索</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -66,7 +66,7 @@
|
||||||
<text>家庭保障</text>
|
<text>家庭保障</text>
|
||||||
</view>
|
</view>
|
||||||
<view class=" flex-align-end" style="margin-top: 10px;">
|
<view class=" flex-align-end" style="margin-top: 10px;">
|
||||||
<text class="value alipay-font">{{ familyProtection }}</text>
|
<text class="value alipay-font">{{ financeInfo.familyProtection || 0 }}</text>
|
||||||
<text class="text">份</text>
|
<text class="text">份</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -194,7 +194,7 @@
|
||||||
<!-- 底部导航栏 -->
|
<!-- 底部导航栏 -->
|
||||||
<view class="bottom-box"
|
<view class="bottom-box"
|
||||||
:class="`navigation-menu-${financeInfo.navigationMenuStyle}`, { 'ios-bottom-box': $system == 'iOS' }">
|
:class="`navigation-menu-${financeInfo.navigationMenuStyle}`, { 'ios-bottom-box': $system == 'iOS' }">
|
||||||
<view class="bottom-item" v-for="item in navigationMenu" :key="item.name">
|
<view class="bottom-item" v-for="item in navigationMenu" :key="item.name" @click="clickBottomItem(item)">
|
||||||
<image
|
<image
|
||||||
:src="`/static/image/finance-management/navigation-menu/${financeInfo.navigationMenuStyle}/${item.icon}.png`">
|
:src="`/static/image/finance-management/navigation-menu/${financeInfo.navigationMenuStyle}/${item.icon}.png`">
|
||||||
</image>
|
</image>
|
||||||
|
|
@ -221,6 +221,20 @@
|
||||||
</view>
|
</view>
|
||||||
</uni-popup>
|
</uni-popup>
|
||||||
|
|
||||||
|
<!-- 修改搜索框提示弹窗 -->
|
||||||
|
<uni-popup ref="searchTextPopup" type="center">
|
||||||
|
<view class="popup-content">
|
||||||
|
<view class="popup-title">修改搜索框提示</view>
|
||||||
|
<view class="input-box">
|
||||||
|
<input class="popup-input" v-model="tempSearchText" placeholder="请输入搜索框提示文字" />
|
||||||
|
</view>
|
||||||
|
<view class="popup-btns">
|
||||||
|
<view class="btn cancel" @click="closeSearchTextDialog">取消</view>
|
||||||
|
<view class="btn confirm" @click="confirmSearchTextDialog">确定</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</uni-popup>
|
||||||
|
|
||||||
<!-- 蒙层 -->
|
<!-- 蒙层 -->
|
||||||
<view v-if="showMask" class="mask" @click="closeMask">
|
<view v-if="showMask" class="mask" @click="closeMask">
|
||||||
<image class="mask-icon" src="/static/image/common/mask-icon.png" mode="widthFix">
|
<image class="mask-icon" src="/static/image/common/mask-icon.png" mode="widthFix">
|
||||||
|
|
@ -277,6 +291,11 @@ const buttonGroup = [{
|
||||||
financeInfo.value.bgImage = ""
|
financeInfo.value.bgImage = ""
|
||||||
uni.setStorageSync('financeInfo', financeInfo.value)
|
uni.setStorageSync('financeInfo', financeInfo.value)
|
||||||
}
|
}
|
||||||
|
}, {
|
||||||
|
name: "修改搜索框提示",
|
||||||
|
click: () => {
|
||||||
|
openSearchTextDialog()
|
||||||
|
}
|
||||||
}]
|
}]
|
||||||
|
|
||||||
const defualtData = {
|
const defualtData = {
|
||||||
|
|
@ -284,7 +303,9 @@ const defualtData = {
|
||||||
cardHolderName: 'XiaoMing',
|
cardHolderName: 'XiaoMing',
|
||||||
vipLevel: "v3",
|
vipLevel: "v3",
|
||||||
navigationMenuStyle: "style-1",
|
navigationMenuStyle: "style-1",
|
||||||
bgImage: "",
|
bgImage: "/static/image/finance-management/bg-style/style-1.png",
|
||||||
|
familyProtection: "3",
|
||||||
|
searchText: "小而美的基金",
|
||||||
fortune: {
|
fortune: {
|
||||||
yuebao: { value: 3.53, labelSub: "100000" },
|
yuebao: { value: 3.53, labelSub: "100000" },
|
||||||
dingqi: { value: 50, labelSub: "100000" },
|
dingqi: { value: 50, labelSub: "100000" },
|
||||||
|
|
@ -303,6 +324,13 @@ const defualtData = {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const clickBottomItem = (item) => {
|
||||||
|
console.log("clickBottomItem", item)
|
||||||
|
if (item.name == "首页") {
|
||||||
|
uni.navigateBack()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onLoad(() => {
|
onLoad(() => {
|
||||||
// 理财页面埋点
|
// 理财页面埋点
|
||||||
proxy.$apiUserEvent('all', {
|
proxy.$apiUserEvent('all', {
|
||||||
|
|
@ -328,7 +356,10 @@ onReady(() => {
|
||||||
})
|
})
|
||||||
|
|
||||||
onShow(() => {
|
onShow(() => {
|
||||||
financeInfo.value = uni.getStorageSync('financeInfo') || defualtData
|
financeInfo.value = {
|
||||||
|
...defualtData,
|
||||||
|
...uni.getStorageSync('financeInfo')
|
||||||
|
}
|
||||||
console.log("financeInfo.value", financeInfo.value)
|
console.log("financeInfo.value", financeInfo.value)
|
||||||
// #ifdef APP-PLUS
|
// #ifdef APP-PLUS
|
||||||
util.setAndroidSystemBarColor(financeInfo.value.navigationMenuStyle == 'style-1' ? '#ffffff' : '#F8F8F8')
|
util.setAndroidSystemBarColor(financeInfo.value.navigationMenuStyle == 'style-1' ? '#ffffff' : '#F8F8F8')
|
||||||
|
|
@ -688,6 +719,8 @@ const closeMask = () => {
|
||||||
|
|
||||||
|
|
||||||
const stylePopup = ref(null)
|
const stylePopup = ref(null)
|
||||||
|
const searchTextPopup = ref(null)
|
||||||
|
const tempSearchText = ref('')
|
||||||
|
|
||||||
const styleList = [{
|
const styleList = [{
|
||||||
label: '样式 1 (默认)',
|
label: '样式 1 (默认)',
|
||||||
|
|
@ -722,6 +755,35 @@ const confirmStyleDialog = (type) => {
|
||||||
const clickTitlePopupButton = (button) => {
|
const clickTitlePopupButton = (button) => {
|
||||||
button.click()
|
button.click()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 打开搜索框提示修改弹窗
|
||||||
|
const openSearchTextDialog = () => {
|
||||||
|
tempSearchText.value = financeInfo.value.searchText || ''
|
||||||
|
searchTextPopup.value.open()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 关闭搜索框提示修改弹窗
|
||||||
|
const closeSearchTextDialog = () => {
|
||||||
|
searchTextPopup.value.close()
|
||||||
|
}
|
||||||
|
|
||||||
|
// 确认搜索框提示修改
|
||||||
|
const confirmSearchTextDialog = () => {
|
||||||
|
if (!tempSearchText.value.trim()) {
|
||||||
|
uni.showToast({
|
||||||
|
title: '请输入提示文字',
|
||||||
|
icon: 'none'
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
financeInfo.value.searchText = tempSearchText.value.trim()
|
||||||
|
uni.setStorageSync('financeInfo', financeInfo.value)
|
||||||
|
searchTextPopup.value.close()
|
||||||
|
uni.showToast({
|
||||||
|
title: '修改成功',
|
||||||
|
icon: 'success'
|
||||||
|
})
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
|
@ -1158,14 +1220,14 @@ const clickTitlePopupButton = (button) => {
|
||||||
background-color: #fff;
|
background-color: #fff;
|
||||||
border-radius: 12px;
|
border-radius: 12px;
|
||||||
width: 600rpx;
|
width: 600rpx;
|
||||||
padding: 20px;
|
padding: 24px 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.popup-title {
|
.popup-title {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
font-size: 18px;
|
font-size: 17px;
|
||||||
font-weight: 500;
|
font-weight: 400;
|
||||||
margin-bottom: 20px;
|
margin-bottom: 18px;
|
||||||
color: #333;
|
color: #333;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1190,20 +1252,46 @@ const clickTitlePopupButton = (button) => {
|
||||||
|
|
||||||
.popup-btns {
|
.popup-btns {
|
||||||
display: flex;
|
display: flex;
|
||||||
|
gap: 12px;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn {
|
.btn {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 44px;
|
height: 40px;
|
||||||
line-height: 44px;
|
line-height: 40px;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
border-radius: 22px;
|
border-radius: 6px;
|
||||||
font-size: 16px;
|
font-size: 15px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.btn.cancel {
|
.btn.cancel {
|
||||||
background-color: #f5f5f5;
|
background-color: #f5f5f5;
|
||||||
color: #666;
|
color: #666;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
.input-box {
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-input {
|
||||||
|
width: 100%;
|
||||||
|
height: 44px;
|
||||||
|
padding: 0 15px;
|
||||||
|
line-height: 44px;
|
||||||
|
border: 1px solid #e0e0e0;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-size: 14px;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-input::placeholder {
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn.confirm {
|
||||||
|
background-color: #1677ff;
|
||||||
|
color: #fff;
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
@ -23,7 +23,8 @@ const defualtData = {
|
||||||
cardHolderName: 'XiaoMing',
|
cardHolderName: 'XiaoMing',
|
||||||
vipLevel: "v3",
|
vipLevel: "v3",
|
||||||
navigationMenuStyle: "style-1",
|
navigationMenuStyle: "style-1",
|
||||||
bgImage: "",
|
bgImage: "/static/image/finance-management/bg-style/style-1.png",
|
||||||
|
searchText: "小而美的基金",
|
||||||
fortune: {
|
fortune: {
|
||||||
yuebao: {
|
yuebao: {
|
||||||
value: 3.53,
|
value: 3.53,
|
||||||
|
|
@ -83,7 +84,10 @@ const data = reactive({
|
||||||
const { bgImage, selectedImage, financeInfo } = toRefs(data)
|
const { bgImage, selectedImage, financeInfo } = toRefs(data)
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
financeInfo.value = uni.getStorageSync('financeInfo') || defualtData
|
financeInfo.value = {
|
||||||
|
...defualtData,
|
||||||
|
...uni.getStorageSync('financeInfo')
|
||||||
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
const handleRightButtonClick = () => {
|
const handleRightButtonClick = () => {
|
||||||
|
|
|
||||||
|
|
@ -3,11 +3,12 @@
|
||||||
<image class="index-bg-img" :style="{ width: windowWidth + 'px' }" src="/static/image/index/index-bg.png"
|
<image class="index-bg-img" :style="{ width: windowWidth + 'px' }" src="/static/image/index/index-bg.png"
|
||||||
mode="widthFix">
|
mode="widthFix">
|
||||||
</image>
|
</image>
|
||||||
<view class="nav-bar-box">
|
<view class="nav-bar-box" :style="{ backgroundColor: data.navBarBgColor }">
|
||||||
<view class="status-box" :style="{ height: statusBarHeight + 'px' }"></view>
|
<view class="status-box" :style="{ height: statusBarHeight + 'px' }"></view>
|
||||||
<view class="nav-box">
|
<view class="nav-box">
|
||||||
<view class="left-box" @click="exit">
|
<view class="left-box" @click="exit" @tap="exit" @touchstart.stop="exit">
|
||||||
<image style="width: 40rpx; height: 40rpx;" src="/static/image/nav-bar/back-black.png"></image>
|
<image style="width: 40rpx; height: 40rpx;" src="/static/image/nav-bar/back-black.png" @click="exit"
|
||||||
|
@tap="exit"></image>
|
||||||
</view>
|
</view>
|
||||||
<view class="title">小宝模拟器</view>
|
<view class="title">小宝模拟器</view>
|
||||||
<view class="right-box"></view>
|
<view class="right-box"></view>
|
||||||
|
|
@ -15,8 +16,24 @@
|
||||||
|
|
||||||
</view>
|
</view>
|
||||||
<view class="content-box" :style="{ height: windowHeight + 'px' }">
|
<view class="content-box" :style="{ height: windowHeight + 'px' }">
|
||||||
|
<scroll-view scroll-y="true"
|
||||||
|
:style="{ height: (windowHeight - statusBarHeight - 44) + 'px', marginTop: (statusBarHeight + 44) + 'px' }"
|
||||||
|
@scroll="handleScroll">
|
||||||
<view>
|
<view>
|
||||||
<view style="background-color: transparent;" :style="{ height: (44 + statusBarHeight) + 'px' }"></view>
|
<!-- iOS专用:透明点击区域覆盖导航栏左上角 -->
|
||||||
|
<view @click="exit" @tap="exit" :style="{
|
||||||
|
position: 'fixed',
|
||||||
|
top: statusBarHeight + 'px',
|
||||||
|
left: '0px',
|
||||||
|
width: '60px',
|
||||||
|
height: '44px',
|
||||||
|
zIndex: 10000,
|
||||||
|
backgroundColor: 'transparent'
|
||||||
|
}">
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- <view style="background-color: transparent;" :style="{ height: (44 + statusBarHeight) + 'px' }">
|
||||||
|
</view> -->
|
||||||
<view class="user-box">
|
<view class="user-box">
|
||||||
<image class="user-bg" :style="{ width: (windowWidth - 32) + 'px' }"
|
<image class="user-bg" :style="{ width: (windowWidth - 32) + 'px' }"
|
||||||
:src="`/static/image/index/${userInfo.vip > 1 ? (userInfo.vip == 3 ? 'lifetime-vip-bg' : 'vip-bg') : 'no-vip-bg'}.png`"
|
:src="`/static/image/index/${userInfo.vip > 1 ? (userInfo.vip == 3 ? 'lifetime-vip-bg' : 'vip-bg') : 'no-vip-bg'}.png`"
|
||||||
|
|
@ -26,11 +43,13 @@
|
||||||
<view class="user-info">
|
<view class="user-info">
|
||||||
<view class="name-box">
|
<view class="name-box">
|
||||||
<text class="phone-text">ID:{{ userInfo.user_id }}</text>
|
<text class="phone-text">ID:{{ userInfo.user_id }}</text>
|
||||||
<image v-if="userInfo.vip > 1" class="vip-logo" src="/static/image/index/vip-logo.png">
|
<image v-if="userInfo.vip > 1" class="vip-logo"
|
||||||
|
src="/static/image/index/vip-logo.png">
|
||||||
</image>
|
</image>
|
||||||
</view>
|
</view>
|
||||||
<view class="">
|
<view class="">
|
||||||
<text v-if="userInfo.vip > 1" class="vip-end-time">会员到期:{{ userInfo.vip_expire }}</text>
|
<text v-if="userInfo.vip > 1" class="vip-end-time">会员到期:{{ userInfo.vip_expire
|
||||||
|
}}</text>
|
||||||
<text v-else class="vip-end-time">开通会员解锁全部功能</text>
|
<text v-else class="vip-end-time">开通会员解锁全部功能</text>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -49,7 +68,8 @@
|
||||||
|
|
||||||
<view ref="noticeContainer" class="notice-content-wrapper">
|
<view ref="noticeContainer" class="notice-content-wrapper">
|
||||||
<view ref="noticeInner" class="notice-inner">
|
<view ref="noticeInner" class="notice-inner">
|
||||||
<text ref="noticeBox" class="notice-content" style="margin-right: 30rpx;">{{ noticeInfo.text
|
<text ref="noticeBox" class="notice-content" style="margin-right: 30rpx;">{{
|
||||||
|
noticeInfo.text
|
||||||
}}</text>
|
}}</text>
|
||||||
<text class="notice-content" style="margin-right: 30rpx;">{{ noticeInfo.text }}</text>
|
<text class="notice-content" style="margin-right: 30rpx;">{{ noticeInfo.text }}</text>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -79,7 +99,8 @@
|
||||||
<image style="width: 108rpx;height: 108rpx; flex-shrink: 0;"
|
<image style="width: 108rpx;height: 108rpx; flex-shrink: 0;"
|
||||||
:src="'/static/image/index/menu-icon/' + item.icon + '.png'"></image>
|
:src="'/static/image/index/menu-icon/' + item.icon + '.png'"></image>
|
||||||
</view>
|
</view>
|
||||||
<image v-if="item.isHot" class="hot-icon" src="/static/image/index/hot-icon.png"></image>
|
<image v-if="item.isHot" class="hot-icon" src="/static/image/index/hot-icon.png">
|
||||||
|
</image>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
@ -89,11 +110,23 @@
|
||||||
src="/static/image/index/alipay-year-bill.png" mode="widthFix"
|
src="/static/image/index/alipay-year-bill.png" mode="widthFix"
|
||||||
@click="util.goPage(`/pages/common/alipay-annual-bill/alipay-annual-bill`)"></image>
|
@click="util.goPage(`/pages/common/alipay-annual-bill/alipay-annual-bill`)"></image>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
|
<view class="group-box">
|
||||||
|
<image class="title-img" src="/static/image/index/qita.png"></image>
|
||||||
|
<view class="video-help-box">
|
||||||
|
<view class="video-help-item" v-for="item in otherList" :key="item.id"
|
||||||
|
@click="clickMenu(item)">
|
||||||
|
<image class="video-help-img" :src="item.icon"></image>
|
||||||
|
<text class="video-help-title">{{ item.name }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
</view>
|
</view>
|
||||||
|
|
||||||
<view class="footer-box" :class="{ 'ios-padding-bottom': platform === 'ios' }">
|
<view class="footer-box" :class="{ 'ios-padding-bottom': platform === 'ios' }">
|
||||||
<text class="vision-text">版本:{{ vision }}</text>
|
<text class="vision-text">版本:{{ vision }}</text>
|
||||||
</view>
|
</view>
|
||||||
|
</scroll-view>
|
||||||
</view>
|
</view>
|
||||||
</view>
|
</view>
|
||||||
</template>
|
</template>
|
||||||
|
|
@ -178,10 +211,33 @@ const menuList = [{
|
||||||
isHot: false,
|
isHot: false,
|
||||||
path: "/pages/ant-credit-pay/index"
|
path: "/pages/ant-credit-pay/index"
|
||||||
},
|
},
|
||||||
|
]
|
||||||
|
|
||||||
|
|
||||||
|
const otherList = [{
|
||||||
|
icon: "/static/image/index/qita/jipiao.png",
|
||||||
|
name: "机票",
|
||||||
|
path: ""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: "/static/image/index/qita/huochepiao.png",
|
||||||
|
name: "火车票",
|
||||||
|
path: ""
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: "/static/image/index/qita/gongzidan.png",
|
||||||
|
name: "工资单",
|
||||||
|
path: "/pages/other/splash/splash"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
icon: "/static/image/index/qita/shipinqunliao.png",
|
||||||
|
name: "视频群聊",
|
||||||
|
path: "/pages/other/video-group-chat/video-group-chat"
|
||||||
|
},
|
||||||
]
|
]
|
||||||
|
|
||||||
const data = reactive({
|
const data = reactive({
|
||||||
|
navBarBgColor: 'transparent',
|
||||||
statusBarHeight: 0,
|
statusBarHeight: 0,
|
||||||
windowWidth: 0,
|
windowWidth: 0,
|
||||||
windowHeight: 0,
|
windowHeight: 0,
|
||||||
|
|
@ -203,6 +259,19 @@ const {
|
||||||
platform
|
platform
|
||||||
} = toRefs(data);
|
} = toRefs(data);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 处理页面滚动事件
|
||||||
|
*/
|
||||||
|
const handleScroll = (e) => {
|
||||||
|
const scrollTop = e.detail.scrollTop
|
||||||
|
// 滚动超过20px时显示蓝色背景,否则显示透明背景
|
||||||
|
if (scrollTop > 20) {
|
||||||
|
data.navBarBgColor = '#DDF0FD'
|
||||||
|
} else {
|
||||||
|
data.navBarBgColor = 'transparent'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
onLoad(async () => {
|
onLoad(async () => {
|
||||||
// 获取平台信息
|
// 获取平台信息
|
||||||
const systemInfo = uni.getSystemInfoSync()
|
const systemInfo = uni.getSystemInfoSync()
|
||||||
|
|
@ -362,6 +431,7 @@ const clickVideoHelp = (item) => {
|
||||||
* 点击公告
|
* 点击公告
|
||||||
*/
|
*/
|
||||||
const clickNotice = () => {
|
const clickNotice = () => {
|
||||||
|
console.log("点击公告", noticeInfo.value)
|
||||||
if (!noticeInfo.value.url) return
|
if (!noticeInfo.value.url) return
|
||||||
const url = noticeInfo.value.url + `&uni_id=${userInfo.value.user_id}`
|
const url = noticeInfo.value.url + `&uni_id=${userInfo.value.user_id}`
|
||||||
util.goPage(`/pages/common/webview/webview?url=${encodeURIComponent(url)}&title=${noticeInfo.value.title}`)
|
util.goPage(`/pages/common/webview/webview?url=${encodeURIComponent(url)}&title=${noticeInfo.value.title}`)
|
||||||
|
|
@ -371,7 +441,22 @@ const clickNotice = () => {
|
||||||
* 退出模拟器
|
* 退出模拟器
|
||||||
*/
|
*/
|
||||||
const exit = () => {
|
const exit = () => {
|
||||||
|
console.log("点击退出按钮")
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
if (typeof plus !== 'undefined' && plus.runtime) {
|
||||||
|
console.log("执行退出应用")
|
||||||
plus.runtime.quit()
|
plus.runtime.quit()
|
||||||
|
} else {
|
||||||
|
console.log("plus对象未定义")
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
// #ifndef APP-PLUS
|
||||||
|
console.log("非APP环境,无法退出")
|
||||||
|
uni.showToast({
|
||||||
|
title: '仅APP环境支持退出',
|
||||||
|
icon: 'none'
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
}
|
}
|
||||||
|
|
||||||
const noticeContainer = ref(null);
|
const noticeContainer = ref(null);
|
||||||
|
|
@ -491,7 +576,7 @@ const runMarqueeAnimation = (containerWidth, textWidth, myId) => {
|
||||||
position: fixed;
|
position: fixed;
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
z-index: 1;
|
z-index: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.nav-bar-box {
|
.nav-bar-box {
|
||||||
|
|
@ -499,7 +584,7 @@ const runMarqueeAnimation = (containerWidth, textWidth, myId) => {
|
||||||
top: 0;
|
top: 0;
|
||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
z-index: 999;
|
z-index: 9999;
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -508,7 +593,7 @@ const runMarqueeAnimation = (containerWidth, textWidth, myId) => {
|
||||||
top: 0rpx;
|
top: 0rpx;
|
||||||
left: 0;
|
left: 0;
|
||||||
right: 0;
|
right: 0;
|
||||||
z-index: 999;
|
z-index: 1;
|
||||||
background-color: transparent;
|
background-color: transparent;
|
||||||
display: flex;
|
display: flex;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|
@ -535,6 +620,8 @@ const runMarqueeAnimation = (containerWidth, textWidth, myId) => {
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
|
position: relative;
|
||||||
|
z-index: 10;
|
||||||
}
|
}
|
||||||
|
|
||||||
.title {
|
.title {
|
||||||
|
|
|
||||||
|
|
@ -0,0 +1,604 @@
|
||||||
|
<template>
|
||||||
|
<view class="container">
|
||||||
|
<!-- 自定义头部导航栏 -->
|
||||||
|
<ZdyNavbar @right-click="edit" isRightButton rightButtonText="编辑" :title="data.navbar.title"
|
||||||
|
:bgColor="data.navbar.bgColor" :isBack="true" />
|
||||||
|
|
||||||
|
<image :src="data.code" mode="widthFix" style="width: 100vw;" @click="previewImage"></image>
|
||||||
|
<view class="button-container">
|
||||||
|
<button class="btn-save-image" @click="saveImage">保存图片</button>
|
||||||
|
</view>
|
||||||
|
<l-painter isCanvasToTempFilePath @success="data.code = $event" hidden
|
||||||
|
:css="`width:${data.width}px;height:${data.width / 4 * 3}px;`">
|
||||||
|
<l-painter-view
|
||||||
|
:css="`position: relative;width:${data.width}px;height:${data.width / 4 * 3}px;background-image: url('/static/image/other/gzd.png');`">
|
||||||
|
<!-- 头部年月 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:480px;top:336px;`">
|
||||||
|
<l-painter-text :css="data.textCssTime" :text="data.form.year" />
|
||||||
|
</l-painter-view>
|
||||||
|
<l-painter-view :css="`position: absolute;left:544px;top:336px;`">
|
||||||
|
<l-painter-text :css="data.textCssTime" :text="data.form.month" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 岗位 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:158px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCss" :text="data.form.work" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 姓名 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:213px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCss" :text="data.form.name" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 基本工资 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:268px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money1 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 绩效工资 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:314px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money2 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 全勤工资 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:358px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money3 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 其他奖金 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:404px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money4 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 合计 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:454px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money5 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 社保缴纳 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:504px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money6 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 个税缴纳 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:554px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money7 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 考勤扣款 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:598px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money8 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 其他扣款 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:640px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCssMoney" :text="data.form.money9 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 合计 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:688px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCss" :text="data.form.money10 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<!-- 实发工资 -->
|
||||||
|
<l-painter-view :css="`position: absolute;left:734px;top:445px;`">
|
||||||
|
<l-painter-text :css="data.textCss" :text="data.form.money11 + ''" />
|
||||||
|
</l-painter-view>
|
||||||
|
<l-painter-view v-if="$isVip()" :css="`position: absolute;right:20px;bottom:20px;`">
|
||||||
|
<l-painter-image src="/static/image/other/shuiying.png" css="width: 170rpx; height: 50rpx;" />
|
||||||
|
</l-painter-view>
|
||||||
|
</l-painter-view>
|
||||||
|
</l-painter>
|
||||||
|
|
||||||
|
<!-- 编辑弹窗 -->
|
||||||
|
<view v-if="showEditPopup" class="popup-overlay">
|
||||||
|
<view class="popup-content">
|
||||||
|
<view class="popup-header">
|
||||||
|
<text class="popup-title">编辑工资条</text>
|
||||||
|
<text class="popup-close" @click="closeEditPopup">×</text>
|
||||||
|
</view>
|
||||||
|
<view class="popup-body">
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">岗位:</text>
|
||||||
|
<input class="form-input" v-model="editForm.work" type="text" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">姓名:</text>
|
||||||
|
<input class="form-input" v-model="editForm.name" type="text" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">年月:</text>
|
||||||
|
<input class="form-input" v-model="editForm.year" type="number" style="width: 100px;" />
|
||||||
|
<text class="form-separator">-</text>
|
||||||
|
<input class="form-input" v-model="editForm.month" type="number" style="width: 100px;" />
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<view class="form-section">
|
||||||
|
<text class="section-title">收入部分</text>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">基本工资:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money1" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">绩效工资:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money2" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">全勤工资:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money3" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">其他奖金:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money4" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row form-total">
|
||||||
|
<text class="form-label">合计:</text>
|
||||||
|
<input class="form-input form-total-input" v-model.number="editForm.money5" type="number"
|
||||||
|
disabled />
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<view class="form-section">
|
||||||
|
<text class="section-title">扣款部分</text>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">社保缴纳:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money6" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">个税缴纳:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money7" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">考勤扣款:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money8" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row">
|
||||||
|
<text class="form-label">其他扣款:</text>
|
||||||
|
<input class="form-input" v-model.number="editForm.money9" type="number"
|
||||||
|
@input="calculateValues" />
|
||||||
|
</view>
|
||||||
|
<view class="form-row form-total">
|
||||||
|
<text class="form-label">合计:</text>
|
||||||
|
<input class="form-input form-total-input" v-model.number="editForm.money10" type="number"
|
||||||
|
disabled />
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<view class="form-row form-total">
|
||||||
|
<text class="form-label">实发工资:</text>
|
||||||
|
<input class="form-input form-total-input" v-model.number="editForm.money11" type="number"
|
||||||
|
disabled />
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="popup-footer">
|
||||||
|
<button class="btn-cancel" @click="closeEditPopup">取消</button>
|
||||||
|
<button class="btn-save" @click="saveEditForm">保存</button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
// 自定义头部
|
||||||
|
import ZdyNavbar from "@/components/nav-bar/nav-bar.vue"
|
||||||
|
|
||||||
|
import {
|
||||||
|
ref,
|
||||||
|
reactive,
|
||||||
|
watch,
|
||||||
|
nextTick,
|
||||||
|
getCurrentInstance
|
||||||
|
} from "vue";
|
||||||
|
import {
|
||||||
|
onLoad,
|
||||||
|
onShow,
|
||||||
|
onReady,
|
||||||
|
onPullDownRefresh,
|
||||||
|
onReachBottom
|
||||||
|
} from "@dcloudio/uni-app";
|
||||||
|
const {
|
||||||
|
appContext,
|
||||||
|
proxy
|
||||||
|
} = getCurrentInstance();
|
||||||
|
const data = reactive({
|
||||||
|
navbar: {
|
||||||
|
title: "工资条",
|
||||||
|
bgColor: '#EDEDED',
|
||||||
|
},
|
||||||
|
width: 375,
|
||||||
|
height: 495,
|
||||||
|
code: '',
|
||||||
|
form: {
|
||||||
|
work: "销售",
|
||||||
|
name: '小明',
|
||||||
|
money1: 5000,
|
||||||
|
money2: 1000,
|
||||||
|
money3: 1000,
|
||||||
|
money4: 1000,
|
||||||
|
money5: 8000,
|
||||||
|
money6: 512,
|
||||||
|
money7: 12,
|
||||||
|
money8: 0,
|
||||||
|
money9: 0,
|
||||||
|
money10: 524,
|
||||||
|
money11: 7476,
|
||||||
|
year: 2026,
|
||||||
|
month: 1,
|
||||||
|
},
|
||||||
|
textCss: ' letter-spacing: 10px;font-family: "SimHei", "Microsoft YaHei", sans-serif;width:50px;text-align: center;color:#505156;transform: scale(0.4);font-size:26rpx;font-weight: bold; mix-blend-mode: overlay;',
|
||||||
|
textCssMoney: 'font-family: "SimHei", "Microsoft YaHei", sans-serif;width:50px;text-align: center;color:#505156;transform: scale(0.4);font-size:12px;font-weight: bold; mix-blend-mode: overlay;',
|
||||||
|
textCssTime: 'font-family: "SimHei", "Microsoft YaHei", sans-serif;width:50px;text-align: center;color:#505156;transform: scale(0.4);font-size:18px;font-weight: bold; mix-blend-mode: overlay;'
|
||||||
|
})
|
||||||
|
|
||||||
|
// 弹窗相关
|
||||||
|
const showEditPopup = ref(false);
|
||||||
|
const editForm = ref({});
|
||||||
|
|
||||||
|
// 打开编辑弹窗
|
||||||
|
function edit() {
|
||||||
|
console.log(data.form)
|
||||||
|
// 复制当前表单数据到编辑表单
|
||||||
|
editForm.value = JSON.parse(JSON.stringify(data.form));
|
||||||
|
// 计算初始值
|
||||||
|
calculateValues();
|
||||||
|
// 显示弹窗
|
||||||
|
showEditPopup.value = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 关闭编辑弹窗
|
||||||
|
function closeEditPopup() {
|
||||||
|
showEditPopup.value = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存编辑表单
|
||||||
|
function saveEditForm() {
|
||||||
|
// 将编辑后的数据复制回原始表单
|
||||||
|
data.form = JSON.parse(JSON.stringify(editForm.value));
|
||||||
|
// 关闭弹窗
|
||||||
|
showEditPopup.value = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 计算值
|
||||||
|
function calculateValues() {
|
||||||
|
// 计算收入合计 (money5 = money1 + money2 + money3 + money4)
|
||||||
|
editForm.value.money5 = (editForm.value.money1 || 0) + (editForm.value.money2 || 0) + (editForm.value.money3 ||
|
||||||
|
0) + (editForm.value.money4 || 0);
|
||||||
|
|
||||||
|
// 计算扣款合计 (money10 = money6 + money7 + money8 + money9)
|
||||||
|
editForm.value.money10 = (editForm.value.money6 || 0) + (editForm.value.money7 || 0) + (editForm.value.money8 ||
|
||||||
|
0) + (editForm.value.money9 || 0);
|
||||||
|
|
||||||
|
// 计算实发工资 (money11 = money5 - money10)
|
||||||
|
editForm.value.money11 = editForm.value.money5 - editForm.value.money10;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 预览图片
|
||||||
|
function previewImage() {
|
||||||
|
if (data.code) {
|
||||||
|
uni.previewImage({
|
||||||
|
urls: [data.code],
|
||||||
|
current: 0
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存图片
|
||||||
|
function saveImage() {
|
||||||
|
if (data.code) {
|
||||||
|
console.log(data.code)
|
||||||
|
uni.showLoading({
|
||||||
|
title: '保存中...'
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
// 检查是否为base64格式
|
||||||
|
if (data.code.startsWith('data:image')) {
|
||||||
|
// 处理base64格式图片
|
||||||
|
console.log('开始处理base64图片');
|
||||||
|
uni.base64ToTempFile({
|
||||||
|
base64: data.code.split(',')[1], // 去除base64前缀
|
||||||
|
success: (res) => {
|
||||||
|
console.log('base64转换成功', res);
|
||||||
|
if (res.tempFilePath) {
|
||||||
|
uni.saveImageToPhotosAlbum({
|
||||||
|
filePath: res.tempFilePath,
|
||||||
|
success: () => {
|
||||||
|
console.log('保存图片成功');
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '保存成功',
|
||||||
|
icon: 'success'
|
||||||
|
});
|
||||||
|
},
|
||||||
|
fail: (err) => {
|
||||||
|
console.log('保存图片失败', err);
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '保存失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('base64转换失败,无临时文件路径');
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '转换失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
fail: (err) => {
|
||||||
|
console.log('base64转换失败', err);
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '转换失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else if (data.code.startsWith('_') || data.code.includes('temp') || data.code.includes('cache') || data
|
||||||
|
.code.includes('_doc') || data.code.includes('_tmp')) {
|
||||||
|
// 处理本地临时文件
|
||||||
|
console.log('开始处理本地临时文件');
|
||||||
|
uni.saveImageToPhotosAlbum({
|
||||||
|
filePath: data.code,
|
||||||
|
success: () => {
|
||||||
|
console.log('保存本地文件成功');
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '保存成功',
|
||||||
|
icon: 'success'
|
||||||
|
});
|
||||||
|
},
|
||||||
|
fail: (err) => {
|
||||||
|
console.log('保存本地文件失败', err);
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '保存失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
// 处理网络图片
|
||||||
|
console.log('开始处理网络图片');
|
||||||
|
uni.downloadFile({
|
||||||
|
url: data.code,
|
||||||
|
success: (res) => {
|
||||||
|
console.log('下载图片成功', res);
|
||||||
|
if (res.statusCode === 200 && res.tempFilePath) {
|
||||||
|
uni.saveImageToPhotosAlbum({
|
||||||
|
filePath: res.tempFilePath,
|
||||||
|
success: () => {
|
||||||
|
console.log('保存图片成功');
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '保存成功',
|
||||||
|
icon: 'success'
|
||||||
|
});
|
||||||
|
},
|
||||||
|
fail: (err) => {
|
||||||
|
console.log('保存图片失败', err);
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '保存失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
console.log('下载图片失败,状态码:', res.statusCode);
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '下载失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
fail: (err) => {
|
||||||
|
console.log('下载图片失败', err);
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '下载失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.log('保存图片异常', error);
|
||||||
|
uni.hideLoading();
|
||||||
|
uni.showToast({
|
||||||
|
title: '保存失败',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
uni.showToast({
|
||||||
|
title: '暂无图片可保存',
|
||||||
|
icon: 'none'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onLoad((option) => { })
|
||||||
|
onReady(() => {
|
||||||
|
|
||||||
|
})
|
||||||
|
onShow(() => { })
|
||||||
|
onPullDownRefresh(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
uni.stopPullDownRefresh();
|
||||||
|
}, 1000);
|
||||||
|
})
|
||||||
|
onReachBottom(() => {
|
||||||
|
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss" scoped>
|
||||||
|
* {
|
||||||
|
box-sizing: content-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
.aadadad {
|
||||||
|
text-align: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.heiti {
|
||||||
|
font-family: "SimHei", "Microsoft YaHei", sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 弹窗样式 */
|
||||||
|
.popup-overlay {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background-color: rgba(0, 0, 0, 0.5);
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
z-index: 9999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-content {
|
||||||
|
background-color: #fff;
|
||||||
|
border-radius: 10px;
|
||||||
|
width: 90%;
|
||||||
|
max-width: 500px;
|
||||||
|
max-height: 80vh;
|
||||||
|
overflow-y: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
padding: 20rpx;
|
||||||
|
border-bottom: 1rpx solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-title {
|
||||||
|
font-size: 32rpx;
|
||||||
|
font-weight: bold;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-close {
|
||||||
|
font-size: 48rpx;
|
||||||
|
color: #999;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-body {
|
||||||
|
padding: 20rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-section {
|
||||||
|
margin-bottom: 30rpx;
|
||||||
|
padding: 20rpx;
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
border-radius: 8rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.section-title {
|
||||||
|
font-size: 28rpx;
|
||||||
|
font-weight: bold;
|
||||||
|
color: #333;
|
||||||
|
margin-bottom: 20rpx;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 20rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-label {
|
||||||
|
width: 200rpx;
|
||||||
|
font-size: 26rpx;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-input {
|
||||||
|
flex: 1;
|
||||||
|
padding: 15rpx;
|
||||||
|
border: 1rpx solid #ddd;
|
||||||
|
border-radius: 4rpx;
|
||||||
|
font-size: 26rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-separator {
|
||||||
|
margin: 0 10rpx;
|
||||||
|
font-size: 26rpx;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-total {
|
||||||
|
margin-top: 10rpx;
|
||||||
|
padding-top: 10rpx;
|
||||||
|
border-top: 1rpx dashed #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-total-input {
|
||||||
|
background-color: #f9f9f9;
|
||||||
|
color: #ff6b35;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-footer {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 20rpx;
|
||||||
|
border-top: 1rpx solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-cancel,
|
||||||
|
.btn-save {
|
||||||
|
width: 48%;
|
||||||
|
padding: 20rpx;
|
||||||
|
border-radius: 4rpx;
|
||||||
|
font-size: 28rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-cancel {
|
||||||
|
background-color: #f5f5f5;
|
||||||
|
color: #333;
|
||||||
|
border: 1rpx solid #ddd;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-save {
|
||||||
|
background-color: #ff6b35;
|
||||||
|
color: #fff;
|
||||||
|
border: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 保存图片按钮 */
|
||||||
|
.button-container {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 30rpx 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-save-image {
|
||||||
|
background-color: #07C160;
|
||||||
|
color: #fff;
|
||||||
|
border: none;
|
||||||
|
padding: 18rpx 60rpx;
|
||||||
|
border-radius: 40rpx;
|
||||||
|
font-size: 26rpx;
|
||||||
|
font-weight: bold;
|
||||||
|
box-shadow: 0 3rpx 10rpx rgba(7, 193, 96, 0.3);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
text-align: center;
|
||||||
|
min-width: 160rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-save-image:hover {
|
||||||
|
transform: translateY(-2rpx);
|
||||||
|
box-shadow: 0 4rpx 12rpx rgba(7, 193, 96, 0.4);
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-save-image:active {
|
||||||
|
transform: translateY(0);
|
||||||
|
box-shadow: 0 2rpx 6rpx rgba(7, 193, 96, 0.3);
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
@ -0,0 +1,776 @@
|
||||||
|
<template>
|
||||||
|
<!-- 水印 -->
|
||||||
|
<view v-if="$isVip()">
|
||||||
|
<watermark :dark="data.dark" />
|
||||||
|
<liu-drag-button :canDocking="false" @clickBtn="$goRechargePage('watermark')">
|
||||||
|
<c-lottie ref="cLottieRef" :src='$watermark()' width="94px" height='74px' :loop="true"></c-lottie>
|
||||||
|
</liu-drag-button>
|
||||||
|
</view>
|
||||||
|
<view class="container">
|
||||||
|
<!-- 导航栏 placeholder -->
|
||||||
|
<NavBar bgColor="transparent" tipLayerType="video-group-chat-tip" isTipLayer tipLayerText="修改聊天信息"
|
||||||
|
@button-click="util.clickTitlePopupButton" :buttonGroup="buttonGroup">
|
||||||
|
<view class="nav-content">
|
||||||
|
<view class="left">
|
||||||
|
<image class="icon" src="/static/image/other/video-call/float.png"></image>
|
||||||
|
</view>
|
||||||
|
<view class="center">
|
||||||
|
<text class="time">{{ videoData.timeText }}</text>
|
||||||
|
</view>
|
||||||
|
<view v-if="data.isEdit" class="right" @click.stop>
|
||||||
|
<view class="button" @click="confirmEdit">完成</view>
|
||||||
|
</view>
|
||||||
|
<view v-else class="right">
|
||||||
|
<image class="icon" src="/static/image/other/video-call/screen-mirroring.png"></image>
|
||||||
|
<image style="margin-left: 30px;" class="icon" src="/static/image/other/video-call/add.png"></image>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
</view>
|
||||||
|
</NavBar>
|
||||||
|
|
||||||
|
<!-- 视频画面区域 -->
|
||||||
|
<view class="video-container">
|
||||||
|
<view class="video-grid" :class="videoGridClass">
|
||||||
|
<view class="video-item" v-for="(item, index) in videoData.videoList" :key="index"
|
||||||
|
:class="{ 'dragging': data.dragState.draggingIndex === index }" :style="getItemStyle(index)"
|
||||||
|
@touchstart="data.isEdit ? handleTouchStart($event, index) : null"
|
||||||
|
@touchmove.prevent="data.isEdit ? handleTouchMove($event, index) : null"
|
||||||
|
@touchend="data.isEdit ? handleTouchEnd($event, index) : null" @click.stop="changeIconType(item)">
|
||||||
|
<image class="video-preview" :src="item.preview" mode="aspectFill"></image>
|
||||||
|
<view class="video-overlay">
|
||||||
|
<image class="mute-icon" v-if="item.iconType > 0"
|
||||||
|
:src="`/static/image/other/video-call/${item.iconType == 1 ? 'mute' : 'unmute'}.png`">
|
||||||
|
</image>
|
||||||
|
</view>
|
||||||
|
<view v-if="data.isEdit" class="close-btn" @click.stop="deleteVideo(index)">
|
||||||
|
<image style="width: 40rpx;height: 40rpx;" src="/static/image/common/tipLayer-close.png">
|
||||||
|
</image>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view v-if="videoData.videoList.length < 9 && data.isEdit" class="video-item"
|
||||||
|
style="background-color: #F3F3F3;" @click="addVideo">
|
||||||
|
<image style="width: 54rpx;height: 54rpx;" src="/static/image/other/video-call/add-image.png"
|
||||||
|
mode="aspectFill">
|
||||||
|
</image>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 底部控制栏 -->
|
||||||
|
<view class="control-bar">
|
||||||
|
<view class="control-buttons">
|
||||||
|
<!-- 麦克风 -->
|
||||||
|
<view class="control-item">
|
||||||
|
<view class="control-btn" :class="{ active: videoData.micOn }" @click="changeInfo('micOn')">
|
||||||
|
<image class="control-icon"
|
||||||
|
:src="videoData.micOn ? '/static/image/other/video-call/mic-on.png' : '/static/image/other/video-call/mic-off.png'">
|
||||||
|
</image>
|
||||||
|
</view>
|
||||||
|
<text class="control-label">{{ videoData.micOn ? '麦克风已开' : '麦克风已关' }}</text>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 扬声器 -->
|
||||||
|
<view class="control-item">
|
||||||
|
<view class="control-btn" :class="{ active: videoData.speakerOn }" @click="changeInfo('speakerOn')">
|
||||||
|
<image class="control-icon"
|
||||||
|
:src="videoData.speakerOn ? '/static/image/other/video-call/speaker-on.png' : '/static/image/other/video-call/speaker-off.png'">
|
||||||
|
</image>
|
||||||
|
</view>
|
||||||
|
<text class="control-label">{{ videoData.speakerOn ? '扬声器已开' : '扬声器已关' }}</text>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 摄像头 -->
|
||||||
|
<view class="control-item">
|
||||||
|
<view class="control-btn" :class="{ active: videoData.cameraOn }" @click="changeInfo('cameraOn')">
|
||||||
|
<image class="control-icon"
|
||||||
|
:src="videoData.cameraOn ? '/static/image/other/video-call/camera-on.png' : '/static/image/other/video-call/camera-off.png'">
|
||||||
|
</image>
|
||||||
|
</view>
|
||||||
|
<text class="control-label">{{ videoData.cameraOn ? '摄像头已开' : '摄像头已关' }}</text>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 挂断按钮 -->
|
||||||
|
<view class="hangup-btn" @click="hangup">
|
||||||
|
<image class="hangup-icon" src="/static/image/other/video-call/hangup.png"></image>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
|
||||||
|
<!-- 时间编辑弹窗 -->
|
||||||
|
<view v-if="data.showTimeEditPopup" class="popup-overlay" @click="closeTimeEditPopup">
|
||||||
|
<view class="popup-content" @click.stop>
|
||||||
|
<view class="popup-header">
|
||||||
|
<text class="popup-title">通话时长</text>
|
||||||
|
</view>
|
||||||
|
<view class="popup-body">
|
||||||
|
<view class="time-edit-row">
|
||||||
|
<text class="time-label">分钟</text>
|
||||||
|
<input class="time-input" type="number" v-model="data.tempMinutes" placeholder="00" maxlength="3" />
|
||||||
|
<text class="time-separator">:</text>
|
||||||
|
<text class="time-label">秒钟</text>
|
||||||
|
<input class="time-input" type="number" v-model="data.tempSeconds" placeholder="00" maxlength="2" />
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
<view class="popup-footer">
|
||||||
|
<button class="btn-cancel" @click="closeTimeEditPopup">取消</button>
|
||||||
|
<button class="btn-save" @click="saveTimeEdit">保存</button>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import NavBar from "@/components/nav-bar/nav-bar.vue"
|
||||||
|
|
||||||
|
import { ref, toRefs, onMounted, onUnmounted, reactive, computed } from 'vue'
|
||||||
|
import { onLoad, onShow } from '@dcloudio/uni-app'
|
||||||
|
import { util } from '@/utils/common.js'
|
||||||
|
|
||||||
|
const buttonGroup = [
|
||||||
|
{
|
||||||
|
name: "编辑时间",
|
||||||
|
click: () => {
|
||||||
|
openTimeEditPopup()
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
name: "编辑聊天人数",
|
||||||
|
click: () => {
|
||||||
|
// 进入编辑模式前备份数据
|
||||||
|
data.videoDataBackup = JSON.parse(JSON.stringify(data.videoData))
|
||||||
|
data.isEdit = true
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
|
||||||
|
const data = reactive({
|
||||||
|
videoData: {
|
||||||
|
micOn: true,
|
||||||
|
speakerOn: true,
|
||||||
|
cameraOn: false,
|
||||||
|
mainVideoIndex: 0,
|
||||||
|
timeText: '125:22',
|
||||||
|
videoList: [
|
||||||
|
{ preview: '/static/image/other/video-call/defualt/video-img1.png', iconType: 0 },
|
||||||
|
{ preview: '/static/image/other/video-call/defualt/video-img2.png', iconType: 1 },
|
||||||
|
{ preview: '/static/image/other/video-call/defualt/video-img3.png', iconType: 2 }
|
||||||
|
]
|
||||||
|
},
|
||||||
|
videoDataBackup: null, // 编辑模式备份
|
||||||
|
isEdit: false,
|
||||||
|
statusBarHeight: 0,
|
||||||
|
// 时间编辑弹窗
|
||||||
|
showTimeEditPopup: false,
|
||||||
|
tempMinutes: '0',
|
||||||
|
tempSeconds: '00',
|
||||||
|
// 拖动状态管理
|
||||||
|
dragState: {
|
||||||
|
draggingIndex: -1, // 当前拖动的索引
|
||||||
|
startY: 0, // 触摸起始Y坐标
|
||||||
|
startX: 0, // 触摸起始X坐标
|
||||||
|
offsetX: 0, // 当前X偏移量
|
||||||
|
offsetY: 0, // 当前Y偏移量
|
||||||
|
longPressTimer: null, // 长按定时器
|
||||||
|
isDragging: false // 是否正在拖动
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
let { videoData, statusBarHeight } = toRefs(data)
|
||||||
|
|
||||||
|
// 视频网格样式计算属性
|
||||||
|
const videoGridClass = computed(() => {
|
||||||
|
const count = videoData.value.videoList.length
|
||||||
|
if (data.isEdit) {
|
||||||
|
if (count <= 1) return 'video-grid-2'
|
||||||
|
if (count <= 3) return 'video-grid-3'
|
||||||
|
} else {
|
||||||
|
if (count <= 2) return 'video-grid-2'
|
||||||
|
if (count <= 4) return 'video-grid-3'
|
||||||
|
}
|
||||||
|
return 'video-grid-5'
|
||||||
|
})
|
||||||
|
|
||||||
|
// 获取当前网格的列数
|
||||||
|
const getGridCols = () => {
|
||||||
|
const count = videoData.value.videoList.length
|
||||||
|
if (data.isEdit) {
|
||||||
|
if (count <= 1) return 2
|
||||||
|
if (count <= 3) return 2
|
||||||
|
} else {
|
||||||
|
if (count <= 2) return 2
|
||||||
|
if (count <= 4) return 2
|
||||||
|
}
|
||||||
|
return 3
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取网格尺寸
|
||||||
|
const getGridSize = () => {
|
||||||
|
const cols = getGridCols()
|
||||||
|
return uni.getSystemInfoSync().windowWidth / cols
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取每个video-item的样式(用于拖动跟随)
|
||||||
|
const getItemStyle = (index) => {
|
||||||
|
if (data.dragState.draggingIndex === index && data.dragState.isDragging) {
|
||||||
|
return {
|
||||||
|
transform: `translate(${data.dragState.offsetX}px, ${data.dragState.offsetY}px)`,
|
||||||
|
transition: 'none'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return {}
|
||||||
|
}
|
||||||
|
|
||||||
|
onMounted(() => {
|
||||||
|
const systemInfo = uni.getSystemInfoSync()
|
||||||
|
statusBarHeight.value = systemInfo.statusBarHeight || 0
|
||||||
|
})
|
||||||
|
|
||||||
|
onLoad(() => {
|
||||||
|
data.videoData = uni.getStorageSync('videoData') || data.videoData
|
||||||
|
})
|
||||||
|
|
||||||
|
onShow(() => {
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
util.setAndroidSystemBarColor('#232323')
|
||||||
|
setTimeout(() => {
|
||||||
|
plus.navigator.setStatusBarStyle("light");
|
||||||
|
}, 500)
|
||||||
|
// #endif
|
||||||
|
})
|
||||||
|
|
||||||
|
// 打开时间编辑弹窗
|
||||||
|
const openTimeEditPopup = () => {
|
||||||
|
// 解析当前时间 (格式: "125:22")
|
||||||
|
const parts = data.videoData.timeText.split(':')
|
||||||
|
data.tempMinutes = parts[0] || '0'
|
||||||
|
data.tempSeconds = parts[1] || '00'
|
||||||
|
data.showTimeEditPopup = true
|
||||||
|
}
|
||||||
|
|
||||||
|
// 保存时间编辑
|
||||||
|
const saveTimeEdit = () => {
|
||||||
|
// 格式化分钟和秒钟
|
||||||
|
const minutes = parseInt(data.tempMinutes) || 0
|
||||||
|
const seconds = parseInt(data.tempSeconds) || 0
|
||||||
|
|
||||||
|
// 秒钟不能超过59
|
||||||
|
const validSeconds = Math.min(59, Math.max(0, seconds))
|
||||||
|
|
||||||
|
// 格式化为两位数
|
||||||
|
const formattedSeconds = validSeconds.toString().padStart(2, '0')
|
||||||
|
|
||||||
|
// 更新时间文本
|
||||||
|
data.videoData.timeText = `${minutes}:${formattedSeconds}`
|
||||||
|
|
||||||
|
// 保存到storage
|
||||||
|
uni.setStorageSync('videoData', data.videoData)
|
||||||
|
|
||||||
|
// 关闭弹窗
|
||||||
|
data.showTimeEditPopup = false
|
||||||
|
}
|
||||||
|
|
||||||
|
// 关闭时间编辑弹窗
|
||||||
|
const closeTimeEditPopup = () => {
|
||||||
|
data.showTimeEditPopup = false
|
||||||
|
}
|
||||||
|
|
||||||
|
const confirmEdit = async () => {
|
||||||
|
// 保存临时图片为永久路径
|
||||||
|
try {
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
const savePromises = data.videoData.videoList.map(async (item) => {
|
||||||
|
// 检查是否为临时路径(通常包含 tmp 或 temp)
|
||||||
|
if (item.preview && (item.preview.includes('tmp') || item.preview.includes('temp'))) {
|
||||||
|
try {
|
||||||
|
const savedFile = await new Promise((resolve, reject) => {
|
||||||
|
uni.saveFile({
|
||||||
|
tempFilePath: item.preview,
|
||||||
|
success: (res) => resolve(res.savedFilePath),
|
||||||
|
fail: (err) => reject(err)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
// 更新为永久路径
|
||||||
|
item.preview = savedFile
|
||||||
|
console.log('✅ 图片已保存:', savedFile)
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ 保存图片失败:', item.preview, error)
|
||||||
|
// 保存失败时保持原路径
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
// 等待所有图片保存完成
|
||||||
|
await Promise.all(savePromises)
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// 保存数据
|
||||||
|
uni.setStorageSync('videoData', data.videoData)
|
||||||
|
console.log('✅ 视频数据已保存')
|
||||||
|
} catch (error) {
|
||||||
|
console.error('❌ 保存过程出错:', error)
|
||||||
|
// 即使出错也保存数据
|
||||||
|
uni.setStorageSync('videoData', data.videoData)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 退出编辑模式
|
||||||
|
data.isEdit = false
|
||||||
|
data.videoDataBackup = null
|
||||||
|
}
|
||||||
|
|
||||||
|
const changeIconType = (item) => {
|
||||||
|
item.iconType = item.iconType == 2 ? 0 : item.iconType + 1
|
||||||
|
// 编辑模式下不立即保存,点击完成后统一保存
|
||||||
|
if (!data.isEdit) {
|
||||||
|
uni.setStorageSync('videoData', data.videoData)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const addVideo = () => {
|
||||||
|
if (data.videoData.videoList.length >= 9) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
uni.chooseImage({
|
||||||
|
count: 9 - data.videoData.videoList.length,
|
||||||
|
sizeType: ['original', 'compressed'],
|
||||||
|
sourceType: ['album', 'camera'],
|
||||||
|
success: (res) => {
|
||||||
|
data.videoData.videoList.push(...res.tempFilePaths.map((item) => ({ preview: item, iconType: 1 })))
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 改变信息
|
||||||
|
const changeInfo = (key) => {
|
||||||
|
data.videoData[key] = !data.videoData[key]
|
||||||
|
uni.setStorageSync('videoData', data.videoData)
|
||||||
|
}
|
||||||
|
|
||||||
|
const deleteVideo = (index) => {
|
||||||
|
data.videoData.videoList.splice(index, 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 触摸开始 - 启动长按检测
|
||||||
|
const handleTouchStart = (event, index) => {
|
||||||
|
if (!data.isEdit) return
|
||||||
|
|
||||||
|
const touch = event.touches[0]
|
||||||
|
data.dragState.startX = touch.clientX
|
||||||
|
data.dragState.startY = touch.clientY
|
||||||
|
|
||||||
|
// 长按300ms后启动拖动模式
|
||||||
|
data.dragState.longPressTimer = setTimeout(() => {
|
||||||
|
data.dragState.isDragging = true
|
||||||
|
data.dragState.draggingIndex = index
|
||||||
|
// 震动反馈(如果支持)
|
||||||
|
// #ifdef APP-PLUS || MP-WEIXIN
|
||||||
|
uni.vibrateShort({ type: 'heavy' })
|
||||||
|
// #endif
|
||||||
|
}, 300)
|
||||||
|
}
|
||||||
|
|
||||||
|
// 触摸移动 - 实时更新位置偏移
|
||||||
|
const handleTouchMove = (event, index) => {
|
||||||
|
if (!data.isEdit || !data.dragState.isDragging) return
|
||||||
|
|
||||||
|
const touch = event.touches[0]
|
||||||
|
const deltaX = touch.clientX - data.dragState.startX
|
||||||
|
const deltaY = touch.clientY - data.dragState.startY
|
||||||
|
|
||||||
|
// 更新偏移量,让元素跟随手指移动
|
||||||
|
data.dragState.offsetX = deltaX
|
||||||
|
data.dragState.offsetY = deltaY
|
||||||
|
}
|
||||||
|
|
||||||
|
// 触摸结束 - 完成拖动并保存
|
||||||
|
const handleTouchEnd = (event, index) => {
|
||||||
|
if (!data.isEdit) return
|
||||||
|
|
||||||
|
// 清除长按定时器
|
||||||
|
if (data.dragState.longPressTimer) {
|
||||||
|
clearTimeout(data.dragState.longPressTimer)
|
||||||
|
data.dragState.longPressTimer = null
|
||||||
|
}
|
||||||
|
|
||||||
|
// 如果发生了拖动,计算最终位置并交换
|
||||||
|
if (data.dragState.isDragging) {
|
||||||
|
const deltaX = data.dragState.offsetX
|
||||||
|
const deltaY = data.dragState.offsetY
|
||||||
|
|
||||||
|
// 计算目标索引
|
||||||
|
const cols = getGridCols()
|
||||||
|
const gridSize = getGridSize()
|
||||||
|
|
||||||
|
console.log('🔍 拖动结束 - cols:', cols, 'gridSize:', gridSize, 'deltaX:', deltaX, 'deltaY:', deltaY)
|
||||||
|
|
||||||
|
const moveX = Math.round(deltaX / gridSize)
|
||||||
|
const moveY = Math.round(deltaY / gridSize)
|
||||||
|
|
||||||
|
console.log('🔍 移动格数 - moveX:', moveX, 'moveY:', moveY)
|
||||||
|
|
||||||
|
const currentRow = Math.floor(data.dragState.draggingIndex / cols)
|
||||||
|
const currentCol = data.dragState.draggingIndex % cols
|
||||||
|
|
||||||
|
const targetRow = currentRow + moveY
|
||||||
|
const targetCol = currentCol + moveX
|
||||||
|
|
||||||
|
console.log('🔍 位置计算 - 当前:', `(${currentRow},${currentCol})`, '目标:', `(${targetRow},${targetCol})`)
|
||||||
|
|
||||||
|
if (targetCol >= 0 && targetCol < cols && targetRow >= 0) {
|
||||||
|
const targetIndex = targetRow * cols + targetCol
|
||||||
|
|
||||||
|
console.log('🔍 索引计算 - draggingIndex:', data.dragState.draggingIndex, 'targetIndex:', targetIndex, 'listLength:', videoData.value.videoList.length)
|
||||||
|
|
||||||
|
if (targetIndex >= 0 && targetIndex < videoData.value.videoList.length && targetIndex !== data.dragState.draggingIndex) {
|
||||||
|
// 交换数组元素
|
||||||
|
const list = videoData.value.videoList
|
||||||
|
const temp = list[data.dragState.draggingIndex]
|
||||||
|
list[data.dragState.draggingIndex] = list[targetIndex]
|
||||||
|
list[targetIndex] = temp
|
||||||
|
|
||||||
|
console.log('✅ 交换成功!')
|
||||||
|
} else {
|
||||||
|
console.log('❌ 目标索引无效,不交换 - 原因:', targetIndex < 0 ? '索引<0' : targetIndex >= videoData.value.videoList.length ? '索引超出' : '索引相同')
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.log('❌ 目标位置超出范围,不交换 - targetCol:', targetCol, 'targetRow:', targetRow)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 重置拖动状态
|
||||||
|
data.dragState.isDragging = false
|
||||||
|
data.dragState.draggingIndex = -1
|
||||||
|
data.dragState.offsetX = 0
|
||||||
|
data.dragState.offsetY = 0
|
||||||
|
}
|
||||||
|
|
||||||
|
const hangup = () => {
|
||||||
|
uni.navigateBack()
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="less" scoped>
|
||||||
|
.container {
|
||||||
|
width: 100%;
|
||||||
|
height: 100vh;
|
||||||
|
background-color: #232323;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: space-between;
|
||||||
|
}
|
||||||
|
|
||||||
|
.nav-content {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
height: 88rpx;
|
||||||
|
padding: 0 16px;
|
||||||
|
|
||||||
|
.icon {
|
||||||
|
width: 24px;
|
||||||
|
height: 24px;
|
||||||
|
flex-shrink: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.left {
|
||||||
|
height: 100%;
|
||||||
|
width: 80px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.center {
|
||||||
|
height: 100%;
|
||||||
|
flex: 1;
|
||||||
|
text-align: center;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
.time {
|
||||||
|
font-size: 16px;
|
||||||
|
color: #ffffff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.right {
|
||||||
|
width: 80px;
|
||||||
|
height: 100%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: flex-end;
|
||||||
|
gap: 12px;
|
||||||
|
|
||||||
|
.button {
|
||||||
|
font-size: 28rpx;
|
||||||
|
color: #ffffff;
|
||||||
|
background-color: #07C160;
|
||||||
|
padding: 10rpx 20rpx;
|
||||||
|
text-align: center;
|
||||||
|
border-radius: 12rpx;
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 视频区域 */
|
||||||
|
.video-container {
|
||||||
|
flex: 1;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
.video-grid-2 {
|
||||||
|
margin-top: 200rpx;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
width: 100%;
|
||||||
|
|
||||||
|
.video-item {
|
||||||
|
width: 50vw;
|
||||||
|
height: 50vw;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-grid-3 {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-content: flex-start;
|
||||||
|
width: 100%;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
|
||||||
|
.video-item {
|
||||||
|
width: 50vw;
|
||||||
|
height: 50vw;
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-grid-5 {
|
||||||
|
display: flex;
|
||||||
|
justify-content: flex-start;
|
||||||
|
align-content: flex-start;
|
||||||
|
width: 100%;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
|
||||||
|
.video-item {
|
||||||
|
width: calc(100vw / 3);
|
||||||
|
height: calc(100vw / 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-item {
|
||||||
|
position: relative;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
|
||||||
|
.video-preview {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.video-overlay {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 8px;
|
||||||
|
left: 8px;
|
||||||
|
width: 26px;
|
||||||
|
height: 26px;
|
||||||
|
border-radius: 50%;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.close-btn {
|
||||||
|
position: absolute;
|
||||||
|
top: 8px;
|
||||||
|
right: 8px;
|
||||||
|
width: 40rpx;
|
||||||
|
height: 40rpx;
|
||||||
|
border-radius: 50%;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 拖动时的样式
|
||||||
|
&.dragging {
|
||||||
|
opacity: 0.7;
|
||||||
|
transform: scale(1.05);
|
||||||
|
z-index: 999;
|
||||||
|
transition: transform 0.2s ease, opacity 0.2s ease;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
.mute-icon {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 控制栏 */
|
||||||
|
.control-bar {
|
||||||
|
flex-shrink: 0;
|
||||||
|
padding: 40rpx 32rpx 80rpx;
|
||||||
|
background-color: transparent;
|
||||||
|
|
||||||
|
.control-buttons {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-around;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-item {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-btn {
|
||||||
|
width: 120rpx;
|
||||||
|
height: 120rpx;
|
||||||
|
border-radius: 50%;
|
||||||
|
background-color: #4a4a4a;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
transition: all 0.3s;
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
background-color: #ffffff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-icon {
|
||||||
|
width: 100%·;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.control-label {
|
||||||
|
font-size: 12px;
|
||||||
|
color: #999999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hangup-btn {
|
||||||
|
width: 120rpx;
|
||||||
|
height: 120rpx;
|
||||||
|
border-radius: 50%;
|
||||||
|
background-color: #ff3b30;
|
||||||
|
margin: 0 auto;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.hangup-icon {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 时间编辑弹窗样式 */
|
||||||
|
.popup-overlay {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background-color: rgba(0, 0, 0, 0.7);
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
z-index: 10000;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-content {
|
||||||
|
background-color: #fff;
|
||||||
|
border-radius: 20rpx;
|
||||||
|
width: 600rpx;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 30rpx 16rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-title {
|
||||||
|
font-size: 32rpx;
|
||||||
|
font-weight: bold;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-close {
|
||||||
|
font-size: 60rpx;
|
||||||
|
color: #999;
|
||||||
|
line-height: 1;
|
||||||
|
padding: 0 10rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-body {
|
||||||
|
padding: 20rpx 0;
|
||||||
|
padding-bottom: 40rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-edit-row {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 20rpx;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-label {
|
||||||
|
font-size: 28rpx;
|
||||||
|
color: #666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-input {
|
||||||
|
width: 120rpx;
|
||||||
|
height: 60rpx;
|
||||||
|
border-radius: 10rpx;
|
||||||
|
text-align: center;
|
||||||
|
font-size: 30rpx;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.time-separator {
|
||||||
|
font-size: 36rpx;
|
||||||
|
font-weight: 500;
|
||||||
|
color: #333;
|
||||||
|
margin: 0 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.popup-footer {
|
||||||
|
display: flex;
|
||||||
|
border-top: 1rpx solid #eee;
|
||||||
|
|
||||||
|
::v-deep uni-button:after {
|
||||||
|
border: none !important;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-cancel,
|
||||||
|
.btn-save {
|
||||||
|
flex: 1;
|
||||||
|
height: 100rpx;
|
||||||
|
line-height: 100rpx;
|
||||||
|
text-align: center;
|
||||||
|
font-size: 32rpx;
|
||||||
|
border: none;
|
||||||
|
background: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-cancel {
|
||||||
|
color: #666;
|
||||||
|
border-right: 1px solid #eee;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-save {
|
||||||
|
color: #07C160;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
After Width: | Height: | Size: 97 KiB |
|
After Width: | Height: | Size: 62 KiB |
|
After Width: | Height: | Size: 675 B |
|
After Width: | Height: | Size: 2.0 KiB |
|
After Width: | Height: | Size: 2.3 KiB |
|
After Width: | Height: | Size: 1.1 KiB |
|
After Width: | Height: | Size: 8.0 KiB |
|
Before Width: | Height: | Size: 1.3 KiB After Width: | Height: | Size: 1.7 KiB |
|
After Width: | Height: | Size: 7.0 KiB |
|
After Width: | Height: | Size: 4.0 KiB |
|
After Width: | Height: | Size: 3.3 KiB |
|
After Width: | Height: | Size: 3.8 KiB |
|
After Width: | Height: | Size: 3.6 KiB |
|
After Width: | Height: | Size: 610 KiB |
|
After Width: | Height: | Size: 4.4 KiB |
|
After Width: | Height: | Size: 189 B |
|
After Width: | Height: | Size: 115 B |
|
After Width: | Height: | Size: 1.3 KiB |
|
After Width: | Height: | Size: 1.1 KiB |
|
After Width: | Height: | Size: 129 KiB |
|
After Width: | Height: | Size: 136 KiB |
|
After Width: | Height: | Size: 103 KiB |
|
After Width: | Height: | Size: 339 B |
|
After Width: | Height: | Size: 1.5 KiB |
|
After Width: | Height: | Size: 1.6 KiB |
|
After Width: | Height: | Size: 1.6 KiB |
|
After Width: | Height: | Size: 994 B |
|
After Width: | Height: | Size: 374 B |
|
After Width: | Height: | Size: 1.6 KiB |
|
After Width: | Height: | Size: 1.4 KiB |
|
After Width: | Height: | Size: 926 B |
|
|
@ -0,0 +1,225 @@
|
||||||
|
## 1.9.6.6(2024-09-25)
|
||||||
|
- fix: 修复background-position无效的问题
|
||||||
|
## 1.9.6.5(2024-04-14)
|
||||||
|
- fix: 修复`nvue`无法生图的问题
|
||||||
|
## 1.9.6.4(2024-03-10)
|
||||||
|
- fix: 修复代理ctx导致H5不能使用ctx.save
|
||||||
|
## 1.9.6.3(2024-03-08)
|
||||||
|
- fix: 修复支付宝真机无法使用的问题
|
||||||
|
## 1.9.6.2(2024-02-22)
|
||||||
|
- fix: 修复使用render函数报错的问题
|
||||||
|
## 1.9.6.1(2023-12-22)
|
||||||
|
- fix: 修复字节小程序非2d字体偏移
|
||||||
|
- fix: 修复`canvasToTempFilePathSync`会触发两次的问题
|
||||||
|
- fix: 修复`parser`图片没有宽度的问题
|
||||||
|
## 1.9.6(2023-12-06)
|
||||||
|
- fix: 修复背景图受padding影响
|
||||||
|
- fix: 修复因字节报错改了代理实现导致微信报错
|
||||||
|
- 1.9.5.8(2023-11-16)
|
||||||
|
- fix: 修复margin问题
|
||||||
|
- fix: 修复borderWidth问题
|
||||||
|
- fix: 修复textBox问题
|
||||||
|
- fix: 修复字节开发工具报`could not be cloned.`问题
|
||||||
|
## 1.9.5.7(2023-07-27)
|
||||||
|
- fix: 去掉多余的方法
|
||||||
|
- chore: 更新文档,增加自定义字体说明
|
||||||
|
## 1.9.5.6(2023-07-21)
|
||||||
|
- feat: 有限的支持富文本
|
||||||
|
- feat: H5和APP 增加 `hidpi` prop,主要用于大尺寸无法生成图片时用
|
||||||
|
- fix: 修复 钉钉小程序 缺少 `measureText` 方法
|
||||||
|
- chore: 由于微信小程序 pc 端的 canvas 2d 时不时抽风,故不使用canvas 2d
|
||||||
|
## 1.9.5.5(2023-06-27)
|
||||||
|
- fix: 修复把`emoji`表情字符拆分成多个字符的情况
|
||||||
|
## 1.9.5.4(2023-06-05)
|
||||||
|
- fix: 修复因`canvasToTempFilePathSync`监听导致重复调用
|
||||||
|
## 1.9.5.3(2023-05-23)
|
||||||
|
- fix: 因isPc错写成了isPC导致小程序PC不能生成图片
|
||||||
|
## 1.9.5.2(2023-05-22)
|
||||||
|
- feat: 删除多余文件
|
||||||
|
## 1.9.5.1(2023-05-22)
|
||||||
|
- fix: 修复 文字行数与`line-clamp`相同但不满一行时也加了省略号的问题
|
||||||
|
## 1.9.5(2023-05-14)
|
||||||
|
- feat: 增加 `text-indent` 和 `calc` 方法
|
||||||
|
- feat: 优化 布局时间
|
||||||
|
## 1.9.4.4(2023-04-15)
|
||||||
|
- fix: 修复无法匹配负值
|
||||||
|
- fix: 修复 Nvue IOS getImageInfo `useCORS` 为 undefined
|
||||||
|
## 1.9.4.3(2023-04-01)
|
||||||
|
- feat: 增加支持文字描边 `text-stroke: '5rpx #fff'`
|
||||||
|
## 1.9.4.2(2023-03-30)
|
||||||
|
- fix: 修复 支付宝小程序 isPC 在手机也为true的问题
|
||||||
|
- feat: 由 微信开发工具 3060 版 无法获取图片尺寸,现 微信开发工具 3220 版 修复该问题,故还原上一版的获取图片方式。
|
||||||
|
## 1.9.4.1(2023-03-28)
|
||||||
|
- fix: 修复固定高度不正确问题
|
||||||
|
## 1.9.4(2023-03-17)
|
||||||
|
- fix: nvue ios getImageInfo缺少this报错
|
||||||
|
- fix: pathType 非2d无效问题
|
||||||
|
- fix: 修复 小米9se 可能会存在多次init 导致画面多次放大
|
||||||
|
- fix: 修复 border 分开写 width style无效问题
|
||||||
|
- fix: 修复 支付宝小程序IOS 再次进入不渲染的问题
|
||||||
|
- fix: 修复 支付宝小程序安卓Zindex排序错乱问题
|
||||||
|
- fix: 修复 微信开发工具 3060 版 无法获取图片的问题
|
||||||
|
- feat: 把 for in 改为 forEach
|
||||||
|
- feat: 增加 hidden
|
||||||
|
- feat: 根节点 box-sizing 默认 `border-box`
|
||||||
|
- feat: 增加支持 `vw` `wh`
|
||||||
|
- chore: pathType 取消 默认值,因为字节开发工具不能显示
|
||||||
|
- chore: 支付宝小程序开发工具不支持 生成图片 请以真机调试为准
|
||||||
|
- bug: 企业微信 2.20.3无法使用
|
||||||
|
## 1.9.3.5(2022-06-29)
|
||||||
|
- feat: justifyContent 增加 `space-around`、`space-between`
|
||||||
|
- feat: canvas 2d 也使用`getImageInfo`
|
||||||
|
- fix: 修复 `text`的 `text-decoration`错位
|
||||||
|
## 1.9.3.4(2022-06-20)
|
||||||
|
- fix: 修复 因创建节点速度问题导致顺序出错。
|
||||||
|
- fix: 修复 微信小程序 PC 无法显示本地图片
|
||||||
|
- fix: 修复 flex-box 对齐问题
|
||||||
|
- feat: 增加 `text-shadow`
|
||||||
|
- feat: 重写 `text` 对齐方式
|
||||||
|
- chore: 更新文档
|
||||||
|
## 1.9.3.3(2022-06-17)
|
||||||
|
- fix: 修复 支付宝小程序 canvas 2d 存在ctx.draw问题导致报错
|
||||||
|
- fix: 修复 支付宝小程序 toDataURL 存在权限问题改用 `toTempFilePath`
|
||||||
|
- fix: 修复 支付宝小程序 image size 问题导致 `objectFit` 无效
|
||||||
|
## 1.9.3.2(2022-06-14)
|
||||||
|
- fix: 修复 image 设置背景色不生效问题
|
||||||
|
- fix: 修复 nvue 环境判断缺少参数问题
|
||||||
|
## 1.9.3.1(2022-06-14)
|
||||||
|
- fix: 修复 bottom 定位不对问题
|
||||||
|
- fix: 修复 因小数导致计算出错换行问题
|
||||||
|
- feat: 增加 `useCORS` h5端图片跨域 在设置请求头无效果后试一下设置这个值
|
||||||
|
- chore: 更新文档
|
||||||
|
## 1.9.3(2022-06-13)
|
||||||
|
- feat: 增加 `zIndex`
|
||||||
|
- feat: 增加 `flex-box` 该功能处于原始阶段,非常简陋。
|
||||||
|
- tips: QQ小程序 vue3 不支持, 为 uni 官方BUG
|
||||||
|
## 1.9.2.9(2022-06-10)
|
||||||
|
- fix: 修复`text-align`及`margin`居中问题
|
||||||
|
## 1.9.2.8(2022-06-10)
|
||||||
|
- fix: 修复 Nvue `canvasToTempFilePathSync` 不生效问题
|
||||||
|
## 1.9.2.7(2022-06-10)
|
||||||
|
- fix: 修复 margin及padding的bug
|
||||||
|
- fix: 修复 Nvue `isCanvasToTempFilePath` 不生效问题
|
||||||
|
## 1.9.2.6(2022-06-09)
|
||||||
|
- fix: 修复 Nvue 不显示
|
||||||
|
- feat: 增加支持字体渐变
|
||||||
|
```html
|
||||||
|
<l-painter-text
|
||||||
|
text="水调歌头\n明月几时有?把酒问青天。不知天上宫阙,今夕是何年。我欲乘风归去,又恐琼楼玉宇,高处不胜寒。起舞弄清影,何似在人间。"
|
||||||
|
css="background: linear-gradient(,#ff971b 0%, #1989fa 100%); background-clip: text" />
|
||||||
|
```
|
||||||
|
## 1.9.2.5(2022-06-09)
|
||||||
|
- chore: 更变获取父级宽度的设定
|
||||||
|
- chore: `pathType` 在canvas 2d 默认为 `url`
|
||||||
|
## 1.9.2.4(2022-06-08)
|
||||||
|
- fix: 修复 `pathType` 不生效问题
|
||||||
|
## 1.9.2.3(2022-06-08)
|
||||||
|
- fix: 修复 `canvasToTempFilePath` 漏写 `success` 参数
|
||||||
|
## 1.9.2.2(2022-06-07)
|
||||||
|
- chore: 更新文档
|
||||||
|
## 1.9.2.1(2022-06-07)
|
||||||
|
- fix: 修复 vue3 赋值给this再传入导致image无法绘制
|
||||||
|
- fix: 修复 `canvasToTempFilePathSync` 时机问题
|
||||||
|
- feat: canvas 2d 更改图片生成方式 `toDataURL`
|
||||||
|
## 1.9.2(2022-05-30)
|
||||||
|
- fix: 修复 `canvasToTempFilePathSync` 在 vue3 下只生成一次
|
||||||
|
## 1.9.1.7(2022-05-28)
|
||||||
|
- fix: 修复 `qrcode`显示不全问题
|
||||||
|
## 1.9.1.6(2022-05-28)
|
||||||
|
- fix: 修复 `canvasToTempFilePathSync` 会重复多次问题
|
||||||
|
- fix: 修复 `view` css `backgroundImage` 图片下载失败导致 子节点不渲染
|
||||||
|
## 1.9.1.5(2022-05-27)
|
||||||
|
- fix: 修正支付宝小程序 canvas 2d版本号 2.7.15
|
||||||
|
## 1.9.1.4(2022-05-22)
|
||||||
|
- fix: 修复字节小程序无法使用xml方式
|
||||||
|
- fix: 修复字节小程序无法使用base64(非2D情况下工具上无法显示)
|
||||||
|
- fix: 修复支付宝小程序 `canvasToTempFilePath` 报错
|
||||||
|
## 1.9.1.3(2022-04-29)
|
||||||
|
- fix: 修复vue3打包后uni对象为空后的报错
|
||||||
|
## 1.9.1.2(2022-04-25)
|
||||||
|
- fix: 删除多余文件
|
||||||
|
## 1.9.1.1(2022-04-25)
|
||||||
|
- fix: 修复图片不显示问题
|
||||||
|
## 1.9.1(2022-04-12)
|
||||||
|
- fix: 因四舍五入导致有些机型错位
|
||||||
|
- fix: 修复无views报错
|
||||||
|
- chore: nvue下因ios无法读取插件内static文件,改由下载方式
|
||||||
|
## 1.9.0(2022-03-20)
|
||||||
|
- fix: 因无法固定尺寸导致生成图片不全
|
||||||
|
- fix: 特定情况下text判断无效
|
||||||
|
- chore: 本地化APP Nvue webview
|
||||||
|
## 1.8.9(2022-02-20)
|
||||||
|
- fix: 修复 小程序下载最多10次并发的问题
|
||||||
|
- fix: 修复 APP端无法获取本地图片
|
||||||
|
- fix: 修复 APP Nvue端不执行问题
|
||||||
|
- chore: 增加图片缓存机制
|
||||||
|
## 1.8.8.8(2022-01-27)
|
||||||
|
- fix: 修复 主动调用尺寸问题
|
||||||
|
## 1.8.8.6(2022-01-26)
|
||||||
|
- fix: 修复 nvue 下无宽度时获取父级宽度
|
||||||
|
- fix: 修复 ios app 无法渲染问题
|
||||||
|
## 1.8.8(2022-01-23)
|
||||||
|
- fix: 修复 主动调用时无节点问题
|
||||||
|
- fix: 修复 `box-shadow` 颜色问题
|
||||||
|
- fix: 修复 `transform:rotate` 角度位置问题
|
||||||
|
- feat: 增加 `overflow:hidden`
|
||||||
|
## 1.8.7(2022-01-07)
|
||||||
|
- fix: 修复 image 方向为 `right` 时原始宽高问题
|
||||||
|
- feat: 支持 view 设置背景图 `background-image: url(xxx)`
|
||||||
|
- chore: 去掉可选链
|
||||||
|
## 1.8.6(2021-11-28)
|
||||||
|
- feat: 支持`view`对`inline-block`的子集使用`text-align`
|
||||||
|
## 1.8.5.5(2021-08-17)
|
||||||
|
- chore: 更新文档,删除 replace
|
||||||
|
- fix: 修复 text 值为 number时报错
|
||||||
|
## 1.8.5.4(2021-08-16)
|
||||||
|
- fix: 字节小程序兼容
|
||||||
|
## 1.8.5.3(2021-08-15)
|
||||||
|
- fix: 修复线性渐变与css现实效果不一致的问题
|
||||||
|
- chore: 更新文档
|
||||||
|
## 1.8.5.2(2021-08-13)
|
||||||
|
- chore: 增加`background-image`、`background-repeat` 能力,主要用于背景纹理的绘制,并不是代替`image`。例如:大面积的重复平铺的水印
|
||||||
|
- 注意:这个功能H5暂时无法使用,因为[官方的API有BUG](https://ask.dcloud.net.cn/question/128793),待官方修复!!!
|
||||||
|
## 1.8.5.1(2021-08-10)
|
||||||
|
- fix: 修复因`margin`报错问题
|
||||||
|
## 1.8.5(2021-08-09)
|
||||||
|
- chore: 增加margin支持`auto`,以达到居中效果
|
||||||
|
## 1.8.4(2021-08-06)
|
||||||
|
- chore: 增加判断缓存文件条件
|
||||||
|
- fix: 修复css 多余空格报错问题
|
||||||
|
## 1.8.3(2021-08-04)
|
||||||
|
- tips: 1.6.x 以下的版本升级到1.8.x后要为每个元素都加上定位:position: 'absolute'
|
||||||
|
- fix: 修复只有一个view子元素时不计算高度的问题
|
||||||
|
## 1.8.2(2021-08-03)
|
||||||
|
- fix: 修复 path-type 为 `url` 无效问题
|
||||||
|
- fix: 修复 qrcode `text` 为空时报错问题
|
||||||
|
- fix: 修复 image `src` 动态设置时不生效问题
|
||||||
|
- feat: 增加 css 属性 `min-width` `max-width`
|
||||||
|
## 1.8.1(2021-08-02)
|
||||||
|
- fix: 修复无法加载本地图片
|
||||||
|
## 1.8.0(2021-08-02)
|
||||||
|
- chore 文档更新
|
||||||
|
- 使用旧版的同学不要升级!
|
||||||
|
## 1.8.0-beta(2021-07-30)
|
||||||
|
- ## 全新布局方式 不兼容旧版!
|
||||||
|
- chore: 布局方式变更
|
||||||
|
- tips: 微信canvas 2d 不支持真机调试
|
||||||
|
## 1.6.6(2021-07-09)
|
||||||
|
- chore: 统一命名规范,无须主动引入组件
|
||||||
|
## 1.6.5(2021-06-08)
|
||||||
|
- chore: 去掉console
|
||||||
|
## 1.6.4(2021-06-07)
|
||||||
|
- fix: 修复 数字 为纯字符串时不转换的BUG
|
||||||
|
## 1.6.3(2021-06-06)
|
||||||
|
- fix: 修复 PC 端放大的BUG
|
||||||
|
## 1.6.2(2021-05-31)
|
||||||
|
- fix: 修复 报`adaptor is not a function`错误
|
||||||
|
- fix: 修复 text 多行高度
|
||||||
|
- fix: 优化 默认文字的基准线
|
||||||
|
- feat: `@progress`事件,监听绘制进度
|
||||||
|
## 1.6.1(2021-02-28)
|
||||||
|
- 删除多余节点
|
||||||
|
## 1.6.0(2021-02-26)
|
||||||
|
- 调整为uni_modules目录规范
|
||||||
|
- 修复:transform的rotate不能为负数问题
|
||||||
|
- 新增:`pathType` 指定生成图片返回的路径类型,可选值有 `base64`、`url`
|
||||||
|
|
@ -0,0 +1,150 @@
|
||||||
|
const styles = (v ='') => v.split(';').filter(v => v && !/^[\n\s]+$/.test(v)).map(v => {
|
||||||
|
const key = v.slice(0, v.indexOf(':'))
|
||||||
|
const value = v.slice(v.indexOf(':')+1)
|
||||||
|
return {
|
||||||
|
[key
|
||||||
|
.replace(/-([a-z])/g, function() { return arguments[1].toUpperCase()})
|
||||||
|
.replace(/\s+/g, '')
|
||||||
|
]: value.replace(/^\s+/, '').replace(/\s+$/, '') || ''
|
||||||
|
}
|
||||||
|
})
|
||||||
|
export function parent(parent) {
|
||||||
|
return {
|
||||||
|
provide() {
|
||||||
|
return {
|
||||||
|
[parent]: this
|
||||||
|
}
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
el: {
|
||||||
|
id: null,
|
||||||
|
css: {},
|
||||||
|
views: []
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
css: {
|
||||||
|
handler(v) {
|
||||||
|
if(this.canvasId) {
|
||||||
|
this.el.css = (typeof v == 'object' ? v : v && Object.assign(...styles(v))) || {}
|
||||||
|
this.canvasWidth = this.el.css && this.el.css.width || this.canvasWidth
|
||||||
|
this.canvasHeight = this.el.css && this.el.css.height || this.canvasHeight
|
||||||
|
}
|
||||||
|
},
|
||||||
|
immediate: true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
export function children(parent, options = {}) {
|
||||||
|
const indexKey = options.indexKey || 'index'
|
||||||
|
return {
|
||||||
|
inject: {
|
||||||
|
[parent]: {
|
||||||
|
default: null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
watch: {
|
||||||
|
el: {
|
||||||
|
handler(v, o) {
|
||||||
|
if(JSON.stringify(v) != JSON.stringify(o))
|
||||||
|
this.bindRelation()
|
||||||
|
},
|
||||||
|
deep: true,
|
||||||
|
immediate: true
|
||||||
|
},
|
||||||
|
src: {
|
||||||
|
handler(v, o) {
|
||||||
|
if(v != o)
|
||||||
|
this.bindRelation()
|
||||||
|
},
|
||||||
|
immediate: true
|
||||||
|
},
|
||||||
|
text: {
|
||||||
|
handler(v, o) {
|
||||||
|
if(v != o) this.bindRelation()
|
||||||
|
},
|
||||||
|
immediate: true
|
||||||
|
},
|
||||||
|
css: {
|
||||||
|
handler(v, o) {
|
||||||
|
if(v != o)
|
||||||
|
this.el.css = (typeof v == 'object' ? v : v && Object.assign(...styles(v))) || {}
|
||||||
|
},
|
||||||
|
immediate: true
|
||||||
|
},
|
||||||
|
replace: {
|
||||||
|
handler(v, o) {
|
||||||
|
if(JSON.stringify(v) != JSON.stringify(o))
|
||||||
|
this.bindRelation()
|
||||||
|
},
|
||||||
|
deep: true,
|
||||||
|
immediate: true
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
if(!this._uid) {
|
||||||
|
this._uid = this._.uid
|
||||||
|
}
|
||||||
|
Object.defineProperty(this, 'parent', {
|
||||||
|
get: () => this[parent] || [],
|
||||||
|
})
|
||||||
|
Object.defineProperty(this, 'index', {
|
||||||
|
get: () => {
|
||||||
|
this.bindRelation();
|
||||||
|
const {parent: {el: {views=[]}={}}={}} = this
|
||||||
|
return views.indexOf(this.el)
|
||||||
|
},
|
||||||
|
});
|
||||||
|
this.el.type = this.type
|
||||||
|
if(this.uid) {
|
||||||
|
this.el.uid = this.uid
|
||||||
|
}
|
||||||
|
this.bindRelation()
|
||||||
|
},
|
||||||
|
// #ifdef VUE3
|
||||||
|
beforeUnmount() {
|
||||||
|
this.removeEl()
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
// #ifdef VUE2
|
||||||
|
beforeDestroy() {
|
||||||
|
this.removeEl()
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
methods: {
|
||||||
|
removeEl() {
|
||||||
|
if (this.parent) {
|
||||||
|
this.parent.el.views = this.parent.el.views.filter(
|
||||||
|
(item) => item._uid !== this._uid
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
bindRelation() {
|
||||||
|
if(!this.el._uid) {
|
||||||
|
this.el._uid = this._uid
|
||||||
|
}
|
||||||
|
if(['text','qrcode'].includes(this.type)) {
|
||||||
|
this.el.text = this.$slots && this.$slots.default && this.$slots.default[0].text || `${this.text || ''}`.replace(/\\n/g, '\n')
|
||||||
|
}
|
||||||
|
if(this.type == 'image') {
|
||||||
|
this.el.src = this.src
|
||||||
|
}
|
||||||
|
if (!this.parent) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let views = this.parent.el.views || [];
|
||||||
|
if(views.indexOf(this.el) !== -1) {
|
||||||
|
this.parent.el.views = views.map(v => v._uid == this._uid ? this.el : v)
|
||||||
|
} else {
|
||||||
|
this.parent.el.views = [...views, this.el];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
// this.bindRelation()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,28 @@
|
||||||
|
<template>
|
||||||
|
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {parent, children} from '../common/relation';
|
||||||
|
export default {
|
||||||
|
name: 'lime-painter-image',
|
||||||
|
mixins:[children('painter')],
|
||||||
|
props: {
|
||||||
|
id: String,
|
||||||
|
css: [String, Object],
|
||||||
|
src: String
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
type: 'image',
|
||||||
|
el: {
|
||||||
|
css: {},
|
||||||
|
src: null
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
</style>
|
||||||
|
|
@ -0,0 +1,27 @@
|
||||||
|
<template>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {parent, children} from '../common/relation';
|
||||||
|
export default {
|
||||||
|
name: 'lime-painter-qrcode',
|
||||||
|
mixins:[children('painter')],
|
||||||
|
props: {
|
||||||
|
id: String,
|
||||||
|
css: [String, Object],
|
||||||
|
text: String
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
type: 'qrcode',
|
||||||
|
el: {
|
||||||
|
css: {},
|
||||||
|
text: null
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
</style>
|
||||||
|
|
@ -0,0 +1,33 @@
|
||||||
|
<template>
|
||||||
|
<text style="opacity: 0;height: 0;"><slot/></text>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {parent, children} from '../common/relation';
|
||||||
|
export default {
|
||||||
|
name: 'lime-painter-text',
|
||||||
|
mixins:[children('painter')],
|
||||||
|
props: {
|
||||||
|
type: {
|
||||||
|
type: String,
|
||||||
|
default: 'text'
|
||||||
|
},
|
||||||
|
uid: String,
|
||||||
|
css: [String, Object],
|
||||||
|
text: [String, Number],
|
||||||
|
replace: Object,
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
// type: 'text',
|
||||||
|
el: {
|
||||||
|
css: {},
|
||||||
|
text: null
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
</style>
|
||||||
|
|
@ -0,0 +1,34 @@
|
||||||
|
<template>
|
||||||
|
<view><slot/></view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import {parent, children} from '../common/relation';
|
||||||
|
export default {
|
||||||
|
name: 'lime-painter-view',
|
||||||
|
mixins:[children('painter'), parent('painter')],
|
||||||
|
props: {
|
||||||
|
id: String,
|
||||||
|
type: {
|
||||||
|
type: String,
|
||||||
|
default: 'view'
|
||||||
|
},
|
||||||
|
css: [String, Object],
|
||||||
|
},
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
// type: 'view',
|
||||||
|
el: {
|
||||||
|
css: {},
|
||||||
|
views:[]
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
mounted() {
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
</style>
|
||||||
|
|
@ -0,0 +1,461 @@
|
||||||
|
<template>
|
||||||
|
<view class="lime-painter" ref="limepainter">
|
||||||
|
<view v-if="canvasId && size" :style="styles">
|
||||||
|
<!-- #ifndef APP-NVUE -->
|
||||||
|
<canvas class="lime-painter__canvas" v-if="use2dCanvas" :id="canvasId" type="2d" :style="size"></canvas>
|
||||||
|
<canvas class="lime-painter__canvas" v-else :id="canvasId" :canvas-id="canvasId" :style="size"
|
||||||
|
:width="boardWidth * dpr" :height="boardHeight * dpr" :hidpi="hidpi"></canvas>
|
||||||
|
|
||||||
|
<!-- #endif -->
|
||||||
|
<!-- #ifdef APP-NVUE -->
|
||||||
|
<web-view :style="size" ref="webview"
|
||||||
|
src="/uni_modules/lime-painter/hybrid/html/index.html"
|
||||||
|
class="lime-painter__canvas" @pagefinish="onPageFinish" @error="onError" @onPostMessage="onMessage">
|
||||||
|
</web-view>
|
||||||
|
<!-- #endif -->
|
||||||
|
</view>
|
||||||
|
<slot />
|
||||||
|
</view>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
import { parent } from '../common/relation'
|
||||||
|
import props from './props'
|
||||||
|
import {toPx, base64ToPath, pathToBase64, isBase64, sleep, getImageInfo }from './utils';
|
||||||
|
// #ifndef APP-NVUE
|
||||||
|
import { canIUseCanvas2d, isPC} from './utils';
|
||||||
|
import Painter from './painter';
|
||||||
|
// import Painter from '@painter'
|
||||||
|
const nvue = {}
|
||||||
|
// #endif
|
||||||
|
// #ifdef APP-NVUE
|
||||||
|
import nvue from './nvue'
|
||||||
|
// #endif
|
||||||
|
export default {
|
||||||
|
name: 'lime-painter',
|
||||||
|
mixins: [props, parent('painter'), nvue],
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
use2dCanvas: false,
|
||||||
|
canvasHeight: 150,
|
||||||
|
canvasWidth: null,
|
||||||
|
parentWidth: 0,
|
||||||
|
inited: false,
|
||||||
|
progress: 0,
|
||||||
|
firstRender: 0,
|
||||||
|
done: false,
|
||||||
|
tasks: []
|
||||||
|
};
|
||||||
|
},
|
||||||
|
computed: {
|
||||||
|
styles() {
|
||||||
|
return `${this.size}${this.customStyle||''};` + (this.hidden && 'position: fixed; left: 1500rpx;')
|
||||||
|
},
|
||||||
|
canvasId() {
|
||||||
|
return `l-painter${this._ && this._.uid || this._uid}`
|
||||||
|
},
|
||||||
|
size() {
|
||||||
|
if (this.boardWidth && this.boardHeight) {
|
||||||
|
return `width:${this.boardWidth}px; height: ${this.boardHeight}px;`;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
dpr() {
|
||||||
|
return this.pixelRatio || uni.getSystemInfoSync().pixelRatio;
|
||||||
|
},
|
||||||
|
boardWidth() {
|
||||||
|
const {width = 0} = (this.elements && this.elements.css) || this.elements || this
|
||||||
|
const w = toPx(width||this.width)
|
||||||
|
return w || Math.max(w, toPx(this.canvasWidth));
|
||||||
|
},
|
||||||
|
boardHeight() {
|
||||||
|
const {height = 0} = (this.elements && this.elements.css) || this.elements || this
|
||||||
|
const h = toPx(height||this.height)
|
||||||
|
return h || Math.max(h, toPx(this.canvasHeight));
|
||||||
|
},
|
||||||
|
hasBoard() {
|
||||||
|
return this.board && Object.keys(this.board).length
|
||||||
|
},
|
||||||
|
elements() {
|
||||||
|
return this.hasBoard ? this.board : JSON.parse(JSON.stringify(this.el))
|
||||||
|
}
|
||||||
|
},
|
||||||
|
created() {
|
||||||
|
this.use2dCanvas = this.type === '2d' && canIUseCanvas2d() && !isPC
|
||||||
|
},
|
||||||
|
async mounted() {
|
||||||
|
await sleep(30)
|
||||||
|
await this.getParentWeith()
|
||||||
|
this.$nextTick(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
this.$watch('elements', this.watchRender, {
|
||||||
|
deep: true,
|
||||||
|
immediate: true
|
||||||
|
});
|
||||||
|
}, 30)
|
||||||
|
})
|
||||||
|
},
|
||||||
|
// #ifdef VUE3
|
||||||
|
unmounted() {
|
||||||
|
this.done = false
|
||||||
|
this.inited = false
|
||||||
|
this.firstRender = 0
|
||||||
|
this.progress = 0
|
||||||
|
this.painter = null
|
||||||
|
clearTimeout(this.rendertimer)
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
// #ifdef VUE2
|
||||||
|
destroyed() {
|
||||||
|
this.done = false
|
||||||
|
this.inited = false
|
||||||
|
this.firstRender = 0
|
||||||
|
this.progress = 0
|
||||||
|
this.painter = null
|
||||||
|
clearTimeout(this.rendertimer)
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
methods: {
|
||||||
|
async watchRender(val, old) {
|
||||||
|
if (!val || !val.views || (!this.firstRender ? !val.views.length : !this.firstRender) || !Object.keys(val).length || JSON.stringify(val) == JSON.stringify(old)) return;
|
||||||
|
this.firstRender = 1
|
||||||
|
this.progress = 0
|
||||||
|
this.done = false
|
||||||
|
clearTimeout(this.rendertimer)
|
||||||
|
this.rendertimer = setTimeout(() => {
|
||||||
|
this.render(val);
|
||||||
|
}, this.beforeDelay)
|
||||||
|
},
|
||||||
|
async setFilePath(path, param) {
|
||||||
|
let filePath = path
|
||||||
|
const {pathType = this.pathType} = param || this
|
||||||
|
if (pathType == 'base64' && !isBase64(path)) {
|
||||||
|
filePath = await pathToBase64(path)
|
||||||
|
} else if (pathType == 'url' && isBase64(path)) {
|
||||||
|
filePath = await base64ToPath(path)
|
||||||
|
}
|
||||||
|
if (param && param.isEmit) {
|
||||||
|
this.$emit('success', filePath);
|
||||||
|
}
|
||||||
|
return filePath
|
||||||
|
},
|
||||||
|
async getSize(args) {
|
||||||
|
const {width} = args.css || args
|
||||||
|
const {height} = args.css || args
|
||||||
|
if (!this.size) {
|
||||||
|
if (width || height) {
|
||||||
|
this.canvasWidth = width || this.canvasWidth
|
||||||
|
this.canvasHeight = height || this.canvasHeight
|
||||||
|
await sleep(30);
|
||||||
|
} else {
|
||||||
|
await this.getParentWeith()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
canvasToTempFilePathSync(args) {
|
||||||
|
// this.stopWatch && this.stopWatch()
|
||||||
|
// this.stopWatch = this.$watch('done', (v) => {
|
||||||
|
// if (v) {
|
||||||
|
// this.canvasToTempFilePath(args)
|
||||||
|
// this.stopWatch && this.stopWatch()
|
||||||
|
// }
|
||||||
|
// }, {
|
||||||
|
// immediate: true
|
||||||
|
// })
|
||||||
|
this.tasks.push(args)
|
||||||
|
if(this.done){
|
||||||
|
this.runTask()
|
||||||
|
}
|
||||||
|
},
|
||||||
|
runTask(){
|
||||||
|
while(this.tasks.length){
|
||||||
|
const task = this.tasks.shift()
|
||||||
|
this.canvasToTempFilePath(task)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// #ifndef APP-NVUE
|
||||||
|
getParentWeith() {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
uni.createSelectorQuery()
|
||||||
|
.in(this)
|
||||||
|
.select(`.lime-painter`)
|
||||||
|
.boundingClientRect()
|
||||||
|
.exec(res => {
|
||||||
|
const {width, height} = res[0]||{}
|
||||||
|
this.parentWidth = Math.ceil(width||0)
|
||||||
|
this.canvasWidth = this.parentWidth || 300
|
||||||
|
this.canvasHeight = height || this.canvasHeight||150
|
||||||
|
resolve(res[0])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
},
|
||||||
|
async render(args = {}) {
|
||||||
|
if(!Object.keys(args).length) {
|
||||||
|
return console.error('空对象')
|
||||||
|
}
|
||||||
|
this.progress = 0
|
||||||
|
this.done = false
|
||||||
|
// #ifdef APP-NVUE
|
||||||
|
this.tempFilePath.length = 0
|
||||||
|
// #endif
|
||||||
|
await this.getSize(args)
|
||||||
|
const ctx = await this.getContext();
|
||||||
|
|
||||||
|
let {
|
||||||
|
use2dCanvas,
|
||||||
|
boardWidth,
|
||||||
|
boardHeight,
|
||||||
|
canvas,
|
||||||
|
afterDelay
|
||||||
|
} = this;
|
||||||
|
if (use2dCanvas && !canvas) {
|
||||||
|
return Promise.reject(new Error('canvas 没创建'));
|
||||||
|
}
|
||||||
|
this.boundary = {
|
||||||
|
top: 0,
|
||||||
|
left: 0,
|
||||||
|
width: boardWidth,
|
||||||
|
height: boardHeight
|
||||||
|
};
|
||||||
|
this.painter = null
|
||||||
|
if (!this.painter) {
|
||||||
|
const {width} = args.css || args
|
||||||
|
const {height} = args.css || args
|
||||||
|
if(!width && this.parentWidth) {
|
||||||
|
Object.assign(args, {width: this.parentWidth})
|
||||||
|
}
|
||||||
|
const param = {
|
||||||
|
context: ctx,
|
||||||
|
canvas,
|
||||||
|
width: boardWidth,
|
||||||
|
height: boardHeight,
|
||||||
|
pixelRatio: this.dpr,
|
||||||
|
useCORS: this.useCORS,
|
||||||
|
createImage: getImageInfo.bind(this),
|
||||||
|
performance: this.performance,
|
||||||
|
listen: {
|
||||||
|
onProgress: (v) => {
|
||||||
|
this.progress = v
|
||||||
|
this.$emit('progress', v)
|
||||||
|
},
|
||||||
|
onEffectFail: (err) => {
|
||||||
|
this.$emit('faill', err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
this.painter = new Painter(param)
|
||||||
|
}
|
||||||
|
try{
|
||||||
|
// vue3 赋值给data会引起图片无法绘制
|
||||||
|
const { width, height } = await this.painter.source(JSON.parse(JSON.stringify(args)))
|
||||||
|
this.boundary.height = this.canvasHeight = height
|
||||||
|
this.boundary.width = this.canvasWidth = width
|
||||||
|
await sleep(this.sleep);
|
||||||
|
await this.painter.render()
|
||||||
|
await new Promise(resolve => this.$nextTick(resolve));
|
||||||
|
if (!use2dCanvas) {
|
||||||
|
await this.canvasDraw();
|
||||||
|
}
|
||||||
|
if (afterDelay && use2dCanvas) {
|
||||||
|
await sleep(afterDelay);
|
||||||
|
}
|
||||||
|
this.$emit('done');
|
||||||
|
this.done = true
|
||||||
|
if (this.isCanvasToTempFilePath) {
|
||||||
|
this.canvasToTempFilePath()
|
||||||
|
.then(res => {
|
||||||
|
this.$emit('success', res.tempFilePath)
|
||||||
|
})
|
||||||
|
.catch(err => {
|
||||||
|
this.$emit('fail', new Error(JSON.stringify(err)));
|
||||||
|
});
|
||||||
|
}
|
||||||
|
this.runTask()
|
||||||
|
return Promise.resolve({
|
||||||
|
ctx,
|
||||||
|
draw: this.painter,
|
||||||
|
node: this.node
|
||||||
|
});
|
||||||
|
}catch(e){
|
||||||
|
//TODO handle the exception
|
||||||
|
}
|
||||||
|
|
||||||
|
},
|
||||||
|
canvasDraw(flag = false) {
|
||||||
|
return new Promise((resolve, reject) => this.ctx.draw(flag, () => setTimeout(() => resolve(), this
|
||||||
|
.afterDelay)));
|
||||||
|
},
|
||||||
|
async getContext() {
|
||||||
|
if (!this.canvasWidth) {
|
||||||
|
this.$emit('fail', 'painter no size')
|
||||||
|
console.error('[lime-painter]: 给画板或父级设置尺寸')
|
||||||
|
return Promise.reject();
|
||||||
|
}
|
||||||
|
if (this.ctx && this.inited) {
|
||||||
|
return Promise.resolve(this.ctx);
|
||||||
|
}
|
||||||
|
const { type, use2dCanvas, dpr, boardWidth, boardHeight } = this;
|
||||||
|
const _getContext = () => {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
uni.createSelectorQuery()
|
||||||
|
.in(this)
|
||||||
|
.select(`#${this.canvasId}`)
|
||||||
|
.boundingClientRect()
|
||||||
|
.exec(res => {
|
||||||
|
if (res) {
|
||||||
|
const ctx = uni.createCanvasContext(this.canvasId, this);
|
||||||
|
if (!this.inited) {
|
||||||
|
this.inited = true;
|
||||||
|
this.use2dCanvas = false;
|
||||||
|
this.canvas = res;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 钉钉小程序框架不支持 measureText 方法,用此方法 mock
|
||||||
|
if (!ctx.measureText) {
|
||||||
|
function strLen(str) {
|
||||||
|
let len = 0;
|
||||||
|
for (let i = 0; i < str.length; i++) {
|
||||||
|
if (str.charCodeAt(i) > 0 && str.charCodeAt(i) < 128) {
|
||||||
|
len++;
|
||||||
|
} else {
|
||||||
|
len += 2;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
ctx.measureText = text => {
|
||||||
|
let fontSize = ctx.state && ctx.state.fontSize || 12;
|
||||||
|
const font = ctx.__font
|
||||||
|
if (font && fontSize == 12) {
|
||||||
|
fontSize = parseInt(font.split(' ')[3], 10);
|
||||||
|
}
|
||||||
|
fontSize /= 2;
|
||||||
|
return {
|
||||||
|
width: strLen(text) * fontSize
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// #ifdef MP-ALIPAY
|
||||||
|
ctx.scale(dpr, dpr);
|
||||||
|
// #endif
|
||||||
|
this.ctx = ctx
|
||||||
|
resolve(this.ctx);
|
||||||
|
} else {
|
||||||
|
console.error('[lime-painter] no node')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
if (!use2dCanvas) {
|
||||||
|
return _getContext();
|
||||||
|
}
|
||||||
|
return new Promise(resolve => {
|
||||||
|
uni.createSelectorQuery()
|
||||||
|
.in(this)
|
||||||
|
.select(`#${this.canvasId}`)
|
||||||
|
.node()
|
||||||
|
.exec(res => {
|
||||||
|
let {node: canvas} = res && res[0]||{};
|
||||||
|
if(canvas) {
|
||||||
|
const ctx = canvas.getContext(type);
|
||||||
|
if (!this.inited) {
|
||||||
|
this.inited = true;
|
||||||
|
this.use2dCanvas = true;
|
||||||
|
this.canvas = canvas;
|
||||||
|
}
|
||||||
|
this.ctx = ctx
|
||||||
|
resolve(this.ctx);
|
||||||
|
} else {
|
||||||
|
console.error('[lime-painter]: no size')
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
},
|
||||||
|
canvasToTempFilePath(args = {}) {
|
||||||
|
return new Promise(async (resolve, reject) => {
|
||||||
|
const { use2dCanvas, canvasId, dpr, fileType, quality } = this;
|
||||||
|
const success = async (res) => {
|
||||||
|
try {
|
||||||
|
const tempFilePath = await this.setFilePath(res.tempFilePath || res, args)
|
||||||
|
const result = Object.assign(res, {tempFilePath})
|
||||||
|
args.success && args.success(result)
|
||||||
|
resolve(result)
|
||||||
|
} catch (e) {
|
||||||
|
this.$emit('fail', e)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let { top: y = 0, left: x = 0, width, height } = this.boundary || this;
|
||||||
|
// let destWidth = width * dpr;
|
||||||
|
// let destHeight = height * dpr;
|
||||||
|
// #ifdef MP-ALIPAY
|
||||||
|
// width = destWidth;
|
||||||
|
// height = destHeight;
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
const copyArgs = Object.assign({
|
||||||
|
// x,
|
||||||
|
// y,
|
||||||
|
// width,
|
||||||
|
// height,
|
||||||
|
// destWidth,
|
||||||
|
// destHeight,
|
||||||
|
canvasId,
|
||||||
|
id: canvasId,
|
||||||
|
fileType,
|
||||||
|
quality,
|
||||||
|
}, args, {success});
|
||||||
|
// if(this.isPC || use2dCanvas) {
|
||||||
|
// copyArgs.canvas = this.canvas
|
||||||
|
// }
|
||||||
|
if (use2dCanvas) {
|
||||||
|
copyArgs.canvas = this.canvas
|
||||||
|
try{
|
||||||
|
// #ifndef MP-ALIPAY
|
||||||
|
const oFilePath = this.canvas.toDataURL(`image/${args.fileType||fileType}`.replace(/pg/, 'peg'), args.quality||quality)
|
||||||
|
if(/data:,/.test(oFilePath)) {
|
||||||
|
uni.canvasToTempFilePath(copyArgs, this);
|
||||||
|
} else {
|
||||||
|
const tempFilePath = await this.setFilePath(oFilePath, args)
|
||||||
|
args.success && args.success({tempFilePath})
|
||||||
|
resolve({tempFilePath})
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-ALIPAY
|
||||||
|
this.canvas.toTempFilePath(copyArgs)
|
||||||
|
// #endif
|
||||||
|
}catch(e){
|
||||||
|
args.fail && args.fail(e)
|
||||||
|
reject(e)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// #ifdef MP-ALIPAY
|
||||||
|
if(this.ctx.toTempFilePath) {
|
||||||
|
// 钉钉
|
||||||
|
const ctx = uni.createCanvasContext(canvasId);
|
||||||
|
ctx.toTempFilePath(copyArgs);
|
||||||
|
} else {
|
||||||
|
my.canvasToTempFilePath(copyArgs);
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
// #ifndef MP-ALIPAY
|
||||||
|
uni.canvasToTempFilePath(copyArgs, this);
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.lime-painter,
|
||||||
|
.lime-painter__canvas {
|
||||||
|
// #ifndef APP-NVUE
|
||||||
|
width: 100%;
|
||||||
|
// #endif
|
||||||
|
// #ifdef APP-NVUE
|
||||||
|
flex: 1;
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
@ -0,0 +1,214 @@
|
||||||
|
// #ifdef APP-NVUE
|
||||||
|
import {
|
||||||
|
sleep,
|
||||||
|
getImageInfo,
|
||||||
|
isBase64,
|
||||||
|
networkReg
|
||||||
|
} from './utils';
|
||||||
|
const dom = weex.requireModule('dom')
|
||||||
|
import {
|
||||||
|
version
|
||||||
|
} from '../../package.json'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
tempFilePath: [],
|
||||||
|
isInitFile: false,
|
||||||
|
osName: uni.getSystemInfoSync().osName
|
||||||
|
}
|
||||||
|
},
|
||||||
|
methods: {
|
||||||
|
getParentWeith() {
|
||||||
|
return new Promise(resolve => {
|
||||||
|
dom.getComponentRect(this.$refs.limepainter, (res) => {
|
||||||
|
this.parentWidth = Math.ceil(res.size.width)
|
||||||
|
this.canvasWidth = this.canvasWidth || this.parentWidth || 300
|
||||||
|
this.canvasHeight = res.size.height || this.canvasHeight || 150
|
||||||
|
resolve(res.size)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
},
|
||||||
|
onPageFinish() {
|
||||||
|
this.webview = this.$refs.webview
|
||||||
|
this.webview.evalJS(`init(${this.dpr})`)
|
||||||
|
},
|
||||||
|
onMessage(e) {
|
||||||
|
const res = e.detail.data[0] || null;
|
||||||
|
if (res.event) {
|
||||||
|
if (res.event == 'inited') {
|
||||||
|
this.inited = true
|
||||||
|
}
|
||||||
|
if (res.event == 'fail') {
|
||||||
|
this.$emit('fail', res)
|
||||||
|
}
|
||||||
|
if (res.event == 'layoutChange') {
|
||||||
|
const data = typeof res.data == 'string' ? JSON.parse(res.data) : res.data
|
||||||
|
this.canvasWidth = Math.ceil(data.width);
|
||||||
|
this.canvasHeight = Math.ceil(data.height);
|
||||||
|
}
|
||||||
|
if (res.event == 'progressChange') {
|
||||||
|
this.progress = res.data * 1
|
||||||
|
}
|
||||||
|
if (res.event == 'file') {
|
||||||
|
this.tempFilePath.push(res.data)
|
||||||
|
if (this.tempFilePath.length > 7) {
|
||||||
|
this.tempFilePath.shift()
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if (res.event == 'success') {
|
||||||
|
if (res.data) {
|
||||||
|
this.tempFilePath.push(res.data)
|
||||||
|
if (this.tempFilePath.length > 8) {
|
||||||
|
this.tempFilePath.shift()
|
||||||
|
}
|
||||||
|
if (this.isCanvasToTempFilePath) {
|
||||||
|
this.setFilePath(this.tempFilePath.join(''), {
|
||||||
|
isEmit: true
|
||||||
|
})
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
this.$emit('fail', 'canvas no data')
|
||||||
|
}
|
||||||
|
return
|
||||||
|
}
|
||||||
|
this.$emit(res.event, JSON.parse(res.data));
|
||||||
|
} else if (res.file) {
|
||||||
|
this.file = res.data;
|
||||||
|
} else {
|
||||||
|
console.info(res[0])
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getWebViewInited() {
|
||||||
|
if (this.inited) return Promise.resolve(this.inited);
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
this.$watch(
|
||||||
|
'inited',
|
||||||
|
async val => {
|
||||||
|
if (val) {
|
||||||
|
resolve(val)
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
immediate: true
|
||||||
|
}
|
||||||
|
);
|
||||||
|
})
|
||||||
|
},
|
||||||
|
getTempFilePath() {
|
||||||
|
if (this.tempFilePath.length == 8) return Promise.resolve(this.tempFilePath)
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
this.$watch(
|
||||||
|
'tempFilePath',
|
||||||
|
async val => {
|
||||||
|
if (val.length == 8) {
|
||||||
|
resolve(val.join(''))
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
deep: true
|
||||||
|
}
|
||||||
|
);
|
||||||
|
})
|
||||||
|
},
|
||||||
|
getWebViewDone() {
|
||||||
|
if (this.progress == 1) return Promise.resolve(this.progress);
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
this.$watch(
|
||||||
|
'progress',
|
||||||
|
async val => {
|
||||||
|
if (val == 1) {
|
||||||
|
this.$emit('done')
|
||||||
|
this.done = true
|
||||||
|
this.runTask()
|
||||||
|
resolve(val)
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
immediate: true
|
||||||
|
}
|
||||||
|
);
|
||||||
|
})
|
||||||
|
},
|
||||||
|
async render(args) {
|
||||||
|
try {
|
||||||
|
await this.getSize(args)
|
||||||
|
const {
|
||||||
|
width
|
||||||
|
} = args.css || args
|
||||||
|
if (!width && this.parentWidth) {
|
||||||
|
Object.assign(args, {
|
||||||
|
width: this.parentWidth
|
||||||
|
})
|
||||||
|
}
|
||||||
|
const newNode = await this.calcImage(args);
|
||||||
|
await this.getWebViewInited()
|
||||||
|
this.webview.evalJS(`source(${JSON.stringify(newNode)})`)
|
||||||
|
await this.getWebViewDone()
|
||||||
|
await sleep(this.afterDelay)
|
||||||
|
if (this.isCanvasToTempFilePath) {
|
||||||
|
const params = {
|
||||||
|
fileType: this.fileType,
|
||||||
|
quality: this.quality
|
||||||
|
}
|
||||||
|
this.webview.evalJS(`save(${JSON.stringify(params)})`)
|
||||||
|
}
|
||||||
|
return Promise.resolve()
|
||||||
|
} catch (e) {
|
||||||
|
this.$emit('fail', e)
|
||||||
|
}
|
||||||
|
},
|
||||||
|
async calcImage(args) {
|
||||||
|
let node = JSON.parse(JSON.stringify(args))
|
||||||
|
const urlReg = /url\((.+)\)/
|
||||||
|
const {
|
||||||
|
backgroundImage
|
||||||
|
} = node.css || {}
|
||||||
|
const isBG = backgroundImage && urlReg.exec(backgroundImage)[1]
|
||||||
|
const url = node.url || node.src || isBG
|
||||||
|
if (['text', 'qrcode'].includes(node.type)) {
|
||||||
|
return node
|
||||||
|
}
|
||||||
|
if ((node.type === "image" || isBG) && url && !isBase64(url) && (this.osName == 'ios' || !networkReg
|
||||||
|
.test(url))) {
|
||||||
|
let {
|
||||||
|
path
|
||||||
|
} = await getImageInfo(url, true)
|
||||||
|
if (isBG) {
|
||||||
|
node.css.backgroundImage = `url(${path})`
|
||||||
|
} else {
|
||||||
|
node.src = path
|
||||||
|
}
|
||||||
|
} else if (node.views && node.views.length) {
|
||||||
|
for (let i = 0; i < node.views.length; i++) {
|
||||||
|
node.views[i] = await this.calcImage(node.views[i])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return node
|
||||||
|
},
|
||||||
|
async canvasToTempFilePath(args = {}) {
|
||||||
|
if (!this.inited) {
|
||||||
|
return this.$emit('fail', 'no init')
|
||||||
|
}
|
||||||
|
this.tempFilePath = []
|
||||||
|
if (args.fileType == 'jpg') {
|
||||||
|
args.fileType = 'jpeg'
|
||||||
|
}
|
||||||
|
|
||||||
|
this.webview.evalJS(`save(${JSON.stringify(args)})`)
|
||||||
|
try {
|
||||||
|
let tempFilePath = await this.getTempFilePath()
|
||||||
|
|
||||||
|
tempFilePath = await this.setFilePath(tempFilePath, args)
|
||||||
|
args.success({
|
||||||
|
errMsg: "canvasToTempFilePath:ok",
|
||||||
|
tempFilePath
|
||||||
|
})
|
||||||
|
} catch (e) {
|
||||||
|
console.log('e', e)
|
||||||
|
args.fail({
|
||||||
|
error: e
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
|
@ -0,0 +1,56 @@
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
board: Object,
|
||||||
|
pathType: String, // 'base64'、'url'
|
||||||
|
fileType: {
|
||||||
|
type: String,
|
||||||
|
default: 'png'
|
||||||
|
},
|
||||||
|
hidden: Boolean,
|
||||||
|
quality: {
|
||||||
|
type: Number,
|
||||||
|
default: 1
|
||||||
|
},
|
||||||
|
css: [String, Object],
|
||||||
|
// styles: [String, Object],
|
||||||
|
width: [Number, String],
|
||||||
|
height: [Number, String],
|
||||||
|
pixelRatio: Number,
|
||||||
|
customStyle: String,
|
||||||
|
isCanvasToTempFilePath: Boolean,
|
||||||
|
// useCanvasToTempFilePath: Boolean,
|
||||||
|
sleep: {
|
||||||
|
type: Number,
|
||||||
|
default: 1000 / 30
|
||||||
|
},
|
||||||
|
beforeDelay: {
|
||||||
|
type: Number,
|
||||||
|
default: 100
|
||||||
|
},
|
||||||
|
afterDelay: {
|
||||||
|
type: Number,
|
||||||
|
default: 100
|
||||||
|
},
|
||||||
|
performance: Boolean,
|
||||||
|
// #ifdef MP-WEIXIN || MP-TOUTIAO || MP-ALIPAY
|
||||||
|
type: {
|
||||||
|
type: String,
|
||||||
|
default: '2d'
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
// #ifdef APP-NVUE
|
||||||
|
hybrid: Boolean,
|
||||||
|
timeout: {
|
||||||
|
type: Number,
|
||||||
|
default: 2000
|
||||||
|
},
|
||||||
|
// #endif
|
||||||
|
// #ifdef H5 || APP-PLUS
|
||||||
|
useCORS: Boolean,
|
||||||
|
hidpi: {
|
||||||
|
type: Boolean,
|
||||||
|
default: true
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,368 @@
|
||||||
|
export const networkReg = /^(http|\/\/)/;
|
||||||
|
export const isBase64 = (path) => /^data:image\/(\w+);base64/.test(path);
|
||||||
|
export function sleep(delay) {
|
||||||
|
return new Promise(resolve => setTimeout(resolve, delay))
|
||||||
|
}
|
||||||
|
let {platform, SDKVersion} = uni.getSystemInfoSync()
|
||||||
|
export const isPC = /windows|mac/.test(platform)
|
||||||
|
// 缓存图片
|
||||||
|
let cache = {}
|
||||||
|
export function isNumber(value) {
|
||||||
|
return /^-?\d+(\.\d+)?$/.test(value);
|
||||||
|
}
|
||||||
|
export function toPx(value, baseSize, isDecimal = false) {
|
||||||
|
// 如果是数字
|
||||||
|
if (typeof value === 'number') {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
// 如果是字符串数字
|
||||||
|
if (isNumber(value)) {
|
||||||
|
return value * 1
|
||||||
|
}
|
||||||
|
// 如果有单位
|
||||||
|
if (typeof value === 'string') {
|
||||||
|
const reg = /^-?([0-9]+)?([.]{1}[0-9]+){0,1}(em|rpx|px|%)$/g
|
||||||
|
const results = reg.exec(value);
|
||||||
|
if (!value || !results) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
const unit = results[3];
|
||||||
|
value = parseFloat(value);
|
||||||
|
let res = 0;
|
||||||
|
if (unit === 'rpx') {
|
||||||
|
res = uni.upx2px(value);
|
||||||
|
} else if (unit === 'px') {
|
||||||
|
res = value * 1;
|
||||||
|
} else if (unit === '%') {
|
||||||
|
res = value * toPx(baseSize) / 100;
|
||||||
|
} else if (unit === 'em') {
|
||||||
|
res = value * toPx(baseSize || 14);
|
||||||
|
}
|
||||||
|
return isDecimal ? res.toFixed(2) * 1 : Math.round(res);
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
// 计算版本
|
||||||
|
export function compareVersion(v1, v2) {
|
||||||
|
v1 = v1.split('.')
|
||||||
|
v2 = v2.split('.')
|
||||||
|
const len = Math.max(v1.length, v2.length)
|
||||||
|
while (v1.length < len) {
|
||||||
|
v1.push('0')
|
||||||
|
}
|
||||||
|
while (v2.length < len) {
|
||||||
|
v2.push('0')
|
||||||
|
}
|
||||||
|
for (let i = 0; i < len; i++) {
|
||||||
|
const num1 = parseInt(v1[i], 10)
|
||||||
|
const num2 = parseInt(v2[i], 10)
|
||||||
|
|
||||||
|
if (num1 > num2) {
|
||||||
|
return 1
|
||||||
|
} else if (num1 < num2) {
|
||||||
|
return -1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
|
||||||
|
function gte(version) {
|
||||||
|
// #ifdef MP-ALIPAY
|
||||||
|
SDKVersion = my.SDKVersion
|
||||||
|
// #endif
|
||||||
|
return compareVersion(SDKVersion, version) >= 0;
|
||||||
|
}
|
||||||
|
export function canIUseCanvas2d() {
|
||||||
|
// #ifdef MP-WEIXIN
|
||||||
|
return gte('2.9.2');
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-ALIPAY
|
||||||
|
return gte('2.7.15');
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-TOUTIAO
|
||||||
|
return gte('1.78.0');
|
||||||
|
// #endif
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
// #ifdef MP
|
||||||
|
export const prefix = () => {
|
||||||
|
// #ifdef MP-TOUTIAO
|
||||||
|
return tt
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-WEIXIN
|
||||||
|
return wx
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-BAIDU
|
||||||
|
return swan
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-ALIPAY
|
||||||
|
return my
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-QQ
|
||||||
|
return qq
|
||||||
|
// #endif
|
||||||
|
// #ifdef MP-360
|
||||||
|
return qh
|
||||||
|
// #endif
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* base64转路径
|
||||||
|
* @param {Object} base64
|
||||||
|
*/
|
||||||
|
export function base64ToPath(base64) {
|
||||||
|
const [, format] = /^data:image\/(\w+);base64,/.exec(base64) || [];
|
||||||
|
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
// #ifdef MP
|
||||||
|
const fs = uni.getFileSystemManager()
|
||||||
|
//自定义文件名
|
||||||
|
if (!format) {
|
||||||
|
reject(new Error('ERROR_BASE64SRC_PARSE'))
|
||||||
|
}
|
||||||
|
const time = new Date().getTime();
|
||||||
|
let pre = prefix()
|
||||||
|
// #ifdef MP-TOUTIAO
|
||||||
|
const filePath = `${pre.getEnvInfoSync().common.USER_DATA_PATH}/${time}.${format}`
|
||||||
|
// #endif
|
||||||
|
// #ifndef MP-TOUTIAO
|
||||||
|
const filePath = `${pre.env.USER_DATA_PATH}/${time}.${format}`
|
||||||
|
// #endif
|
||||||
|
fs.writeFile({
|
||||||
|
filePath,
|
||||||
|
data: base64.split(',')[1],
|
||||||
|
encoding: 'base64',
|
||||||
|
success() {
|
||||||
|
resolve(filePath)
|
||||||
|
},
|
||||||
|
fail(err) {
|
||||||
|
console.error(err)
|
||||||
|
reject(err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifdef H5
|
||||||
|
// mime类型
|
||||||
|
let mimeString = base64.split(',')[0].split(':')[1].split(';')[0];
|
||||||
|
//base64 解码
|
||||||
|
let byteString = atob(base64.split(',')[1]);
|
||||||
|
//创建缓冲数组
|
||||||
|
let arrayBuffer = new ArrayBuffer(byteString.length);
|
||||||
|
//创建视图
|
||||||
|
let intArray = new Uint8Array(arrayBuffer);
|
||||||
|
for (let i = 0; i < byteString.length; i++) {
|
||||||
|
intArray[i] = byteString.charCodeAt(i);
|
||||||
|
}
|
||||||
|
resolve(URL.createObjectURL(new Blob([intArray], {
|
||||||
|
type: mimeString
|
||||||
|
})))
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
const bitmap = new plus.nativeObj.Bitmap('bitmap' + Date.now())
|
||||||
|
bitmap.loadBase64Data(base64, () => {
|
||||||
|
if (!format) {
|
||||||
|
reject(new Error('ERROR_BASE64SRC_PARSE'))
|
||||||
|
}
|
||||||
|
const time = new Date().getTime();
|
||||||
|
const filePath = `_doc/uniapp_temp/${time}.${format}`
|
||||||
|
bitmap.save(filePath, {},
|
||||||
|
() => {
|
||||||
|
bitmap.clear()
|
||||||
|
resolve(filePath)
|
||||||
|
},
|
||||||
|
(error) => {
|
||||||
|
bitmap.clear()
|
||||||
|
reject(error)
|
||||||
|
})
|
||||||
|
}, (error) => {
|
||||||
|
bitmap.clear()
|
||||||
|
reject(error)
|
||||||
|
})
|
||||||
|
// #endif
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 路径转base64
|
||||||
|
* @param {Object} string
|
||||||
|
*/
|
||||||
|
export function pathToBase64(path) {
|
||||||
|
if (/^data:/.test(path)) return path
|
||||||
|
return new Promise((resolve, reject) => {
|
||||||
|
// #ifdef H5
|
||||||
|
let image = new Image();
|
||||||
|
image.setAttribute("crossOrigin", 'Anonymous');
|
||||||
|
image.onload = function() {
|
||||||
|
let canvas = document.createElement('canvas');
|
||||||
|
canvas.width = this.naturalWidth;
|
||||||
|
canvas.height = this.naturalHeight;
|
||||||
|
canvas.getContext('2d').drawImage(image, 0, 0);
|
||||||
|
let result = canvas.toDataURL('image/png')
|
||||||
|
resolve(result);
|
||||||
|
canvas.height = canvas.width = 0
|
||||||
|
}
|
||||||
|
image.src = path + '?v=' + Math.random()
|
||||||
|
image.onerror = (error) => {
|
||||||
|
reject(error);
|
||||||
|
};
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifdef MP
|
||||||
|
if (uni.canIUse('getFileSystemManager')) {
|
||||||
|
uni.getFileSystemManager().readFile({
|
||||||
|
filePath: path,
|
||||||
|
encoding: 'base64',
|
||||||
|
success: (res) => {
|
||||||
|
resolve('data:image/png;base64,' + res.data)
|
||||||
|
},
|
||||||
|
fail: (error) => {
|
||||||
|
console.error({error, path})
|
||||||
|
reject(error)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
plus.io.resolveLocalFileSystemURL(getLocalFilePath(path), (entry) => {
|
||||||
|
entry.file((file) => {
|
||||||
|
const fileReader = new plus.io.FileReader()
|
||||||
|
fileReader.onload = (data) => {
|
||||||
|
resolve(data.target.result)
|
||||||
|
}
|
||||||
|
fileReader.onerror = (error) => {
|
||||||
|
reject(error)
|
||||||
|
}
|
||||||
|
fileReader.readAsDataURL(file)
|
||||||
|
}, reject)
|
||||||
|
}, reject)
|
||||||
|
// #endif
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
export function getImageInfo(path, useCORS) {
|
||||||
|
const isCanvas2D = this && this.canvas && this.canvas.createImage
|
||||||
|
return new Promise(async (resolve, reject) => {
|
||||||
|
// let time = +new Date()
|
||||||
|
let src = path.replace(/^@\//,'/')
|
||||||
|
if (cache[path] && cache[path].errMsg) {
|
||||||
|
resolve(cache[path])
|
||||||
|
} else {
|
||||||
|
try {
|
||||||
|
// #ifdef MP || APP-PLUS
|
||||||
|
if (isBase64(path) && (isCanvas2D ? isPC : true)) {
|
||||||
|
src = await base64ToPath(path)
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
// #ifdef H5
|
||||||
|
if(useCORS) {
|
||||||
|
src = await pathToBase64(path)
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
} catch (error) {
|
||||||
|
reject({
|
||||||
|
...error,
|
||||||
|
src
|
||||||
|
})
|
||||||
|
}
|
||||||
|
// #ifndef APP-NVUE
|
||||||
|
if(isCanvas2D && !isPC) {
|
||||||
|
const img = this.canvas.createImage()
|
||||||
|
img.onload = function() {
|
||||||
|
const image = {
|
||||||
|
path: img,
|
||||||
|
width: img.width,
|
||||||
|
height: img.height
|
||||||
|
}
|
||||||
|
cache[path] = image
|
||||||
|
resolve(cache[path])
|
||||||
|
}
|
||||||
|
img.onerror = function(err) {
|
||||||
|
reject({err,path})
|
||||||
|
}
|
||||||
|
img.src = src
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
uni.getImageInfo({
|
||||||
|
src,
|
||||||
|
success: (image) => {
|
||||||
|
const localReg = /^\.|^\/(?=[^\/])/;
|
||||||
|
// #ifdef MP-WEIXIN || MP-BAIDU || MP-QQ || MP-TOUTIAO
|
||||||
|
image.path = localReg.test(src) ? `/${image.path}` : image.path;
|
||||||
|
// #endif
|
||||||
|
if(isCanvas2D) {
|
||||||
|
const img = this.canvas.createImage()
|
||||||
|
img.onload = function() {
|
||||||
|
image.path = img
|
||||||
|
cache[path] = image
|
||||||
|
resolve(cache[path])
|
||||||
|
}
|
||||||
|
img.onerror = function(err) {
|
||||||
|
reject({err,path})
|
||||||
|
}
|
||||||
|
img.src = src
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
// console.log('getImageInfo', +new Date() - time)
|
||||||
|
// ios 比较严格 可能需要设置跨域
|
||||||
|
if(uni.getSystemInfoSync().osName == 'ios' && useCORS) {
|
||||||
|
pathToBase64(image.path).then(base64 => {
|
||||||
|
image.path = base64
|
||||||
|
cache[path] = image
|
||||||
|
resolve(cache[path])
|
||||||
|
}).catch(err => {
|
||||||
|
console.error({err, path})
|
||||||
|
reject({err,path})
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
cache[path] = image
|
||||||
|
resolve(cache[path])
|
||||||
|
},
|
||||||
|
fail(err) {
|
||||||
|
console.error({err, path})
|
||||||
|
reject({err,path})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// #ifdef APP-PLUS
|
||||||
|
const getLocalFilePath = (path) => {
|
||||||
|
if (path.indexOf('_www') === 0 || path.indexOf('_doc') === 0 || path.indexOf('_documents') === 0 || path
|
||||||
|
.indexOf('_downloads') === 0) {
|
||||||
|
return path
|
||||||
|
}
|
||||||
|
if (path.indexOf('file://') === 0) {
|
||||||
|
return path
|
||||||
|
}
|
||||||
|
if (path.indexOf('/storage/emulated/0/') === 0) {
|
||||||
|
return path
|
||||||
|
}
|
||||||
|
if (path.indexOf('/') === 0) {
|
||||||
|
const localFilePath = plus.io.convertAbsoluteFileSystem(path)
|
||||||
|
if (localFilePath !== path) {
|
||||||
|
return localFilePath
|
||||||
|
} else {
|
||||||
|
path = path.substr(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return '_www/' + path
|
||||||
|
}
|
||||||
|
// #endif
|
||||||
|
|
||||||
|
|
||||||
|
|
@ -0,0 +1,119 @@
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="zh">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="ie=edge">
|
||||||
|
<title></title>
|
||||||
|
<style type="text/css">
|
||||||
|
html,
|
||||||
|
body,
|
||||||
|
canvas {
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
overflow-y: hidden;
|
||||||
|
background-color: transparent;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<canvas id="lime-painter"></canvas>
|
||||||
|
<script type="text/javascript" src="./uni.webview.1.5.3.js"></script>
|
||||||
|
<script type="text/javascript" src="./painter.js"></script>
|
||||||
|
<script>
|
||||||
|
var cache = [];
|
||||||
|
var painter = null;
|
||||||
|
var canvas = null;
|
||||||
|
var context = null;
|
||||||
|
var timer = null;
|
||||||
|
var pixelRatio = 1;
|
||||||
|
console.log = function (...args) {
|
||||||
|
postMessage(args);
|
||||||
|
};
|
||||||
|
// function stringify(key, value) {
|
||||||
|
// if (typeof value === 'object' && value !== null) {
|
||||||
|
// if (cache.indexOf(value) !== -1) {
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
// cache.push(value);
|
||||||
|
// }
|
||||||
|
// return value;
|
||||||
|
// };
|
||||||
|
|
||||||
|
function emit(event, data) {
|
||||||
|
postMessage({
|
||||||
|
event,
|
||||||
|
data: (typeof data !== 'object' && data !== null ? data : JSON.stringify(data))
|
||||||
|
});
|
||||||
|
cache = [];
|
||||||
|
};
|
||||||
|
function postMessage(data) {
|
||||||
|
uni.postMessage({
|
||||||
|
data
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
function init(dpr) {
|
||||||
|
canvas = document.querySelector('#lime-painter');
|
||||||
|
context = canvas.getContext('2d');
|
||||||
|
pixelRatio = dpr || window.devicePixelRatio;
|
||||||
|
painter = new Painter({
|
||||||
|
id: 'lime-painter',
|
||||||
|
context,
|
||||||
|
canvas,
|
||||||
|
pixelRatio,
|
||||||
|
width: canvas.offsetWidth,
|
||||||
|
height: canvas.offsetHeight,
|
||||||
|
listen: {
|
||||||
|
onProgress(v) {
|
||||||
|
emit('progressChange', v);
|
||||||
|
},
|
||||||
|
onEffectFail(err) {
|
||||||
|
//console.error(err)
|
||||||
|
emit('fail', err);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
emit('inited', true);
|
||||||
|
};
|
||||||
|
function save(args) {
|
||||||
|
delete args.success;
|
||||||
|
delete args.fail;
|
||||||
|
clearTimeout(timer);
|
||||||
|
timer = setTimeout(() => {
|
||||||
|
const path = painter.save(args);
|
||||||
|
if (typeof path == 'string') {
|
||||||
|
const index = Math.ceil(path.length / 8);
|
||||||
|
for (var i = 0; i < 8; i++) {
|
||||||
|
if (i == 7) {
|
||||||
|
emit('success', path.substr(i * index, index));
|
||||||
|
} else {
|
||||||
|
emit('file', path.substr(i * index, index));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
// console.log('canvas no data')
|
||||||
|
emit('fail', 'canvas no data');
|
||||||
|
};
|
||||||
|
}, 30);
|
||||||
|
};
|
||||||
|
async function source(args) {
|
||||||
|
let size = await painter.source(args);
|
||||||
|
emit('layoutChange', size);
|
||||||
|
if(!canvas.height) {
|
||||||
|
console.log('canvas no size')
|
||||||
|
emit('fail', 'canvas no size');
|
||||||
|
}
|
||||||
|
painter.render().catch(err => {
|
||||||
|
// console.error(err)
|
||||||
|
emit('fail', err);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
|
|
@ -0,0 +1,93 @@
|
||||||
|
{
|
||||||
|
"id": "lime-painter",
|
||||||
|
"displayName": "海报画板",
|
||||||
|
"version": "1.9.6.6",
|
||||||
|
"description": "一款canvas海报组件,更优雅的海报生成方案,有限的支持富文本",
|
||||||
|
"keywords": [
|
||||||
|
"海报",
|
||||||
|
"富文本",
|
||||||
|
"生成海报",
|
||||||
|
"生成二维码",
|
||||||
|
"JSON"
|
||||||
|
],
|
||||||
|
"repository": "https://gitee.com/liangei/lime-painter",
|
||||||
|
"engines": {
|
||||||
|
"HBuilderX": "^3.4.14"
|
||||||
|
},
|
||||||
|
"dcloudext": {
|
||||||
|
"sale": {
|
||||||
|
"regular": {
|
||||||
|
"price": "0.00"
|
||||||
|
},
|
||||||
|
"sourcecode": {
|
||||||
|
"price": "0.00"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"contact": {
|
||||||
|
"qq": "305716444"
|
||||||
|
},
|
||||||
|
"declaration": {
|
||||||
|
"ads": "无",
|
||||||
|
"data": "无",
|
||||||
|
"permissions": "无"
|
||||||
|
},
|
||||||
|
"npmurl": "",
|
||||||
|
"type": "component-vue"
|
||||||
|
},
|
||||||
|
"uni_modules": {
|
||||||
|
"dependencies": [],
|
||||||
|
"encrypt": [],
|
||||||
|
"platforms": {
|
||||||
|
"cloud": {
|
||||||
|
"tcb": "y",
|
||||||
|
"aliyun": "y",
|
||||||
|
"alipay": "n"
|
||||||
|
},
|
||||||
|
"client": {
|
||||||
|
"App": {
|
||||||
|
"app-vue": "y",
|
||||||
|
"app-nvue": "y"
|
||||||
|
},
|
||||||
|
"H5-mobile": {
|
||||||
|
"Safari": "y",
|
||||||
|
"Android Browser": "y",
|
||||||
|
"微信浏览器(Android)": "y",
|
||||||
|
"QQ浏览器(Android)": "y"
|
||||||
|
},
|
||||||
|
"H5-pc": {
|
||||||
|
"Chrome": "y",
|
||||||
|
"IE": "u",
|
||||||
|
"Edge": "u",
|
||||||
|
"Firefox": "u",
|
||||||
|
"Safari": "y"
|
||||||
|
},
|
||||||
|
"小程序": {
|
||||||
|
"微信": "y",
|
||||||
|
"阿里": "y",
|
||||||
|
"百度": "y",
|
||||||
|
"字节跳动": "y",
|
||||||
|
"QQ": "y",
|
||||||
|
"钉钉": "u",
|
||||||
|
"快手": "u",
|
||||||
|
"飞书": "u",
|
||||||
|
"京东": "u"
|
||||||
|
},
|
||||||
|
"快应用": {
|
||||||
|
"华为": "u",
|
||||||
|
"联盟": "u"
|
||||||
|
},
|
||||||
|
"Vue": {
|
||||||
|
"vue2": "y",
|
||||||
|
"vue3": "y"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"name": "lime-painter",
|
||||||
|
"main": "index.js",
|
||||||
|
"scripts": {
|
||||||
|
"test": "echo \"Error: no test specified\" && exit 1"
|
||||||
|
},
|
||||||
|
"author": "",
|
||||||
|
"license": "ISC"
|
||||||
|
}
|
||||||
|
|
@ -0,0 +1,388 @@
|
||||||
|
/*
|
||||||
|
* HTML5 Parser By Sam Blowes
|
||||||
|
*
|
||||||
|
* Designed for HTML5 documents
|
||||||
|
*
|
||||||
|
* Original code by John Resig (ejohn.org)
|
||||||
|
* http://ejohn.org/blog/pure-javascript-html-parser/
|
||||||
|
* Original code by Erik Arvidsson, Mozilla Public License
|
||||||
|
* http://erik.eae.net/simplehtmlparser/simplehtmlparser.js
|
||||||
|
*
|
||||||
|
* ----------------------------------------------------------------------------
|
||||||
|
* License
|
||||||
|
* ----------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* This code is triple licensed using Apache Software License 2.0,
|
||||||
|
* Mozilla Public License or GNU Public License
|
||||||
|
*
|
||||||
|
* ////////////////////////////////////////////////////////////////////////////
|
||||||
|
*
|
||||||
|
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
||||||
|
* use this file except in compliance with the License. You may obtain a copy
|
||||||
|
* of the License at http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
*
|
||||||
|
* ////////////////////////////////////////////////////////////////////////////
|
||||||
|
*
|
||||||
|
* The contents of this file are subject to the Mozilla Public License
|
||||||
|
* Version 1.1 (the "License"); you may not use this file except in
|
||||||
|
* compliance with the License. You may obtain a copy of the License at
|
||||||
|
* http://www.mozilla.org/MPL/
|
||||||
|
*
|
||||||
|
* Software distributed under the License is distributed on an "AS IS"
|
||||||
|
* basis, WITHOUT WARRANTY OF ANY KIND, either express or implied. See the
|
||||||
|
* License for the specific language governing rights and limitations
|
||||||
|
* under the License.
|
||||||
|
*
|
||||||
|
* The Original Code is Simple HTML Parser.
|
||||||
|
*
|
||||||
|
* The Initial Developer of the Original Code is Erik Arvidsson.
|
||||||
|
* Portions created by Erik Arvidssson are Copyright (C) 2004. All Rights
|
||||||
|
* Reserved.
|
||||||
|
*
|
||||||
|
* ////////////////////////////////////////////////////////////////////////////
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, write to the Free Software
|
||||||
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
||||||
|
*
|
||||||
|
* ----------------------------------------------------------------------------
|
||||||
|
* Usage
|
||||||
|
* ----------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* // Use like so:
|
||||||
|
* HTMLParser(htmlString, {
|
||||||
|
* start: function(tag, attrs, unary) {},
|
||||||
|
* end: function(tag) {},
|
||||||
|
* chars: function(text) {},
|
||||||
|
* comment: function(text) {}
|
||||||
|
* });
|
||||||
|
*
|
||||||
|
* // or to get an XML string:
|
||||||
|
* HTMLtoXML(htmlString);
|
||||||
|
*
|
||||||
|
* // or to get an XML DOM Document
|
||||||
|
* HTMLtoDOM(htmlString);
|
||||||
|
*
|
||||||
|
* // or to inject into an existing document/DOM node
|
||||||
|
* HTMLtoDOM(htmlString, document);
|
||||||
|
* HTMLtoDOM(htmlString, document.body);
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
// Regular Expressions for parsing tags and attributes
|
||||||
|
var startTag = /^<([-A-Za-z0-9_]+)((?:\s+[a-zA-Z_:][-a-zA-Z0-9_:.]*(?:\s*=\s*(?:(?:"[^"]*")|(?:'[^']*')|[^>\s]+))?)*)\s*(\/?)>/;
|
||||||
|
var endTag = /^<\/([-A-Za-z0-9_]+)[^>]*>/;
|
||||||
|
var attr = /([a-zA-Z_:][-a-zA-Z0-9_:.]*)(?:\s*=\s*(?:(?:"((?:\\.|[^"])*)")|(?:'((?:\\.|[^'])*)')|([^>\s]+)))?/g; // Empty Elements - HTML 5
|
||||||
|
|
||||||
|
var empty = makeMap('area,base,basefont,br,col,frame,hr,img,input,link,meta,param,embed,command,keygen,source,track,wbr'); // Block Elements - HTML 5
|
||||||
|
// fixed by xxx 将 ins 标签从块级名单中移除
|
||||||
|
|
||||||
|
var block = makeMap('a,address,article,applet,aside,audio,blockquote,button,canvas,center,dd,del,dir,div,dl,dt,fieldset,figcaption,figure,footer,form,frameset,h1,h2,h3,h4,h5,h6,header,hgroup,hr,iframe,isindex,li,map,menu,noframes,noscript,object,ol,output,p,pre,section,script,table,tbody,td,tfoot,th,thead,tr,ul,video'); // Inline Elements - HTML 5
|
||||||
|
|
||||||
|
var inline = makeMap('abbr,acronym,applet,b,basefont,bdo,big,br,button,cite,code,del,dfn,em,font,i,iframe,img,input,ins,kbd,label,map,object,q,s,samp,script,select,small,span,strike,strong,sub,sup,textarea,tt,u,var'); // Elements that you can, intentionally, leave open
|
||||||
|
// (and which close themselves)
|
||||||
|
|
||||||
|
var closeSelf = makeMap('colgroup,dd,dt,li,options,p,td,tfoot,th,thead,tr'); // Attributes that have their values filled in disabled="disabled"
|
||||||
|
|
||||||
|
var fillAttrs = makeMap('checked,compact,declare,defer,disabled,ismap,multiple,nohref,noresize,noshade,nowrap,readonly,selected'); // Special Elements (can contain anything)
|
||||||
|
|
||||||
|
var special = makeMap('script,style');
|
||||||
|
function HTMLParser(html, handler) {
|
||||||
|
var index;
|
||||||
|
var chars;
|
||||||
|
var match;
|
||||||
|
var stack = [];
|
||||||
|
var last = html;
|
||||||
|
|
||||||
|
stack.last = function () {
|
||||||
|
return this[this.length - 1];
|
||||||
|
};
|
||||||
|
|
||||||
|
while (html) {
|
||||||
|
chars = true; // Make sure we're not in a script or style element
|
||||||
|
|
||||||
|
if (!stack.last() || !special[stack.last()]) {
|
||||||
|
// Comment
|
||||||
|
if (html.indexOf('<!--') == 0) {
|
||||||
|
index = html.indexOf('-->');
|
||||||
|
|
||||||
|
if (index >= 0) {
|
||||||
|
if (handler.comment) {
|
||||||
|
handler.comment(html.substring(4, index));
|
||||||
|
}
|
||||||
|
|
||||||
|
html = html.substring(index + 3);
|
||||||
|
chars = false;
|
||||||
|
} // end tag
|
||||||
|
|
||||||
|
} else if (html.indexOf('</') == 0) {
|
||||||
|
match = html.match(endTag);
|
||||||
|
|
||||||
|
if (match) {
|
||||||
|
html = html.substring(match[0].length);
|
||||||
|
match[0].replace(endTag, parseEndTag);
|
||||||
|
chars = false;
|
||||||
|
} // start tag
|
||||||
|
|
||||||
|
} else if (html.indexOf('<') == 0) {
|
||||||
|
match = html.match(startTag);
|
||||||
|
|
||||||
|
if (match) {
|
||||||
|
html = html.substring(match[0].length);
|
||||||
|
match[0].replace(startTag, parseStartTag);
|
||||||
|
chars = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (chars) {
|
||||||
|
index = html.indexOf('<');
|
||||||
|
var text = index < 0 ? html : html.substring(0, index);
|
||||||
|
html = index < 0 ? '' : html.substring(index);
|
||||||
|
|
||||||
|
if (handler.chars) {
|
||||||
|
handler.chars(text);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
html = html.replace(new RegExp('([\\s\\S]*?)<\/' + stack.last() + '[^>]*>'), function (all, text) {
|
||||||
|
text = text.replace(/<!--([\s\S]*?)-->|<!\[CDATA\[([\s\S]*?)]]>/g, '$1$2');
|
||||||
|
|
||||||
|
if (handler.chars) {
|
||||||
|
handler.chars(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
|
});
|
||||||
|
parseEndTag('', stack.last());
|
||||||
|
}
|
||||||
|
|
||||||
|
if (html == last) {
|
||||||
|
throw 'Parse Error: ' + html;
|
||||||
|
}
|
||||||
|
|
||||||
|
last = html;
|
||||||
|
} // Clean up any remaining tags
|
||||||
|
|
||||||
|
|
||||||
|
parseEndTag();
|
||||||
|
|
||||||
|
function parseStartTag(tag, tagName, rest, unary) {
|
||||||
|
tagName = tagName.toLowerCase();
|
||||||
|
if (block[tagName]) {
|
||||||
|
while (stack.last() && inline[stack.last()]) {
|
||||||
|
parseEndTag('', stack.last());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (closeSelf[tagName] && stack.last() == tagName) {
|
||||||
|
parseEndTag('', tagName);
|
||||||
|
}
|
||||||
|
|
||||||
|
unary = empty[tagName] || !!unary;
|
||||||
|
|
||||||
|
if (!unary) {
|
||||||
|
stack.push(tagName);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (handler.start) {
|
||||||
|
var attrs = [];
|
||||||
|
rest.replace(attr, function (match, name) {
|
||||||
|
var value = arguments[2] ? arguments[2] : arguments[3] ? arguments[3] : arguments[4] ? arguments[4] : fillAttrs[name] ? name : '';
|
||||||
|
attrs.push({
|
||||||
|
name: name,
|
||||||
|
value: value,
|
||||||
|
escaped: value.replace(/(^|[^\\])"/g, '$1\\\"') // "
|
||||||
|
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
if (handler.start) {
|
||||||
|
handler.start(tagName, attrs, unary);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseEndTag(tag, tagName) {
|
||||||
|
// If no tag name is provided, clean shop
|
||||||
|
if (!tagName) {
|
||||||
|
var pos = 0;
|
||||||
|
} // Find the closest opened tag of the same type
|
||||||
|
else {
|
||||||
|
for (var pos = stack.length - 1; pos >= 0; pos--) {
|
||||||
|
if (stack[pos] == tagName) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (pos >= 0) {
|
||||||
|
// Close all the open elements, up the stack
|
||||||
|
for (var i = stack.length - 1; i >= pos; i--) {
|
||||||
|
if (handler.end) {
|
||||||
|
handler.end(stack[i]);
|
||||||
|
}
|
||||||
|
} // Remove the open elements from the stack
|
||||||
|
|
||||||
|
|
||||||
|
stack.length = pos;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function makeMap(str) {
|
||||||
|
var obj = {};
|
||||||
|
var items = str.split(',');
|
||||||
|
|
||||||
|
for (var i = 0; i < items.length; i++) {
|
||||||
|
obj[items[i]] = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
return obj;
|
||||||
|
}
|
||||||
|
|
||||||
|
function removeDOCTYPE(html) {
|
||||||
|
return html.replace(/<\?xml.*\?>\n/, '').replace(/<!doctype.*>\n/, '').replace(/<!DOCTYPE.*>\n/, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
function parseAttrs(attrs) {
|
||||||
|
return attrs.reduce(function (pre, attr) {
|
||||||
|
var value = attr.value;
|
||||||
|
var name = attr.name;
|
||||||
|
if (pre[name]) {
|
||||||
|
pre[name] = pre[name] + " " + value;
|
||||||
|
} else {
|
||||||
|
pre[name] = value;
|
||||||
|
}
|
||||||
|
|
||||||
|
return pre;
|
||||||
|
}, {});
|
||||||
|
}
|
||||||
|
function convertStyleStringToJSON(styleString) {
|
||||||
|
var styles = styleString.split(";"); // 通过分号将样式字符串分割为多个样式声明
|
||||||
|
var result = {};
|
||||||
|
|
||||||
|
styles.forEach(function(style) {
|
||||||
|
var styleParts = style.split(":"); // 通过冒号将样式声明分割为属性和值
|
||||||
|
var property = styleParts[0].trim();
|
||||||
|
var value = styleParts[1] && styleParts[1].trim();
|
||||||
|
|
||||||
|
if (property && value) {
|
||||||
|
result[property] = value; // 将属性和值添加到结果对象中
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
function parseHtml(html) {
|
||||||
|
html = removeDOCTYPE(html);
|
||||||
|
var stacks = [];
|
||||||
|
var results = {
|
||||||
|
node: 'root',
|
||||||
|
children: []
|
||||||
|
};
|
||||||
|
HTMLParser(html, {
|
||||||
|
start: function start(tag, attrs, unary) {
|
||||||
|
var node = {
|
||||||
|
name: tag
|
||||||
|
};
|
||||||
|
|
||||||
|
if (attrs.length !== 0) {
|
||||||
|
node.attrs = parseAttrs(attrs);
|
||||||
|
node.styles = node.attrs.style ? convertStyleStringToJSON(node.attrs.style) : {}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!node.type) {
|
||||||
|
if(inline[node.name] && node.name !== 'img' ) {
|
||||||
|
node.type = 'text';
|
||||||
|
if(node.name == 'br') {
|
||||||
|
node.text = '\n'
|
||||||
|
} else if(node.name == 'strong'){
|
||||||
|
node.styles.fontWeight = 'bold'
|
||||||
|
}
|
||||||
|
} else if(node.name == 'img'){
|
||||||
|
node.type = 'image'
|
||||||
|
node.src = node.attrs.src
|
||||||
|
} else {
|
||||||
|
node.type = 'view'
|
||||||
|
if(['h1','h2','h3','h4','h5','h6'].includes(node.name)) {
|
||||||
|
node.styles.fontWeight = 'bold'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (unary) {
|
||||||
|
var parent = stacks[0] || results;
|
||||||
|
|
||||||
|
if (!parent.children) {
|
||||||
|
parent.children = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
parent.children.push(node);
|
||||||
|
} else {
|
||||||
|
stacks.unshift(node);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
end: function end(tag) {
|
||||||
|
var node = stacks.shift();
|
||||||
|
if (node.name !== tag) console.error('invalid state: mismatch end tag');
|
||||||
|
if (stacks.length === 0) {
|
||||||
|
results.children.push(node);
|
||||||
|
} else {
|
||||||
|
var parent = stacks[0];
|
||||||
|
|
||||||
|
if (!parent.children) {
|
||||||
|
parent.children = [];
|
||||||
|
}
|
||||||
|
parent.children.push(node);
|
||||||
|
}
|
||||||
|
const isTextBox = node.children && node.children.length > 1 && node.children.every(child => {
|
||||||
|
return ['text','image'].includes(child.type)
|
||||||
|
})
|
||||||
|
if(isTextBox) {
|
||||||
|
node.type = 'textBox'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
chars: function chars(text) {
|
||||||
|
var node = {
|
||||||
|
type: 'text',
|
||||||
|
text: text
|
||||||
|
};
|
||||||
|
|
||||||
|
if (stacks.length === 0) {
|
||||||
|
results.children.push(node);
|
||||||
|
} else {
|
||||||
|
var parent = stacks[0];
|
||||||
|
|
||||||
|
if (!parent.children) {
|
||||||
|
parent.children = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
parent.children.push(node);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
comment: function comment(text) {
|
||||||
|
var node = {
|
||||||
|
node: 'comment',
|
||||||
|
text: text
|
||||||
|
};
|
||||||
|
var parent = stacks[0];
|
||||||
|
|
||||||
|
if (!parent.children) {
|
||||||
|
parent.children = [];
|
||||||
|
}
|
||||||
|
|
||||||
|
parent.children.push(node);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return results.children;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default parseHtml;
|
||||||
|
|
@ -0,0 +1,961 @@
|
||||||
|
# Painter 画板 测试版
|
||||||
|
|
||||||
|
> uniapp 海报画板,更优雅的海报生成方案
|
||||||
|
> [查看更多](https://limeui.qcoon.cn/#/painter)
|
||||||
|
|
||||||
|
## 平台兼容
|
||||||
|
|
||||||
|
| H5 | 微信小程序 | 支付宝小程序 | 百度小程序 | 头条小程序 | QQ 小程序 | App |
|
||||||
|
| --- | ---------- | ------------ | ---------- | ---------- | --------- | --- |
|
||||||
|
| √ | √ | √ | 未测 | √ | √ | √ |
|
||||||
|
|
||||||
|
## 安装
|
||||||
|
在市场导入**[海报画板](https://ext.dcloud.net.cn/plugin?id=2389)uni_modules**版本的即可,无需`import`
|
||||||
|
|
||||||
|
## 代码演示
|
||||||
|
|
||||||
|
### 插件demo
|
||||||
|
- lime-painter 为 demo
|
||||||
|
- 位于 uni_modules/lime-painter/components/lime-painter
|
||||||
|
- 导入插件后直接使用可查看demo
|
||||||
|
```vue
|
||||||
|
<lime-painter />
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### 基本用法
|
||||||
|
|
||||||
|
- 插件提供 JSON 及 Template 的方式绘制海报
|
||||||
|
- 参考 css 块状流布局模拟 css schema。
|
||||||
|
- 另外flex布局还不是成完善,请谨慎使用,普通的流布局我觉得已经够用了。
|
||||||
|
|
||||||
|
#### 方式一 Template
|
||||||
|
|
||||||
|
- 提供`l-painter-view`、`l-painter-text`、`l-painter-image`、`l-painter-qrcode`四种类型组件
|
||||||
|
- 通过 `css` 属性绘制样式,与 style 使用方式保持一致。
|
||||||
|
```html
|
||||||
|
<l-painter>
|
||||||
|
//如果使用Template出现顺序错乱,可使用`template` 等所有变量完成再显示
|
||||||
|
<template v-if="show">
|
||||||
|
<l-painter-view
|
||||||
|
css="background: #07c160; height: 120rpx; width: 120rpx; display: inline-block"
|
||||||
|
></l-painter-view>
|
||||||
|
<l-painter-view
|
||||||
|
css="background: #1989fa; height: 120rpx; width: 120rpx; border-top-right-radius: 60rpx; border-bottom-left-radius: 60rpx; display: inline-block; margin: 0 30rpx;"
|
||||||
|
></l-painter-view>
|
||||||
|
<l-painter-view
|
||||||
|
css="background: #ff9d00; height: 120rpx; width: 120rpx; border-radius: 50%; display: inline-block"
|
||||||
|
></l-painter-view>
|
||||||
|
<template>
|
||||||
|
</l-painter>
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 方式二 JSON
|
||||||
|
|
||||||
|
- 在 json 里四种类型组件的`type`为`view`、`text`、`image`、`qrcode`
|
||||||
|
- 通过 `board` 设置海报所需的 JSON 数据进行绘制或`ref`获取组件实例调用组件内的`render(json)`
|
||||||
|
- 所有类型的 schema 都具有`css`字段,css 的 key 值使用**驼峰**如:`lineHeight`
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter :board="poster"/>
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
poster: {
|
||||||
|
css: {
|
||||||
|
// 根节点若无尺寸,自动获取父级节点
|
||||||
|
width: '750rpx'
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
css: {
|
||||||
|
background: "#07c160",
|
||||||
|
height: "120rpx",
|
||||||
|
width: "120rpx",
|
||||||
|
display: "inline-block"
|
||||||
|
},
|
||||||
|
type: "view"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
css: {
|
||||||
|
background: "#1989fa",
|
||||||
|
height: "120rpx",
|
||||||
|
width: "120rpx",
|
||||||
|
borderTopRightRadius: "60rpx",
|
||||||
|
borderBottomLeftRadius: "60rpx",
|
||||||
|
display: "inline-block",
|
||||||
|
margin: "0 30rpx"
|
||||||
|
},
|
||||||
|
views: [],
|
||||||
|
type: "view"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
css: {
|
||||||
|
background: "#ff9d00",
|
||||||
|
height: "120rpx",
|
||||||
|
width: "120rpx",
|
||||||
|
borderRadius: "50%",
|
||||||
|
display: "inline-block"
|
||||||
|
},
|
||||||
|
views: [],
|
||||||
|
type: "view"
|
||||||
|
},
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### View 容器
|
||||||
|
|
||||||
|
- 类似于 `div` 可以嵌套承载更多的 view、text、image,qrcode 共同构建一颗完整的节点树
|
||||||
|
- 在 JSON 里具有 `views` 的数组字段,用于嵌套承载节点。
|
||||||
|
|
||||||
|
#### 方式一 Template
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter>
|
||||||
|
<l-painter-view css="background: #f0f0f0; padding-top: 100rpx;">
|
||||||
|
<l-painter-view
|
||||||
|
css="background: #d9d9d9; width: 33.33%; height: 100rpx; display: inline-block"
|
||||||
|
></l-painter-view>
|
||||||
|
<l-painter-view
|
||||||
|
css="background: #bfbfbf; width: 66.66%; height: 100rpx; display: inline-block"
|
||||||
|
></l-painter-view>
|
||||||
|
</l-painter-view>
|
||||||
|
</l-painter>
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 方式二 JSON
|
||||||
|
|
||||||
|
```js
|
||||||
|
{
|
||||||
|
css: {},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
type: 'view',
|
||||||
|
css: {
|
||||||
|
background: '#f0f0f0',
|
||||||
|
paddingTop: '100rpx'
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
type: 'view',
|
||||||
|
css: {
|
||||||
|
background: '#d9d9d9',
|
||||||
|
width: '33.33%',
|
||||||
|
height: '100rpx',
|
||||||
|
display: 'inline-block'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'view',
|
||||||
|
css: {
|
||||||
|
background: '#bfbfbf',
|
||||||
|
width: '66.66%',
|
||||||
|
height: '100rpx',
|
||||||
|
display: 'inline-block'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Text 文本
|
||||||
|
|
||||||
|
- 通过 `text` 属性填写文本内容。
|
||||||
|
- 支持`\n`换行符
|
||||||
|
- 支持省略号,使用 css 的`line-clamp`设置行数,当文字内容超过会显示省略号。
|
||||||
|
- 支持`text-decoration`
|
||||||
|
|
||||||
|
#### 方式一 Template
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter>
|
||||||
|
<l-painter-view css="background: #e0e2db; padding: 30rpx; color: #222a29">
|
||||||
|
<l-painter-text
|
||||||
|
text="登鹳雀楼\n白日依山尽,黄河入海流\n欲穷千里目,更上一层楼"
|
||||||
|
/>
|
||||||
|
<l-painter-text
|
||||||
|
text="登鹳雀楼\n白日依山尽,黄河入海流\n欲穷千里目,更上一层楼"
|
||||||
|
css="text-align:center; padding-top: 20rpx; text-decoration: line-through "
|
||||||
|
/>
|
||||||
|
<l-painter-text
|
||||||
|
text="登鹳雀楼\n白日依山尽,黄河入海流\n欲穷千里目,更上一层楼"
|
||||||
|
css="text-align:right; padding-top: 20rpx"
|
||||||
|
/>
|
||||||
|
<l-painter-text
|
||||||
|
text="水调歌头\n明月几时有?把酒问青天。不知天上宫阙,今夕是何年。我欲乘风归去,又恐琼楼玉宇,高处不胜寒。起舞弄清影,何似在人间。"
|
||||||
|
css="line-clamp: 3; padding-top: 20rpx; background: linear-gradient(,#ff971b 0%, #ff5000 100%); background-clip: text"
|
||||||
|
/>
|
||||||
|
</l-painter-view>
|
||||||
|
</l-painter>
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 方式二 JSON
|
||||||
|
|
||||||
|
```js
|
||||||
|
// 基础用法
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: '登鹳雀楼\n白日依山尽,黄河入海流\n欲穷千里目,更上一层楼',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: '登鹳雀楼\n白日依山尽,黄河入海流\n欲穷千里目,更上一层楼',
|
||||||
|
css: {
|
||||||
|
// 设置居中对齐
|
||||||
|
textAlign: 'center',
|
||||||
|
// 设置中划线
|
||||||
|
textDecoration: 'line-through'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: '登鹳雀楼\n白日依山尽,黄河入海流\n欲穷千里目,更上一层楼',
|
||||||
|
css: {
|
||||||
|
// 设置右对齐
|
||||||
|
textAlign: 'right',
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'text',
|
||||||
|
text: '登鹳雀楼\n白日依山尽,黄河入海流\n欲穷千里目,更上一层楼',
|
||||||
|
css: {
|
||||||
|
// 设置行数,超出显示省略号
|
||||||
|
lineClamp: 3,
|
||||||
|
// 渐变文字
|
||||||
|
background: 'linear-gradient(,#ff971b 0%, #1989fa 100%)',
|
||||||
|
backgroundClip: 'text'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Image 图片
|
||||||
|
|
||||||
|
- 通过 `src` 属性填写图片路径。
|
||||||
|
- 图片路径支持:网络图片,本地 static 里的图片路径,缓存路径,**字节的static目录是写相对路径**
|
||||||
|
- 通过 `css` 的 `object-fit`属性可以设置图片的填充方式,可选值见下方 CSS 表格。
|
||||||
|
- 通过 `css` 的 `object-position`配合 `object-fit` 可以设置图片的对齐方式,类似于`background-position`,详情见下方 CSS 表格。
|
||||||
|
- 使用网络图片时:小程序需要去公众平台配置 [downloadFile](https://mp.weixin.qq.com/) 域名
|
||||||
|
- 使用网络图片时:**H5 和 Nvue 需要决跨域问题**
|
||||||
|
|
||||||
|
#### 方式一 Template
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter>
|
||||||
|
<!-- 基础用法 -->
|
||||||
|
<l-painter-image
|
||||||
|
src="https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg"
|
||||||
|
css="width: 200rpx; height: 200rpx"
|
||||||
|
/>
|
||||||
|
<!-- 填充方式 -->
|
||||||
|
<!-- css object-fit 设置 填充方式 见下方表格-->
|
||||||
|
<l-painter-image
|
||||||
|
src="https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg"
|
||||||
|
css="width: 200rpx; height: 200rpx; object-fit: contain; background: #eee"
|
||||||
|
/>
|
||||||
|
<!-- css object-position 设置 图片的对齐方式-->
|
||||||
|
<l-painter-image
|
||||||
|
src="https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg"
|
||||||
|
css="width: 200rpx; height: 200rpx; object-fit: contain; object-position: 50% 50%; background: #eee"
|
||||||
|
/>
|
||||||
|
</l-painter>
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 方式二 JSON
|
||||||
|
|
||||||
|
```js
|
||||||
|
// 基础用法
|
||||||
|
{
|
||||||
|
type: 'image',
|
||||||
|
src: 'https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg',
|
||||||
|
css: {
|
||||||
|
width: '200rpx',
|
||||||
|
height: '200rpx'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// 填充方式
|
||||||
|
// css objectFit 设置 填充方式 见下方表格
|
||||||
|
{
|
||||||
|
type: 'image',
|
||||||
|
src: 'https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg',
|
||||||
|
css: {
|
||||||
|
width: '200rpx',
|
||||||
|
height: '200rpx',
|
||||||
|
objectFit: 'contain'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// css objectPosition 设置 图片的对齐方式
|
||||||
|
{
|
||||||
|
type: 'image',
|
||||||
|
src: 'https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg',
|
||||||
|
css: {
|
||||||
|
width: '200rpx',
|
||||||
|
height: '200rpx',
|
||||||
|
objectFit: 'contain',
|
||||||
|
objectPosition: '50% 50%'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### Qrcode 二维码
|
||||||
|
|
||||||
|
- 通过`text`属性填写需要生成二维码的文本。
|
||||||
|
- 通过 `css` 里的 `color` 可设置生成码点的颜色。
|
||||||
|
- 通过 `css` 里的 `background`可设置背景色。
|
||||||
|
- 通过 `css `里的 `width`、`height`设置尺寸。
|
||||||
|
|
||||||
|
#### 方式一 Template
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter>
|
||||||
|
<l-painter-qrcode
|
||||||
|
text="limeui.qcoon.cn"
|
||||||
|
css="width: 200rpx; height: 200rpx"
|
||||||
|
/>
|
||||||
|
</l-painter>
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 方式二 JSON
|
||||||
|
|
||||||
|
```js
|
||||||
|
{
|
||||||
|
type: 'qrcode',
|
||||||
|
text: 'limeui.qcoon.cn',
|
||||||
|
css: {
|
||||||
|
width: '200rpx',
|
||||||
|
height: '200rpx',
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
### 富文本
|
||||||
|
- 这是一个有限支持的测试能力,只能通过JSON方式,不要抱太大希望!
|
||||||
|
- 首先需要把富文本转成JSON,这需要引入`parser`这个包,如果你不使用是不会进入主包
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter ref="painter"/>
|
||||||
|
```
|
||||||
|
```js
|
||||||
|
import parseHtml from '@/uni_modules/lime-painter/parser'
|
||||||
|
const json = parseHtml(`<p><span>测试测试</span><img src="/static/logo.png"/></p>`)
|
||||||
|
this.$refs.painter.render(json)
|
||||||
|
```
|
||||||
|
|
||||||
|
### 生成图片
|
||||||
|
|
||||||
|
- 方式1、通过设置`isCanvasToTempFilePath`自动生成图片并在 `@success` 事件里接收海报临时路径
|
||||||
|
- 方式2、通过调用内部方法生成图片:
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter ref="painter">...code</l-painter>
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
this.$refs.painter.canvasToTempFilePathSync({
|
||||||
|
fileType: "jpg",
|
||||||
|
// 如果返回的是base64是无法使用 saveImageToPhotosAlbum,需要设置 pathType为url
|
||||||
|
pathType: 'url',
|
||||||
|
quality: 1,
|
||||||
|
success: (res) => {
|
||||||
|
console.log(res.tempFilePath);
|
||||||
|
// 非H5 保存到相册
|
||||||
|
// H5 提示用户长按图另存
|
||||||
|
uni.saveImageToPhotosAlbum({
|
||||||
|
filePath: res.tempFilePath,
|
||||||
|
success: function () {
|
||||||
|
console.log('save success');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### 主动调用方式
|
||||||
|
|
||||||
|
- 通过获取组件实例内部的`render`函数 传递`JSON`即可
|
||||||
|
|
||||||
|
```html
|
||||||
|
<l-painter ref="painter" />
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
// 渲染
|
||||||
|
this.$refs.painter.render(jsonSchema);
|
||||||
|
// 生成图片
|
||||||
|
this.$refs.painter.canvasToTempFilePathSync({
|
||||||
|
fileType: "jpg",
|
||||||
|
// 如果返回的是base64是无法使用 saveImageToPhotosAlbum,需要设置 pathType为url
|
||||||
|
pathType: 'url',
|
||||||
|
quality: 1,
|
||||||
|
success: (res) => {
|
||||||
|
console.log(res.tempFilePath);
|
||||||
|
// 非H5 保存到相册
|
||||||
|
uni.saveImageToPhotosAlbum({
|
||||||
|
filePath: res.tempFilePath,
|
||||||
|
success: function () {
|
||||||
|
console.log('save success');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### H5跨域
|
||||||
|
- 一般是需要后端或管理OSS资源的大佬处理
|
||||||
|
- 一般OSS的处理方式:
|
||||||
|
|
||||||
|
1、设置来源
|
||||||
|
```cmd
|
||||||
|
*
|
||||||
|
```
|
||||||
|
|
||||||
|
2、允许Methods
|
||||||
|
```html
|
||||||
|
GET
|
||||||
|
```
|
||||||
|
|
||||||
|
3、允许Headers
|
||||||
|
```html
|
||||||
|
access-control-allow-origin:*
|
||||||
|
```
|
||||||
|
|
||||||
|
4、最后如果还是不行,可试下给插件设置`useCORS`
|
||||||
|
```html
|
||||||
|
<l-painter useCORS>
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
### 海报示例
|
||||||
|
|
||||||
|
- 提供一份示例,只把插件当成生成图片的工具,非必要不要在弹窗里使用。
|
||||||
|
- 通过设置`isCanvasToTempFilePath`主动生成图片,再由 `@success` 事件接收海报临时路径
|
||||||
|
- 设置`hidden`隐藏画板。
|
||||||
|
请注意,示例用到了图片,海报的渲染是包括下载图片的时间,也许在某天图片会失效或访问超级慢,请更换为你的图片再查看,另外如果你是小程序请在使用示例时把**不校验合法域名**勾上!!!!!不然不显示还以为是插件的锅,求求了大佬们!
|
||||||
|
#### 方式一 Template
|
||||||
|
|
||||||
|
```html
|
||||||
|
<image :src="path" mode="widthFix"></image>
|
||||||
|
<l-painter
|
||||||
|
isCanvasToTempFilePath
|
||||||
|
@success="path = $event"
|
||||||
|
hidden
|
||||||
|
css="width: 750rpx; padding-bottom: 40rpx; background: linear-gradient(,#ff971b 0%, #ff5000 100%)"
|
||||||
|
>
|
||||||
|
<l-painter-image
|
||||||
|
src="https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg"
|
||||||
|
css="margin-left: 40rpx; margin-top: 40rpx; width: 84rpx; height: 84rpx; border-radius: 50%;"
|
||||||
|
/>
|
||||||
|
<l-painter-view
|
||||||
|
css="margin-top: 40rpx; padding-left: 20rpx; display: inline-block"
|
||||||
|
>
|
||||||
|
<l-painter-text
|
||||||
|
text="隔壁老王"
|
||||||
|
css="display: block; padding-bottom: 10rpx; color: #fff; font-size: 32rpx; fontWeight: bold"
|
||||||
|
/>
|
||||||
|
<l-painter-text
|
||||||
|
text="为您挑选了一个好物"
|
||||||
|
css="color: rgba(255,255,255,.7); font-size: 24rpx"
|
||||||
|
/>
|
||||||
|
</l-painter-view>
|
||||||
|
<l-painter-view
|
||||||
|
css="margin-left: 40rpx; margin-top: 30rpx; padding: 32rpx; box-sizing: border-box; background: #fff; border-radius: 16rpx; width: 670rpx; box-shadow: 0 20rpx 58rpx rgba(0,0,0,.15)"
|
||||||
|
>
|
||||||
|
<l-painter-image
|
||||||
|
src="https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg"
|
||||||
|
css="object-fit: cover; object-position: 50% 50%; width: 606rpx; height: 606rpx; border-radius: 12rpx;"
|
||||||
|
/>
|
||||||
|
<l-painter-view
|
||||||
|
css="margin-top: 32rpx; color: #FF0000; font-weight: bold; font-size: 28rpx; line-height: 1em;"
|
||||||
|
>
|
||||||
|
<l-painter-text text="¥" css="vertical-align: bottom" />
|
||||||
|
<l-painter-text
|
||||||
|
text="39"
|
||||||
|
css="vertical-align: bottom; font-size: 58rpx"
|
||||||
|
/>
|
||||||
|
<l-painter-text text=".39" css="vertical-align: bottom" />
|
||||||
|
<l-painter-text
|
||||||
|
text="¥59.99"
|
||||||
|
css="vertical-align: bottom; padding-left: 10rpx; font-weight: normal; text-decoration: line-through; color: #999999"
|
||||||
|
/>
|
||||||
|
</l-painter-view>
|
||||||
|
<l-painter-view css="margin-top: 32rpx; font-size: 26rpx; color: #8c5400">
|
||||||
|
<l-painter-text text="自营" css="color: #212121; background: #ffb400;" />
|
||||||
|
<l-painter-text
|
||||||
|
text="30天最低价"
|
||||||
|
css="margin-left: 16rpx; background: #fff4d9; text-decoration: line-through;"
|
||||||
|
/>
|
||||||
|
<l-painter-text
|
||||||
|
text="满减优惠"
|
||||||
|
css="margin-left: 16rpx; background: #fff4d9"
|
||||||
|
/>
|
||||||
|
<l-painter-text
|
||||||
|
text="超高好评"
|
||||||
|
css="margin-left: 16rpx; background: #fff4d9"
|
||||||
|
/>
|
||||||
|
</l-painter-view>
|
||||||
|
<l-painter-view css="margin-top: 30rpx">
|
||||||
|
<l-painter-text
|
||||||
|
css="line-clamp: 2; color: #333333; line-height: 1.8em; font-size: 36rpx; width: 478rpx; padding-right:32rpx; box-sizing: border-box"
|
||||||
|
text="360儿童电话手表9X 智能语音问答定位支付手表 4G全网通20米游泳级防水视频通话拍照手表男女孩星空蓝"
|
||||||
|
></l-painter-text>
|
||||||
|
<l-painter-qrcode
|
||||||
|
css="width: 128rpx; height: 128rpx;"
|
||||||
|
text="limeui.qcoon.cn"
|
||||||
|
></l-painter-qrcode>
|
||||||
|
</l-painter-view>
|
||||||
|
</l-painter-view>
|
||||||
|
</l-painter>
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
path: ''
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
#### 方式二 JSON
|
||||||
|
|
||||||
|
```html
|
||||||
|
<image :src="path" mode="widthFix"></image>
|
||||||
|
<l-painter
|
||||||
|
:board="poster"
|
||||||
|
isCanvasToTempFilePath
|
||||||
|
@success="path = $event"
|
||||||
|
hidden
|
||||||
|
/>
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
path: '',
|
||||||
|
poster: {
|
||||||
|
css: {
|
||||||
|
width: "750rpx",
|
||||||
|
paddingBottom: "40rpx",
|
||||||
|
background: "linear-gradient(,#000 0%, #ff5000 100%)"
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
src: "https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg",
|
||||||
|
type: "image",
|
||||||
|
css: {
|
||||||
|
background: "#fff",
|
||||||
|
objectFit: "cover",
|
||||||
|
marginLeft: "40rpx",
|
||||||
|
marginTop: "40rpx",
|
||||||
|
width: "84rpx",
|
||||||
|
border: "2rpx solid #fff",
|
||||||
|
boxSizing: "border-box",
|
||||||
|
height: "84rpx",
|
||||||
|
borderRadius: "50%"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "view",
|
||||||
|
css: {
|
||||||
|
marginTop: "40rpx",
|
||||||
|
paddingLeft: "20rpx",
|
||||||
|
display: "inline-block"
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
text: "隔壁老王",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
display: "block",
|
||||||
|
paddingBottom: "10rpx",
|
||||||
|
color: "#fff",
|
||||||
|
fontSize: "32rpx",
|
||||||
|
fontWeight: "bold"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
text: "为您挑选了一个好物",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
color: "rgba(255,255,255,.7)",
|
||||||
|
fontSize: "24rpx"
|
||||||
|
},
|
||||||
|
}
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
css: {
|
||||||
|
marginLeft: "40rpx",
|
||||||
|
marginTop: "30rpx",
|
||||||
|
padding: "32rpx",
|
||||||
|
boxSizing: "border-box",
|
||||||
|
background: "#fff",
|
||||||
|
borderRadius: "16rpx",
|
||||||
|
width: "670rpx",
|
||||||
|
boxShadow: "0 20rpx 58rpx rgba(0,0,0,.15)"
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
src: "https://m.360buyimg.com/babel/jfs/t1/196317/32/13733/288158/60f4ea39E6fb378ed/d69205b1a8ed3c97.jpg",
|
||||||
|
type: "image",
|
||||||
|
css: {
|
||||||
|
objectFit: "cover",
|
||||||
|
objectPosition: "50% 50%",
|
||||||
|
width: "606rpx",
|
||||||
|
height: "606rpx"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
css: {
|
||||||
|
marginTop: "32rpx",
|
||||||
|
color: "#FF0000",
|
||||||
|
fontWeight: "bold",
|
||||||
|
fontSize: "28rpx",
|
||||||
|
lineHeight: "1em"
|
||||||
|
},
|
||||||
|
views: [{
|
||||||
|
text: "¥",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
verticalAlign: "bottom"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
text: "39",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
verticalAlign: "bottom",
|
||||||
|
fontSize: "58rpx"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
text: ".39",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
verticalAlign: "bottom"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
text: "¥59.99",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
verticalAlign: "bottom",
|
||||||
|
paddingLeft: "10rpx",
|
||||||
|
fontWeight: "normal",
|
||||||
|
textDecoration: "line-through",
|
||||||
|
color: "#999999"
|
||||||
|
}
|
||||||
|
}],
|
||||||
|
|
||||||
|
type: "view"
|
||||||
|
}, {
|
||||||
|
css: {
|
||||||
|
marginTop: "32rpx",
|
||||||
|
fontSize: "26rpx",
|
||||||
|
color: "#8c5400"
|
||||||
|
},
|
||||||
|
views: [{
|
||||||
|
text: "自营",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
color: "#212121",
|
||||||
|
background: "#ffb400"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
text: "30天最低价",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
marginLeft: "16rpx",
|
||||||
|
background: "#fff4d9",
|
||||||
|
textDecoration: "line-through"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
text: "满减优惠",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
marginLeft: "16rpx",
|
||||||
|
background: "#fff4d9"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
text: "超高好评",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
marginLeft: "16rpx",
|
||||||
|
background: "#fff4d9"
|
||||||
|
},
|
||||||
|
|
||||||
|
}],
|
||||||
|
|
||||||
|
type: "view"
|
||||||
|
}, {
|
||||||
|
css: {
|
||||||
|
marginTop: "30rpx"
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
text: "360儿童电话手表9X 智能语音问答定位支付手表 4G全网通20米游泳级防水视频通话拍照手表男女孩星空蓝",
|
||||||
|
type: "text",
|
||||||
|
css: {
|
||||||
|
paddingRight: "32rpx",
|
||||||
|
boxSizing: "border-box",
|
||||||
|
lineClamp: 2,
|
||||||
|
color: "#333333",
|
||||||
|
lineHeight: "1.8em",
|
||||||
|
fontSize: "36rpx",
|
||||||
|
width: "478rpx"
|
||||||
|
},
|
||||||
|
}, {
|
||||||
|
text: "limeui.qcoon.cn",
|
||||||
|
type: "qrcode",
|
||||||
|
css: {
|
||||||
|
width: "128rpx",
|
||||||
|
height: "128rpx",
|
||||||
|
},
|
||||||
|
|
||||||
|
}],
|
||||||
|
type: "view"
|
||||||
|
}],
|
||||||
|
type: "view"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### 自定义字体
|
||||||
|
- 需要平台的支持,已知微信小程序支持,其它的没试过,如果可行请告之
|
||||||
|
|
||||||
|
```
|
||||||
|
// 需要在app.vue中下载字体
|
||||||
|
uni.loadFontFace({
|
||||||
|
global:true,
|
||||||
|
scopes: ['native'],
|
||||||
|
family: '自定义字体名称',
|
||||||
|
source: 'url("https://sungd.github.io/Pacifico.ttf")',
|
||||||
|
|
||||||
|
success() {
|
||||||
|
console.log('success')
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
// 然后就可以在插件的css中写font-family: '自定义字体名称'
|
||||||
|
```
|
||||||
|
|
||||||
|
|
||||||
|
### Nvue
|
||||||
|
- 必须为HBX 3.4.11及以上
|
||||||
|
|
||||||
|
|
||||||
|
### 原生小程序
|
||||||
|
|
||||||
|
- 插件里的`painter.js`支持在原生小程序中使用
|
||||||
|
- new Painter 之后在`source`里传入 JSON
|
||||||
|
- 再调用`render`绘制海报
|
||||||
|
- 如需生成图片,请查看微信小程序 cavnas 的[canvasToTempFilePath](https://developers.weixin.qq.com/miniprogram/dev/api/canvas/wx.canvasToTempFilePath.html)
|
||||||
|
|
||||||
|
```html
|
||||||
|
<canvas type="2d" id="painter" style="width: 100%"></canvas>
|
||||||
|
```
|
||||||
|
|
||||||
|
```js
|
||||||
|
import { Painter } from "./painter";
|
||||||
|
page({
|
||||||
|
data: {
|
||||||
|
poster: {
|
||||||
|
css: {
|
||||||
|
width: "750rpx",
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
type: "view",
|
||||||
|
css: {
|
||||||
|
background: "#d2d4c8",
|
||||||
|
paddingTop: "100rpx",
|
||||||
|
},
|
||||||
|
views: [
|
||||||
|
{
|
||||||
|
type: "view",
|
||||||
|
css: {
|
||||||
|
background: "#5f7470",
|
||||||
|
width: "33.33%",
|
||||||
|
height: "100rpx",
|
||||||
|
display: "inline-block",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "view",
|
||||||
|
css: {
|
||||||
|
background: "#889696",
|
||||||
|
width: "33.33%",
|
||||||
|
height: "100rpx",
|
||||||
|
display: "inline-block",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: "view",
|
||||||
|
css: {
|
||||||
|
background: "#b8bdb5",
|
||||||
|
width: "33.33%",
|
||||||
|
height: "100rpx",
|
||||||
|
display: "inline-block",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
},
|
||||||
|
async onLoad() {
|
||||||
|
const res = await this.getCentext();
|
||||||
|
const painter = new Painter(res);
|
||||||
|
// 返回计算布局后的整个内容尺寸
|
||||||
|
const { width, height } = await painter.source(this.data.poster);
|
||||||
|
// 得到计算后的尺寸后 可给canvas尺寸赋值,达到动态响应效果
|
||||||
|
// 渲染
|
||||||
|
await painter.render();
|
||||||
|
},
|
||||||
|
// 获取canvas 2d
|
||||||
|
// 非2d 需要传一个 createImage 方法用于获取图片信息 即把 getImageInfo 的 success 通过 promise resolve 返回
|
||||||
|
getCentext() {
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
wx.createSelectorQuery()
|
||||||
|
.select(`#painter`)
|
||||||
|
.node()
|
||||||
|
.exec((res) => {
|
||||||
|
let { node: canvas } = res[0];
|
||||||
|
resolve({
|
||||||
|
canvas,
|
||||||
|
context: canvas.getContext("2d"),
|
||||||
|
width: canvas.width,
|
||||||
|
height: canvas.height,
|
||||||
|
// createImage: getImageInfo()
|
||||||
|
pixelRatio: 2,
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
},
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
### 旧版(1.6.x)更新
|
||||||
|
|
||||||
|
- 由于 1.8.x 版放弃了以定位的方式,所以 1.6.x 版更新之后要每个样式都加上`position: absolute`
|
||||||
|
- 旧版的 `image` mode 模式被放弃,使用`object-fit`
|
||||||
|
- 旧版的 `isRenderImage` 改成 `is-canvas-to-temp-file-path`
|
||||||
|
- 旧版的 `maxLines` 改成 `line-clamp`
|
||||||
|
|
||||||
|
## API
|
||||||
|
|
||||||
|
### Props
|
||||||
|
|
||||||
|
| 参数 | 说明 | 类型 | 默认值 |
|
||||||
|
| -------------------------- | ------------------------------------------------------------ | ---------------- | ------------ |
|
||||||
|
| board | JSON 方式的海报元素对象集 | <em>object</em> | - |
|
||||||
|
| css | 海报内容最外层的样式,可以理解为`body` | <em>object</em> | 参数请向下看 |
|
||||||
|
| custom-style | canvas 元素的样式 | <em>string</em> | |
|
||||||
|
| hidden | 隐藏画板 | <em>boolean</em> | `false` |
|
||||||
|
| is-canvas-to-temp-file-path | 是否生成图片,在`@success`事件接收图片地址 | <em>boolean</em> | `false` |
|
||||||
|
| after-delay | 生成图片错乱,可延时生成图片 | <em>number</em> | `100` |
|
||||||
|
| type | canvas 类型,对微信头条支付宝小程序可有效,可选值:`2d`,`''` | <em>string</em> | `2d` |
|
||||||
|
| file-type | 生成图片的后缀类型, 可选值:`png`、`jpg` | <em>string</em> | `png` |
|
||||||
|
| path-type | 生成图片路径类型,可选值`url`、`base64` | <em>string</em> | `-` |
|
||||||
|
| pixel-ratio | 生成图片的像素密度,默认为对应手机的像素密度,`nvue`无效 | <em>number</em> | `-` |
|
||||||
|
| hidpi | H5和APP是否使用高清处理 | <em>boolean</em> | `true` |
|
||||||
|
| width | **废弃** 画板的宽度,一般只用于通过内部方法时加上 | <em>number</em> | `` |
|
||||||
|
| height | **废弃** 画板的高度 ,同上 | <em>number</em> | `` |
|
||||||
|
|
||||||
|
### css
|
||||||
|
| 属性名 | 支持的值或类型 | 默认值 |
|
||||||
|
| ----------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | -------- |
|
||||||
|
| (min\max)width | 支持`%`、`rpx`、`px` | - |
|
||||||
|
| height | 同上 | - |
|
||||||
|
| color | `string` | - |
|
||||||
|
| position | 定位,可选值:`absolute`、`fixed` | - |
|
||||||
|
| ↳ left、top、right、bottom | 配合`position`才生效,支持`%`、`rpx`、`px` | - |
|
||||||
|
| margin | 可简写或各方向分别写,如:`margin-top`,支持`auto`、`rpx`、`px` | - |
|
||||||
|
| padding | 可简写或各方向分别写,支持`rpx`、`px` | - |
|
||||||
|
| border | 可简写或各个值分开写:`border-width`、`border-style` 、`border-color`,简写请按顺序写 | - |
|
||||||
|
| line-clamp | `number`,超过行数显示省略号 | - |
|
||||||
|
| vertical-align | 文字垂直对齐,可选值:`bottom`、`top`、`middle` | `middle` |
|
||||||
|
| line-height | 文字行高,支持`rpx`、`px`、`em` | `1.4em` |
|
||||||
|
| font-weight | 文字粗细,可选值:`normal`、`bold` | `normal` |
|
||||||
|
| font-size | 文字大小,`string`,支持`rpx`、`px` | `14px` |
|
||||||
|
| text-decoration | 文本修饰,可选值:`underline` 、`line-through`、`overline` | - |
|
||||||
|
| text-stroke | 文字描边,可简写或各个值分开写,如:`text-stroke-color`, `text-stroke-width` | - |
|
||||||
|
| text-align | 文本水平对齐,可选值:`right` 、`center` | `left` |
|
||||||
|
| display | 框类型,可选值:`block`、`inline-block`、`flex`、`none`,当为`none`时是不渲染该段, `flex`功能简陋。 | - |
|
||||||
|
| flex | 配合 display: flex; 属性定义了在分配多余空间,目前只用为数值如: flex: 1 | - |
|
||||||
|
| align-self | 配合 display: flex; 单个项目垂直轴对齐方式: `flex-start` `flex-end` `center` | `flex-start` |
|
||||||
|
| justify-content | 配合 display: flex; 水平轴对齐方式: `flex-start` `flex-end` `center` | `flex-start` |
|
||||||
|
| align-items | 配合 display: flex; 垂直轴对齐方式: `flex-start` `flex-end` `center` | `flex-start` |
|
||||||
|
| border-radius | 圆角边框,支持`%`、`rpx`、`px` | - |
|
||||||
|
| box-sizing | 可选值:`border-box` | - |
|
||||||
|
| box-shadow | 投影 | - |
|
||||||
|
| background(color) | 支持渐变,但必须写百分比!如:`linear-gradient(,#ff971b 0%, #ff5000 100%)`、`radial-gradient(#0ff 15%, #f0f 60%)`,目前 radial-gradient 渐变的圆心为元素中点,半径为最长边,不支持设置 | - |
|
||||||
|
| background-clip | 文字渐变,配合`background`背景渐变,设置`background-clip: text` 达到文字渐变效果 | - |
|
||||||
|
| background-image | view 元素背景:`url(src)`,若只是设置背景图,请不要设置`background-repeat` | - |
|
||||||
|
| background-repeat | 设置是否及如何重复背景纹理,可选值:`repeat`、`repeat-x`、`repeat-y`、`no-repeat` | `repeat` |
|
||||||
|
| [object-fit](https://developer.mozilla.org/zh-CN/docs/Web/CSS/object-fit/) | 图片元素适应容器方式,类似于`mode`,可选值:`cover`、 `contain`、 `fill`、 `none` | - |
|
||||||
|
| [object-position](https://developer.mozilla.org/zh-CN/docs/Web/CSS/object-position) | 图片的对齐方式,配合`object-fit`使用 | - |
|
||||||
|
|
||||||
|
### 图片填充模式 object-fit
|
||||||
|
|
||||||
|
| 名称 | 含义 |
|
||||||
|
| ------- | ------------------------------------------------------ |
|
||||||
|
| contain | 保持宽高缩放图片,使图片的长边能完全显示出来 |
|
||||||
|
| cover | 保持宽高缩放图片,使图片的短边能完全显示出来,裁剪长边 |
|
||||||
|
| fill | 拉伸图片,使图片填满元素 |
|
||||||
|
| none | 保持图片原有尺寸 |
|
||||||
|
|
||||||
|
### 事件 Events
|
||||||
|
|
||||||
|
| 事件名 | 说明 | 返回值 |
|
||||||
|
| -------- | ---------------------------------------------------------------- | ------ |
|
||||||
|
| success | 生成图片成功,若使用`is-canvas-to-temp-filePath` 可以接收图片地址 | path |
|
||||||
|
| fail | 生成图片失败 | error |
|
||||||
|
| done | 绘制成功 | |
|
||||||
|
| progress | 绘制进度 | number |
|
||||||
|
|
||||||
|
### 暴露函数 Expose
|
||||||
|
| 事件名 | 说明 | 返回值 |
|
||||||
|
| -------- | ---------------------------------------------------------------- | ------ |
|
||||||
|
| render(object) | 渲染器,传入JSON 绘制海报 | promise |
|
||||||
|
| [canvasToTempFilePath](https://uniapp.dcloud.io/api/canvas/canvasToTempFilePath.html#canvastotempfilepath)(object) | 把当前画布指定区域的内容导出生成指定大小的图片,并返回文件临时路径。 | |
|
||||||
|
| canvasToTempFilePathSync(object) | 同步接口,同上 | |
|
||||||
|
|
||||||
|
|
||||||
|
## 常见问题
|
||||||
|
|
||||||
|
- 1、H5 端使用网络图片需要解决跨域问题。
|
||||||
|
- 2、小程序使用网络图片需要去公众平台增加下载白名单!二级域名也需要配!
|
||||||
|
- 3、H5 端生成图片是 base64,有时显示只有一半可以使用原生标签`<IMG/>`
|
||||||
|
- 4、发生保存图片倾斜变形或提示 native buffer exceed size limit 时,使用 pixel-ratio="2"参数,降分辨率。
|
||||||
|
- 5、h5 保存图片不需要调接口,提示用户长按图片保存。
|
||||||
|
- 6、画板不能隐藏,包括`v-if`,`v-show`、`display:none`、`opacity:0`,另外也不要把画板放在弹窗里。如果需要隐藏画板请设置 `custom-style="position: fixed; left: 200%"`
|
||||||
|
- 7、微信小程序真机调试请使用 **真机调试2.0**,不支持1.0。
|
||||||
|
- 8、微信小程序打开调试时可以生但并闭无法生成时,这种情况一般是没有在公众号配置download域名
|
||||||
|
- 9、HBX 3.4.5之前的版本不支持vue3
|
||||||
|
- 10、在微信开发工具上 canvas 层级最高无法zindex,并不影响真机
|
||||||
|
- 11、请不要导入非uni_modules插件
|
||||||
|
- 12、关于QQ小程序 报 Propertyor method"toJSON"is not defined 请把基础库调到 1.50.3
|
||||||
|
- 13、支付宝小程序 IDE 不支持 生成图片 请以真机调试结果为准
|
||||||
|
- 14、返回值为字符串 `data:,` 大概是尺寸超过限制,设置 pixel-ratio="2"
|
||||||
|
- 华为手机 APP 上无法生成图片,请使用 HBX2.9.11++(已过时,忽略这条)
|
||||||
|
- IOS APP 请勿使用 HBX2.9.3.20201014 的版本!这个版本无法生成图片。(已过时,忽略这条)
|
||||||
|
- 苹果微信 7.0.20 存在闪退和图片无法 onload 为微信 bug(已过时,忽略这条)
|
||||||
|
- 微信小程序 IOS 旧接口 如父级设置圆角,子级也设会导致子级的失效,为旧接口BUG。
|
||||||
|
- 微信小程序 安卓 旧接口 如使用图片必须加背景色,为旧接口BUG。
|
||||||
|
- 微信小程序 安卓端 [图片可能在首次可以加载成功,再次加载会不触发任何事件](https://developers.weixin.qq.com/community/develop/doc/000ee2b8dacf4009337f51f4556800?highLine=canvas%25202d%2520createImage),临时解决方法是给图片加个时间戳
|
||||||
|
## 打赏
|
||||||
|
|
||||||
|
如果你觉得本插件,解决了你的问题,赠人玫瑰,手留余香。
|
||||||
|
|
||||||
|

|
||||||
|

|
||||||
|
|
@ -432,6 +432,14 @@ export const util = {
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 点击标题栏按钮
|
||||||
|
* @param {*} button
|
||||||
|
*/
|
||||||
|
clickTitlePopupButton(button) {
|
||||||
|
button.click()
|
||||||
|
},
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 页面跳转
|
* 页面跳转
|
||||||
* @param {*} url
|
* @param {*} url
|
||||||
|
|
@ -448,15 +456,6 @@ export const util = {
|
||||||
*/
|
*/
|
||||||
goBack() {
|
goBack() {
|
||||||
uni.navigateBack();
|
uni.navigateBack();
|
||||||
},
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* 点击标题弹出按钮
|
|
||||||
* @param e
|
|
||||||
*/
|
|
||||||
clickTitlePopupButton(button) {
|
|
||||||
button.click()
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
||||||