quanyawei
2023-10-31 5fecc9e46d448df3de987504440c4fdd582f858e
fix:小程序秒级数据和登录授权
7 files deleted
27 files added
8 files modified
3417 ■■■■■ changed files
.env.dev.js 2 ●●● patch | view | raw | blame | history
manifest.json 148 ●●●● patch | view | raw | blame | history
pages.json 7 ●●●●● patch | view | raw | blame | history
pages/actionChange/agencyPage/index.vue 2 ●●● patch | view | raw | blame | history
pages/actionChange/components/rectificationInfor.vue 7 ●●●● patch | view | raw | blame | history
pages/actionChange/myInfor/index.vue 2 ●●●●● patch | view | raw | blame | history
pages/actionChange/newPage/index.vue 4 ●●●● patch | view | raw | blame | history
pages/actionChange/secondLevelData/index.vue 596 ●●●●● patch | view | raw | blame | history
pages/actionChange/secondLevelData/parameter.json 44 ●●●●● patch | view | raw | blame | history
pages/login/login.vue 18 ●●●●● patch | view | raw | blame | history
static/img/headSculpture.png patch | view | raw | blame | history
static/img/shouye.png patch | view | raw | blame | history
static/img/shouyeClick.png patch | view | raw | blame | history
static/img/wode-.png patch | view | raw | blame | history
static/img/wodeClick.png patch | view | raw | blame | history
static/img/xinjian.png patch | view | raw | blame | history
static/img/xinjianClick.png patch | view | raw | blame | history
uni_modules/uni-popup/changelog.md 68 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup-dialog/keypress.js 45 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup-dialog/uni-popup-dialog.vue 275 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup-message/uni-popup-message.vue 143 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup-share/uni-popup-share.vue 187 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup/i18n/en.json 7 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup/i18n/index.js 8 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup/i18n/zh-Hans.json 7 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup/i18n/zh-Hant.json 7 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup/keypress.js 45 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup/popup.js 26 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/components/uni-popup/uni-popup.vue 473 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/package.json 87 ●●●●● patch | view | raw | blame | history
uni_modules/uni-popup/readme.md 17 ●●●●● patch | view | raw | blame | history
uni_modules/uni-transition/changelog.md 22 ●●●●● patch | view | raw | blame | history
uni_modules/uni-transition/components/uni-transition/createAnimation.js 131 ●●●●● patch | view | raw | blame | history
uni_modules/uni-transition/components/uni-transition/uni-transition.vue 286 ●●●●● patch | view | raw | blame | history
uni_modules/uni-transition/package.json 84 ●●●●● patch | view | raw | blame | history
uni_modules/uni-transition/readme.md 11 ●●●●● patch | view | raw | blame | history
uni_modules/ws-wx-privacy/changelog.md 22 ●●●●● patch | view | raw | blame | history
uni_modules/ws-wx-privacy/components/ws-wx-privacy/util.js 51 ●●●●● patch | view | raw | blame | history
uni_modules/ws-wx-privacy/components/ws-wx-privacy/ws-wx-privacy.vue 287 ●●●●● patch | view | raw | blame | history
uni_modules/ws-wx-privacy/package.json 85 ●●●●● patch | view | raw | blame | history
uni_modules/ws-wx-privacy/readme.md 122 ●●●●● patch | view | raw | blame | history
utils/websoket.js 91 ●●●●● patch | view | raw | blame | history
.env.dev.js
@@ -1,6 +1,6 @@
const config = {
    // baseUrl: "http://120.26.43.34:8081/api/", //测试
    baseUrl: "https://qx.7drlb.com/api", //生产
    // baseUrl: "http://192.168.0.9:8081/api", //生产
    // baseUrl: "http://192.168.0.12:8081/api", //生产
};
module.exports = config;
manifest.json
@@ -1,75 +1,75 @@
{
    "name" : "test",
    "appid" : "__UNI__93C3197",
    "description" : "",
    "versionName" : "1.0.0",
    "versionCode" : "100",
    "transformPx" : false,
    /* 5+App特有相关 */
    "app-plus" : {
        "usingComponents" : true,
        "nvueStyleCompiler" : "uni-app",
        "compilerVersion" : 3,
        "splashscreen" : {
            "alwaysShowBeforeRender" : true,
            "waiting" : true,
            "autoclose" : true,
            "delay" : 0
        },
        /* 模块配置 */
        "modules" : {},
        /* 应用发布信息 */
        "distribute" : {
            /* android打包配置 */
            "android" : {
                "permissions" : [
                    "<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>",
                    "<uses-permission android:name=\"android.permission.VIBRATE\"/>",
                    "<uses-permission android:name=\"android.permission.READ_LOGS\"/>",
                    "<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>",
                    "<uses-feature android:name=\"android.hardware.camera.autofocus\"/>",
                    "<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.CAMERA\"/>",
                    "<uses-permission android:name=\"android.permission.GET_ACCOUNTS\"/>",
                    "<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>",
                    "<uses-permission android:name=\"android.permission.FLASHLIGHT\"/>",
                    "<uses-feature android:name=\"android.hardware.camera\"/>",
                    "<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>"
                ]
            },
            /* ios打包配置 */
            "ios" : {},
            /* SDK配置 */
            "sdkConfigs" : {}
        }
    },
    /* 快应用特有相关 */
    "quickapp" : {},
    /* 小程序特有相关 */
    "mp-weixin" : {
        "appid" : "wx41f4c3c007545088",
        "setting" : {
            "urlCheck" : false,
            "es6" : true,
            "postcss" : true,
            "minified" : true
        },
        "usingComponents" : true
    },
    "mp-alipay" : {
        "usingComponents" : true
    },
    "mp-baidu" : {
        "usingComponents" : true
    },
    "mp-toutiao" : {
        "usingComponents" : true
    },
    "uniStatistics" : {
        "enable" : false
    },
    "vueVersion" : "2"
}
    "name": "test",
    "appid": "__UNI__93C3197",
    "description": "",
    "versionName": "1.0.0",
    "versionCode": "100",
    "transformPx": false,
    /* 5+App特有相关 */
    "app-plus": {
        "usingComponents": true,
        "nvueStyleCompiler": "uni-app",
        "compilerVersion": 3,
        "splashscreen": {
            "alwaysShowBeforeRender": true,
            "waiting": true,
            "autoclose": true,
            "delay": 0
        },
        /* 模块配置 */
        "modules": {},
        /* 应用发布信息 */
        "distribute": {
            /* android打包配置 */
            "android": {
                "permissions": ["<uses-permission android:name=\"android.permission.CHANGE_NETWORK_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.MOUNT_UNMOUNT_FILESYSTEMS\"/>",
                    "<uses-permission android:name=\"android.permission.VIBRATE\"/>",
                    "<uses-permission android:name=\"android.permission.READ_LOGS\"/>",
                    "<uses-permission android:name=\"android.permission.ACCESS_WIFI_STATE\"/>",
                    "<uses-feature android:name=\"android.hardware.camera.autofocus\"/>",
                    "<uses-permission android:name=\"android.permission.ACCESS_NETWORK_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.CAMERA\"/>",
                    "<uses-permission android:name=\"android.permission.GET_ACCOUNTS\"/>",
                    "<uses-permission android:name=\"android.permission.READ_PHONE_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.CHANGE_WIFI_STATE\"/>",
                    "<uses-permission android:name=\"android.permission.WAKE_LOCK\"/>",
                    "<uses-permission android:name=\"android.permission.FLASHLIGHT\"/>",
                    "<uses-feature android:name=\"android.hardware.camera\"/>",
                    "<uses-permission android:name=\"android.permission.WRITE_SETTINGS\"/>"
                ]
            },
            /* ios打包配置 */
            "ios": {},
            /* SDK配置 */
            "sdkConfigs": {}
        }
    },
    /* 快应用特有相关 */
    "quickapp": {},
    /* 小程序特有相关 */
    "mp-weixin": {
        "appid": "wxf95fb77e0b1f8c09",
        "__usePrivacyCheck__": true,
        "setting": {
            "urlCheck": false,
            "es6": true,
            "postcss": true,
            "minified": true
        },
        "usingComponents": true
    },
    "mp-alipay": {
        "usingComponents": true
    },
    "mp-baidu": {
        "usingComponents": true
    },
    "mp-toutiao": {
        "usingComponents": true
    },
    "uniStatistics": {
        "enable": false
    },
    "vueVersion": "2"
}
pages.json
@@ -44,6 +44,13 @@
                "disableSwipeBack": true,
                "navigationStyle": "custom"
            }
        }, {
            "path": "pages/actionChange/secondLevelData/index",
            "style": {
                "navigationBarTitleText": "秒级数据",
                "enablePullDownRefresh": false,
                "disableSwipeBack": true
            }
        }
    ]
    // "tabBar": {
pages/actionChange/agencyPage/index.vue
@@ -349,7 +349,7 @@
                    id: e.allocationId
                }).then(res => {
                    let data = res.data
                    data.pageState = data.state === 50 || data.state === 40 ? 'view' : pageState
                    data.pageState = e.stateName === '待处理' ? pageState : 'view'
                    let myData = JSON.stringify(data)
                    uni.navigateTo({
                        url: '/pages/actionChange/workOrderDetails/index?infor=' + myData,
pages/actionChange/components/rectificationInfor.vue
@@ -145,8 +145,11 @@
                    let name = item.fileType === 1 ? 'name.png' : 'name.mp4'
                    this.fileList.push(`${this.baseUrl}/file/preview/${item.fileId}?${name}`) // 原图
                })
                console.log('this.fileList', this.basicInfor.fileBaseList)
                console.log('this.fileList', this.fileList)
            }
            let userInfor = JSON.parse(uni.getStorageSync('userInfor') || '{}')
            console.log('userInfor', userInfor)
            if (this.basicInfor.pageState === 'edit') {
                this.form.changeName = userInfor.userName
            }
        },
        methods: {
pages/actionChange/myInfor/index.vue
@@ -11,6 +11,8 @@
                <u-cell icon="account-fill" title="账号" :value="userInfor.account"></u-cell>
                <u-cell icon="integral-fill" title="责任单位" :value="unitName"></u-cell>
                <u-cell icon="phone-fill" title="手机号" :value="userInfor.mobile"></u-cell>
                <u-cell icon="clock-fill" title="查看秒级数据" v-if="userInfor.device===1" isLink
                    url="/pages/actionChange/secondLevelData/index"></u-cell>
            </u-cell-group>
        </view>
        <view class="bunts">
pages/actionChange/newPage/index.vue
@@ -265,8 +265,8 @@
                    this.form.escalationUnitId = data
                    this.sumbitForm.escalationUnitId = userInfor.unitId
                }
                this.form.escalationName = userInfor.name || ''
                this.sumbitForm.escalationName = userInfor.name || ''
                this.form.escalationName = userInfor.userName || ''
                this.sumbitForm.escalationName = userInfor.userName || ''
                this.form.investigationType = 1
                this.sumbitForm.investigationType = 1
                console.log('this.sumbitForm', this.sumbitForm)
pages/actionChange/secondLevelData/index.vue
New file
@@ -0,0 +1,596 @@
<template>
    <view class="rootBg">
        <view class="">
            <view class="ridoa">
                <u-radio-group v-model="value" placement="row" :size='radioOption.size'
                    :labelColor='radioOption.labelColor' :labelSize='radioOption.labelSize' @change='changeSearchType'>
                    <u-radio activeColor="#19be6b" label="通过站点查询" name="0" :labelSize='radioOption.labelSize'
                        :labelColor='radioOption.labelColor'></u-radio>
                    <u-radio activeColor="#19be6b" label="通过设查号询" name="1" :labelSize='radioOption.labelSize'
                        :labelColor='radioOption.labelColor'></u-radio>
                </u-radio-group>
            </view>
            <view class="" v-if="value==='1'">
                <u-input placeholder="请输入Mac号" :placeholderStyle='placeholderStyle' color="#fff" v-model='macName'>
                    <template slot="suffix">
                        <u-button @tap="searchMacList" text="查询" type="success" size="small"></u-button>
                    </template>
                </u-input>
            </view>
            <view class="" v-else>
                <u-form ref="uForm" label-width="80" :model="form">
                    <u-form-item border-bottom @click="showeEscalationTime = true ;hideKeyboard('orgName','orgId')">
                        <u-input v-model="form.orgName" disabled disabled-color="#ffffff" placeholder="请选择组织"
                            type="select" />
                    </u-form-item>
                    <u-form-item border-bottom @click="selectSite">
                        <u-input v-model="form.siteName" disabled disabled-color="#ffffff" placeholder="请选择站点"
                            type="select" />
                    </u-form-item>
                </u-form>
                <u-picker :show="showeEscalationTime" :columns="columns" @confirm="selectBack" keyName="name"
                    @cancel="showeEscalationTime = false"></u-picker>
            </view>
        </view>
        <view v-if="wsData2">
            <u-cell title="风向:" :value="windDirData"></u-cell>
        </view>
        <view class="boxPadd">
            <u-list v-if="showData.length>0">
                <u-list-item v-for="(item, index) in showData" :key="index">
                    <u-row customStyle="margin-bottom: 10px">
                        <u-col span="3">
                            <view>{{ item.sensorCode | sensorFilter}}</view>
                        </u-col>
                        <u-col span="6" textAlign="right">
                            <u-line-progress :percentage="item.alarm" :activeColor="item.colour"></u-line-progress>
                        </u-col>
                        <u-col span="3" textAlign="right">
                            <view>{{ item.value || ''}} </view>
                        </u-col>
                    </u-row>
                </u-list-item>
            </u-list>
        </view>
        <u-toast ref="uToast"></u-toast>
        <u-modal :show="showModel" title='请选择Mac号' showCancelButton @cancel='showModel=false' @confirm='selectMac'>
            <view class="slot-content">
                <u-radio-group v-model="modeleMacSelect" iconPlacement="right" placement='column' borderBottom='true'>
                    <view v-for="(item,index) in macList" :key="index" class="ridioBox">
                        <u-radio activeColor="#19be6b" :label="item.mac" :name="item.mac"></u-radio>
                    </view>
                </u-radio-group>
            </view>
        </u-modal>
    </view>
</template>
<script>
    import json from '@/pages/actionChange/secondLevelData/parameter.json'
    import wsRequest from '@/utils/websoket.js'
    export default {
        data() {
            return {
                modeleMacSelect: '',
                value: '0',
                showModel: false,
                showeEscalationTime: false,
                columns: [],
                form: {
                    orgName: '',
                    orgId: '',
                    siteName: '',
                    siteId: ''
                },
                selectForm: {
                    name: '',
                    id: ''
                },
                macList: [],
                radioOption: {
                    size: '20px',
                    labelColor: '#fff',
                    labelSize: '18px'
                },
                placeholderStyle: 'color: #fff',
                macName: '',
                defaultMonitorItems: [],
                chartSensorKey: [],
                coreMonitorItems: [],
                fixedMonitorItems: [],
                totalArray: [],
                alarmLevel: null,
                aqi: 0,
                baseUrl: '',
                windDir: 'null',
                windDeg: 180,
                wsData2: null,
                // 报警进度条
                alarmProgress: {},
                // 报警颜色
                alarmColour: {},
                // 报警圆环图片
                alarmBg: {},
                timer: '',
                isLoading: true
            }
        },
        filters: {
            // 过滤器替换websocket实时数据的key值
            sensorFilter: function(value) {
                if (!value) return ''
                return json[value]
            }
        },
        computed: {
            windDirData() {
                if (this.wsData2) {
                    console.log('this.windDir', this.windDir)
                    return `${this.windDir} (${this.wsData2.a01008})`
                } else {
                    return ''
                }
            },
            showData() {
                return [...this.defaultMonitorItems, ...this.coreMonitorItems]
            }
        },
        created() {
            this.getOrgs()
        },
        onUnload() {
            let that = this
            this.socketTask.close()
            that.isLoading = false
            that.timer = ''
        },
        methods: {
            getData() {
                this.queryAlarmByMac()
                this.getHourlyAqi()
                this.sensorLayout()
            },
            changeSearchType() {
                this.macName = ''
            },
            selectMac() {
                this.macName = this.modeleMacSelect
                this.showModel = false
                if (this.socketTask) {
                    this.socketTask.onClose()
                }
                this.getData()
            },
            searchMacList() {
                this.$http.httpGet('/AppDevice/fuzzySearch', {
                    mac: this.macName
                }).then(res => {
                    this.showModel = true
                    if (res.data && res.data.length > 10) {
                        this.macList = res.data.slice(0, 10)
                    }
                }).catch(res => {
                    uni.$u.toast(res)
                })
            },
            selectSite() {
                if (this.form.orgId === '') {
                    uni.showToast({
                        title: '请先选择组织!',
                        icon: 'none',
                    })
                } else {
                    this.selectForm = {
                            name: 'siteName',
                            id: 'siteId'
                        },
                        this.getSites()
                }
            },
            hideKeyboard(name, id) {
                this.selectForm = {
                    'name': name,
                    'id': id
                }
                this.getOrgs()
            },
            selectBack(e) {
                if (this.selectForm.name === 'orgName') {
                    this.form = {
                        orgName: '',
                        orgId: '',
                        siteName: '',
                        siteId: ''
                    }
                } else {
                    this.macName = e.value[0].mac
                    if (this.socketTask) {
                        this.socketTask.onClose()
                    }
                    this.getData()
                }
                this.form[this.selectForm.name] = e.value[0].name
                this.form[this.selectForm.id] = e.value[0].id
                this.showeEscalationTime = false
            },
            getOrgs() {
                this.$http.httpGet('/AppDevice/selectOrganization').then(res => {
                    this.columns = []
                    this.columns.push(res.data)
                }).catch(res => {
                    uni.$u.toast(res)
                })
            },
            getSites() {
                let params = {
                    organizationId: this.form.orgId
                }
                this.$http.httpGet('/AppDevice/selectDevice', params).then(res => {
                    console.log(res)
                    this.columns = []
                    this.columns.push(res.data)
                    this.showeEscalationTime = true
                }).catch(res => {
                    uni.$u.toast(res)
                    console.log(res)
                })
            },
            sensorLayout() {
                let params = {
                    mac: this.macName
                }
                this.$http.httpGet('/organizationLayout/getLayoutByMac', params).then(res => {
                    this.chartSensorKey = res.data.chartSensorKey
                    this.coreMonitorItems = res.data.coreMonitorItems
                    this.defaultMonitorItems = res.data.defaultMonitorItems
                    this.fixedMonitorItems = res.data.fixedMonitorItems
                    this.totalArray.push(...this.coreMonitorItems, ...this.defaultMonitorItems, ...this
                        .fixedMonitorItems)
                    this.getSensorMonthAvg()
                    this.connectSocket()
                }).catch(res => {
                    uni.$u.toast(res)
                    console.log(res)
                })
            },
            // 请求单台设备某参数月平均值
            getSensorMonthAvg() {
                let params = {
                    sensorCode: this.chartSensorKey[0].sensorCode,
                    mac: this.macName
                }
                this.$http.httpGet('/deviceInfo/getMonthAvg', params).then(res => {
                    this.average = res.data.avg
                }).catch(res => {
                    uni.$u.toast(res)
                    console.log(res)
                })
            },
            // 设备因子报警值
            queryAlarmByMac() {
                let params = {
                    mac: this.macName
                }
                this.$http.httpGet('/deviceInfo/queryAlarmByMac', params).then(res => {
                    this.alarmLevel = res.data.alarmLevel
                }).catch(res => {
                    uni.$u.toast(res)
                    console.log(res)
                })
            },
            // 请求单台设备一个小时
            getHourlyAqi() {
                let params = {
                    mac: this.macName
                }
                this.$http.httpGet('/deviceInfo/getHourlyAqi', params).then(res => {
                    this.aqi = res.data.AQI
                }).catch(res => {
                    uni.$u.toast(res)
                    console.log(res)
                })
            },
            handlerMsg(msg) {
                this.wsData2 = JSON.parse(msg.data)
                if (this.wsData2.dustld) {
                    this.wsData2.dustld = this.wsData2.dustld.replace(/g/, 'ug')
                }
                // 风向算法
                if (this.wsData2.a01008) {
                    var windDirs = Number(this.wsData2.a01008.substr(0, this.wsData2.a01008.length - 1))
                    this.windDeg = windDirs
                    if (windDirs === 0) {
                        this.windDir = '北风'
                    } else if (windDirs > 0 && windDirs < 90) {
                        this.windDir = '东北风'
                    } else if (windDirs === 90) {
                        this.windDir = '东风'
                    } else if (windDirs > 90 && windDirs < 180) {
                        this.windDir = '东南风'
                    } else if (windDirs === 180) {
                        this.windDir = '南风'
                    } else if (windDirs > 180 && windDirs < 270) {
                        this.windDir = '西南风'
                    } else if (windDirs === 270) {
                        this.windDir = '西风'
                    } else if (windDirs > 270 && windDirs < 360) {
                        this.windDir = '西北风'
                    }
                }
                this.alarmColour = {}
                this.alarmProgress = {}
                this.alarmBg = {}
                // 报警等级
                for (const key in this.wsData2) {
                    if (this.chartSensorKey && key === this.chartSensorKey[0].sensorCode) {
                        this.chartSensorKey[0].value = this.wsData2[key]
                    }
                    for (const keys in this.alarmLevel) {
                        // 遍历进度条颜色
                        if (key === keys && JSON.parse(this.alarmLevel[keys])) {
                            if (parseFloat(this.wsData2[key]) >= parseFloat(JSON.parse(this.alarmLevel[keys])[5])) {
                                this.alarmColour[keys] = '#000000'
                                this.alarmBg[keys] = 5
                            } else if (parseFloat(this.wsData2[key]) >= parseFloat(JSON.parse(this.alarmLevel[keys])[4])) {
                                this.alarmColour[keys] = '#c00261'
                                this.alarmBg[keys] = 4
                            } else if (parseFloat(this.wsData2[key]) >= parseFloat(JSON.parse(this.alarmLevel[keys])[3])) {
                                this.alarmColour[keys] = '#fc0101'
                                this.alarmBg[keys] = 3
                            } else if (parseFloat(this.wsData2[key]) >= parseFloat(JSON.parse(this.alarmLevel[keys])[2])) {
                                this.alarmColour[keys] = '#ff8202'
                                this.alarmBg[keys] = 2
                            } else if (parseFloat(this.wsData2[key]) >= parseFloat(JSON.parse(this.alarmLevel[keys])[1])) {
                                this.alarmColour[keys] = '#fdff00'
                                this.alarmBg[keys] = 1
                            } else {
                                this.alarmColour[keys] = '#00ff01'
                                this.alarmBg[keys] = 0
                            }
                        } else if (key === keys) {
                            this.alarmColour[keys] = '#00ff01'
                            this.alarmBg[keys] = 0
                        }
                        // 遍历进度条百分比
                        if (key === keys && JSON.parse(this.alarmLevel[keys])) {
                            if (JSON.parse(this.alarmLevel[keys])[5]) {
                                const percentage = Math.round(
                                    (parseFloat(this.wsData2[key]) / parseFloat(JSON.parse(this.alarmLevel[keys])[
                                    5])) * 100)
                                this.alarmProgress[keys] = percentage
                            } else if (JSON.parse(this.alarmLevel[keys])[4]) {
                                const percentage = Math.round(
                                    (parseFloat(this.wsData2[key]) / parseFloat(JSON.parse(this.alarmLevel[keys])[
                                    4])) * 100)
                                this.alarmProgress[keys] = percentage
                            } else if (JSON.parse(this.alarmLevel[keys])[3]) {
                                const percentage = Math.round(
                                    (parseFloat(this.wsData2[key]) / parseFloat(JSON.parse(this.alarmLevel[keys])[
                                    3])) * 100)
                                this.alarmProgress[keys] = percentage
                            } else if (JSON.parse(this.alarmLevel[keys])[2]) {
                                const percentage = Math.round(
                                    (parseFloat(this.wsData2[key]) / parseFloat(JSON.parse(this.alarmLevel[keys])[
                                    2])) * 100)
                                this.alarmProgress[keys] = percentage
                            } else if (JSON.parse(this.alarmLevel[keys])[1]) {
                                const percentage = Math.round(
                                    (parseFloat(this.wsData2[key]) / parseFloat(JSON.parse(this.alarmLevel[keys])[
                                    1])) * 100)
                                this.alarmProgress[keys] = percentage
                            } else if (JSON.parse(this.alarmLevel[keys])[0]) {
                                const percentage = Math.round(
                                    (parseFloat(this.wsData2[key]) / parseFloat(JSON.parse(this.alarmLevel[keys])[
                                    0])) * 100)
                                this.alarmProgress[keys] = percentage
                            }
                        } else if (key === keys) {
                            this.alarmProgress[keys] = 0
                        }
                        if (this.alarmProgress[keys] > 100) {
                            this.alarmProgress[keys] = 100
                        }
                    }
                }
                // 实时数据改变合并数组的值
                for (let i = 0; i < this.totalArray.length; i++) {
                    for (const key in this.wsData2) {
                        if (key !== 'time') {
                            // 获取数据的小数
                            var tempDecimal = this.wsData2[key].replace(/[^\d.]/g, '').split('.')[1]
                            if (this.totalArray[i].sensorCode === key) {
                                // 臭气和湿度保留整数
                                if (this.totalArray[i].sensorCode === 'a19002') {
                                    this.totalArray[i].value = parseInt(this.wsData2[key])
                                } else if (this.totalArray[i].sensorCode === 'a01002') {
                                    this.totalArray[i].value = parseInt(this.wsData2[key]) + '%'
                                    // 除了TVOCQ其他因子数据小数是零的就去掉小数
                                } else if (this.totalArray[i].sensorCode !== 'a99054' && Number(tempDecimal) === 0) {
                                    this.totalArray[i].value = parseInt(this.wsData2[key]) + this.wsData2[key].split(' ')[
                                        1]
                                } else {
                                    this.totalArray[i].value = this.wsData2[key]
                                }
                            }
                        }
                    }
                    for (const key in this.alarmProgress) {
                        if (this.totalArray[i].sensorCode === key) {
                            this.totalArray[i].alarm = this.alarmProgress[key]
                        }
                    }
                    console.log('this.totalArray', this.totalArray)
                    console.log('this.alarmColour', this.alarmColour)
                    for (const key in this.alarmColour) {
                        if (this.totalArray[i].sensorCode === key) {
                            this.totalArray[i].colour = this.alarmColour[key]
                        }
                    }
                    for (const key in this.alarmBg) {
                        if (this.totalArray[i].sensorCode === key) {
                            this.totalArray[i].bg = this.alarmBg[key]
                        }
                    }
                }
            },
            //连接websocket
            connectSocket() {
                if (!this.isLoading) {
                    this.socketTask.close()
                    this.socketTask = null
                }
                this.baseUrl = this.$storage.get('baseUrl')
                let that = this
                let socketUrl = this.baseUrl + '/singleDevice/' + this.macName
                socketUrl = socketUrl.replace('https', 'wss').replace('http', 'ws')
                console.log('调用连接websocket')
                this.socketTask = uni.connectSocket({
                    url: socketUrl,
                    success(res) {
                        console.log('websocket连接成功')
                        that.isLoading = true
                    },
                    fail(err) {
                        console.log('报错', err)
                    }
                }, )
                this.socketTask.onOpen(function(res) {
                    console.log('WebSocket连接已打开!')
                    that.isLoading = true
                    // that.getStatus()
                    // that.heart()
                })
                this.socketTask.onMessage(function(res) {
                    console.log('收到服务器内容:' + res.data)
                    that.handlerMsg(res) //这里是对获取到的数据进行操作
                })
                this.socketTask.onError(function(res) {
                    console.log('WebSocket连接打开失败,请检查!')
                    console.log(res)
                    // this.isSuccess = false
                    // that.connectSocket()
                    //进入重新连接
                    that.reconnect()
                })
                // // 监听连接关闭 -
                this.socketTask.onClose(e => {
                    console.log('WebSocket连接关闭!')
                    clearInterval(that.timer)
                    that.timer = ''
                    if (!that.isClose) {
                        that.reconnect()
                    }
                })
                console.log(this.socketTask)
            },
            //进入重新连接
            reconnect() {
                console.log('进入断线重连')
                this.socketTask.close()
                this.socketTask = null
                if (this.isLoading) {
                    this.connectSocket()
                }
            },
            //发送消息
            sendSocketMessage(msg) {
                console.log('发送信息')
                console.log(msg)
                return new Promise((reslove, reject) => {
                    this.socketTask.send({
                        data: msg,
                        success(res) {
                            console.log('发送成功')
                            reslove(res)
                        },
                        fail(res) {
                            console.log('发送失败')
                            console.log(res)
                            reject(res)
                        }
                    })
                })
            },
            //心跳
            heart() {
                let that = this
                clearInterval(this.timer)
                this.timer = ''
                let msg = {
                    'type': 'heartbeat',
                }
                this.timer = setInterval(() => {
                    that.sendSocketMessage(JSON.stringify(msg)).then(res => {
                        console.log('心跳成功')
                    }).catch(res => {
                        console.log('发送失败')
                        console.log((res))
                    })
                }, 2000)
            },
            beforeDestroy() {
                if (this.socketTask) {
                    this.socketTask.close()
                    this.socketTask = null
                }
                if (this.timer) {
                    clearInterval(this.timer) // 在Vue实例销毁前,清除我们的定时器
                }
            },
        }
    }
</script>
<style scoped lang="scss">
    .rootBg {
        min-height: 100%;
        color: #fff;
        /* 背景图垂直、水平均居中 */
        background-position: center center;
        /* 背景图不平铺 */
        background-repeat: no-repeat;
        /* 当内容高度大于图片高度时,背景图像的位置相对于viewport固定 */
        background-attachment: fixed;
        /* 让背景图基于容器大小伸缩 */
        background-size: cover;
        background-image: url('');
    }
    .boxPadd {
        padding: 5px 10px;
    }
    /deep/.u-line-progress__text {
        color: #000 !important;
    }
    /deep/.u-radio-group--row {
        justify-content: center;
    }
    .ridoa {
        text-align: center;
        padding: 10px 0px;
    }
    /deep/.placeholderColor {
        color: #fff;
    }
    /deep/.u-form {
        display: flex;
        justify-content: space-evenly;
    }
    .slot-content {
        width: 80%;
    }
    /deep/.u-cell__body,
    /deep/.u-cell__title-text,
    /deep/.u-cell__value {
        color: #fff !important;
    }
    .ridioBox {
        margin-bottom: 10px;
    }
</style>
pages/actionChange/secondLevelData/parameter.json
New file
@@ -0,0 +1,44 @@
{
    "flylon": "经度",
    "a00e12": "光照强度",
    "a00e34": "TSP",
    "a00e13": "噪音",
    "a21005": "一氧化碳",
    "a21004": "二氧化氮",
    "a21026": "二氧化硫",
    "a21028": "硫化氢",
    "flyhig": "高度",
    "a21001": "氨气",
    "a40002": "苯乙烯",
    "a21022": "氯气",
    "a00e19": "二氧化碳",
    "a40001": "烟气流量",
    "a01001": "温度",
    "a25005": "二甲苯",
    "a05024": "臭氧",
    "a21024": "氯化氢",
    "a40003": "环氧乙烷",
    "a25003": "甲苯",
    "a01002": "湿度",
    "a25002": "苯",
    "a23001": "酚类",
    "a01007": "风速",
    "a01006": "气压",
    "a01008": "风向",
    "a30001": "甲醇",
    "a19002": "臭气浓度",
    "a19001": "氧气",
    "a34002": "PM10",
    "a34004": "PM2.5",
    "dustld": "尘负荷",
    "PORRST": "开机标识",
    "a00e03": "颗粒物0.3",
    "a00e04": "颗粒物2.5",
    "a06001": "雨量",
    "a99054": "TVOC",
    "a35e21": "核辐射",
    "a01011": "烟气流速",
    "a24088": "非甲烷总烃",
    "flylat": "纬度",
    "a31001": "甲醛"
}
pages/login/login.vue
@@ -29,6 +29,7 @@
                <u-button @click="submit">登录</u-button>
            </view>
        </view>
        <ws-wx-privacy id="privacy-popup" enableAutoProtocol></ws-wx-privacy>
    </view>
</template>
<script>
@@ -44,6 +45,7 @@
    export default {
        data() {
            return {
                titlePrivacy: '用户隐私保护提示',
                labelStyle: {
                    color: '#fff'
                },
@@ -81,9 +83,23 @@
            uni.hideHomeButton()
        },
        methods: {
            doRequire() {
                uni.requirePrivacyAuthorize({
                    success: () => {
                        console.log('同意')
                        // 用户同意授权
                        // 继续小程序逻辑
                        this.getlogin()
                    },
                    fail: () => {
                        console.log('拒绝')
                    }, // 用户拒绝授权
                    complete: () => {}
                })
            },
            submit() {
                this.$refs.uForm.validate().then(res => {
                    this.getlogin()
                    this.doRequire()
                }).catch(errors => {})
            },
            //登录
static/img/headSculpture.png
Binary files differ
static/img/shouye.png
Binary files differ
static/img/shouyeClick.png
Binary files differ
static/img/wode-.png
Binary files differ
static/img/wodeClick.png
Binary files differ
static/img/xinjian.png
Binary files differ
static/img/xinjianClick.png
Binary files differ
uni_modules/uni-popup/changelog.md
New file
@@ -0,0 +1,68 @@
## 1.8.3(2023-04-17)
- 修复 uni-popup 重复打开时的 bug
## 1.8.2(2023-02-02)
- uni-popup-dialog 组件新增 inputType 属性
## 1.8.1(2022-12-01)
- 修复 nvue 下 v-show 报错
## 1.8.0(2022-11-29)
- 优化 主题样式
## 1.7.9(2022-04-02)
- 修复 弹出层内部无法滚动的bug
## 1.7.8(2022-03-28)
- 修复 小程序中高度错误的bug
## 1.7.7(2022-03-17)
- 修复 快速调用open出现问题的Bug
## 1.7.6(2022-02-14)
- 修复 safeArea 属性不能设置为false的bug
## 1.7.5(2022-01-19)
- 修复 isMaskClick 失效的bug
## 1.7.4(2022-01-19)
- 新增 cancelText \ confirmText 属性 ,可自定义文本
- 新增 maskBackgroundColor 属性 ,可以修改蒙版颜色
- 优化 maskClick属性 更新为 isMaskClick ,解决微信小程序警告的问题
## 1.7.3(2022-01-13)
- 修复 设置 safeArea 属性不生效的bug
## 1.7.2(2021-11-26)
- 优化 组件示例
## 1.7.1(2021-11-26)
- 修复 vuedoc 文字错误
## 1.7.0(2021-11-19)
- 优化 组件UI,并提供设计资源,详见:[https://uniapp.dcloud.io/component/uniui/resource](https://uniapp.dcloud.io/component/uniui/resource)
- 文档迁移,详见:[https://uniapp.dcloud.io/component/uniui/uni-popup](https://uniapp.dcloud.io/component/uniui/uni-popup)
## 1.6.2(2021-08-24)
- 新增 支持国际化
## 1.6.1(2021-07-30)
- 优化 vue3下事件警告的问题
## 1.6.0(2021-07-13)
- 组件兼容 vue3,如何创建vue3项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834)
## 1.5.0(2021-06-23)
- 新增 mask-click 遮罩层点击事件
## 1.4.5(2021-06-22)
- 修复 nvue 平台中间弹出后,点击内容,再点击遮罩无法关闭的Bug
## 1.4.4(2021-06-18)
- 修复 H5平台中间弹出后,点击内容,再点击遮罩无法关闭的Bug
## 1.4.3(2021-06-08)
- 修复 错误的 watch 字段
- 修复 safeArea 属性不生效的问题
- 修复 点击内容,再点击遮罩无法关闭的Bug
## 1.4.2(2021-05-12)
- 新增 组件示例地址
## 1.4.1(2021-04-29)
- 修复 组件内放置 input 、textarea 组件,无法聚焦的问题
## 1.4.0 (2021-04-29)
- 新增 type 属性的 left\right 值,支持左右弹出
- 新增 open(String:type) 方法参数 ,可以省略 type 属性 ,直接传入类型打开指定弹窗
- 新增 backgroundColor 属性,可定义主窗口背景色,默认不显示背景色
- 新增 safeArea 属性,是否适配底部安全区
- 修复 App\h5\微信小程序底部安全区占位不对的Bug
- 修复 App 端弹出等待的Bug
- 优化 提升低配设备性能,优化动画卡顿问题
- 优化 更简单的组件自定义方式
## 1.2.9(2021-02-05)
- 优化 组件引用关系,通过uni_modules引用组件
## 1.2.8(2021-02-05)
- 调整为uni_modules目录规范
## 1.2.7(2021-02-05)
- 调整为uni_modules目录规范
- 新增 支持 PC 端
- 新增 uni-popup-message 、uni-popup-dialog扩展组件支持 PC 端
uni_modules/uni-popup/components/uni-popup-dialog/keypress.js
New file
@@ -0,0 +1,45 @@
// #ifdef H5
export default {
  name: 'Keypress',
  props: {
    disable: {
      type: Boolean,
      default: false
    }
  },
  mounted () {
    const keyNames = {
      esc: ['Esc', 'Escape'],
      tab: 'Tab',
      enter: 'Enter',
      space: [' ', 'Spacebar'],
      up: ['Up', 'ArrowUp'],
      left: ['Left', 'ArrowLeft'],
      right: ['Right', 'ArrowRight'],
      down: ['Down', 'ArrowDown'],
      delete: ['Backspace', 'Delete', 'Del']
    }
    const listener = ($event) => {
      if (this.disable) {
        return
      }
      const keyName = Object.keys(keyNames).find(key => {
        const keyName = $event.key
        const value = keyNames[key]
        return value === keyName || (Array.isArray(value) && value.includes(keyName))
      })
      if (keyName) {
        // 避免和其他按键事件冲突
        setTimeout(() => {
          this.$emit(keyName, {})
        }, 0)
      }
    }
    document.addEventListener('keyup', listener)
    this.$once('hook:beforeDestroy', () => {
      document.removeEventListener('keyup', listener)
    })
  },
    render: () => {}
}
// #endif
uni_modules/uni-popup/components/uni-popup-dialog/uni-popup-dialog.vue
New file
@@ -0,0 +1,275 @@
<template>
    <view class="uni-popup-dialog">
        <view class="uni-dialog-title">
            <text class="uni-dialog-title-text" :class="['uni-popup__'+dialogType]">{{titleText}}</text>
        </view>
        <view v-if="mode === 'base'" class="uni-dialog-content">
            <slot>
                <text class="uni-dialog-content-text">{{content}}</text>
            </slot>
        </view>
        <view v-else class="uni-dialog-content">
            <slot>
                <input class="uni-dialog-input" v-model="val" :type="inputType" :placeholder="placeholderText" :focus="focus" >
            </slot>
        </view>
        <view class="uni-dialog-button-group">
            <view class="uni-dialog-button" @click="closeDialog">
                <text class="uni-dialog-button-text">{{closeText}}</text>
            </view>
            <view class="uni-dialog-button uni-border-left" @click="onOk">
                <text class="uni-dialog-button-text uni-button-color">{{okText}}</text>
            </view>
        </view>
    </view>
</template>
<script>
    import popup from '../uni-popup/popup.js'
    import {
    initVueI18n
    } from '@dcloudio/uni-i18n'
    import messages from '../uni-popup/i18n/index.js'
    const {    t } = initVueI18n(messages)
    /**
     * PopUp 弹出层-对话框样式
     * @description 弹出层-对话框样式
     * @tutorial https://ext.dcloud.net.cn/plugin?id=329
     * @property {String} value input 模式下的默认值
     * @property {String} placeholder input 模式下输入提示
     * @property {String} type = [success|warning|info|error] 主题样式
     *  @value success 成功
     *     @value warning 提示
     *     @value info 消息
     *     @value error 错误
     * @property {String} mode = [base|input] 模式、
     *     @value base 基础对话框
     *     @value input 可输入对话框
     * @property {String} content 对话框内容
     * @property {Boolean} beforeClose 是否拦截取消事件
     * @event {Function} confirm 点击确认按钮触发
     * @event {Function} close 点击取消按钮触发
     */
    export default {
        name: "uniPopupDialog",
        mixins: [popup],
        emits:['confirm','close'],
        props: {
            inputType:{
                type: String,
                default: 'text'
            },
            value: {
                type: [String, Number],
                default: ''
            },
            placeholder: {
                type: [String, Number],
                default: ''
            },
            type: {
                type: String,
                default: 'error'
            },
            mode: {
                type: String,
                default: 'base'
            },
            title: {
                type: String,
                default: ''
            },
            content: {
                type: String,
                default: ''
            },
            beforeClose: {
                type: Boolean,
                default: false
            },
            cancelText:{
                type: String,
                default: ''
            },
            confirmText:{
                type: String,
                default: ''
            }
        },
        data() {
            return {
                dialogType: 'error',
                focus: false,
                val: ""
            }
        },
        computed: {
            okText() {
                return this.confirmText || t("uni-popup.ok")
            },
            closeText() {
                return this.cancelText || t("uni-popup.cancel")
            },
            placeholderText() {
                return this.placeholder || t("uni-popup.placeholder")
            },
            titleText() {
                return this.title || t("uni-popup.title")
            }
        },
        watch: {
            type(val) {
                this.dialogType = val
            },
            mode(val) {
                if (val === 'input') {
                    this.dialogType = 'info'
                }
            },
            value(val) {
                this.val = val
            }
        },
        created() {
            // 对话框遮罩不可点击
            this.popup.disableMask()
            // this.popup.closeMask()
            if (this.mode === 'input') {
                this.dialogType = 'info'
                this.val = this.value
            } else {
                this.dialogType = this.type
            }
        },
        mounted() {
            this.focus = true
        },
        methods: {
            /**
             * 点击确认按钮
             */
            onOk() {
                if (this.mode === 'input'){
                    this.$emit('confirm', this.val)
                }else{
                    this.$emit('confirm')
                }
                if(this.beforeClose) return
                this.popup.close()
            },
            /**
             * 点击取消按钮
             */
            closeDialog() {
                this.$emit('close')
                if(this.beforeClose) return
                this.popup.close()
            },
            close(){
                this.popup.close()
            }
        }
    }
</script>
<style lang="scss" >
    .uni-popup-dialog {
        width: 300px;
        border-radius: 11px;
        background-color: #fff;
    }
    .uni-dialog-title {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        justify-content: center;
        padding-top: 25px;
    }
    .uni-dialog-title-text {
        font-size: 16px;
        font-weight: 500;
    }
    .uni-dialog-content {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        justify-content: center;
        align-items: center;
        padding: 20px;
    }
    .uni-dialog-content-text {
        font-size: 14px;
        color: #6C6C6C;
    }
    .uni-dialog-button-group {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        border-top-color: #f5f5f5;
        border-top-style: solid;
        border-top-width: 1px;
    }
    .uni-dialog-button {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex: 1;
        flex-direction: row;
        justify-content: center;
        align-items: center;
        height: 45px;
    }
    .uni-border-left {
        border-left-color: #f0f0f0;
        border-left-style: solid;
        border-left-width: 1px;
    }
    .uni-dialog-button-text {
        font-size: 16px;
        color: #333;
    }
    .uni-button-color {
        color: #007aff;
    }
    .uni-dialog-input {
        flex: 1;
        font-size: 14px;
        border: 1px #eee solid;
        height: 40px;
        padding: 0 10px;
        border-radius: 5px;
        color: #555;
    }
    .uni-popup__success {
        color: #4cd964;
    }
    .uni-popup__warn {
        color: #f0ad4e;
    }
    .uni-popup__error {
        color: #dd524d;
    }
    .uni-popup__info {
        color: #909399;
    }
</style>
uni_modules/uni-popup/components/uni-popup-message/uni-popup-message.vue
New file
@@ -0,0 +1,143 @@
<template>
    <view class="uni-popup-message">
        <view class="uni-popup-message__box fixforpc-width" :class="'uni-popup__'+type">
            <slot>
                <text class="uni-popup-message-text" :class="'uni-popup__'+type+'-text'">{{message}}</text>
            </slot>
        </view>
    </view>
</template>
<script>
    import popup from '../uni-popup/popup.js'
    /**
     * PopUp 弹出层-消息提示
     * @description 弹出层-消息提示
     * @tutorial https://ext.dcloud.net.cn/plugin?id=329
     * @property {String} type = [success|warning|info|error] 主题样式
     *  @value success 成功
     *     @value warning 提示
     *     @value info 消息
     *     @value error 错误
     * @property {String} message 消息提示文字
     * @property {String} duration 显示时间,设置为 0 则不会自动关闭
     */
    export default {
        name: 'uniPopupMessage',
        mixins:[popup],
        props: {
            /**
             * 主题 success/warning/info/error      默认 success
             */
            type: {
                type: String,
                default: 'success'
            },
            /**
             * 消息文字
             */
            message: {
                type: String,
                default: ''
            },
            /**
             * 显示时间,设置为 0 则不会自动关闭
             */
            duration: {
                type: Number,
                default: 3000
            },
            maskShow:{
                type:Boolean,
                default:false
            }
        },
        data() {
            return {}
        },
        created() {
            this.popup.maskShow = this.maskShow
            this.popup.messageChild = this
        },
        methods: {
            timerClose(){
                if(this.duration === 0) return
                clearTimeout(this.timer)
                this.timer = setTimeout(()=>{
                    this.popup.close()
                },this.duration)
            }
        }
    }
</script>
<style lang="scss" >
    .uni-popup-message {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        justify-content: center;
    }
    .uni-popup-message__box {
        background-color: #e1f3d8;
        padding: 10px 15px;
        border-color: #eee;
        border-style: solid;
        border-width: 1px;
        flex: 1;
    }
    @media screen and (min-width: 500px) {
        .fixforpc-width {
            margin-top: 20px;
            border-radius: 4px;
            flex: none;
            min-width: 380px;
            /* #ifndef APP-NVUE */
            max-width: 50%;
            /* #endif */
            /* #ifdef APP-NVUE */
            max-width: 500px;
            /* #endif */
        }
    }
    .uni-popup-message-text {
        font-size: 14px;
        padding: 0;
    }
    .uni-popup__success {
        background-color: #e1f3d8;
    }
    .uni-popup__success-text {
        color: #67C23A;
    }
    .uni-popup__warn {
        background-color: #faecd8;
    }
    .uni-popup__warn-text {
        color: #E6A23C;
    }
    .uni-popup__error {
        background-color: #fde2e2;
    }
    .uni-popup__error-text {
        color: #F56C6C;
    }
    .uni-popup__info {
        background-color: #F2F6FC;
    }
    .uni-popup__info-text {
        color: #909399;
    }
</style>
uni_modules/uni-popup/components/uni-popup-share/uni-popup-share.vue
New file
@@ -0,0 +1,187 @@
<template>
    <view class="uni-popup-share">
        <view class="uni-share-title"><text class="uni-share-title-text">{{shareTitleText}}</text></view>
        <view class="uni-share-content">
            <view class="uni-share-content-box">
                <view class="uni-share-content-item" v-for="(item,index) in bottomData" :key="index" @click.stop="select(item,index)">
                    <image class="uni-share-image" :src="item.icon" mode="aspectFill"></image>
                    <text class="uni-share-text">{{item.text}}</text>
                </view>
            </view>
        </view>
        <view class="uni-share-button-box">
            <button class="uni-share-button" @click="close">{{cancelText}}</button>
        </view>
    </view>
</template>
<script>
    import popup from '../uni-popup/popup.js'
    import {
    initVueI18n
    } from '@dcloudio/uni-i18n'
    import messages from '../uni-popup/i18n/index.js'
    const {    t    } = initVueI18n(messages)
    export default {
        name: 'UniPopupShare',
        mixins:[popup],
        emits:['select'],
        props: {
            title: {
                type: String,
                default: ''
            },
            beforeClose: {
                type: Boolean,
                default: false
            }
        },
        data() {
            return {
                bottomData: [{
                        text: '微信',
                        icon: 'https://vkceyugu.cdn.bspapp.com/VKCEYUGU-dc-site/c2b17470-50be-11eb-b680-7980c8a877b8.png',
                        name: 'wx'
                    },
                    {
                        text: '支付宝',
                        icon: 'https://vkceyugu.cdn.bspapp.com/VKCEYUGU-dc-site/d684ae40-50be-11eb-8ff1-d5dcf8779628.png',
                        name: 'wx'
                    },
                    {
                        text: 'QQ',
                        icon: 'https://vkceyugu.cdn.bspapp.com/VKCEYUGU-dc-site/e7a79520-50be-11eb-b997-9918a5dda011.png',
                        name: 'qq'
                    },
                    {
                        text: '新浪',
                        icon: 'https://vkceyugu.cdn.bspapp.com/VKCEYUGU-dc-site/0dacdbe0-50bf-11eb-8ff1-d5dcf8779628.png',
                        name: 'sina'
                    },
                    // {
                    //     text: '百度',
                    //     icon: 'https://vkceyugu.cdn.bspapp.com/VKCEYUGU-dc-site/1ec6e920-50bf-11eb-8a36-ebb87efcf8c0.png',
                    //     name: 'copy'
                    // },
                    // {
                    //     text: '其他',
                    //     icon: 'https://vkceyugu.cdn.bspapp.com/VKCEYUGU-dc-site/2e0fdfe0-50bf-11eb-b997-9918a5dda011.png',
                    //     name: 'more'
                    // }
                ]
            }
        },
        created() {},
        computed: {
            cancelText() {
                return t("uni-popup.cancel")
            },
        shareTitleText() {
                return this.title || t("uni-popup.shareTitle")
            }
        },
        methods: {
            /**
             * 选择内容
             */
            select(item, index) {
                this.$emit('select', {
                    item,
                    index
                })
                this.close()
            },
            /**
             * 关闭窗口
             */
            close() {
                if(this.beforeClose) return
                this.popup.close()
            }
        }
    }
</script>
<style lang="scss" >
    .uni-popup-share {
        background-color: #fff;
        border-top-left-radius: 11px;
        border-top-right-radius: 11px;
    }
    .uni-share-title {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        align-items: center;
        justify-content: center;
        height: 40px;
    }
    .uni-share-title-text {
        font-size: 14px;
        color: #666;
    }
    .uni-share-content {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        justify-content: center;
        padding-top: 10px;
    }
    .uni-share-content-box {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        flex-wrap: wrap;
        width: 360px;
    }
    .uni-share-content-item {
        width: 90px;
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: column;
        justify-content: center;
        padding: 10px 0;
        align-items: center;
    }
    .uni-share-content-item:active {
        background-color: #f5f5f5;
    }
    .uni-share-image {
        width: 30px;
        height: 30px;
    }
    .uni-share-text {
        margin-top: 10px;
        font-size: 14px;
        color: #3B4144;
    }
    .uni-share-button-box {
        /* #ifndef APP-NVUE */
        display: flex;
        /* #endif */
        flex-direction: row;
        padding: 10px 15px;
    }
    .uni-share-button {
        flex: 1;
        border-radius: 50px;
        color: #666;
        font-size: 16px;
    }
    .uni-share-button::after {
        border-radius: 50px;
    }
</style>
uni_modules/uni-popup/components/uni-popup/i18n/en.json
New file
@@ -0,0 +1,7 @@
{
    "uni-popup.cancel": "cancel",
    "uni-popup.ok": "ok",
    "uni-popup.placeholder": "pleace enter",
    "uni-popup.title": "Hint",
    "uni-popup.shareTitle": "Share to"
}
uni_modules/uni-popup/components/uni-popup/i18n/index.js
New file
@@ -0,0 +1,8 @@
import en from './en.json'
import zhHans from './zh-Hans.json'
import zhHant from './zh-Hant.json'
export default {
    en,
    'zh-Hans': zhHans,
    'zh-Hant': zhHant
}
uni_modules/uni-popup/components/uni-popup/i18n/zh-Hans.json
New file
@@ -0,0 +1,7 @@
{
    "uni-popup.cancel": "取消",
    "uni-popup.ok": "确定",
    "uni-popup.placeholder": "请输入",
        "uni-popup.title": "提示",
        "uni-popup.shareTitle": "分享到"
}
uni_modules/uni-popup/components/uni-popup/i18n/zh-Hant.json
New file
@@ -0,0 +1,7 @@
{
    "uni-popup.cancel": "取消",
    "uni-popup.ok": "確定",
    "uni-popup.placeholder": "請輸入",
    "uni-popup.title": "提示",
    "uni-popup.shareTitle": "分享到"
}
uni_modules/uni-popup/components/uni-popup/keypress.js
New file
@@ -0,0 +1,45 @@
// #ifdef H5
export default {
  name: 'Keypress',
  props: {
    disable: {
      type: Boolean,
      default: false
    }
  },
  mounted () {
    const keyNames = {
      esc: ['Esc', 'Escape'],
      tab: 'Tab',
      enter: 'Enter',
      space: [' ', 'Spacebar'],
      up: ['Up', 'ArrowUp'],
      left: ['Left', 'ArrowLeft'],
      right: ['Right', 'ArrowRight'],
      down: ['Down', 'ArrowDown'],
      delete: ['Backspace', 'Delete', 'Del']
    }
    const listener = ($event) => {
      if (this.disable) {
        return
      }
      const keyName = Object.keys(keyNames).find(key => {
        const keyName = $event.key
        const value = keyNames[key]
        return value === keyName || (Array.isArray(value) && value.includes(keyName))
      })
      if (keyName) {
        // 避免和其他按键事件冲突
        setTimeout(() => {
          this.$emit(keyName, {})
        }, 0)
      }
    }
    document.addEventListener('keyup', listener)
    // this.$once('hook:beforeDestroy', () => {
    //   document.removeEventListener('keyup', listener)
    // })
  },
    render: () => {}
}
// #endif
uni_modules/uni-popup/components/uni-popup/popup.js
New file
@@ -0,0 +1,26 @@
export default {
    data() {
        return {
        }
    },
    created(){
        this.popup = this.getParent()
    },
    methods:{
        /**
         * 获取父元素实例
         */
        getParent(name = 'uniPopup') {
            let parent = this.$parent;
            let parentName = parent.$options.name;
            while (parentName !== name) {
                parent = parent.$parent;
                if (!parent) return false
                parentName = parent.$options.name;
            }
            return parent;
        },
    }
}
uni_modules/uni-popup/components/uni-popup/uni-popup.vue
New file
@@ -0,0 +1,473 @@
<template>
    <view v-if="showPopup" class="uni-popup" :class="[popupstyle, isDesktop ? 'fixforpc-z-index' : '']">
        <view @touchstart="touchstart">
            <uni-transition key="1" v-if="maskShow" name="mask" mode-class="fade" :styles="maskClass"
                :duration="duration" :show="showTrans" @click="onTap" />
            <uni-transition key="2" :mode-class="ani" name="content" :styles="transClass" :duration="duration"
                :show="showTrans" @click="onTap">
                <view class="uni-popup__wrapper" :style="{ backgroundColor: bg }" :class="[popupstyle]" @click="clear">
                    <slot />
                </view>
            </uni-transition>
        </view>
        <!-- #ifdef H5 -->
        <keypress v-if="maskShow" @esc="onTap" />
        <!-- #endif -->
    </view>
</template>
<script>
    // #ifdef H5
    import keypress from './keypress.js'
    // #endif
    /**
     * PopUp 弹出层
     * @description 弹出层组件,为了解决遮罩弹层的问题
     * @tutorial https://ext.dcloud.net.cn/plugin?id=329
     * @property {String} type = [top|center|bottom|left|right|message|dialog|share] 弹出方式
     *     @value top 顶部弹出
     *     @value center 中间弹出
     *     @value bottom 底部弹出
     *     @value left        左侧弹出
     *     @value right  右侧弹出
     *     @value message 消息提示
     *     @value dialog 对话框
     *     @value share 底部分享示例
     * @property {Boolean} animation = [true|false] 是否开启动画
     * @property {Boolean} maskClick = [true|false] 蒙版点击是否关闭弹窗(废弃)
     * @property {Boolean} isMaskClick = [true|false] 蒙版点击是否关闭弹窗
     * @property {String}  backgroundColor 主窗口背景色
     * @property {String}  maskBackgroundColor 蒙版颜色
     * @property {Boolean} safeArea           是否适配底部安全区
     * @event {Function} change 打开关闭弹窗触发,e={show: false}
     * @event {Function} maskClick 点击遮罩触发
     */
    export default {
        name: 'uniPopup',
        components: {
            // #ifdef H5
            keypress
            // #endif
        },
        emits: ['change', 'maskClick'],
        props: {
            // 开启动画
            animation: {
                type: Boolean,
                default: true
            },
            // 弹出层类型,可选值,top: 顶部弹出层;bottom:底部弹出层;center:全屏弹出层
            // message: 消息提示 ; dialog : 对话框
            type: {
                type: String,
                default: 'center'
            },
            // maskClick
            isMaskClick: {
                type: Boolean,
                default: null
            },
            // TODO 2 个版本后废弃属性 ,使用 isMaskClick
            maskClick: {
                type: Boolean,
                default: null
            },
            backgroundColor: {
                type: String,
                default: 'none'
            },
            safeArea: {
                type: Boolean,
                default: true
            },
            maskBackgroundColor: {
                type: String,
                default: 'rgba(0, 0, 0, 0.4)'
            },
        },
        watch: {
            /**
             * 监听type类型
             */
            type: {
                handler: function(type) {
                    if (!this.config[type]) return
                    this[this.config[type]](true)
                },
                immediate: true
            },
            isDesktop: {
                handler: function(newVal) {
                    if (!this.config[newVal]) return
                    this[this.config[this.type]](true)
                },
                immediate: true
            },
            /**
             * 监听遮罩是否可点击
             * @param {Object} val
             */
            maskClick: {
                handler: function(val) {
                    this.mkclick = val
                },
                immediate: true
            },
            isMaskClick: {
                handler: function(val) {
                    this.mkclick = val
                },
                immediate: true
            },
            // H5 下禁止底部滚动
            showPopup(show) {
                // #ifdef H5
                // fix by mehaotian 处理 h5 滚动穿透的问题
                document.getElementsByTagName('body')[0].style.overflow = show ? 'hidden' : 'visible'
                // #endif
            }
        },
        data() {
            return {
                duration: 300,
                ani: [],
                showPopup: false,
                showTrans: false,
                popupWidth: 0,
                popupHeight: 0,
                config: {
                    top: 'top',
                    bottom: 'bottom',
                    center: 'center',
                    left: 'left',
                    right: 'right',
                    message: 'top',
                    dialog: 'center',
                    share: 'bottom'
                },
                maskClass: {
                    position: 'fixed',
                    bottom: 0,
                    top: 0,
                    left: 0,
                    right: 0,
                    backgroundColor: 'rgba(0, 0, 0, 0.4)'
                },
                transClass: {
                    position: 'fixed',
                    left: 0,
                    right: 0
                },
                maskShow: true,
                mkclick: true,
                popupstyle: this.isDesktop ? 'fixforpc-top' : 'top'
            }
        },
        computed: {
            isDesktop() {
                return this.popupWidth >= 500 && this.popupHeight >= 500
            },
            bg() {
                if (this.backgroundColor === '' || this.backgroundColor === 'none') {
                    return 'transparent'
                }
                return this.backgroundColor
            }
        },
        mounted() {
            const fixSize = () => {
                const {
                    windowWidth,
                    windowHeight,
                    windowTop,
                    safeArea,
                    screenHeight,
                    safeAreaInsets
                } = uni.getSystemInfoSync()
                this.popupWidth = windowWidth
                this.popupHeight = windowHeight + (windowTop || 0)
                // TODO fix by mehaotian 是否适配底部安全区 ,目前微信ios 、和 app ios 计算有差异,需要框架修复
                if (safeArea && this.safeArea) {
                    // #ifdef MP-WEIXIN
                    this.safeAreaInsets = screenHeight - safeArea.bottom
                    // #endif
                    // #ifndef MP-WEIXIN
                    this.safeAreaInsets = safeAreaInsets.bottom
                    // #endif
                } else {
                    this.safeAreaInsets = 0
                }
            }
            fixSize()
            // #ifdef H5
            // window.addEventListener('resize', fixSize)
            // this.$once('hook:beforeDestroy', () => {
            //     window.removeEventListener('resize', fixSize)
            // })
            // #endif
        },
        // #ifndef VUE3
        // TODO vue2
        destroyed() {
            this.setH5Visible()
        },
        // #endif
        // #ifdef VUE3
        // TODO vue3
        unmounted() {
            this.setH5Visible()
        },
        // #endif
        created() {
            // this.mkclick =  this.isMaskClick || this.maskClick
            if (this.isMaskClick === null && this.maskClick === null) {
                this.mkclick = true
            } else {
                this.mkclick = this.isMaskClick !== null ? this.isMaskClick : this.maskClick
            }
            if (this.animation) {
                this.duration = 300
            } else {
                this.duration = 0
            }
            // TODO 处理 message 组件生命周期异常的问题
            this.messageChild = null
            // TODO 解决头条冒泡的问题
            this.clearPropagation = false
            this.maskClass.backgroundColor = this.maskBackgroundColor
        },
        methods: {
            setH5Visible() {
                // #ifdef H5
                // fix by mehaotian 处理 h5 滚动穿透的问题
                document.getElementsByTagName('body')[0].style.overflow = 'visible'
                // #endif
            },
            /**
             * 公用方法,不显示遮罩层
             */
            closeMask() {
                this.maskShow = false
            },
            /**
             * 公用方法,遮罩层禁止点击
             */
            disableMask() {
                this.mkclick = false
            },
            // TODO nvue 取消冒泡
            clear(e) {
                // #ifndef APP-NVUE
                e.stopPropagation()
                // #endif
                this.clearPropagation = true
            },
            open(direction) {
                // fix by mehaotian 处理快速打开关闭的情况
                if (this.showPopup) {
                    return
                }
                let innerType = ['top', 'center', 'bottom', 'left', 'right', 'message', 'dialog', 'share']
                if (!(direction && innerType.indexOf(direction) !== -1)) {
                    direction = this.type
                }
                if (!this.config[direction]) {
                    console.error('缺少类型:', direction)
                    return
                }
                this[this.config[direction]]()
                this.$emit('change', {
                    show: true,
                    type: direction
                })
            },
            close(type) {
                this.showTrans = false
                this.$emit('change', {
                    show: false,
                    type: this.type
                })
                clearTimeout(this.timer)
                // // 自定义关闭事件
                // this.customOpen && this.customClose()
                this.timer = setTimeout(() => {
                    this.showPopup = false
                }, 300)
            },
            // TODO 处理冒泡事件,头条的冒泡事件有问题 ,先这样兼容
            touchstart() {
                this.clearPropagation = false
            },
            onTap() {
                if (this.clearPropagation) {
                    // fix by mehaotian 兼容 nvue
                    this.clearPropagation = false
                    return
                }
                this.$emit('maskClick')
                if (!this.mkclick) return
                this.close()
            },
            /**
             * 顶部弹出样式处理
             */
            top(type) {
                this.popupstyle = this.isDesktop ? 'fixforpc-top' : 'top'
                this.ani = ['slide-top']
                this.transClass = {
                    position: 'fixed',
                    left: 0,
                    right: 0,
                    backgroundColor: this.bg
                }
                // TODO 兼容 type 属性 ,后续会废弃
                if (type) return
                this.showPopup = true
                this.showTrans = true
                this.$nextTick(() => {
                    if (this.messageChild && this.type === 'message') {
                        this.messageChild.timerClose()
                    }
                })
            },
            /**
             * 底部弹出样式处理
             */
            bottom(type) {
                this.popupstyle = 'bottom'
                this.ani = ['slide-bottom']
                this.transClass = {
                    position: 'fixed',
                    left: 0,
                    right: 0,
                    bottom: 0,
                    paddingBottom: this.safeAreaInsets + 'px',
                    backgroundColor: this.bg
                }
                // TODO 兼容 type 属性 ,后续会废弃
                if (type) return
                this.showPopup = true
                this.showTrans = true
            },
            /**
             * 中间弹出样式处理
             */
            center(type) {
                this.popupstyle = 'center'
                this.ani = ['zoom-out', 'fade']
                this.transClass = {
                    position: 'fixed',
                    /* #ifndef APP-NVUE */
                    display: 'flex',
                    flexDirection: 'column',
                    /* #endif */
                    bottom: 0,
                    left: 0,
                    right: 0,
                    top: 0,
                    justifyContent: 'center',
                    alignItems: 'center'
                }
                // TODO 兼容 type 属性 ,后续会废弃
                if (type) return
                this.showPopup = true
                this.showTrans = true
            },
            left(type) {
                this.popupstyle = 'left'
                this.ani = ['slide-left']
                this.transClass = {
                    position: 'fixed',
                    left: 0,
                    bottom: 0,
                    top: 0,
                    backgroundColor: this.bg,
                    /* #ifndef APP-NVUE */
                    display: 'flex',
                    flexDirection: 'column'
                    /* #endif */
                }
                // TODO 兼容 type 属性 ,后续会废弃
                if (type) return
                this.showPopup = true
                this.showTrans = true
            },
            right(type) {
                this.popupstyle = 'right'
                this.ani = ['slide-right']
                this.transClass = {
                    position: 'fixed',
                    bottom: 0,
                    right: 0,
                    top: 0,
                    backgroundColor: this.bg,
                    /* #ifndef APP-NVUE */
                    display: 'flex',
                    flexDirection: 'column'
                    /* #endif */
                }
                // TODO 兼容 type 属性 ,后续会废弃
                if (type) return
                this.showPopup = true
                this.showTrans = true
            }
        }
    }
</script>
<style lang="scss">
    .uni-popup {
        position: fixed;
        /* #ifndef APP-NVUE */
        z-index: 99;
        /* #endif */
        &.top,
        &.left,
        &.right {
            /* #ifdef H5 */
            top: var(--window-top);
            /* #endif */
            /* #ifndef H5 */
            top: 0;
            /* #endif */
        }
        .uni-popup__wrapper {
            /* #ifndef APP-NVUE */
            display: block;
            /* #endif */
            position: relative;
            /* iphonex 等安全区设置,底部安全区适配 */
            /* #ifndef APP-NVUE */
            // padding-bottom: constant(safe-area-inset-bottom);
            // padding-bottom: env(safe-area-inset-bottom);
            /* #endif */
            &.left,
            &.right {
                /* #ifdef H5 */
                padding-top: var(--window-top);
                /* #endif */
                /* #ifndef H5 */
                padding-top: 0;
                /* #endif */
                flex: 1;
            }
        }
    }
    .fixforpc-z-index {
        /* #ifndef APP-NVUE */
        z-index: 999;
        /* #endif */
    }
    .fixforpc-top {
        top: 0;
    }
</style>
uni_modules/uni-popup/package.json
New file
@@ -0,0 +1,87 @@
{
    "id": "uni-popup",
    "displayName": "uni-popup 弹出层",
    "version": "1.8.3",
    "description": " Popup 组件,提供常用的弹层",
    "keywords": [
        "uni-ui",
        "弹出层",
        "弹窗",
        "popup",
        "弹框"
    ],
    "repository": "https://github.com/dcloudio/uni-ui",
    "engines": {
        "HBuilderX": ""
    },
    "directories": {
        "example": "../../temps/example_temps"
    },
    "dcloudext": {
        "sale": {
            "regular": {
                "price": "0.00"
            },
            "sourcecode": {
                "price": "0.00"
            }
        },
        "contact": {
            "qq": ""
        },
        "declaration": {
            "ads": "无",
            "data": "无",
            "permissions": "无"
        },
        "npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui",
        "type": "component-vue"
    },
    "uni_modules": {
        "dependencies": [
            "uni-scss",
            "uni-transition"
        ],
        "encrypt": [],
        "platforms": {
            "cloud": {
                "tcb": "y",
                "aliyun": "y"
            },
            "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": "y",
                    "Edge": "y",
                    "Firefox": "y",
                    "Safari": "y"
                },
                "小程序": {
                    "微信": "y",
                    "阿里": "y",
                    "百度": "y",
                    "字节跳动": "y",
                    "QQ": "y"
                },
                "快应用": {
                    "华为": "u",
                    "联盟": "u"
                },
                "Vue": {
                    "vue2": "y",
                    "vue3": "y"
                }
            }
        }
    }
}
uni_modules/uni-popup/readme.md
New file
@@ -0,0 +1,17 @@
## Popup 弹出层
> **组件名:uni-popup**
> 代码块: `uPopup`
> 关联组件:`uni-transition`
弹出层组件,在应用中弹出一个消息提示窗口、提示框等
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-popup)
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839
uni_modules/uni-transition/changelog.md
New file
@@ -0,0 +1,22 @@
## 1.3.2(2023-05-04)
- 修复 NVUE 平台报错的问题
## 1.3.1(2021-11-23)
- 修复 init 方法初始化问题
## 1.3.0(2021-11-19)
- 优化 组件UI,并提供设计资源,详见:[https://uniapp.dcloud.io/component/uniui/resource](https://uniapp.dcloud.io/component/uniui/resource)
- 文档迁移,详见:[https://uniapp.dcloud.io/component/uniui/uni-transition](https://uniapp.dcloud.io/component/uniui/uni-transition)
## 1.2.1(2021-09-27)
- 修复 init 方法不生效的 Bug
## 1.2.0(2021-07-30)
- 组件兼容 vue3,如何创建 vue3 项目,详见 [uni-app 项目支持 vue3 介绍](https://ask.dcloud.net.cn/article/37834)
## 1.1.1(2021-05-12)
- 新增 示例地址
- 修复 示例项目缺少组件的 Bug
## 1.1.0(2021-04-22)
- 新增 通过方法自定义动画
- 新增 custom-class 非 NVUE 平台支持自定义 class 定制样式
- 优化 动画触发逻辑,使动画更流畅
- 优化 支持单独的动画类型
- 优化 文档示例
## 1.0.2(2021-02-05)
- 调整为 uni_modules 目录规范
uni_modules/uni-transition/components/uni-transition/createAnimation.js
New file
@@ -0,0 +1,131 @@
// const defaultOption = {
//     duration: 300,
//     timingFunction: 'linear',
//     delay: 0,
//     transformOrigin: '50% 50% 0'
// }
// #ifdef APP-NVUE
const nvueAnimation = uni.requireNativePlugin('animation')
// #endif
class MPAnimation {
    constructor(options, _this) {
        this.options = options
        // 在iOS10+QQ小程序平台下,传给原生的对象一定是个普通对象而不是Proxy对象,否则会报parameter should be Object instead of ProxyObject的错误
        this.animation = uni.createAnimation({
            ...options
        })
        this.currentStepAnimates = {}
        this.next = 0
        this.$ = _this
    }
    _nvuePushAnimates(type, args) {
        let aniObj = this.currentStepAnimates[this.next]
        let styles = {}
        if (!aniObj) {
            styles = {
                styles: {},
                config: {}
            }
        } else {
            styles = aniObj
        }
        if (animateTypes1.includes(type)) {
            if (!styles.styles.transform) {
                styles.styles.transform = ''
            }
            let unit = ''
            if(type === 'rotate'){
                unit = 'deg'
            }
            styles.styles.transform += `${type}(${args+unit}) `
        } else {
            styles.styles[type] = `${args}`
        }
        this.currentStepAnimates[this.next] = styles
    }
    _animateRun(styles = {}, config = {}) {
        let ref = this.$.$refs['ani'].ref
        if (!ref) return
        return new Promise((resolve, reject) => {
            nvueAnimation.transition(ref, {
                styles,
                ...config
            }, res => {
                resolve()
            })
        })
    }
    _nvueNextAnimate(animates, step = 0, fn) {
        let obj = animates[step]
        if (obj) {
            let {
                styles,
                config
            } = obj
            this._animateRun(styles, config).then(() => {
                step += 1
                this._nvueNextAnimate(animates, step, fn)
            })
        } else {
            this.currentStepAnimates = {}
            typeof fn === 'function' && fn()
            this.isEnd = true
        }
    }
    step(config = {}) {
        // #ifndef APP-NVUE
        this.animation.step(config)
        // #endif
        // #ifdef APP-NVUE
        this.currentStepAnimates[this.next].config = Object.assign({}, this.options, config)
        this.currentStepAnimates[this.next].styles.transformOrigin = this.currentStepAnimates[this.next].config.transformOrigin
        this.next++
        // #endif
        return this
    }
    run(fn) {
        // #ifndef APP-NVUE
        this.$.animationData = this.animation.export()
        this.$.timer = setTimeout(() => {
            typeof fn === 'function' && fn()
        }, this.$.durationTime)
        // #endif
        // #ifdef APP-NVUE
        this.isEnd = false
        let ref = this.$.$refs['ani'] && this.$.$refs['ani'].ref
        if(!ref) return
        this._nvueNextAnimate(this.currentStepAnimates, 0, fn)
        this.next = 0
        // #endif
    }
}
const animateTypes1 = ['matrix', 'matrix3d', 'rotate', 'rotate3d', 'rotateX', 'rotateY', 'rotateZ', 'scale', 'scale3d',
    'scaleX', 'scaleY', 'scaleZ', 'skew', 'skewX', 'skewY', 'translate', 'translate3d', 'translateX', 'translateY',
    'translateZ'
]
const animateTypes2 = ['opacity', 'backgroundColor']
const animateTypes3 = ['width', 'height', 'left', 'right', 'top', 'bottom']
animateTypes1.concat(animateTypes2, animateTypes3).forEach(type => {
    MPAnimation.prototype[type] = function(...args) {
        // #ifndef APP-NVUE
        this.animation[type](...args)
        // #endif
        // #ifdef APP-NVUE
        this._nvuePushAnimates(type, args)
        // #endif
        return this
    }
})
export function createAnimation(option, _this) {
    if(!_this) return
    clearTimeout(_this.timer)
    return new MPAnimation(option, _this)
}
uni_modules/uni-transition/components/uni-transition/uni-transition.vue
New file
@@ -0,0 +1,286 @@
<template>
  <!-- #ifndef APP-NVUE -->
  <view v-show="isShow" ref="ani" :animation="animationData" :class="customClass" :style="transformStyles" @click="onClick"><slot></slot></view>
  <!-- #endif -->
  <!-- #ifdef APP-NVUE -->
  <view v-if="isShow" ref="ani" :animation="animationData" :class="customClass" :style="transformStyles" @click="onClick"><slot></slot></view>
  <!-- #endif -->
</template>
<script>
import { createAnimation } from './createAnimation'
/**
 * Transition 过渡动画
 * @description 简单过渡动画组件
 * @tutorial https://ext.dcloud.net.cn/plugin?id=985
 * @property {Boolean} show = [false|true] 控制组件显示或隐藏
 * @property {Array|String} modeClass = [fade|slide-top|slide-right|slide-bottom|slide-left|zoom-in|zoom-out] 过渡动画类型
 *  @value fade 渐隐渐出过渡
 *  @value slide-top 由上至下过渡
 *  @value slide-right 由右至左过渡
 *  @value slide-bottom 由下至上过渡
 *  @value slide-left 由左至右过渡
 *  @value zoom-in 由小到大过渡
 *  @value zoom-out 由大到小过渡
 * @property {Number} duration 过渡动画持续时间
 * @property {Object} styles 组件样式,同 css 样式,注意带’-‘连接符的属性需要使用小驼峰写法如:`backgroundColor:red`
 */
export default {
    name: 'uniTransition',
    emits:['click','change'],
    props: {
        show: {
            type: Boolean,
            default: false
        },
        modeClass: {
            type: [Array, String],
            default() {
                return 'fade'
            }
        },
        duration: {
            type: Number,
            default: 300
        },
        styles: {
            type: Object,
            default() {
                return {}
            }
        },
        customClass:{
            type: String,
            default: ''
        },
        onceRender:{
            type:Boolean,
            default:false
        },
    },
    data() {
        return {
            isShow: false,
            transform: '',
            opacity: 1,
            animationData: {},
            durationTime: 300,
            config: {}
        }
    },
    watch: {
        show: {
            handler(newVal) {
                if (newVal) {
                    this.open()
                } else {
                    // 避免上来就执行 close,导致动画错乱
                    if (this.isShow) {
                        this.close()
                    }
                }
            },
            immediate: true
        }
    },
    computed: {
        // 生成样式数据
        stylesObject() {
            let styles = {
                ...this.styles,
                'transition-duration': this.duration / 1000 + 's'
            }
            let transform = ''
            for (let i in styles) {
                let line = this.toLine(i)
                transform += line + ':' + styles[i] + ';'
            }
            return transform
        },
        // 初始化动画条件
        transformStyles() {
            return 'transform:' + this.transform + ';' + 'opacity:' + this.opacity + ';' + this.stylesObject
        }
    },
    created() {
        // 动画默认配置
        this.config = {
            duration: this.duration,
            timingFunction: 'ease',
            transformOrigin: '50% 50%',
            delay: 0
        }
        this.durationTime = this.duration
    },
    methods: {
        /**
         *  ref 触发 初始化动画
         */
        init(obj = {}) {
            if (obj.duration) {
                this.durationTime = obj.duration
            }
            this.animation = createAnimation(Object.assign(this.config, obj),this)
        },
        /**
         * 点击组件触发回调
         */
        onClick() {
            this.$emit('click', {
                detail: this.isShow
            })
        },
        /**
         * ref 触发 动画分组
         * @param {Object} obj
         */
        step(obj, config = {}) {
            if (!this.animation) return
            for (let i in obj) {
                try {
                    if(typeof obj[i] === 'object'){
                        this.animation[i](...obj[i])
                    }else{
                        this.animation[i](obj[i])
                    }
                } catch (e) {
                    console.error(`方法 ${i} 不存在`)
                }
            }
            this.animation.step(config)
            return this
        },
        /**
         *  ref 触发 执行动画
         */
        run(fn) {
            if (!this.animation) return
            this.animation.run(fn)
        },
        // 开始过度动画
        open() {
            clearTimeout(this.timer)
            this.transform = ''
            this.isShow = true
            let { opacity, transform } = this.styleInit(false)
            if (typeof opacity !== 'undefined') {
                this.opacity = opacity
            }
            this.transform = transform
            // 确保动态样式已经生效后,执行动画,如果不加 nextTick ,会导致 wx 动画执行异常
            this.$nextTick(() => {
                // TODO 定时器保证动画完全执行,目前有些问题,后面会取消定时器
                this.timer = setTimeout(() => {
                    this.animation = createAnimation(this.config, this)
                    this.tranfromInit(false).step()
                    this.animation.run()
                    this.$emit('change', {
                        detail: this.isShow
                    })
                }, 20)
            })
        },
        // 关闭过度动画
        close(type) {
            if (!this.animation) return
            this.tranfromInit(true)
                .step()
                .run(() => {
                    this.isShow = false
                    this.animationData = null
                    this.animation = null
                    let { opacity, transform } = this.styleInit(false)
                    this.opacity = opacity || 1
                    this.transform = transform
                    this.$emit('change', {
                        detail: this.isShow
                    })
                })
        },
        // 处理动画开始前的默认样式
        styleInit(type) {
            let styles = {
                transform: ''
            }
            let buildStyle = (type, mode) => {
                if (mode === 'fade') {
                    styles.opacity = this.animationType(type)[mode]
                } else {
                    styles.transform += this.animationType(type)[mode] + ' '
                }
            }
            if (typeof this.modeClass === 'string') {
                buildStyle(type, this.modeClass)
            } else {
                this.modeClass.forEach(mode => {
                    buildStyle(type, mode)
                })
            }
            return styles
        },
        // 处理内置组合动画
        tranfromInit(type) {
            let buildTranfrom = (type, mode) => {
                let aniNum = null
                if (mode === 'fade') {
                    aniNum = type ? 0 : 1
                } else {
                    aniNum = type ? '-100%' : '0'
                    if (mode === 'zoom-in') {
                        aniNum = type ? 0.8 : 1
                    }
                    if (mode === 'zoom-out') {
                        aniNum = type ? 1.2 : 1
                    }
                    if (mode === 'slide-right') {
                        aniNum = type ? '100%' : '0'
                    }
                    if (mode === 'slide-bottom') {
                        aniNum = type ? '100%' : '0'
                    }
                }
                this.animation[this.animationMode()[mode]](aniNum)
            }
            if (typeof this.modeClass === 'string') {
                buildTranfrom(type, this.modeClass)
            } else {
                this.modeClass.forEach(mode => {
                    buildTranfrom(type, mode)
                })
            }
            return this.animation
        },
        animationType(type) {
            return {
                fade: type ? 1 : 0,
                'slide-top': `translateY(${type ? '0' : '-100%'})`,
                'slide-right': `translateX(${type ? '0' : '100%'})`,
                'slide-bottom': `translateY(${type ? '0' : '100%'})`,
                'slide-left': `translateX(${type ? '0' : '-100%'})`,
                'zoom-in': `scaleX(${type ? 1 : 0.8}) scaleY(${type ? 1 : 0.8})`,
                'zoom-out': `scaleX(${type ? 1 : 1.2}) scaleY(${type ? 1 : 1.2})`
            }
        },
        // 内置动画类型与实际动画对应字典
        animationMode() {
            return {
                fade: 'opacity',
                'slide-top': 'translateY',
                'slide-right': 'translateX',
                'slide-bottom': 'translateY',
                'slide-left': 'translateX',
                'zoom-in': 'scale',
                'zoom-out': 'scale'
            }
        },
        // 驼峰转中横线
        toLine(name) {
            return name.replace(/([A-Z])/g, '-$1').toLowerCase()
        }
    }
}
</script>
<style></style>
uni_modules/uni-transition/package.json
New file
@@ -0,0 +1,84 @@
{
  "id": "uni-transition",
  "displayName": "uni-transition 过渡动画",
  "version": "1.3.2",
  "description": "元素的简单过渡动画",
  "keywords": [
    "uni-ui",
    "uniui",
    "动画",
    "过渡",
    "过渡动画"
],
  "repository": "https://github.com/dcloudio/uni-ui",
  "engines": {
    "HBuilderX": ""
  },
  "directories": {
    "example": "../../temps/example_temps"
  },
"dcloudext": {
    "sale": {
      "regular": {
        "price": "0.00"
      },
      "sourcecode": {
        "price": "0.00"
      }
    },
    "contact": {
      "qq": ""
    },
    "declaration": {
      "ads": "无",
      "data": "无",
      "permissions": "无"
    },
    "npmurl": "https://www.npmjs.com/package/@dcloudio/uni-ui",
    "type": "component-vue"
  },
  "uni_modules": {
    "dependencies": ["uni-scss"],
    "encrypt": [],
    "platforms": {
      "cloud": {
        "tcb": "y",
        "aliyun": "y"
      },
      "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": "y",
          "Edge": "y",
          "Firefox": "y",
          "Safari": "y"
        },
        "小程序": {
          "微信": "y",
          "阿里": "y",
          "百度": "y",
          "字节跳动": "y",
          "QQ": "y"
        },
        "快应用": {
          "华为": "u",
          "联盟": "u"
        },
        "Vue": {
            "vue2": "y",
            "vue3": "y"
        }
      }
    }
  }
}
uni_modules/uni-transition/readme.md
New file
@@ -0,0 +1,11 @@
## Transition 过渡动画
> **组件名:uni-transition**
> 代码块: `uTransition`
元素过渡动画
### [查看文档](https://uniapp.dcloud.io/component/uniui/uni-transition)
#### 如使用过程中有任何问题,或者您对uni-ui有一些好的建议,欢迎加入 uni-ui 交流群:871950839
uni_modules/ws-wx-privacy/changelog.md
New file
@@ -0,0 +1,22 @@
## 1.0.10(2023-09-12)
支持配置自动获取微信小程序后台的隐私协议名称
## 1.0.9(2023-09-11)
增加一些常见问题的解决办法
## 1.0.8(2023-09-11)
支持自定义主题色和背景色
## 1.0.7(2023-09-08)
优化文档示例
## 1.0.6(2023-09-07)
支持不允许不同意协议的功能,disagreeEnabled为false时,不同意按钮点击会提示disagreePromptText的内容
## 1.0.5(2023-09-05)
## README中增加关于授权的说明
## 1.0.4(2023-09-02)
强化文档中关于隐私指引调试配置的提醒
## 1.0.3(2023-09-01)
# 请注意 微信后台的用户隐私保护指引必须审核通过之后才可以调试
## 1.0.2(2023-09-01)
更新隐私保护指引调试前的配置说明
## 1.0.1(2023-08-31)
修复文案中【您点击同意并开始时用产品服务时】出现错别字的问题
## 1.0.0(2023-08-30)
微信隐私保护弹出框,支持vue2和vue3
uni_modules/ws-wx-privacy/components/ws-wx-privacy/util.js
New file
@@ -0,0 +1,51 @@
/**
 * 获取当前页面上下文
 * @returns 页面对象
 */
export function getContext() {
    // eslint-disable-next-line no-undef
    const pages = getCurrentPages()
    return pages[pages.length - 1]
}
/**
 * 获取上下文中指定节点组件
 * @param context 选择器的选择范围,可以传入自定义组件的 this 作为上下文
 * @param selector 自定义节点选择器
 */
export function getComponent(context, selector ) {
    let component = null
    // #ifdef H5
    context.$children.forEach((child) => {
        if (`#${child.$attrs.id}` === selector) {
            component = child
        } else if (child.$children && child.$children.length) {
            if (getComponent(child, selector)) {
                component = getComponent(child, selector)
            }
        }
        if (component) {
            return component
        }
    })
    // #endif
    // #ifdef MP-WEIXIN
    component = context.selectComponent && context.selectComponent(selector) && context.selectComponent(selector).$vm
    // #endif
    // #ifdef MP-ALIPAY
    const alipay = context.$children ? context.$children : context.$vm && context.$vm.$children ? context.$vm
        .$children : []
    component = alipay.find((component) => {
        return `#${component.$scope.props.id}` === selector
    })
    // #endif
    // #ifdef APP-PLUS
    const app = context.$children ? context.$children : context.$vm && context.$vm.$children ? context.$vm.$children :
    []
    component = app.find((component) => {
        return `#${component.$attrs.id}` === selector
    })
    // #endif
    return component
}
uni_modules/ws-wx-privacy/components/ws-wx-privacy/ws-wx-privacy.vue
New file
@@ -0,0 +1,287 @@
<template>
    <uni-popup id="privacy" type="center" ref="privacyPopup" :maskClick="false">
        <view class="ws-privacy-popup" :style="rootStyle">
            <view class="ws-privacy-popup__header">
                <!--标题-->
                <view class="ws-picker__title">{{ title }}</view>
            </view>
            <view class="ws-privacy-popup__container">
                <text>{{ desc }}</text>
                <text class="ws-privacy-popup__container-protocol" :style="protocolStyle"
                    @click="openPrivacyContract">{{ privacyContractName||protocol }}</text>
                <text>{{ subDesc }}</text>
            </view>
            <view class="ws-privacy-popup__footer">
                <button class="is-agree" :style="agreeStyle" id="agree-btn" open-type="agreePrivacyAuthorization"
                    @agreeprivacyauthorization="handleAgree">
                    {{agreeText}}
                </button>
                <button class="is-disagree" id="disagree-btn" @click="handleDisagree">
                    {{disagreeText}}
                </button>
            </view>
        </view>
    </uni-popup>
</template>
<script>
    import {
        getContext,
        getComponent
    } from './util'
    const privacyResolves = new Set() // onNeedPrivacyAuthorization的reslove
    let privacyHandler = null
    // 注册监听
    if (uni.onNeedPrivacyAuthorization) {
        uni.onNeedPrivacyAuthorization((resolve) => {
            if (typeof privacyHandler === 'function') {
                privacyHandler(resolve)
            }
        })
    }
    export default {
        name: 'wsWxPrivacy',
        emits: ['disagree', 'agree'],
        props: {
            // 标题
            title: {
                type: String,
                default: '用户隐私保护提示'
            },
            // 描述
            desc: {
                type: String,
                default: '感谢您使用本应用,您使用本应用的服务之前请仔细阅读并同意'
            },
            // 自定义隐私保护指引名称
            protocol: {
                type: String,
                default: '《用户隐私保护指引》'
            },
            // 是否自动获取隐私保护指引名称(开启后调用getPrivacySetting获取名称)
            enableAutoProtocol: {
                type: Boolean,
                default: false, // 默认为使用自定义隐私指引名称
            },
            // 子描述
            subDesc: {
                type: String,
                default: '。当您点击同意并开始使用产品服务时,即表示你已理解并同意该条款内容,该条款将对您产生法律约束力。如您拒绝,将无法使用相应服务。'
            },
            /**
             * 控制是否可以点击不同意按钮并显示提示。
             * 如果设置为 true,用户可以点击不同意按钮执行后续逻辑。
             * 如果设置为 false,点击不同意按钮会显示提示信息,但不会执行后续逻辑。
             * 默认为 true
             */
            disagreeEnabled: {
                type: Boolean,
                default: true, // 默认为可以点击
            },
            /**
             * 配置不同意按钮的提示消息内容。
             */
            disagreePromptText: {
                type: String,
                default: '请先仔细阅读并同意隐私协议', // 默认提示消息
            },
            // 拒绝按钮文字
            disagreeText: {
                type: String,
                default: '不同意'
            },
            // 同意按钮文字
            agreeText: {
                type: String,
                default: '同意并继续'
            },
            // 自定义背景颜色
            bgColor: {
                type: String,
                default: ''
            },
            // 自定义主题颜色(控制同意按钮和隐私协议名称的颜色)
            themeColor: {
                type: String,
                default: ''
            }
        },
        data() {
            return {
                privacyContractName: '',
            }
        },
        computed: {
            rootStyle() {
                if (this.bgColor) {
                    return `background:${this.bgColor}`
                } else {
                    return ''
                }
            },
            protocolStyle() {
                if (this.themeColor) {
                    return `color:${this.themeColor}`
                } else {
                    return ''
                }
            },
            agreeStyle() {
                if (this.themeColor) {
                    return `background:${this.themeColor}`
                } else {
                    return ''
                }
            }
        },
        created() {
            privacyHandler = (resolve) => {
                const context = getContext()
                const privacyPopup = getComponent(context, '#privacy-popup')
                if (privacyPopup) {
                    const privacy = getComponent(privacyPopup, '#privacy')
                    if (privacy && privacy.open) {
                        privacy.open()
                    }
                }
                privacyResolves.add(resolve)
            }
            if (this.enableAutoProtocol && uni.getPrivacySetting) {
                uni.getPrivacySetting({
                    success: res => {
                        if (res.privacyContractName) {
                            this.privacyContractName = res.privacyContractName
                        }
                    },
                    fail: () => {},
                    complete: () => {}
                })
            }
        },
        methods: {
            /**
             * 打开隐私协议
             */
            openPrivacyContract() {
                wx.openPrivacyContract({
                    success: (res) => {
                        console.log('openPrivacyContract success')
                    },
                    fail: (res) => {
                        console.error('openPrivacyContract fail', res)
                    }
                })
            },
            /**
             * 拒绝隐私协议
             */
            handleDisagree() {
                if (this.disagreeEnabled) {
                    this.$refs.privacyPopup.close()
                    privacyResolves.forEach((resolve) => {
                        resolve({
                            event: 'disagree'
                        })
                    })
                    privacyResolves.clear()
                    this.$emit('disagree')
                } else {
                    uni.showToast({
                        icon: 'none',
                        title: this.disagreePromptText
                    })
                }
            },
            /**
             * 同意隐私协议
             */
            handleAgree() {
                this.$refs.privacyPopup.close()
                privacyResolves.forEach((resolve) => {
                    resolve({
                        event: 'agree',
                        buttonId: 'agree-btn'
                    })
                })
                privacyResolves.clear()
                this.$emit('agree')
            }
        }
    }
</script>
<style lang="scss" scoped>
    .ws-privacy-popup {
        padding: 48rpx;
        box-sizing: border-box;
        overflow: hidden;
        width: 560rpx;
        background: linear-gradient(180deg, #e5edff 0%, #ffffff 100%);
        border-radius: 24rpx;
        &__header {
            display: flex;
            align-items: center;
            justify-content: center;
            width: 100%;
            height: 52rpx;
            font-size: 36rpx;
            font-family: PingFangSC-Medium, PingFang SC;
            font-weight: 550;
            color: #1a1a1a;
            line-height: 52rpx;
            margin-bottom: 48rpx;
        }
        &__container {
            width: 100%;
            box-sizing: border-box;
            font-size: 28rpx;
            font-family: PingFangSC-Regular, PingFang SC;
            font-weight: 400;
            color: #333333;
            line-height: 48rpx;
            margin-bottom: 48rpx;
            &-protocol {
                font-weight: 550;
                color: #4D80F0;
            }
        }
        &__footer {
            display: flex;
            flex-direction: column;
            .is-disagree,
            .is-agree {
                width: 100%;
                height: 88rpx;
                background: #ffffff;
                border-radius: 44rpx;
                font-size: 32rpx;
                font-family: PingFangSC-Regular, PingFang SC;
                font-weight: 400;
                color: #666666;
            }
            .is-agree {
                background: #4D80F0;
                color: #ffffff;
                margin-bottom: 18rpx;
            }
            button {
                border: none;
                outline: none;
                &::after {
                    border: none;
                }
            }
        }
    }
</style>
uni_modules/ws-wx-privacy/package.json
New file
@@ -0,0 +1,85 @@
{
  "id": "ws-wx-privacy",
  "displayName": "ws-wx-privacy 微信隐私保护弹出框 隐私协议弹出框",
  "version": "1.0.10",
  "description": "微信隐私保护弹出框 隐私协议弹出框,支持vue2和vue3,支持自定义文字和颜色 ws-wx-privacy,有问题评论区沟通,恶意差评走开。",
  "keywords": [
    "微信小程序",
    "隐私保护",
    "隐私保护指引",
    "隐私协议",
    "隐私"
],
  "repository": "",
  "engines": {
    "HBuilderX": "^3.8.6"
  },
  "dcloudext": {
    "type": "component-vue",
    "sale": {
      "regular": {
        "price": "0.00"
      },
      "sourcecode": {
        "price": "0.00"
      }
    },
    "contact": {
      "qq": ""
    },
    "declaration": {
      "ads": "无",
      "data": "插件不采集任何数据",
      "permissions": "无"
    },
    "npmurl": ""
  },
  "uni_modules": {
    "dependencies": ["uni-popup"],
    "encrypt": [],
    "platforms": {
      "cloud": {
        "tcb": "y",
        "aliyun": "y"
      },
      "client": {
        "Vue": {
          "vue2": "y",
          "vue3": "y"
        },
        "App": {
          "app-vue": "n",
          "app-nvue": "n"
        },
        "H5-mobile": {
          "Safari": "n",
          "Android Browser": "n",
          "微信浏览器(Android)": "n",
          "QQ浏览器(Android)": "n"
        },
        "H5-pc": {
          "Chrome": "n",
          "IE": "n",
          "Edge": "n",
          "Firefox": "n",
          "Safari": "n"
        },
        "小程序": {
          "微信": "y",
          "阿里": "n",
          "百度": "n",
          "字节跳动": "n",
          "QQ": "n",
          "钉钉": "n",
          "快手": "n",
          "飞书": "n",
          "京东": "n"
        },
        "快应用": {
          "华为": "n",
          "联盟": "n"
        }
      }
    }
  }
}
uni_modules/ws-wx-privacy/readme.md
New file
@@ -0,0 +1,122 @@
# ws-wx-privacy  隐私保护弹窗组件
本组件基于微信小程序官方政策开发,[关于小程序隐私保护指引设置的公告](https://developers.weixin.qq.com/community/develop/doc/00042e3ef54940ce8520e38db61801?blockType=1&page=1#comment-list)
`wsWxPrivacy` 组件用于显示隐私保护提示的弹窗,用户可以选择同意或不同意隐私协议。
# 请注意 微信后台的用户隐私保护指引必须审核通过之后才可以调试
### 注意事项
1. 请使用HbuilderX导入,因为本组件依赖`uni-popup`直接下载会丢失依赖。
2. 部分隐私接口需要用`wx.requirePrivacyAuthorize`接口包装,请仔细阅读[微信小程序隐私协议开发指南](https://developers.weixin.qq.com/miniprogram/dev/framework/user-privacy/PrivacyAuthorize.html)
3. popup可以自行替换为自己使用的组件,以减小小程序包体积,不过需要注意使用`util.js`中的方法获取组件并打开,使用ref可能会因为this指向的问题导致无法打开弹出框
4. 导入demo项目请选择vue3版本,导入组件不限。
### 隐私指引调试配置(必须配置完成才可以顺利调试)
执行完以下三个步骤之后才可以顺利调试隐私指引内容,如有问题请先自查下边三个步骤是否都完成
#### 1.更新用户隐私保护指引(必需)
小程序管理员或开发者可以根据具体小程序涉及到的[隐私相关接口](https://developers.weixin.qq.com/miniprogram/dev/framework/user-privacy/miniprogram-intro.html)来更新微信小程序后台的`用户隐私保护指引`,更新并审核通过后就可以进行相关的开发调试工作。
`需要注意的是,仅有在指引中声明所处理的用户信息,才可以调用平台提供的对应接口或组件。若未声明,对应接口或组件将直接禁用`
`微信后台的用户隐私保护指引必须审核通过之后才可以调试,必须审核通过!审核通过!审核通过!`
#### 2.配置调试字段__usePrivacyCheck__ (必需)
参考 [微信小程序隐私协议开发指南](https://developers.weixin.qq.com/miniprogram/dev/framework/user-privacy/PrivacyAuthorize.html)中的介绍
目前 `getPrivacySetting`、`onNeedPrivacyAuthorization`、`requirePrivacyAuthorize` 等接口目前可以正常接入调试。调试说明:
- 在 2023 年 9 月 15 号之前,在 app.json 中配置 `"__usePrivacyCheck__": true` 后,会启用隐私相关功能,如果不配置或者配置为 false 则不会启用。
- 在 2023 年 9 月 15 号之后,不论 app.json 中是否有配置 **usePrivacyCheck**,隐私相关功能都会启用。
所以在基于`uni-app`开发时,我们在 2023 年 9 月 15 号之前进行相关开发调试则需要在`manifest.json`文件`mp-weixin`中添加`"__usePrivacyCheck__": true`
#### 3.配置微信开发工具基础库(必需)
将调试基础库改为3.0.0以上。具体路径为:
微信开发者工具->详情->本地设置->调试基础库
#### 4.如果三个步骤都做完了,还是无法调起弹出框
可以使用getPrivacySetting接口查询一下是否已经授权,已经授权就不会调起弹出框了,这时候需要将缓存清除。
### Props
| 属性                | 类型        | 默认值                            | 描述                                                                                                                                                                    |
|------------        |--------    |----------------------------------    |------------------------------------------------------------------------------------------                                                                                |
| title                | String    | 用户隐私保护提示                    | 弹窗标题。                                                                                                                                                            |
| desc                | String    | 感谢您使用本应用,...                | 弹窗描述。                                                                                                                                                            |
| protocol            | String    | 用户隐私保护指引                    | 隐私保护指引的名称。                                                                                                                                                    |
| subDesc            | String    | 。当您点击同意并开始时...            | 子描述,进一步解释同意隐私协议的含义。                                                                                                                                |
| disagreeEnabled    | Boolean    | true                                | 是否允许拒绝。控制是否可以点击不同意按钮并执行后续逻辑。如果设置为 true,用户可以点击不同意按钮执行后续逻辑。如果设置为 false,点击不同意按钮会显示提示信息,但不会执行后续逻辑。    |
| disagreePromptText| String    | 请先仔细阅读并同意隐私协议        | 不同意按钮的提示消息内容。                                                                                                                                        |
| bgColor           | String   |  -                                | 自定义背景颜色。                                                                                                                                   |
| themeColor        | String   |  -                               | 自定义主题颜色(控制同意按钮和隐私协议名称的颜色)。                                                                                              |
### Events
| 事件      | 参数                                   | 描述                                                                                   |
|-----------|----------------------------------------|----------------------------------------------------------------------------------------|
| disagree  | 无参数                                 | 用户选择不同意隐私协议时触发的事件。                                                       |
| agree     | 无参数                                 | 用户选择同意隐私协议时触发的事件。                                                       |
### 使用示例
请确保你已经阅读或了解过[uni-modules 规范](https://uniapp.dcloud.net.cn/plugin/uni_modules.html#uni-modules)
使用hbuildX导入组件后可以直接使用,无需import。
```vue
<template>
  <view>
  <!-- #ifdef MP-WEIXIN -->
  <ws-wx-privacy id="privacy-popup"></ws-wx-privacy>
  <!-- #endif -->
  </view>
</template>
<script>
export default {
  components: {},
  methods: {
    handleDisagree() {
      // 处理用户不同意隐私协议的逻辑
    },
    handleAgree() {
      // 处理用户同意隐私协议的逻辑
    }
  }
}
</script>
```
请注意,上述示例中的组件路径和事件处理逻辑可能需要根据实际情况进行调整。其中`id="privacy-popup"`必填,目的是为了解决多个连续页面同时引入,造成的this指向错误的问题。
## 授权协议
```
1. 企业或个人可以免费商用。
2. 企业或个人不得复制、改造传播并分发本组件。
3. 如果你想分享本组件及教程,请保留本组件的下载地址以及作者名称。
```
## 常见问题
### 1.弹出框弹不出来?
#### 一、请先自检"隐私指引调试配置(必须配置完成才可以顺利调试)"章节的配置是否完成(隐私协议一定要包含调用的API且隐私协议要审核通过)
#### 二、然后看看微信开发者工具是否最新
#### 三、其次使用getPrivacySetting接口看一下是否已经授权过了
### 2.示例项目跑不起来?
#### 请先看看是否选择了vue3下载,如果不是重新选择vue3下载。
### 3.组件是否支持vue2
#### 支持
### 4.chooseImage同意了但是没有弹出选图片的界面?
#### 也是参考常见问题1,隐私协议是否审核通过且包含选取图片的API
### 5.如何知道隐私协议里面该配哪些权限
#### 在[隐私相关接口](https://developers.weixin.qq.com/miniprogram/dev/framework/user-privacy/miniprogram-intro.html)里面找你用到的API,例如uni.chooseImage对应“收集你选中的照片或视频信息    ”
## Vue3高颜值组件库推荐
这里推荐一下[wot-design-uni](https://ext.dcloud.net.cn/plugin?id=13889),支持暗黑模式自定义主题的高颜值组件库,提供50+组件。
utils/websoket.js
New file
@@ -0,0 +1,91 @@
class websocketUtil {
    constructor(url, time) {
        this.is_open_socket = false //避免重复连接
        this.url = url //地址
        this.data = null
        //心跳检测
        this.timeout = time //多少秒执行检测
        this.heartbeatInterval = null //检测服务器端是否还活着
        this.reconnectTimeOut = null //重连之后多久再次重连
        try {
            return this.connectSocketInit()
        } catch (e) {
            console.log('catch');
            this.is_open_socket = false
            this.reconnect();
        }
    }
    // 进入这个页面的时候创建websocket连接【整个页面随时使用】
    connectSocketInit() {
        this.socketTask = uni.connectSocket({
            url: this.url,
            success: () => {
                console.log("正准备建立websocket中...");
                // 返回实例
                return this.socketTask
            },
        });
        this.socketTask.onOpen((res) => {
            console.log("WebSocket连接正常!");
            clearTimeout(this.reconnectTimeOut)
            clearTimeout(this.heartbeatInterval)
            this.is_open_socket = true;
            this.start();
            // 注:只有连接正常打开中 ,才能正常收到消息
            this.socketTask.onMessage((res) => {
                console.log(res.data)
            });
        })
        // 监听连接失败,这里代码我注释掉的原因是因为如果服务器关闭后,和下面的onclose方法一起发起重连操作,这样会导致重复连接
        // uni.onSocketError((res) => {
        //     console.log('WebSocket连接打开失败,请检查!');
        //     this.is_open_socket = false;
        //     this.reconnect();
        // });
        // 这里仅是事件监听【如果socket关闭了会执行】
        this.socketTask.onClose(() => {
            console.log("已经被关闭了")
            this.is_open_socket = false;
            this.reconnect();
        })
    }
    //发送消息
    send(value) {
        // 注:只有连接正常打开中 ,才能正常成功发送消息
        this.socketTask.send({
            data: value,
            async success() {
                console.log("消息发送成功");
            },
        });
    }
    //开启心跳检测
    start() {
        this.heartbeatInterval = setTimeout(() => {
            this.data = {
                value: "传输内容",
                method: "方法名称"
            }
            console.log(this.data)
            this.send(JSON.stringify(this.data));
        }, this.timeout)
    }
    //重新连接
    reconnect() {
        //停止发送心跳
        clearInterval(this.heartbeatInterval)
        //如果不是人为关闭的话,进行重连
        if (!this.is_open_socket) {
            this.reconnectTimeOut = setTimeout(() => {
                this.connectSocketInit();
            }, 3000)
        }
    }
    //外部获取消息
    getMessage(callback) {
        this.socketTask.onMessage((res) => {
            return callback(res)
        })
    }
}
module.exports = websocketUtil