修改首页部分逻辑;修复部分已知问题;首页新增社区菜单;公告详情页面样式调整
9 files modified
48 files added
4519 ■■■■■ changed files
common/setting.js 3 ●●●● patch | view | raw | blame | history
pages/home/index.vue 159 ●●●● patch | view | raw | blame | history
pages/user/center.vue 2 ●●● patch | view | raw | blame | history
store/index.js 2 ●●● patch | view | raw | blame | history
subPackage/article/detail.vue 140 ●●●● patch | view | raw | blame | history
subPackage/house/family/index.vue 23 ●●●●● patch | view | raw | blame | history
subPackage/house/roomDetails/index.vue 12 ●●●●● patch | view | raw | blame | history
subPackage/label/form.vue 1 ●●●● patch | view | raw | blame | history
subPackage/label/index.vue 19 ●●●● patch | view | raw | blame | history
uni_modules/uv-line-progress/changelog.md 7 ●●●●● patch | view | raw | blame | history
uni_modules/uv-line-progress/components/uv-line-progress/props.js 29 ●●●●● patch | view | raw | blame | history
uni_modules/uv-line-progress/components/uv-line-progress/uv-line-progress.vue 146 ●●●●● patch | view | raw | blame | history
uni_modules/uv-line-progress/package.json 87 ●●●●● patch | view | raw | blame | history
uni_modules/uv-line-progress/readme.md 11 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/changelog.md 68 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/components/uv-ui-tools/uv-ui-tools.vue 6 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/index.js 79 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/index.scss 7 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/config/config.js 34 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/css/color.scss 32 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/css/common.scss 100 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/css/components.scss 23 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/css/variable.scss 111 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/css/vue.scss 40 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/function/colorGradient.js 134 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/function/debounce.js 29 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/function/digit.js 167 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/function/index.js 734 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/function/platform.js 75 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/function/test.js 287 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/function/throttle.js 30 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/adapters/index.js 97 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/core/InterceptorManager.js 50 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/core/Request.js 198 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/core/buildFullPath.js 20 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/core/defaults.js 29 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/core/dispatchRequest.js 3 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/core/mergeConfig.js 103 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/core/settle.js 16 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/helpers/buildURL.js 69 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/helpers/combineURLs.js 14 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/helpers/isAbsoluteURL.js 14 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/index.d.ts 116 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/index.js 3 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/utils.js 131 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/luch-request/utils/clone.js 264 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/mixin/button.js 13 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/mixin/mixin.js 166 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/mixin/mpMixin.js 8 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/mixin/mpShare.js 13 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/mixin/openType.js 47 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/mixin/touch.js 59 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/util/dayjs.js 216 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/libs/util/route.js 126 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/package.json 81 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/readme.md 23 ●●●●● patch | view | raw | blame | history
uni_modules/uv-ui-tools/theme.scss 43 ●●●●● patch | view | raw | blame | history
common/setting.js
@@ -12,7 +12,8 @@
    // devUrl: 'https://sk.hubeishuiyi.cn',
    // devUrl: 'http://192.168.1.156:9528',
    // devUrl:'http://192.168.1.50:9528',
    devUrl: 'http://192.168.0.102:9528',
    // devUrl: 'http://192.168.0.102:9528',
    devUrl:'https://srgdjczzxtpt.com:2080/api',
    // devUrl: 'http://192.168.0.102:9528',
    // devUrl: 'https://srgdjczzxtpt.com:2080/api',
    minioBaseUrl: "https://srgdjczzxtpt.com:2080/gminio/jczz/",
pages/home/index.vue
@@ -10,7 +10,7 @@
        </u-navbar>
        <view class="content">
            <view class="swiper mb-30">
                <u-swiper :list="bannerList"    height="260rpx"></u-swiper>
                <u-swiper :list="bannerList" height="260rpx"></u-swiper>
            </view>
            <view class="">
                <u-grid :border="false" :col="liveList.length">
@@ -68,7 +68,23 @@
                </view>
            </view>
            <view class="" v-if="jobList.length">
            <view class="mb-20"    v-if="roleType == 2">
                <caption-row title="社区" />
                <view class="mt-20">
                    <u-grid :border="false" col="4">
                        <u-grid-item @click="navigatorPage('/subPackage/article/list')">
                            <u-icon name="/static/icon/nav-09.png" width="90rpx" height="70rpx"></u-icon>
                            <text class="f-26  mt-10">公益报名</text>
                        </u-grid-item>
                        <u-grid-item @click="navigatorPage('/subPackage/article/list')">
                            <u-icon name="/static/icon/nav-10.png" width="90rpx" height="70rpx"></u-icon>
                            <text class="f-26  mt-10">选举调查</text>
                        </u-grid-item>
                    </u-grid>
                </view>
            </view>
            <view class="" v-if="jobList.length && roleType == 1">
                <caption-row title="工作台" />
                <u-grid :border="false" col="4">
                    <u-grid-item @click="navigatorPage(item.path)" v-for="(item, index) in jobList" :index="index"
@@ -80,6 +96,23 @@
                    </u-grid-item>
                </u-grid>
            </view>
            <view class="" v-if="roleType == 2 && curSelectSite.addressType == 2">
                <caption-row title="工作台" />
                <u-grid :border="false" col="4">
                    <u-grid-item @click="navigatorPage('/subPackage/workbench/views/editExamine')"
                        :customStyle="{paddingTop:20+'rpx'}">
                        <view class="grid-item flex f-d-c a-i-c"
                            style="background:linear-gradient(133deg, #F9FAFE 0%, rgba(249,250,254,0) 100%)">
                            <u-icon name="/static/icon/menu-05.png" width="60rpx" height="60rpx"></u-icon>
                            <text class="grid-text f-24 fw mt-10">场所维护</text>
                        </view>
                    </u-grid-item>
                </u-grid>
            </view>
            <view class="notic" v-if="isShowNotice">
                <caption-row title="通知公告" :is-link="true" url="/subPackage/article/list" />
@@ -107,7 +140,7 @@
        getBannerList
    } from "@/api/system/index"
    import {
        uploadUrl
        minioBaseUrl
    } from "@/common/setting"
    import noticeList from "@/components/noticeList/noticeList";
    import captionRow from "@/components/caption/caption.vue"
@@ -119,46 +152,6 @@
        },
        data() {
            return {
                swiperList: [
                    'http://220.176.210.217:9000/jczz/upload/20231111/cc093afc5e21639ef5b251eb865c24f4.png',
                    'http://220.176.210.217:9000/jczz/upload/20231111/cc093afc5e21639ef5b251eb865c24f4.png',
                    'http://220.176.210.217:9000/jczz/upload/20231111/cc093afc5e21639ef5b251eb865c24f4.png'
                ],
                cellList: [{
                        icon: "/static/icon/nav-01.png",
                        text: "标签报事",
                        path: '/subPackage/label/index'
                    },
                    {
                        icon: "/static/icon/nav-02.png",
                        text: "报事报修",
                        path: '/subPackage/bs/views/repair'
                    },
                    {
                        icon: "/static/icon/nav-03.png",
                        text: "便民热线",
                        path: '/subPackage/hotLine/index'
                    },
                ],
                navList: [{
                        title: "物业",
                        name: "张三",
                        icon: "/static/icon/tel-green.png",
                        bgImg: "/static/icon/nav-bg-01.png"
                    },
                    {
                        title: "综治网格",
                        name: "张三",
                        icon: "/static/icon/tel-orange.png",
                        bgImg: "/static/icon/nav-bg-02.png"
                    },
                    {
                        title: "公安网格",
                        name: "张三",
                        icon: "/static/icon/tel-blue.png",
                        bgImg: "/static/icon/nav-bg-03.png"
                    },
                ],
                jobList: [{
                        icon: "/static/icon/menu-01.png",
                        text: "场所记录",
@@ -253,27 +246,13 @@
                    //     }
                    // }
                ],
                taskList: [{
                        icon: "/static/icon/cell-01.png",
                        title: "综治任务",
                        status: "待处理0"
                    },
                    {
                        icon: "/static/icon/cell-02.png",
                        title: "住建任务",
                        status: "待处理0"
                    },
                    {
                        icon: "/static/icon/cell-03.png",
                        title: "公安任务",
                        status: "待处理0"
                    }
                ],
                roleType: 2, // 1网格员/系统管理员 、  2居民
                roleTypeName: "街道社区网格",
                selectBoxShow: false,
                siteColumns: [],
                curSelectSite: {},
                curSelectSite: {
                    addressType: 1
                },
                columnData: [],
                houseDataList: [{
                    title: '-',
@@ -286,9 +265,8 @@
                liveList: [], //生活菜单
                contactList: [], //便民服务联系方式
                isShowHouse: false, //显示楼盘表
                isShowNotice: false ,//显示通知公告
                bannerList:[]
                isShowNotice: false, //显示通知公告
                bannerList: []
            }
        },
@@ -297,16 +275,19 @@
            this.init()
            uni.$on('refresh', () => {
                this.selectSiteIndex = [0, 0]
                this.curSelectSite = {addressType:1}
                this.init()
            })
        },
        onReady() {
            this.$store.dispatch("getMenuList").then(res => {
                this.getMenuList();
            });
            if(!uni.getStorageSync("userInfo")){
                uni.redirectTo({
                    url: "/pages/login/login-account"
                })
            }
        },
        computed: {
            hasMenu() {
                return (name) => {
@@ -335,30 +316,29 @@
                handler(newVal, oldVal) {
                    if (newVal) {
                        this.$store.dispatch("getMenuList").then(res => {
                            this.getMenuList();
                            this.getMenuList()
                        });
                    }
                },
                deep: true,
                immediate: true
                // immediate: true
            }
        },
        onPullDownRefresh() {
            this.$store.dispatch("getMenuList").then(res => {
                this.getMenuList();
                this.getMenuList()
            });
            setTimeout(()=>{
            setTimeout(() => {
                uni.stopPullDownRefresh();
            },300)
            }, 300)
        },
        methods: {
            //初始化
            init() {
                this.curSelectSite = {}
                this.selectRole = uni.getStorageSync("activeRole")
                // this.getMenuList()
                this.getMenuList()
                this.getSiteList()
                // this.getNoticeList()
            },
@@ -371,7 +351,7 @@
                    let data = res.data.records;
                    if (data.length) {
                        for (let i of data) {
                            i.url = `${uploadUrl}${i.url}`
                            i.url = `${minioBaseUrl}${i.url}`
                        }
                        this.bannerList = data;
                    }
@@ -484,14 +464,24 @@
            //获取首页菜单
            getMenuList() {
                this.menuList = uni.getStorageSync("menu")
                this.liveList = this.hasMenu('生活')
                this.contactList = this.hasMenu('快捷拨号');
                this.jobList = this.hasMenu('工作台')
                if (this.menuList.length) {
                    this.getHouseMenu();
                    this.getNoticeMenu();
                }
                // this.$store.dispatch("getMenuList").then(res => {
                //     this.menuList = res;
                //     setTimeout(() => {
                //         this.liveList = this.hasMenu('生活')
                //         this.contactList = this.hasMenu('快捷拨号');
                //         this.jobList = this.hasMenu('工作台')
                //         this.getHouseMenu();
                //         this.getNoticeMenu();
                //     }, 100)
                // });
                    this.menuList = uni.getStorageSync("menu")
                    this.liveList = this.hasMenu('生活')
                    this.contactList = this.hasMenu('快捷拨号');
                    this.jobList = this.hasMenu('工作台')
                    if (this.menuList.length) {
                        this.getHouseMenu();
                        this.getNoticeMenu();
                    }
            },
            getHouseMenu() {
@@ -511,7 +501,6 @@
            // 页面跳转
            pushPage(item) {
                console.log(item, 8888)
                const {
                    id,
                    title,
pages/user/center.vue
@@ -71,7 +71,7 @@
                navList: [{
                        icon: "/static/icon/nav-center-01.png",
                        title: "我的房屋",
                        url: ""
                        url: "/subPackage/house/roomDetails/index"
                    },
                    {
                        icon: "/static/icon/nav-center-02.png",
store/index.js
@@ -151,7 +151,7 @@
                                }
                            }
                        }
                        resolve("success")
                        resolve(menu)
                    }else {
                        reject("fail")
                    }
subPackage/article/detail.vue
@@ -1,25 +1,88 @@
<template>
    <view class="container">
        <view class="f-32 fw">{{detailInfo.title}}</view>
        <view class="row flex j-c-s-b a-i-c c-99">
            <view class="flex">
                <!-- <view class="flex a-i-c mr-40">
    <view class="">
        <view class="container">
            <view class="f-32 fw">{{detailInfo.title}}</view>
            <view class="row flex j-c-s-b a-i-c c-99">
                <view class="flex">
                    <!-- <view class="flex a-i-c mr-40">
                    <u-icon name="thumb-up-fill" color="#CECECE"></u-icon>
                    <text class="f-22 ml-10">120</text>
                </view> -->
                <view class="flex a-i-c">
                    <tag-cell  :text="detailInfo.dictValue"></tag-cell>
                    <view class="flex a-i-c ml-20">
                        <u-icon name="eye-fill" color="#CECECE"></u-icon>
                        <text class="f-22 ml-10">{{detailInfo.viewNumber}}</text>
                    <view class="flex a-i-c">
                        <tag-cell :text="detailInfo.dictValue"></tag-cell>
                        <view class="flex a-i-c ml-20">
                            <u-icon name="eye-fill" color="#CECECE"></u-icon>
                            <text class="f-22 ml-10">{{detailInfo.viewNumber}}</text>
                        </view>
                    </view>
                </view>
                <text class="f-24">{{detailInfo.createTime}}</text>
            </view>
            <view class="content f-28">
                <u-parse :content="detailInfo.content"></u-parse>
            </view>
        </view>
        <view class=""  v-if="false">
            <view class="explain-row c-main  f-28">
                报名和投票先指定好房屋
            </view>
            <view class="select-row flex j-c-s-b a-i-c">
                <text class="f-28">指定房屋</text>
                <view class="select-row-content flex a-i-c">
                    <text class="f-26 c-99 mr-20"></text>
                    <u-icon name="arrow-right"  color="#999"></u-icon>
                </view>
            </view>
            <!-- <view class="apply">
                <view class="flex f-d-c a-i-c">
                    <view class="flex mb-20">
                        <image class="apply-user" src="/static/icon/nav-02.png" mode=""></image>
                        <image class="apply-user" src="/static/icon/nav-02.png" mode=""></image>
                        <image class="apply-user" src="/static/icon/nav-02.png" mode=""></image>
                    </view>
                    <text class="f-28 c-99">共3人</text>
                </view>
                <button class="apply-btn bgc-main c-ff f-30">报名</button>
            </view> -->
            <view class="mt-20"  >
                <view class="explain-row c-main  f-28">
                    投票进行中
                </view>
                <view class="vote bgc-ff">
                    <view class="vote-title f-30 fw">
                        是否同意安装充电桩
                    </view>
                    <view class="">
                        <u-radio-group
                            iconPlacement="right"
                            activeColor="#017BFC"
                            placement="column"
                            >
                            <view class="mb-20">
                                <u-radio  label="同意"  ></u-radio>
                                <view class="flex j-c-s-b a-i-c mt-10">
                                    <uv-line-progress :percentage="30"  height="10" ></uv-line-progress>
                                    <text class="c-99 f-28">10</text>
                                </view>
                            </view>
                            <view class="">
                                <u-radio  label="不同意"  ></u-radio>
                                <view class="flex j-c-s-b a-i-c mt-10">
                                    <uv-line-progress   height="10" ></uv-line-progress>
                                    <text class="c-99 f-28">10</text>
                                </view>
                            </view>
                        </u-radio-group>
                    </view>
                </view>
            </view>
            <text class="f-24">{{detailInfo.createTime}}</text>
        </view>
        <view class="content f-28">
            <u-parse :content="detailInfo.content"></u-parse>
        </view>
        <!-- <view class="action flex flex_base">
            <view class="action-item flex_base bgc-main">
                <u-icon name="thumb-up-fill"  size="22"  color="#fff"></u-icon>
@@ -37,7 +100,7 @@
    } from '@/api/article/article.js'
    import tagCell from '@/components/noticeList/tagCell.vue'
    export default {
        components:{
        components: {
            tagCell
        },
        data() {
@@ -60,9 +123,13 @@
    }
</script>
<style>
<style lang="scss">
    page{
        background-color: #f5f5f5;
    }
    .container {
        padding: 0 30rpx;
        background-color: #fff;
    }
    .row {
@@ -70,7 +137,7 @@
    }
    .content {
        margin-bottom: 90rpx;
        padding-bottom:30rpx;
    }
    .action-item {
@@ -83,4 +150,43 @@
    .bgc-gray {
        background-color: #C4C4C4;
    }
    .explain-row {
        background-color: rgb(236, 244, 255);
        padding: 20rpx;
    }
    .select-row{
        padding:30rpx;
        background-color:#fff;
        box-shadow: 0px 3px 3px rgba(0, 0, 0, 0.2);
        .select-row-content{
            flex:1;
            justify-content: flex-end;
        }
    }
    .apply{
        padding:30rpx;
        .apply-user{
            width:80rpx;
            height:80rpx;
            border-radius: 50%;
            margin-right:-15rpx;
            border:1px solid #fff;
        }
        .apply-btn{
            width:100%;
            height:80rpx;
            line-height: 80rpx;
            margin: 30rpx auto 0;
            border:none;
            border-radius: 40rpx;
        }
    }
    .vote{
        padding:0 40rpx 30rpx;
        .vote-title{
            padding:30rpx 0;
        }
    }
</style>
subPackage/house/family/index.vue
@@ -21,8 +21,8 @@
                            <view class="room-content">
                                <u-grid class="flex flex-wrap" col="2" :border="true">
                                    <u-grid-item v-for="(scItem, scIndex) in cItem.children" :key="scItem.id"
                                        @click="pushPage($event, scItem)">
                                        <view class="room-box flex f-d-c">
                                         @click="pushPage(scIndex,cItem.children)">
                                        <view class="room-box flex f-d-c"  >
                                            <view class="flex a-i-c j-c-s-b">
                                                <view class="l">
                                                    <u-icon name="/static/icon/person.png" size="16"></u-icon>
@@ -165,12 +165,9 @@
                    })
                }
            },
            pushPage(name, scItem) {
                const {
                    addressCode
                } = scItem
                let url = `/subPackage/house/roomDetails/index?id=${addressCode}`
            pushPage(index,item) {
                let code = item[index].addressCode;
                let url = `/subPackage/house/roomDetails/index?id=${code}`
                this.$u.func.globalNavigator(url, "navTo")
            },
            // 跳转到商铺页面
@@ -265,20 +262,19 @@
            /deep/ .floor-content {
                // border-bottom: 1rpx solid #f5f5f5;
                // border-bottom:2px solid #F9F9FA;
                border-bottom: 1px solid #999;
                border-bottom: 1px solid #DBDBDB;
                .floor-num-box {
                    width: 50rpx;
                    padding: 0 20rpx !important;
                    // background: #FAFBFE;
                    background-color: #F5F5F5;
                    background-color: #F4F4F4;
                }
                .room-content {
                    width: 0;
                    flex: 1;
                    padding: 6rpx;
                    padding: 12rpx;
                    .room-box {
                        padding: 10rpx;
                        justify-content: center;
@@ -287,7 +283,8 @@
                        min-height: 110rpx;
                        box-sizing: border-box;
                        // background-color: #FAFBFE;
                        background-color: rgb(236, 244, 255);
                        // background-color: rgb(236, 244, 255);
                        background-color: #F4F4F4;
                        // margin:2rpx;
                        &>view {
subPackage/house/roomDetails/index.vue
@@ -186,10 +186,14 @@
            }
        },
        onLoad(e) {
            const {
                id
            } = e
            this.currentId = id
            if(e.id){
                const {
                    id
                } = e
                this.currentId = id
            }else {
                this.currentId = uni.getStorageSync("siteInfo").houseCode
            }
        },
        onShow() {
            this.getHouseRentInfoList()
subPackage/label/form.vue
@@ -124,6 +124,7 @@
            }
        },
        onLoad(option) {
            console.log(option);
            if (option.type != 1) {
                this.eventType = option.type;
                uni.setNavigationBarTitle({
subPackage/label/index.vue
@@ -121,15 +121,16 @@
            },
    
            navTo(path,index){
                let type = 1;
                if(index == 1){
                    type == 1
                }else if (index == 3){
                    type = 3
                }else if(index == 4) {
                    type = 2
                }
                this.$u.func.globalNavigator(`${path}?type=${type}`, "navTo")
                // let type = 1;
                // if(index == 1){
                //     type == 1
                // }else if (index == 3){
                //     type = 3
                // }else if(index == 4) {
                //     type = 2
                // }
                // this.$u.func.globalNavigator(`${path}?type=${type}`, "navTo")
                this.$u.func.globalNavigator(path, "navTo")
                
            }
        }
uni_modules/uv-line-progress/changelog.md
New file
@@ -0,0 +1,7 @@
## 1.0.2(2023-06-20)
1. 适配height参数携带单位
## 1.0.1(2023-05-16)
1. 优化组件依赖,修改后无需全局引入,组件导入即可使用
2. 优化部分功能
## 1.0.0(2023-05-10)
uv-line-progress 线形进度条
uni_modules/uv-line-progress/components/uv-line-progress/props.js
New file
@@ -0,0 +1,29 @@
export default {
    props: {
        // 激活部分的颜色
        activeColor: {
            type: String,
            default: '#19be6b'
        },
        inactiveColor: {
            type: String,
            default: '#ececec'
        },
        // 进度百分比,数值
        percentage: {
            type: [String, Number],
            default: 0
        },
        // 是否在进度条内部显示百分比的值
        showText: {
            type: Boolean,
            default: true
        },
        // 进度条的高度,单位px
        height: {
            type: [String, Number],
            default: 12
        },
        ...uni.$uv?.props?.lineProgress
    }
}
uni_modules/uv-line-progress/components/uv-line-progress/uv-line-progress.vue
New file
@@ -0,0 +1,146 @@
<template>
    <view
        class="uv-line-progress"
        :style="[$uv.addStyle(customStyle)]"
    >
        <view
            class="uv-line-progress__background"
            ref="uv-line-progress__background"
            :style="[{
                backgroundColor: inactiveColor,
                height: $uv.addUnit($uv.getPx(height))
            }]"
        >
        </view>
        <view
            class="uv-line-progress__line"
            :style="[progressStyle]"
        >
            <slot>
                <text v-if="showText && percentage >= 10" class="uv-line-progress__text">{{innserPercentage + '%'}}</text>
            </slot>
        </view>
    </view>
</template>
<script>
    import mpMixin from '@/uni_modules/uv-ui-tools/libs/mixin/mpMixin.js'
    import mixin from '@/uni_modules/uv-ui-tools/libs/mixin/mixin.js'
    import props from './props.js';
    // #ifdef APP-NVUE
    const dom = uni.requireNativePlugin('dom')
    // #endif
    /**
     * lineProgress 线型进度条
     * @description 展示操作或任务的当前进度,比如上传文件,是一个线形的进度条。
     * @tutorial https://www.uvui.cn/components/lineProgress.html
     * @property {String}            activeColor        激活部分的颜色 ( 默认 '#19be6b' )
     * @property {String}            inactiveColor    背景色 ( 默认 '#ececec' )
     * @property {String | Number}    percentage        进度百分比,数值 ( 默认 0 )
     * @property {Boolean}            showText        是否在进度条内部显示百分比的值 ( 默认 true )
     * @property {String | Number}    height            进度条的高度,单位px ( 默认 12 )
     *
     * @example <uv-line-progress :percent="70" :show-percent="true"></uv-line-progress>
     */
    export default {
        name: "uv-line-progress",
        mixins: [mpMixin, mixin, props],
        data() {
            return {
                lineWidth: 0,
            }
        },
        watch: {
            percentage(n) {
                this.resizeProgressWidth()
            }
        },
        computed: {
            progressStyle() {
                let style = {}
                style.width = this.lineWidth
                style.backgroundColor = this.activeColor
                style.height = this.$uv.addUnit(this.$uv.getPx(this.height))
                return style
            },
            innserPercentage() {
                // 控制范围在0-100之间
                return this.$uv.range(0, 100, this.percentage)
            }
        },
        mounted() {
            this.init()
        },
        methods: {
            init() {
                this.$uv.sleep(20).then(() => {
                    this.resizeProgressWidth()
                })
            },
            getProgressWidth() {
                // #ifndef APP-NVUE
                return this.$uvGetRect('.uv-line-progress__background')
                // #endif
                // #ifdef APP-NVUE
                // 返回一个promise
                return new Promise(resolve => {
                    dom.getComponentRect(this.$refs['uv-line-progress__background'], (res) => {
                        resolve(res.size)
                    })
                })
                // #endif
            },
            resizeProgressWidth() {
                this.getProgressWidth().then(size => {
                    const {
                        width
                    } = size
                    // 通过设置的percentage值,计算其所占总长度的百分比
                    this.lineWidth = width * this.innserPercentage / 100 + 'px'
                })
            }
        }
    }
</script>
<style lang="scss" scoped>
    @import '@/uni_modules/uv-ui-tools/libs/css/components.scss';
    .uv-line-progress {
        align-items: stretch;
        position: relative;
        @include flex(row);
        flex: 1;
        overflow: hidden;
        border-radius: 100px;
        &__background {
            background-color: #ececec;
            border-radius: 100px;
            flex: 1;
        }
        &__line {
            position: absolute;
            top: 0;
            left: 0;
            bottom: 0;
            align-items: center;
            @include flex(row);
            color: #ffffff;
            border-radius: 100px;
            transition: width 0.5s ease;
            justify-content: flex-end;
        }
        &__text {
            font-size: 10px;
            align-items: center;
            text-align: right;
            color: #FFFFFF;
            margin-right: 5px;
            transform: scale(0.9);
        }
    }
</style>
uni_modules/uv-line-progress/package.json
New file
@@ -0,0 +1,87 @@
{
  "id": "uv-line-progress",
  "displayName": "uv-line-progress 线形进度条  全面兼容小程序、nvue、vue2、vue3等多端",
  "version": "1.0.2",
  "description": "uv-line-progress 该组件展示操作或任务的当前进度,比如上传文件,是一个线形的进度条。",
  "keywords": [
    "uv-line-progress",
    "uvui",
    "uv-ui",
    "progress",
    "进度条"
],
  "repository": "",
  "engines": {
    "HBuilderX": "^3.1.0"
  },
  "dcloudext": {
    "type": "component-vue",
    "sale": {
      "regular": {
        "price": "0.00"
      },
      "sourcecode": {
        "price": "0.00"
      }
    },
    "contact": {
      "qq": ""
    },
    "declaration": {
        "ads": "无",
        "data": "插件不采集任何数据",
        "permissions": "无"
    },
    "npmurl": ""
  },
  "uni_modules": {
    "dependencies": [
            "uv-ui-tools"
        ],
    "encrypt": [],
    "platforms": {
            "cloud": {
                "tcb": "y",
                "aliyun": "y"
            },
            "client": {
                "Vue": {
                    "vue2": "y",
                    "vue3": "y"
                },
                "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",
                    "飞书": "u",
                    "京东": "u"
                },
                "快应用": {
                    "华为": "u",
                    "联盟": "u"
                }
            }
        }
  }
}
uni_modules/uv-line-progress/readme.md
New file
@@ -0,0 +1,11 @@
## LineProgress 线形进度条
> **组件名:uv-line-progress**
展示操作或任务的当前进度,比如上传文件,是一个线形的进度条。
### <a href="https://www.uvui.cn/components/lineProgress.html" target="_blank">查看文档</a>
### [完整示例项目下载 | 关注更多组件](https://ext.dcloud.net.cn/plugin?name=uv-ui)
#### 如使用过程中有任何问题,或者您对uv-ui有一些好的建议,欢迎加入 uv-ui 交流群:<a href="https://ext.dcloud.net.cn/plugin?id=12287" target="_blank">uv-ui</a>、<a href="https://www.uvui.cn/components/addQQGroup.html" target="_blank">官方QQ群</a>
uni_modules/uv-ui-tools/changelog.md
New file
@@ -0,0 +1,68 @@
## 1.1.21(2023-11-10)
1. 1.1.17版本
## 1.1.20(2023-10-30)
1. 1.1.16版本
## 1.1.19(2023-10-13)
1. 兼容vue3
## 1.1.18(2023-10-12)
1. 1.1.15版本
## 1.1.17(2023-09-27)
1. 1.1.14版本发布
## 1.1.16(2023-09-15)
1. 1.1.13版本发布
## 1.1.15(2023-09-15)
1. 更新button.js相关按钮支持open-type="agreePrivacyAuthorization"
## 1.1.14(2023-09-14)
1. 优化dayjs
## 1.1.13(2023-09-13)
1. 优化,$uv中增加unit参数,方便组件中使用
## 1.1.12(2023-09-10)
1. 升级版本
## 1.1.11(2023-09-04)
1. 1.1.11版本
## 1.1.10(2023-08-31)
1. 修复customStyle和customClass存在冲突的问题
## 1.1.9(2023-08-27)
1. 版本升级
2. 优化
## 1.1.8(2023-08-24)
1. 版本升级
## 1.1.7(2023-08-22)
1. 版本升级
## 1.1.6(2023-08-18)
uvui版本:1.1.6
## 1.0.15(2023-08-14)
1. 更新uvui版本号
## 1.0.13(2023-08-06)
1. 优化
## 1.0.12(2023-08-06)
1. 修改版本号
## 1.0.11(2023-08-06)
1. 路由增加events参数
2. 路由拦截修复
## 1.0.10(2023-08-01)
1. 优化
## 1.0.9(2023-06-28)
优化openType.js
## 1.0.8(2023-06-15)
1. 修改支付宝报错的BUG
## 1.0.7(2023-06-07)
1. 解决微信小程序使用uvui提示 Some selectors are not allowed in component wxss, including tag name selectors, ID selectors, and attribute selectors
2. 解决上述提示,需要在uni.scss配置$uvui-nvue-style: false; 然后在APP.vue下面引入uvui内置的基础样式:@import '@/uni_modules/uv-ui-tools/index.scss';
## 1.0.6(2023-06-04)
1.  uv-ui-tools 优化工具组件,兼容更多功能
2.  小程序分享功能优化等
## 1.0.5(2023-06-02)
1. 修改扩展使用mixin中方法的问题
## 1.0.4(2023-05-23)
1. 兼容百度小程序修改bem函数
## 1.0.3(2023-05-16)
1. 优化组件依赖,修改后无需全局引入,组件导入即可使用
2. 优化部分功能
## 1.0.2(2023-05-10)
1. 增加Http请求封装
2. 优化
## 1.0.1(2023-05-04)
1. 修改名称及备注
## 1.0.0(2023-05-04)
1. uv-ui工具集首次发布
uni_modules/uv-ui-tools/components/uv-ui-tools/uv-ui-tools.vue
New file
@@ -0,0 +1,6 @@
<template>
</template>
<script>
</script>
<style>
</style>
uni_modules/uv-ui-tools/index.js
New file
@@ -0,0 +1,79 @@
// 全局挂载引入http相关请求拦截插件
import Request from './libs/luch-request'
// 引入全局mixin
import mixin from './libs/mixin/mixin.js'
// 小程序特有的mixin
import mpMixin from './libs/mixin/mpMixin.js'
// #ifdef MP
import mpShare from '@/uni_modules/uv-ui-tools/libs/mixin/mpShare.js'
// #endif
// 路由封装
import route from './libs/util/route.js'
// 公共工具函数
import * as index from './libs/function/index.js'
// 防抖方法
import debounce from './libs/function/debounce.js'
// 节流方法
import throttle from './libs/function/throttle.js'
// 规则检验
import * as test from './libs/function/test.js'
// 颜色渐变相关,colorGradient-颜色渐变,hexToRgb-十六进制颜色转rgb颜色,rgbToHex-rgb转十六进制
import * as colorGradient from './libs/function/colorGradient.js'
// 配置信息
import config from './libs/config/config.js'
// 平台
import platform from './libs/function/platform'
const $uv = {
    route,
    config,
    test,
    date: index.timeFormat, // 另名date
    ...index,
    colorGradient: colorGradient.colorGradient,
    hexToRgb: colorGradient.hexToRgb,
    rgbToHex: colorGradient.rgbToHex,
    colorToRgba: colorGradient.colorToRgba,
    http: new Request(),
    debounce,
    throttle,
    platform,
    mixin,
    mpMixin
}
uni.$uv = $uv;
const install = (Vue,options={}) => {
        // #ifndef APP-NVUE
        const cloneMixin = index.deepClone(mixin);
        delete cloneMixin?.props?.customClass;
        delete cloneMixin?.props?.customStyle;
        Vue.mixin(cloneMixin);
        // #ifdef MP
        if(options.mpShare){
            Vue.mixin(mpShare);
        }
        // #endif
        // #endif
        // #ifdef VUE2
        // 时间格式化,同时两个名称,date和timeFormat
        Vue.filter('timeFormat', (timestamp, format) => uni.$uv.timeFormat(timestamp, format));
        Vue.filter('date', (timestamp, format) => uni.$uv.timeFormat(timestamp, format));
        // 将多久以前的方法,注入到全局过滤器
        Vue.filter('timeFrom', (timestamp, format) => uni.$uv.timeFrom(timestamp, format));
        // 同时挂载到uni和Vue.prototype中
        // #ifndef APP-NVUE
        // 只有vue,挂载到Vue.prototype才有意义,因为nvue中全局Vue.prototype和Vue.mixin是无效的
        Vue.prototype.$uv = $uv;
        // #endif
        // #endif
        // #ifdef VUE3
        Vue.config.globalProperties.$uv = $uv;
        // #endif
}
export default {
    install
}
uni_modules/uv-ui-tools/index.scss
New file
@@ -0,0 +1,7 @@
// 引入公共基础类
@import "./libs/css/common.scss";
// 非nvue的样式
/* #ifndef APP-NVUE */
@import "./libs/css/vue.scss";
/* #endif */
uni_modules/uv-ui-tools/libs/config/config.js
New file
@@ -0,0 +1,34 @@
// 此版本发布于2023-11-10
const version = '1.1.17'
// 开发环境才提示,生产环境不会提示
if (process.env.NODE_ENV === 'development') {
    console.log(`\n %c uvui V${version} https://www.uvui.cn/ \n\n`, 'color: #ffffff; background: #3c9cff; padding:5px 0; border-radius: 5px;');
}
export default {
    v: version,
    version,
    // 主题名称
    type: [
        'primary',
        'success',
        'info',
        'error',
        'warning'
    ],
    // 颜色部分,本来可以通过scss的:export导出供js使用,但是奈何nvue不支持
    color: {
        'uv-primary': '#2979ff',
        'uv-warning': '#ff9900',
        'uv-success': '#19be6b',
        'uv-error': '#fa3534',
        'uv-info': '#909399',
        'uv-main-color': '#303133',
        'uv-content-color': '#606266',
        'uv-tips-color': '#909399',
        'uv-light-color': '#c0c4cc'
    },
    // 默认单位,可以通过配置为rpx,那么在用于传入组件大小参数为数值时,就默认为rpx
    unit: 'px'
}
uni_modules/uv-ui-tools/libs/css/color.scss
New file
@@ -0,0 +1,32 @@
$uv-main-color: #303133 !default;
$uv-content-color: #606266 !default;
$uv-tips-color: #909193 !default;
$uv-light-color: #c0c4cc !default;
$uv-border-color: #dadbde !default;
$uv-bg-color: #f3f4f6 !default;
$uv-disabled-color: #c8c9cc !default;
$uv-primary: #3c9cff !default;
$uv-primary-dark: #398ade !default;
$uv-primary-disabled: #9acafc !default;
$uv-primary-light: #ecf5ff !default;
$uv-warning: #f9ae3d !default;
$uv-warning-dark: #f1a532 !default;
$uv-warning-disabled: #f9d39b !default;
$uv-warning-light: #fdf6ec !default;
$uv-success: #5ac725 !default;
$uv-success-dark: #53c21d !default;
$uv-success-disabled: #a9e08f !default;
$uv-success-light: #f5fff0;
$uv-error: #f56c6c !default;
$uv-error-dark: #e45656 !default;
$uv-error-disabled: #f7b2b2 !default;
$uv-error-light: #fef0f0 !default;
$uv-info: #909399 !default;
$uv-info-dark: #767a82 !default;
$uv-info-disabled: #c4c6c9 !default;
$uv-info-light: #f4f4f5 !default;
uni_modules/uv-ui-tools/libs/css/common.scss
New file
@@ -0,0 +1,100 @@
// 超出行数,自动显示行尾省略号,最多5行
// 来自uvui的温馨提示:当您在控制台看到此报错,说明需要在App.vue的style标签加上【lang="scss"】
@for $i from 1 through 5 {
    .uv-line-#{$i} {
        /* #ifdef APP-NVUE */
        // nvue下,可以直接使用lines属性,这是weex特有样式
        lines: $i;
        text-overflow: ellipsis;
        overflow: hidden;
        flex: 1;
        /* #endif */
        /* #ifndef APP-NVUE */
        // vue下,单行和多行显示省略号需要单独处理
        @if $i == '1' {
            overflow: hidden;
            white-space: nowrap;
            text-overflow: ellipsis;
        } @else {
            display: -webkit-box!important;
            overflow: hidden;
            text-overflow: ellipsis;
            word-break: break-all;
            -webkit-line-clamp: $i;
            -webkit-box-orient: vertical!important;
        }
        /* #endif */
    }
}
$uv-bordercolor: #dadbde;
@if variable-exists(uv-border-color) {
    $uv-bordercolor: $uv-border-color;
}
// 此处加上!important并非随意乱用,而是因为目前*.nvue页面编译到H5时,
// App.vue的样式会被uni-app的view元素的自带border属性覆盖,导致无效
// 综上,这是uni-app的缺陷导致我们为了多端兼容,而必须要加上!important
// 移动端兼容性较好,直接使用0.5px去实现细边框,不使用伪元素形式实现
.uv-border {
    border-width: 0.5px!important;
    border-color: $uv-bordercolor!important;
    border-style: solid;
}
.uv-border-top {
    border-top-width: 0.5px!important;
    border-color: $uv-bordercolor!important;
    border-top-style: solid;
}
.uv-border-left {
    border-left-width: 0.5px!important;
    border-color: $uv-bordercolor!important;
    border-left-style: solid;
}
.uv-border-right {
    border-right-width: 0.5px!important;
    border-color: $uv-bordercolor!important;
    border-right-style: solid;
}
.uv-border-bottom {
    border-bottom-width: 0.5px!important;
    border-color: $uv-bordercolor!important;
    border-bottom-style: solid;
}
.uv-border-top-bottom {
    border-top-width: 0.5px!important;
    border-bottom-width: 0.5px!important;
    border-color: $uv-bordercolor!important;
    border-top-style: solid;
    border-bottom-style: solid;
}
// 去除button的所有默认样式,让其表现跟普通的view、text元素一样
.uv-reset-button {
    padding: 0;
    background-color: transparent;
    /* #ifndef APP-PLUS */
    font-size: inherit;
    line-height: inherit;
    color: inherit;
    /* #endif */
    /* #ifdef APP-NVUE */
    border-width: 0;
    /* #endif */
}
/* #ifndef APP-NVUE */
.uv-reset-button::after {
   border: none;
}
/* #endif */
.uv-hover-class {
    opacity: 0.7;
}
uni_modules/uv-ui-tools/libs/css/components.scss
New file
@@ -0,0 +1,23 @@
@mixin flex($direction: row) {
    /* #ifndef APP-NVUE */
    display: flex;
    /* #endif */
    flex-direction: $direction;
}
/* #ifndef APP-NVUE */
// 由于uvui是基于nvue环境进行开发的,此环境中普通元素默认为flex-direction: column;
// 所以在非nvue中,需要对元素进行重置为flex-direction: column; 否则可能会表现异常
$uvui-nvue-style: true !default;
@if $uvui-nvue-style == true {
    view, scroll-view, swiper-item {
        display: flex;
        flex-direction: column;
        flex-shrink: 0;
        flex-grow: 0;
        flex-basis: auto;
        align-items: stretch;
        align-content: flex-start;
    }
}
/* #endif */
uni_modules/uv-ui-tools/libs/css/variable.scss
New file
@@ -0,0 +1,111 @@
// 超出行数,自动显示行尾省略号,最多5行
// 来自uvui的温馨提示:当您在控制台看到此报错,说明需要在App.vue的style标签加上【lang="scss"】
@if variable-exists(show-lines) {
    @for $i from 1 through 5 {
        .uv-line-#{$i} {
            /* #ifdef APP-NVUE */
            // nvue下,可以直接使用lines属性,这是weex特有样式
            lines: $i;
            text-overflow: ellipsis;
            overflow: hidden;
            flex: 1;
            /* #endif */
            /* #ifndef APP-NVUE */
            // vue下,单行和多行显示省略号需要单独处理
            @if $i == '1' {
                overflow: hidden;
                white-space: nowrap;
                text-overflow: ellipsis;
            } @else {
                display: -webkit-box!important;
                overflow: hidden;
                text-overflow: ellipsis;
                word-break: break-all;
                -webkit-line-clamp: $i;
                -webkit-box-orient: vertical!important;
            }
            /* #endif */
        }
    }
}
@if variable-exists(show-border) {
    $uv-bordercolor: #dadbde;
    @if variable-exists(uv-border-color) {
        $uv-bordercolor: $uv-border-color;
    }
    // 此处加上!important并非随意乱用,而是因为目前*.nvue页面编译到H5时,
    // App.vue的样式会被uni-app的view元素的自带border属性覆盖,导致无效
    // 综上,这是uni-app的缺陷导致我们为了多端兼容,而必须要加上!important
    // 移动端兼容性较好,直接使用0.5px去实现细边框,不使用伪元素形式实现
    @if variable-exists(show-border-surround) {
        .uv-border {
            border-width: 0.5px!important;
            border-color: $uv-bordercolor!important;
            border-style: solid;
        }
    }
    @if variable-exists(show-border-top) {
        .uv-border-top {
            border-top-width: 0.5px!important;
            border-color: $uv-bordercolor!important;
            border-top-style: solid;
        }
    }
    @if variable-exists(show-border-left) {
        .uv-border-left {
            border-left-width: 0.5px!important;
            border-color: $uv-bordercolor!important;
            border-left-style: solid;
        }
    }
    @if variable-exists(show-border-right) {
        .uv-border-right {
            border-right-width: 0.5px!important;
            border-color: $uv-bordercolor!important;
            border-right-style: solid;
        }
    }
    @if variable-exists(show-border-bottom) {
        .uv-border-bottom {
            border-bottom-width: 0.5px!important;
            border-color: $uv-bordercolor!important;
                border-bottom-style: solid;
        }
    }
    @if variable-exists(show-border-top-bottom) {
        .uv-border-top-bottom {
            border-top-width: 0.5px!important;
            border-bottom-width: 0.5px!important;
            border-color: $uv-bordercolor!important;
            border-top-style: solid;
            border-bottom-style: solid;
        }
    }
}
@if variable-exists(show-reset-button) {
    // 去除button的所有默认样式,让其表现跟普通的view、text元素一样
    .uv-reset-button {
        padding: 0;
        background-color: transparent;
        /* #ifndef APP-PLUS */
        font-size: inherit;
        line-height: inherit;
        color: inherit;
        /* #endif */
        /* #ifdef APP-NVUE */
        border-width: 0;
        /* #endif */
    }
    /* #ifndef APP-NVUE */
    .uv-reset-button::after {
         border: none;
    }
    /* #endif */
}
@if variable-exists(show-hover) {
    .uv-hover-class {
        opacity: 0.7;
    }
}
uni_modules/uv-ui-tools/libs/css/vue.scss
New file
@@ -0,0 +1,40 @@
// 历遍生成4个方向的底部安全区
@each $d in top, right, bottom, left {
    .uv-safe-area-inset-#{$d} {
        padding-#{$d}: 0;
        padding-#{$d}: constant(safe-area-inset-#{$d});
        padding-#{$d}: env(safe-area-inset-#{$d});
    }
}
//提升H5端uni.toast()的层级,避免被uvui的modal等遮盖
/* #ifdef H5 */
uni-toast {
    z-index: 10090;
}
uni-toast .uni-toast {
   z-index: 10090;
}
/* #endif */
// 隐藏scroll-view的滚动条
::-webkit-scrollbar {
    display: none;
    width: 0 !important;
    height: 0 !important;
    -webkit-appearance: none;
    background: transparent;
}
$uvui-nvue-style: true !default;
@if $uvui-nvue-style == false {
    view, scroll-view, swiper-item {
        display: flex;
        flex-direction: column;
        flex-shrink: 0;
        flex-grow: 0;
        flex-basis: auto;
        align-items: stretch;
        align-content: flex-start;
    }
}
uni_modules/uv-ui-tools/libs/function/colorGradient.js
New file
@@ -0,0 +1,134 @@
/**
 * 求两个颜色之间的渐变值
 * @param {string} startColor 开始的颜色
 * @param {string} endColor 结束的颜色
 * @param {number} step 颜色等分的份额
 * */
function colorGradient(startColor = 'rgb(0, 0, 0)', endColor = 'rgb(255, 255, 255)', step = 10) {
    const startRGB = hexToRgb(startColor, false) // 转换为rgb数组模式
    const startR = startRGB[0]
    const startG = startRGB[1]
    const startB = startRGB[2]
    const endRGB = hexToRgb(endColor, false)
    const endR = endRGB[0]
    const endG = endRGB[1]
    const endB = endRGB[2]
    const sR = (endR - startR) / step // 总差值
    const sG = (endG - startG) / step
    const sB = (endB - startB) / step
    const colorArr = []
    for (let i = 0; i < step; i++) {
        // 计算每一步的hex值
        let hex = rgbToHex(`rgb(${Math.round((sR * i + startR))},${Math.round((sG * i + startG))},${Math.round((sB
            * i + startB))})`)
        // 确保第一个颜色值为startColor的值
        if (i === 0) hex = rgbToHex(startColor)
        // 确保最后一个颜色值为endColor的值
        if (i === step - 1) hex = rgbToHex(endColor)
        colorArr.push(hex)
    }
    return colorArr
}
// 将hex表示方式转换为rgb表示方式(这里返回rgb数组模式)
function hexToRgb(sColor, str = true) {
    const reg = /^#([0-9a-fA-f]{3}|[0-9a-fA-f]{6})$/
    sColor = String(sColor).toLowerCase()
    if (sColor && reg.test(sColor)) {
        if (sColor.length === 4) {
            let sColorNew = '#'
            for (let i = 1; i < 4; i += 1) {
                sColorNew += sColor.slice(i, i + 1).concat(sColor.slice(i, i + 1))
            }
            sColor = sColorNew
        }
        // 处理六位的颜色值
        const sColorChange = []
        for (let i = 1; i < 7; i += 2) {
            sColorChange.push(parseInt(`0x${sColor.slice(i, i + 2)}`))
        }
        if (!str) {
            return sColorChange
        }
        return `rgb(${sColorChange[0]},${sColorChange[1]},${sColorChange[2]})`
    } if (/^(rgb|RGB)/.test(sColor)) {
        const arr = sColor.replace(/(?:\(|\)|rgb|RGB)*/g, '').split(',')
        return arr.map((val) => Number(val))
    }
    return sColor
}
// 将rgb表示方式转换为hex表示方式
function rgbToHex(rgb) {
    const _this = rgb
    const reg = /^#([0-9a-fA-f]{3}|[0-9a-fA-f]{6})$/
    if (/^(rgb|RGB)/.test(_this)) {
        const aColor = _this.replace(/(?:\(|\)|rgb|RGB)*/g, '').split(',')
        let strHex = '#'
        for (let i = 0; i < aColor.length; i++) {
            let hex = Number(aColor[i]).toString(16)
            hex = String(hex).length == 1 ? `${0}${hex}` : hex // 保证每个rgb的值为2位
            if (hex === '0') {
                hex += hex
            }
            strHex += hex
        }
        if (strHex.length !== 7) {
            strHex = _this
        }
        return strHex
    } if (reg.test(_this)) {
        const aNum = _this.replace(/#/, '').split('')
        if (aNum.length === 6) {
            return _this
        } if (aNum.length === 3) {
            let numHex = '#'
            for (let i = 0; i < aNum.length; i += 1) {
                numHex += (aNum[i] + aNum[i])
            }
            return numHex
        }
    } else {
        return _this
    }
}
/**
* JS颜色十六进制转换为rgb或rgba,返回的格式为 rgba(255,255,255,0.5)字符串
* sHex为传入的十六进制的色值
* alpha为rgba的透明度
*/
function colorToRgba(color, alpha) {
    color = rgbToHex(color)
    // 十六进制颜色值的正则表达式
    const reg = /^#([0-9a-fA-f]{3}|[0-9a-fA-f]{6})$/
    /* 16进制颜色转为RGB格式 */
    let sColor = String(color).toLowerCase()
    if (sColor && reg.test(sColor)) {
        if (sColor.length === 4) {
            let sColorNew = '#'
            for (let i = 1; i < 4; i += 1) {
                sColorNew += sColor.slice(i, i + 1).concat(sColor.slice(i, i + 1))
            }
            sColor = sColorNew
        }
        // 处理六位的颜色值
        const sColorChange = []
        for (let i = 1; i < 7; i += 2) {
            sColorChange.push(parseInt(`0x${sColor.slice(i, i + 2)}`))
        }
        // return sColorChange.join(',')
        return `rgba(${sColorChange.join(',')},${alpha})`
    }
    return sColor
}
export {
    colorGradient,
    hexToRgb,
    rgbToHex,
    colorToRgba
}
uni_modules/uv-ui-tools/libs/function/debounce.js
New file
@@ -0,0 +1,29 @@
let timeout = null
/**
 * 防抖原理:一定时间内,只有最后一次操作,再过wait毫秒后才执行函数
 *
 * @param {Function} func 要执行的回调函数
 * @param {Number} wait 延时的时间
 * @param {Boolean} immediate 是否立即执行
 * @return null
 */
function debounce(func, wait = 500, immediate = false) {
    // 清除定时器
    if (timeout !== null) clearTimeout(timeout)
    // 立即执行,此类情况一般用不到
    if (immediate) {
        const callNow = !timeout
        timeout = setTimeout(() => {
            timeout = null
        }, wait)
        if (callNow) typeof func === 'function' && func()
    } else {
        // 设置定时器,当最后一次操作后,timeout不会再被清除,所以在延时wait毫秒后执行func回调方法
        timeout = setTimeout(() => {
            typeof func === 'function' && func()
        }, wait)
    }
}
export default debounce
uni_modules/uv-ui-tools/libs/function/digit.js
New file
@@ -0,0 +1,167 @@
let _boundaryCheckingState = true; // 是否进行越界检查的全局开关
/**
 * 把错误的数据转正
 * @private
 * @example strip(0.09999999999999998)=0.1
 */
function strip(num, precision = 15) {
  return +parseFloat(Number(num).toPrecision(precision));
}
/**
 * Return digits length of a number
 * @private
 * @param {*number} num Input number
 */
function digitLength(num) {
  // Get digit length of e
  const eSplit = num.toString().split(/[eE]/);
  const len = (eSplit[0].split('.')[1] || '').length - +(eSplit[1] || 0);
  return len > 0 ? len : 0;
}
/**
 * 把小数转成整数,如果是小数则放大成整数
 * @private
 * @param {*number} num 输入数
 */
function float2Fixed(num) {
  if (num.toString().indexOf('e') === -1) {
    return Number(num.toString().replace('.', ''));
  }
  const dLen = digitLength(num);
  return dLen > 0 ? strip(Number(num) * Math.pow(10, dLen)) : Number(num);
}
/**
 * 检测数字是否越界,如果越界给出提示
 * @private
 * @param {*number} num 输入数
 */
function checkBoundary(num) {
  if (_boundaryCheckingState) {
    if (num > Number.MAX_SAFE_INTEGER || num < Number.MIN_SAFE_INTEGER) {
      console.warn(`${num} 超出了精度限制,结果可能不正确`);
    }
  }
}
/**
 * 把递归操作扁平迭代化
 * @param {number[]} arr 要操作的数字数组
 * @param {function} operation 迭代操作
 * @private
 */
function iteratorOperation(arr, operation) {
  const [num1, num2, ...others] = arr;
  let res = operation(num1, num2);
  others.forEach((num) => {
    res = operation(res, num);
  });
  return res;
}
/**
 * 高精度乘法
 * @export
 */
export function times(...nums) {
  if (nums.length > 2) {
    return iteratorOperation(nums, times);
  }
  const [num1, num2] = nums;
  const num1Changed = float2Fixed(num1);
  const num2Changed = float2Fixed(num2);
  const baseNum = digitLength(num1) + digitLength(num2);
  const leftValue = num1Changed * num2Changed;
  checkBoundary(leftValue);
  return leftValue / Math.pow(10, baseNum);
}
/**
 * 高精度加法
 * @export
 */
export function plus(...nums) {
  if (nums.length > 2) {
    return iteratorOperation(nums, plus);
  }
  const [num1, num2] = nums;
  // 取最大的小数位
  const baseNum = Math.pow(10, Math.max(digitLength(num1), digitLength(num2)));
  // 把小数都转为整数然后再计算
  return (times(num1, baseNum) + times(num2, baseNum)) / baseNum;
}
/**
 * 高精度减法
 * @export
 */
export function minus(...nums) {
  if (nums.length > 2) {
    return iteratorOperation(nums, minus);
  }
  const [num1, num2] = nums;
  const baseNum = Math.pow(10, Math.max(digitLength(num1), digitLength(num2)));
  return (times(num1, baseNum) - times(num2, baseNum)) / baseNum;
}
/**
 * 高精度除法
 * @export
 */
export function divide(...nums) {
  if (nums.length > 2) {
    return iteratorOperation(nums, divide);
  }
  const [num1, num2] = nums;
  const num1Changed = float2Fixed(num1);
  const num2Changed = float2Fixed(num2);
  checkBoundary(num1Changed);
  checkBoundary(num2Changed);
  // 重要,这里必须用strip进行修正
  return times(num1Changed / num2Changed, strip(Math.pow(10, digitLength(num2) - digitLength(num1))));
}
/**
 * 四舍五入
 * @export
 */
export function round(num, ratio) {
  const base = Math.pow(10, ratio);
  let result = divide(Math.round(Math.abs(times(num, base))), base);
  if (num < 0 && result !== 0) {
    result = times(result, -1);
  }
  // 位数不足则补0
  return result;
}
/**
 * 是否进行边界检查,默认开启
 * @param flag 标记开关,true 为开启,false 为关闭,默认为 true
 * @export
 */
export function enableBoundaryChecking(flag = true) {
  _boundaryCheckingState = flag;
}
export default {
  times,
  plus,
  minus,
  divide,
  round,
  enableBoundaryChecking,
};
uni_modules/uv-ui-tools/libs/function/index.js
New file
@@ -0,0 +1,734 @@
import { number, empty } from './test.js'
import { round } from './digit.js'
/**
 * @description 如果value小于min,取min;如果value大于max,取max
 * @param {number} min
 * @param {number} max
 * @param {number} value
 */
function range(min = 0, max = 0, value = 0) {
    return Math.max(min, Math.min(max, Number(value)))
}
/**
 * @description 用于获取用户传递值的px值  如果用户传递了"xxpx"或者"xxrpx",取出其数值部分,如果是"xxxrpx"还需要用过uni.upx2px进行转换
 * @param {number|string} value 用户传递值的px值
 * @param {boolean} unit
 * @returns {number|string}
 */
function getPx(value, unit = false) {
    if (number(value)) {
        return unit ? `${value}px` : Number(value)
    }
    // 如果带有rpx,先取出其数值部分,再转为px值
    if (/(rpx|upx)$/.test(value)) {
        return unit ? `${uni.upx2px(parseInt(value))}px` : Number(uni.upx2px(parseInt(value)))
    }
    return unit ? `${parseInt(value)}px` : parseInt(value)
}
/**
 * @description 进行延时,以达到可以简写代码的目的 比如: await uni.$uv.sleep(20)将会阻塞20ms
 * @param {number} value 堵塞时间 单位ms 毫秒
 * @returns {Promise} 返回promise
 */
function sleep(value = 30) {
    return new Promise((resolve) => {
        setTimeout(() => {
            resolve()
        }, value)
    })
}
/**
 * @description 运行期判断平台
 * @returns {string} 返回所在平台(小写)
 * @link 运行期判断平台 https://uniapp.dcloud.io/frame?id=判断平台
 */
function os() {
    return uni.getSystemInfoSync().platform.toLowerCase()
}
/**
 * @description 获取系统信息同步接口
 * @link 获取系统信息同步接口 https://uniapp.dcloud.io/api/system/info?id=getsysteminfosync
 */
function sys() {
    return uni.getSystemInfoSync()
}
/**
 * @description 取一个区间数
 * @param {Number} min 最小值
 * @param {Number} max 最大值
 */
function random(min, max) {
    if (min >= 0 && max > 0 && max >= min) {
        const gab = max - min + 1
        return Math.floor(Math.random() * gab + min)
    }
    return 0
}
/**
 * @param {Number} len uuid的长度
 * @param {Boolean} firstU 将返回的首字母置为"u"
 * @param {Nubmer} radix 生成uuid的基数(意味着返回的字符串都是这个基数),2-二进制,8-八进制,10-十进制,16-十六进制
 */
function guid(len = 32, firstU = true, radix = null) {
    const chars = '0123456789ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz'.split('')
    const uuid = []
    radix = radix || chars.length
    if (len) {
        // 如果指定uuid长度,只是取随机的字符,0|x为位运算,能去掉x的小数位,返回整数位
        for (let i = 0; i < len; i++) uuid[i] = chars[0 | Math.random() * radix]
    } else {
        let r
        // rfc4122标准要求返回的uuid中,某些位为固定的字符
        uuid[8] = uuid[13] = uuid[18] = uuid[23] = '-'
        uuid[14] = '4'
        for (let i = 0; i < 36; i++) {
            if (!uuid[i]) {
                r = 0 | Math.random() * 16
                uuid[i] = chars[(i == 19) ? (r & 0x3) | 0x8 : r]
            }
        }
    }
    // 移除第一个字符,并用u替代,因为第一个字符为数值时,该guuid不能用作id或者class
    if (firstU) {
        uuid.shift()
        return `u${uuid.join('')}`
    }
    return uuid.join('')
}
/**
* @description 获取父组件的参数,因为支付宝小程序不支持provide/inject的写法
   this.$parent在非H5中,可以准确获取到父组件,但是在H5中,需要多次this.$parent.$parent.xxx
   这里默认值等于undefined有它的含义,因为最顶层元素(组件)的$parent就是undefined,意味着不传name
   值(默认为undefined),就是查找最顶层的$parent
*  @param {string|undefined} name 父组件的参数名
*/
function $parent(name = undefined) {
    let parent = this.$parent
    // 通过while历遍,这里主要是为了H5需要多层解析的问题
    while (parent) {
        // 父组件
        if (parent.$options && parent.$options.name !== name) {
            // 如果组件的name不相等,继续上一级寻找
            parent = parent.$parent
        } else {
            return parent
        }
    }
    return false
}
/**
 * @description 样式转换
 * 对象转字符串,或者字符串转对象
 * @param {object | string} customStyle 需要转换的目标
 * @param {String} target 转换的目的,object-转为对象,string-转为字符串
 * @returns {object|string}
 */
function addStyle(customStyle, target = 'object') {
    // 字符串转字符串,对象转对象情形,直接返回
    if (empty(customStyle) || typeof(customStyle) === 'object' && target === 'object' || target === 'string' &&
        typeof(customStyle) === 'string') {
        return customStyle
    }
    // 字符串转对象
    if (target === 'object') {
        // 去除字符串样式中的两端空格(中间的空格不能去掉,比如padding: 20px 0如果去掉了就错了),空格是无用的
        customStyle = trim(customStyle)
        // 根据";"将字符串转为数组形式
        const styleArray = customStyle.split(';')
        const style = {}
        // 历遍数组,拼接成对象
        for (let i = 0; i < styleArray.length; i++) {
            // 'font-size:20px;color:red;',如此最后字符串有";"的话,会导致styleArray最后一个元素为空字符串,这里需要过滤
            if (styleArray[i]) {
                const item = styleArray[i].split(':')
                style[trim(item[0])] = trim(item[1])
            }
        }
        return style
    }
    // 这里为对象转字符串形式
    let string = ''
    for (const i in customStyle) {
        // 驼峰转为中划线的形式,否则css内联样式,无法识别驼峰样式属性名
        const key = i.replace(/([A-Z])/g, '-$1').toLowerCase()
        string += `${key}:${customStyle[i]};`
    }
    // 去除两端空格
    return trim(string)
}
/**
 * @description 添加单位,如果有rpx,upx,%,px等单位结尾或者值为auto,直接返回,否则加上px单位结尾
 * @param {string|number} value 需要添加单位的值
 * @param {string} unit 添加的单位名 比如px
 */
function addUnit(value = 'auto', unit = uni?.$uv?.config?.unit ? uni?.$uv?.config?.unit : 'px') {
    value = String(value)
    // 用uvui内置验证规则中的number判断是否为数值
    return number(value) ? `${value}${unit}` : value
}
/**
 * @description 深度克隆
 * @param {object} obj 需要深度克隆的对象
 * @param cache 缓存
 * @returns {*} 克隆后的对象或者原值(不是对象)
 */
function deepClone(obj, cache = new WeakMap()) {
    if (obj === null || typeof obj !== 'object') return obj;
    if (cache.has(obj)) return cache.get(obj);
    let clone;
    if (obj instanceof Date) {
        clone = new Date(obj.getTime());
    } else if (obj instanceof RegExp) {
        clone = new RegExp(obj);
    } else if (obj instanceof Map) {
        clone = new Map(Array.from(obj, ([key, value]) => [key, deepClone(value, cache)]));
    } else if (obj instanceof Set) {
        clone = new Set(Array.from(obj, value => deepClone(value, cache)));
    } else if (Array.isArray(obj)) {
        clone = obj.map(value => deepClone(value, cache));
    } else if (Object.prototype.toString.call(obj) === '[object Object]') {
        clone = Object.create(Object.getPrototypeOf(obj));
        cache.set(obj, clone);
        for (const [key, value] of Object.entries(obj)) {
            clone[key] = deepClone(value, cache);
        }
    } else {
        clone = Object.assign({}, obj);
    }
    cache.set(obj, clone);
    return clone;
}
/**
 * @description JS对象深度合并
 * @param {object} target 需要拷贝的对象
 * @param {object} source 拷贝的来源对象
 * @returns {object|boolean} 深度合并后的对象或者false(入参有不是对象)
 */
function deepMerge(target = {}, source = {}) {
    target = deepClone(target)
    if (typeof target !== 'object' || target === null || typeof source !== 'object' || source === null) return target;
    const merged = Array.isArray(target) ? target.slice() : Object.assign({}, target);
    for (const prop in source) {
        if (!source.hasOwnProperty(prop)) continue;
        const sourceValue = source[prop];
        const targetValue = merged[prop];
        if (sourceValue instanceof Date) {
            merged[prop] = new Date(sourceValue);
        } else if (sourceValue instanceof RegExp) {
            merged[prop] = new RegExp(sourceValue);
        } else if (sourceValue instanceof Map) {
            merged[prop] = new Map(sourceValue);
        } else if (sourceValue instanceof Set) {
            merged[prop] = new Set(sourceValue);
        } else if (typeof sourceValue === 'object' && sourceValue !== null) {
            merged[prop] = deepMerge(targetValue, sourceValue);
        } else {
            merged[prop] = sourceValue;
        }
    }
    return merged;
}
/**
 * @description error提示
 * @param {*} err 错误内容
 */
function error(err) {
    // 开发环境才提示,生产环境不会提示
    if (process.env.NODE_ENV === 'development') {
        console.error(`uvui提示:${err}`)
    }
}
/**
 * @description 打乱数组
 * @param {array} array 需要打乱的数组
 * @returns {array} 打乱后的数组
 */
function randomArray(array = []) {
    // 原理是sort排序,Math.random()产生0<= x < 1之间的数,会导致x-0.05大于或者小于0
    return array.sort(() => Math.random() - 0.5)
}
// padStart 的 polyfill,因为某些机型或情况,还无法支持es7的padStart,比如电脑版的微信小程序
// 所以这里做一个兼容polyfill的兼容处理
if (!String.prototype.padStart) {
    // 为了方便表示这里 fillString 用了ES6 的默认参数,不影响理解
    String.prototype.padStart = function(maxLength, fillString = ' ') {
        if (Object.prototype.toString.call(fillString) !== '[object String]') {
            throw new TypeError(
                'fillString must be String'
            )
        }
        const str = this
        // 返回 String(str) 这里是为了使返回的值是字符串字面量,在控制台中更符合直觉
        if (str.length >= maxLength) return String(str)
        const fillLength = maxLength - str.length
        let times = Math.ceil(fillLength / fillString.length)
        while (times >>= 1) {
            fillString += fillString
            if (times === 1) {
                fillString += fillString
            }
        }
        return fillString.slice(0, fillLength) + str
    }
}
/**
 * @description 格式化时间
 * @param {String|Number} dateTime 需要格式化的时间戳
 * @param {String} fmt 格式化规则 yyyy:mm:dd|yyyy:mm|yyyy年mm月dd日|yyyy年mm月dd日 hh时MM分等,可自定义组合 默认yyyy-mm-dd
 * @returns {string} 返回格式化后的字符串
 */
function timeFormat(dateTime = null, formatStr = 'yyyy-mm-dd') {
    let date
    // 若传入时间为假值,则取当前时间
    if (!dateTime) {
        date = new Date()
    }
    // 若为unix秒时间戳,则转为毫秒时间戳(逻辑有点奇怪,但不敢改,以保证历史兼容)
    else if (/^\d{10}$/.test(dateTime?.toString().trim())) {
        date = new Date(dateTime * 1000)
    }
    // 若用户传入字符串格式时间戳,new Date无法解析,需做兼容
    else if (typeof dateTime === 'string' && /^\d+$/.test(dateTime.trim())) {
        date = new Date(Number(dateTime))
    }
    // 处理平台性差异,在Safari/Webkit中,new Date仅支持/作为分割符的字符串时间
    // 处理 '2022-07-10 01:02:03',跳过 '2022-07-10T01:02:03'
    else if (typeof dateTime === 'string' && dateTime.includes('-') && !dateTime.includes('T')) {
        date = new Date(dateTime.replace(/-/g, '/'))
    }
    // 其他都认为符合 RFC 2822 规范
    else {
        date = new Date(dateTime)
    }
    const timeSource = {
        'y': date.getFullYear().toString(), // 年
        'm': (date.getMonth() + 1).toString().padStart(2, '0'), // 月
        'd': date.getDate().toString().padStart(2, '0'), // 日
        'h': date.getHours().toString().padStart(2, '0'), // 时
        'M': date.getMinutes().toString().padStart(2, '0'), // 分
        's': date.getSeconds().toString().padStart(2, '0') // 秒
        // 有其他格式化字符需求可以继续添加,必须转化成字符串
    }
    for (const key in timeSource) {
        const [ret] = new RegExp(`${key}+`).exec(formatStr) || []
        if (ret) {
            // 年可能只需展示两位
            const beginIndex = key === 'y' && ret.length === 2 ? 2 : 0
            formatStr = formatStr.replace(ret, timeSource[key].slice(beginIndex))
        }
    }
    return formatStr
}
/**
 * @description 时间戳转为多久之前
 * @param {String|Number} timestamp 时间戳
 * @param {String|Boolean} format
 * 格式化规则如果为时间格式字符串,超出一定时间范围,返回固定的时间格式;
 * 如果为布尔值false,无论什么时间,都返回多久以前的格式
 * @returns {string} 转化后的内容
 */
function timeFrom(timestamp = null, format = 'yyyy-mm-dd') {
    if (timestamp == null) timestamp = Number(new Date())
    timestamp = parseInt(timestamp)
    // 判断用户输入的时间戳是秒还是毫秒,一般前端js获取的时间戳是毫秒(13位),后端传过来的为秒(10位)
    if (timestamp.toString().length == 10) timestamp *= 1000
    let timer = (new Date()).getTime() - timestamp
    timer = parseInt(timer / 1000)
    // 如果小于5分钟,则返回"刚刚",其他以此类推
    let tips = ''
    switch (true) {
        case timer < 300:
            tips = '刚刚'
            break
        case timer >= 300 && timer < 3600:
            tips = `${parseInt(timer / 60)}分钟前`
            break
        case timer >= 3600 && timer < 86400:
            tips = `${parseInt(timer / 3600)}小时前`
            break
        case timer >= 86400 && timer < 2592000:
            tips = `${parseInt(timer / 86400)}天前`
            break
        default:
            // 如果format为false,则无论什么时间戳,都显示xx之前
            if (format === false) {
                if (timer >= 2592000 && timer < 365 * 86400) {
                    tips = `${parseInt(timer / (86400 * 30))}个月前`
                } else {
                    tips = `${parseInt(timer / (86400 * 365))}年前`
                }
            } else {
                tips = timeFormat(timestamp, format)
            }
    }
    return tips
}
/**
 * @description 去除空格
 * @param String str 需要去除空格的字符串
 * @param String pos both(左右)|left|right|all 默认both
 */
function trim(str, pos = 'both') {
    str = String(str)
    if (pos == 'both') {
        return str.replace(/^\s+|\s+$/g, '')
    }
    if (pos == 'left') {
        return str.replace(/^\s*/, '')
    }
    if (pos == 'right') {
        return str.replace(/(\s*$)/g, '')
    }
    if (pos == 'all') {
        return str.replace(/\s+/g, '')
    }
    return str
}
/**
 * @description 对象转url参数
 * @param {object} data,对象
 * @param {Boolean} isPrefix,是否自动加上"?"
 * @param {string} arrayFormat 规则 indices|brackets|repeat|comma
 */
function queryParams(data = {}, isPrefix = true, arrayFormat = 'brackets') {
    const prefix = isPrefix ? '?' : ''
    const _result = []
    if (['indices', 'brackets', 'repeat', 'comma'].indexOf(arrayFormat) == -1) arrayFormat = 'brackets'
    for (const key in data) {
        const value = data[key]
        // 去掉为空的参数
        if (['', undefined, null].indexOf(value) >= 0) {
            continue
        }
        // 如果值为数组,另行处理
        if (value.constructor === Array) {
            // e.g. {ids: [1, 2, 3]}
            switch (arrayFormat) {
                case 'indices':
                    // 结果: ids[0]=1&ids[1]=2&ids[2]=3
                    for (let i = 0; i < value.length; i++) {
                        _result.push(`${key}[${i}]=${value[i]}`)
                    }
                    break
                case 'brackets':
                    // 结果: ids[]=1&ids[]=2&ids[]=3
                    value.forEach((_value) => {
                        _result.push(`${key}[]=${_value}`)
                    })
                    break
                case 'repeat':
                    // 结果: ids=1&ids=2&ids=3
                    value.forEach((_value) => {
                        _result.push(`${key}=${_value}`)
                    })
                    break
                case 'comma':
                    // 结果: ids=1,2,3
                    let commaStr = ''
                    value.forEach((_value) => {
                        commaStr += (commaStr ? ',' : '') + _value
                    })
                    _result.push(`${key}=${commaStr}`)
                    break
                default:
                    value.forEach((_value) => {
                        _result.push(`${key}[]=${_value}`)
                    })
            }
        } else {
            _result.push(`${key}=${value}`)
        }
    }
    return _result.length ? prefix + _result.join('&') : ''
}
/**
 * 显示消息提示框
 * @param {String} title 提示的内容,长度与 icon 取值有关。
 * @param {Number} duration 提示的延迟时间,单位毫秒,默认:2000
 */
function toast(title, duration = 2000) {
    uni.showToast({
        title: String(title),
        icon: 'none',
        duration
    })
}
/**
 * @description 根据主题type值,获取对应的图标
 * @param {String} type 主题名称,primary|info|error|warning|success
 * @param {boolean} fill 是否使用fill填充实体的图标
 */
function type2icon(type = 'success', fill = false) {
    // 如果非预置值,默认为success
    if (['primary', 'info', 'error', 'warning', 'success'].indexOf(type) == -1) type = 'success'
    let iconName = ''
    // 目前(2019-12-12),info和primary使用同一个图标
    switch (type) {
        case 'primary':
            iconName = 'info-circle'
            break
        case 'info':
            iconName = 'info-circle'
            break
        case 'error':
            iconName = 'close-circle'
            break
        case 'warning':
            iconName = 'error-circle'
            break
        case 'success':
            iconName = 'checkmark-circle'
            break
        default:
            iconName = 'checkmark-circle'
    }
    // 是否是实体类型,加上-fill,在icon组件库中,实体的类名是后面加-fill的
    if (fill) iconName += '-fill'
    return iconName
}
/**
 * @description 数字格式化
 * @param {number|string} number 要格式化的数字
 * @param {number} decimals 保留几位小数
 * @param {string} decimalPoint 小数点符号
 * @param {string} thousandsSeparator 千分位符号
 * @returns {string} 格式化后的数字
 */
function priceFormat(number, decimals = 0, decimalPoint = '.', thousandsSeparator = ',') {
    number = (`${number}`).replace(/[^0-9+-Ee.]/g, '')
    const n = !isFinite(+number) ? 0 : +number
    const prec = !isFinite(+decimals) ? 0 : Math.abs(decimals)
    const sep = (typeof thousandsSeparator === 'undefined') ? ',' : thousandsSeparator
    const dec = (typeof decimalPoint === 'undefined') ? '.' : decimalPoint
    let s = ''
    s = (prec ? round(n, prec) + '' : `${Math.round(n)}`).split('.')
    const re = /(-?\d+)(\d{3})/
    while (re.test(s[0])) {
        s[0] = s[0].replace(re, `$1${sep}$2`)
    }
    if ((s[1] || '').length < prec) {
        s[1] = s[1] || ''
        s[1] += new Array(prec - s[1].length + 1).join('0')
    }
    return s.join(dec)
}
/**
 * @description 获取duration值
 * 如果带有ms或者s直接返回,如果大于一定值,认为是ms单位,小于一定值,认为是s单位
 * 比如以30位阈值,那么300大于30,可以理解为用户想要的是300ms,而不是想花300s去执行一个动画
 * @param {String|number} value 比如: "1s"|"100ms"|1|100
 * @param {boolean} unit  提示: 如果是false 默认返回number
 * @return {string|number}
 */
function getDuration(value, unit = true) {
    const valueNum = parseInt(value)
    if (unit) {
        if (/s$/.test(value)) return value
        return value > 30 ? `${value}ms` : `${value}s`
    }
    if (/ms$/.test(value)) return valueNum
    if (/s$/.test(value)) return valueNum > 30 ? valueNum : valueNum * 1000
    return valueNum
}
/**
 * @description 日期的月或日补零操作
 * @param {String} value 需要补零的值
 */
function padZero(value) {
    return `00${value}`.slice(-2)
}
/**
 * @description 在uv-form的子组件内容发生变化,或者失去焦点时,尝试通知uv-form执行校验方法
 * @param {*} instance
 * @param {*} event
 */
function formValidate(instance, event) {
    const formItem = $parent.call(instance, 'uv-form-item')
    const form = $parent.call(instance, 'uv-form')
    // 如果发生变化的input或者textarea等,其父组件中有uv-form-item或者uv-form等,就执行form的validate方法
    // 同时将form-item的pros传递给form,让其进行精确对象验证
    if (formItem && form) {
        form.validateField(formItem.prop, () => {}, event)
    }
}
/**
 * @description 获取某个对象下的属性,用于通过类似'a.b.c'的形式去获取一个对象的的属性的形式
 * @param {object} obj 对象
 * @param {string} key 需要获取的属性字段
 * @returns {*}
 */
function getProperty(obj, key) {
    if (!obj) {
        return
    }
    if (typeof key !== 'string' || key === '') {
        return ''
    }
    if (key.indexOf('.') !== -1) {
        const keys = key.split('.')
        let firstObj = obj[keys[0]] || {}
        for (let i = 1; i < keys.length; i++) {
            if (firstObj) {
                firstObj = firstObj[keys[i]]
            }
        }
        return firstObj
    }
    return obj[key]
}
/**
 * @description 设置对象的属性值,如果'a.b.c'的形式进行设置
 * @param {object} obj 对象
 * @param {string} key 需要设置的属性
 * @param {string} value 设置的值
 */
function setProperty(obj, key, value) {
    if (!obj) {
        return
    }
    // 递归赋值
    const inFn = function(_obj, keys, v) {
        // 最后一个属性key
        if (keys.length === 1) {
            _obj[keys[0]] = v
            return
        }
        // 0~length-1个key
        while (keys.length > 1) {
            const k = keys[0]
            if (!_obj[k] || (typeof _obj[k] !== 'object')) {
                _obj[k] = {}
            }
            const key = keys.shift()
            // 自调用判断是否存在属性,不存在则自动创建对象
            inFn(_obj[k], keys, v)
        }
    }
    if (typeof key !== 'string' || key === '') {
    } else if (key.indexOf('.') !== -1) { // 支持多层级赋值操作
        const keys = key.split('.')
        inFn(obj, keys, value)
    } else {
        obj[key] = value
    }
}
/**
 * @description 获取当前页面路径
 */
function page() {
    const pages = getCurrentPages();
    const route = pages[pages.length - 1]?.route;
    // 某些特殊情况下(比如页面进行redirectTo时的一些时机),pages可能为空数组
    return `/${route ? route : ''}`
}
/**
 * @description 获取当前路由栈实例数组
 */
function pages() {
    const pages = getCurrentPages()
    return pages
}
/**
 * 获取页面历史栈指定层实例
 * @param back {number} [0] - 0或者负数,表示获取历史栈的哪一层,0表示获取当前页面实例,-1 表示获取上一个页面实例。默认0。
 */
function getHistoryPage(back = 0) {
    const pages = getCurrentPages()
    const len = pages.length
    return pages[len - 1 + back]
}
/**
 * @description 修改uvui内置属性值
 * @param {object} props 修改内置props属性
 * @param {object} config 修改内置config属性
 * @param {object} color 修改内置color属性
 * @param {object} zIndex 修改内置zIndex属性
 */
function setConfig({
    props = {},
    config = {},
    color = {},
    zIndex = {}
}) {
    const {
        deepMerge,
    } = uni.$uv
    uni.$uv.config = deepMerge(uni.$uv.config, config)
    uni.$uv.props = deepMerge(uni.$uv.props, props)
    uni.$uv.color = deepMerge(uni.$uv.color, color)
    uni.$uv.zIndex = deepMerge(uni.$uv.zIndex, zIndex)
}
export {
    range,
    getPx,
    sleep,
    os,
    sys,
    random,
    guid,
    $parent,
    addStyle,
    addUnit,
    deepClone,
    deepMerge,
    error,
    randomArray,
    timeFormat,
    timeFrom,
    trim,
    queryParams,
    toast,
    type2icon,
    priceFormat,
    getDuration,
    padZero,
    formValidate,
    getProperty,
    setProperty,
    page,
    pages,
    getHistoryPage,
    setConfig
}
uni_modules/uv-ui-tools/libs/function/platform.js
New file
@@ -0,0 +1,75 @@
/**
 * 注意:
 * 此部分内容,在vue-cli模式下,需要在vue.config.js加入如下内容才有效:
 * module.exports = {
 *     transpileDependencies: ['uview-v2']
 * }
 */
let platform = 'none'
// #ifdef VUE3
platform = 'vue3'
// #endif
// #ifdef VUE2
platform = 'vue2'
// #endif
// #ifdef APP-PLUS
platform = 'plus'
// #endif
// #ifdef APP-NVUE
platform = 'nvue'
// #endif
// #ifdef H5
platform = 'h5'
// #endif
// #ifdef MP-WEIXIN
platform = 'weixin'
// #endif
// #ifdef MP-ALIPAY
platform = 'alipay'
// #endif
// #ifdef MP-BAIDU
platform = 'baidu'
// #endif
// #ifdef MP-TOUTIAO
platform = 'toutiao'
// #endif
// #ifdef MP-QQ
platform = 'qq'
// #endif
// #ifdef MP-KUAISHOU
platform = 'kuaishou'
// #endif
// #ifdef MP-360
platform = '360'
// #endif
// #ifdef MP
platform = 'mp'
// #endif
// #ifdef QUICKAPP-WEBVIEW
platform = 'quickapp-webview'
// #endif
// #ifdef QUICKAPP-WEBVIEW-HUAWEI
platform = 'quickapp-webview-huawei'
// #endif
// #ifdef QUICKAPP-WEBVIEW-UNION
platform = 'quckapp-webview-union'
// #endif
export default platform
uni_modules/uv-ui-tools/libs/function/test.js
New file
@@ -0,0 +1,287 @@
/**
 * 验证电子邮箱格式
 */
function email(value) {
    return /^\w+((-\w+)|(\.\w+))*\@[A-Za-z0-9]+((\.|-)[A-Za-z0-9]+)*\.[A-Za-z0-9]+$/.test(value)
}
/**
 * 验证手机格式
 */
function mobile(value) {
    return /^1([3589]\d|4[5-9]|6[1-2,4-7]|7[0-8])\d{8}$/.test(value)
}
/**
 * 验证URL格式
 */
function url(value) {
    return /^((https|http|ftp|rtsp|mms):\/\/)(([0-9a-zA-Z_!~*'().&=+$%-]+: )?[0-9a-zA-Z_!~*'().&=+$%-]+@)?(([0-9]{1,3}.){3}[0-9]{1,3}|([0-9a-zA-Z_!~*'()-]+.)*([0-9a-zA-Z][0-9a-zA-Z-]{0,61})?[0-9a-zA-Z].[a-zA-Z]{2,6})(:[0-9]{1,4})?((\/?)|(\/[0-9a-zA-Z_!~*'().;?:@&=+$,%#-]+)+\/?)$/
        .test(value)
}
/**
 * 验证日期格式
 */
function date(value) {
    if (!value) return false
    // 判断是否数值或者字符串数值(意味着为时间戳),转为数值,否则new Date无法识别字符串时间戳
    if (number(value)) value = +value
    return !/Invalid|NaN/.test(new Date(value).toString())
}
/**
 * 验证ISO类型的日期格式
 */
function dateISO(value) {
    return /^\d{4}[\/\-](0?[1-9]|1[012])[\/\-](0?[1-9]|[12][0-9]|3[01])$/.test(value)
}
/**
 * 验证十进制数字
 */
function number(value) {
    return /^[\+-]?(\d+\.?\d*|\.\d+|\d\.\d+e\+\d+)$/.test(value)
}
/**
 * 验证字符串
 */
function string(value) {
    return typeof value === 'string'
}
/**
 * 验证整数
 */
function digits(value) {
    return /^\d+$/.test(value)
}
/**
 * 验证身份证号码
 */
function idCard(value) {
    return /^[1-9]\d{5}[1-9]\d{3}((0\d)|(1[0-2]))(([0|1|2]\d)|3[0-1])\d{3}([0-9]|X)$/.test(
        value
    )
}
/**
 * 是否车牌号
 */
function carNo(value) {
    // 新能源车牌
    const xreg = /^[京津沪渝冀豫云辽黑湘皖鲁新苏浙赣鄂桂甘晋蒙陕吉闽贵粤青藏川宁琼使领A-Z]{1}[A-Z]{1}(([0-9]{5}[DF]$)|([DF][A-HJ-NP-Z0-9][0-9]{4}$))/
    // 旧车牌
    const creg = /^[京津沪渝冀豫云辽黑湘皖鲁新苏浙赣鄂桂甘晋蒙陕吉闽贵粤青藏川宁琼使领A-Z]{1}[A-Z]{1}[A-HJ-NP-Z0-9]{4}[A-HJ-NP-Z0-9挂学警港澳]{1}$/
    if (value.length === 7) {
        return creg.test(value)
    } if (value.length === 8) {
        return xreg.test(value)
    }
    return false
}
/**
 * 金额,只允许2位小数
 */
function amount(value) {
    // 金额,只允许保留两位小数
    return /^[1-9]\d*(,\d{3})*(\.\d{1,2})?$|^0\.\d{1,2}$/.test(value)
}
/**
 * 中文
 */
function chinese(value) {
    const reg = /^[\u4e00-\u9fa5]+$/gi
    return reg.test(value)
}
/**
 * 只能输入字母
 */
function letter(value) {
    return /^[a-zA-Z]*$/.test(value)
}
/**
 * 只能是字母或者数字
 */
function enOrNum(value) {
    // 英文或者数字
    const reg = /^[0-9a-zA-Z]*$/g
    return reg.test(value)
}
/**
 * 验证是否包含某个值
 */
function contains(value, param) {
    return value.indexOf(param) >= 0
}
/**
 * 验证一个值范围[min, max]
 */
function range(value, param) {
    return value >= param[0] && value <= param[1]
}
/**
 * 验证一个长度范围[min, max]
 */
function rangeLength(value, param) {
    return value.length >= param[0] && value.length <= param[1]
}
/**
 * 是否固定电话
 */
function landline(value) {
    const reg = /^\d{3,4}-\d{7,8}(-\d{3,4})?$/
    return reg.test(value)
}
/**
 * 判断是否为空
 */
function empty(value) {
    switch (typeof value) {
    case 'undefined':
        return true
    case 'string':
        if (value.replace(/(^[ \t\n\r]*)|([ \t\n\r]*$)/g, '').length == 0) return true
        break
    case 'boolean':
        if (!value) return true
        break
    case 'number':
        if (value === 0 || isNaN(value)) return true
        break
    case 'object':
        if (value === null || value.length === 0) return true
        for (const i in value) {
            return false
        }
        return true
    }
    return false
}
/**
 * 是否json字符串
 */
function jsonString(value) {
    if (typeof value === 'string') {
        try {
            const obj = JSON.parse(value)
            if (typeof obj === 'object' && obj) {
                return true
            }
            return false
        } catch (e) {
            return false
        }
    }
    return false
}
/**
 * 是否数组
 */
function array(value) {
    if (typeof Array.isArray === 'function') {
        return Array.isArray(value)
    }
    return Object.prototype.toString.call(value) === '[object Array]'
}
/**
 * 是否对象
 */
function object(value) {
    return Object.prototype.toString.call(value) === '[object Object]'
}
/**
 * 是否短信验证码
 */
function code(value, len = 6) {
    return new RegExp(`^\\d{${len}}$`).test(value)
}
/**
 * 是否函数方法
 * @param {Object} value
 */
function func(value) {
    return typeof value === 'function'
}
/**
 * 是否promise对象
 * @param {Object} value
 */
function promise(value) {
    return object(value) && func(value.then) && func(value.catch)
}
/** 是否图片格式
 * @param {Object} value
 */
function image(value) {
    const newValue = value.split('?')[0]
    const IMAGE_REGEXP = /\.(jpeg|jpg|gif|png|svg|webp|jfif|bmp|dpg)/i
    return IMAGE_REGEXP.test(newValue)
}
/**
 * 是否视频格式
 * @param {Object} value
 */
function video(value) {
    const VIDEO_REGEXP = /\.(mp4|mpg|mpeg|dat|asf|avi|rm|rmvb|mov|wmv|flv|mkv|m3u8)/i
    return VIDEO_REGEXP.test(value)
}
/**
 * 是否为正则对象
 * @param {Object}
 * @return {Boolean}
 */
function regExp(o) {
    return o && Object.prototype.toString.call(o) === '[object RegExp]'
}
export {
    email,
    mobile,
    url,
    date,
    dateISO,
    number,
    digits,
    idCard,
    carNo,
    amount,
    chinese,
    letter,
    enOrNum,
    contains,
    range,
    rangeLength,
    empty,
    jsonString,
    landline,
    object,
    array,
    code,
    func,
    promise,
    video,
    image,
    regExp,
    string
}
uni_modules/uv-ui-tools/libs/function/throttle.js
New file
@@ -0,0 +1,30 @@
let timer; let
    flag
/**
 * 节流原理:在一定时间内,只能触发一次
 *
 * @param {Function} func 要执行的回调函数
 * @param {Number} wait 延时的时间
 * @param {Boolean} immediate 是否立即执行
 * @return null
 */
function throttle(func, wait = 500, immediate = true) {
    if (immediate) {
        if (!flag) {
            flag = true
            // 如果是立即执行,则在wait毫秒内开始时执行
            typeof func === 'function' && func()
            timer = setTimeout(() => {
                flag = false
            }, wait)
        }
    } else if (!flag) {
        flag = true
        // 如果是非立即执行,则在wait毫秒内的结束处执行
        timer = setTimeout(() => {
            flag = false
            typeof func === 'function' && func()
        }, wait)
    }
}
export default throttle
uni_modules/uv-ui-tools/libs/luch-request/adapters/index.js
New file
@@ -0,0 +1,97 @@
import buildURL from '../helpers/buildURL'
import buildFullPath from '../core/buildFullPath'
import settle from '../core/settle'
import { isUndefined } from '../utils'
/**
 * 返回可选值存在的配置
 * @param {Array} keys - 可选值数组
 * @param {Object} config2 - 配置
 * @return {{}} - 存在的配置项
 */
const mergeKeys = (keys, config2) => {
    const config = {}
    keys.forEach((prop) => {
        if (!isUndefined(config2[prop])) {
            config[prop] = config2[prop]
        }
    })
    return config
}
export default (config) => new Promise((resolve, reject) => {
    const fullPath = buildURL(buildFullPath(config.baseURL, config.url), config.params)
    const _config = {
        url: fullPath,
        header: config.header,
        complete: (response) => {
            config.fullPath = fullPath
            response.config = config
            try {
                // 对可能字符串不是json 的情况容错
                if (typeof response.data === 'string') {
                    response.data = JSON.parse(response.data)
                }
                // eslint-disable-next-line no-empty
            } catch (e) {
            }
            settle(resolve, reject, response)
        }
    }
    let requestTask
    if (config.method === 'UPLOAD') {
        delete _config.header['content-type']
        delete _config.header['Content-Type']
        const otherConfig = {
        // #ifdef MP-ALIPAY
            fileType: config.fileType,
            // #endif
            filePath: config.filePath,
            name: config.name
        }
        const optionalKeys = [
        // #ifdef APP-PLUS || H5
            'files',
            // #endif
            // #ifdef H5
            'file',
            // #endif
            // #ifdef H5 || APP-PLUS
            'timeout',
            // #endif
            'formData'
        ]
        requestTask = uni.uploadFile({ ..._config, ...otherConfig, ...mergeKeys(optionalKeys, config) })
    } else if (config.method === 'DOWNLOAD') {
        // #ifdef H5 || APP-PLUS
        if (!isUndefined(config.timeout)) {
            _config.timeout = config.timeout
        }
        // #endif
        requestTask = uni.downloadFile(_config)
    } else {
        const optionalKeys = [
            'data',
            'method',
            // #ifdef H5 || APP-PLUS || MP-ALIPAY || MP-WEIXIN
            'timeout',
            // #endif
            'dataType',
            // #ifndef MP-ALIPAY
            'responseType',
            // #endif
            // #ifdef APP-PLUS
            'sslVerify',
            // #endif
            // #ifdef H5
            'withCredentials',
            // #endif
            // #ifdef APP-PLUS
            'firstIpv4'
        // #endif
        ]
        requestTask = uni.request({ ..._config, ...mergeKeys(optionalKeys, config) })
    }
    if (config.getTask) {
        config.getTask(requestTask, config)
    }
})
uni_modules/uv-ui-tools/libs/luch-request/core/InterceptorManager.js
New file
@@ -0,0 +1,50 @@
'use strict'
function InterceptorManager() {
    this.handlers = []
}
/**
 * Add a new interceptor to the stack
 *
 * @param {Function} fulfilled The function to handle `then` for a `Promise`
 * @param {Function} rejected The function to handle `reject` for a `Promise`
 *
 * @return {Number} An ID used to remove interceptor later
 */
InterceptorManager.prototype.use = function use(fulfilled, rejected) {
    this.handlers.push({
        fulfilled,
        rejected
    })
    return this.handlers.length - 1
}
/**
 * Remove an interceptor from the stack
 *
 * @param {Number} id The ID that was returned by `use`
 */
InterceptorManager.prototype.eject = function eject(id) {
    if (this.handlers[id]) {
        this.handlers[id] = null
    }
}
/**
 * Iterate over all the registered interceptors
 *
 * This method is particularly useful for skipping over any
 * interceptors that may have become `null` calling `eject`.
 *
 * @param {Function} fn The function to call for each interceptor
 */
InterceptorManager.prototype.forEach = function forEach(fn) {
    this.handlers.forEach((h) => {
        if (h !== null) {
            fn(h)
        }
    })
}
export default InterceptorManager
uni_modules/uv-ui-tools/libs/luch-request/core/Request.js
New file
@@ -0,0 +1,198 @@
/**
 * @Class Request
 * @description luch-request http请求插件
 * @version 3.0.7
 * @Author lu-ch
 * @Date 2021-09-04
 * @Email webwork.s@qq.com
 * 文档: https://www.quanzhan.co/luch-request/
 * github: https://github.com/lei-mu/luch-request
 * DCloud: http://ext.dcloud.net.cn/plugin?id=392
 * HBuilderX: beat-3.0.4 alpha-3.0.4
 */
import dispatchRequest from './dispatchRequest'
import InterceptorManager from './InterceptorManager'
import mergeConfig from './mergeConfig'
import defaults from './defaults'
import { isPlainObject } from '../utils'
import clone from '../utils/clone'
export default class Request {
    /**
   * @param {Object} arg - 全局配置
   * @param {String} arg.baseURL - 全局根路径
   * @param {Object} arg.header - 全局header
   * @param {String} arg.method = [GET|POST|PUT|DELETE|CONNECT|HEAD|OPTIONS|TRACE] - 全局默认请求方式
   * @param {String} arg.dataType = [json] - 全局默认的dataType
   * @param {String} arg.responseType = [text|arraybuffer] - 全局默认的responseType。支付宝小程序不支持
   * @param {Object} arg.custom - 全局默认的自定义参数
   * @param {Number} arg.timeout - 全局默认的超时时间,单位 ms。默认60000。H5(HBuilderX 2.9.9+)、APP(HBuilderX 2.9.9+)、微信小程序(2.10.0)、支付宝小程序
   * @param {Boolean} arg.sslVerify - 全局默认的是否验证 ssl 证书。默认true.仅App安卓端支持(HBuilderX 2.3.3+)
   * @param {Boolean} arg.withCredentials - 全局默认的跨域请求时是否携带凭证(cookies)。默认false。仅H5支持(HBuilderX 2.6.15+)
   * @param {Boolean} arg.firstIpv4 - 全DNS解析时优先使用ipv4。默认false。仅 App-Android 支持 (HBuilderX 2.8.0+)
   * @param {Function(statusCode):Boolean} arg.validateStatus - 全局默认的自定义验证器。默认statusCode >= 200 && statusCode < 300
   */
    constructor(arg = {}) {
        if (!isPlainObject(arg)) {
            arg = {}
            console.warn('设置全局参数必须接收一个Object')
        }
        this.config = clone({ ...defaults, ...arg })
        this.interceptors = {
            request: new InterceptorManager(),
            response: new InterceptorManager()
        }
    }
    /**
   * @Function
   * @param {Request~setConfigCallback} f - 设置全局默认配置
   */
    setConfig(f) {
        this.config = f(this.config)
    }
    middleware(config) {
        config = mergeConfig(this.config, config)
        const chain = [dispatchRequest, undefined]
        let promise = Promise.resolve(config)
        this.interceptors.request.forEach((interceptor) => {
            chain.unshift(interceptor.fulfilled, interceptor.rejected)
        })
        this.interceptors.response.forEach((interceptor) => {
            chain.push(interceptor.fulfilled, interceptor.rejected)
        })
        while (chain.length) {
            promise = promise.then(chain.shift(), chain.shift())
        }
        return promise
    }
    /**
   * @Function
   * @param {Object} config - 请求配置项
   * @prop {String} options.url - 请求路径
   * @prop {Object} options.data - 请求参数
   * @prop {Object} [options.responseType = config.responseType] [text|arraybuffer] - 响应的数据类型
   * @prop {Object} [options.dataType = config.dataType] - 如果设为 json,会尝试对返回的数据做一次 JSON.parse
   * @prop {Object} [options.header = config.header] - 请求header
   * @prop {Object} [options.method = config.method] - 请求方法
   * @returns {Promise<unknown>}
   */
    request(config = {}) {
        return this.middleware(config)
    }
    get(url, options = {}) {
        return this.middleware({
            url,
            method: 'GET',
            ...options
        })
    }
    post(url, data, options = {}) {
        return this.middleware({
            url,
            data,
            method: 'POST',
            ...options
        })
    }
    // #ifndef MP-ALIPAY
    put(url, data, options = {}) {
        return this.middleware({
            url,
            data,
            method: 'PUT',
            ...options
        })
    }
    // #endif
    // #ifdef APP-PLUS || H5 || MP-WEIXIN || MP-BAIDU
    delete(url, data, options = {}) {
        return this.middleware({
            url,
            data,
            method: 'DELETE',
            ...options
        })
    }
    // #endif
    // #ifdef H5 || MP-WEIXIN
    connect(url, data, options = {}) {
        return this.middleware({
            url,
            data,
            method: 'CONNECT',
            ...options
        })
    }
    // #endif
    // #ifdef  H5 || MP-WEIXIN || MP-BAIDU
    head(url, data, options = {}) {
        return this.middleware({
            url,
            data,
            method: 'HEAD',
            ...options
        })
    }
    // #endif
    // #ifdef APP-PLUS || H5 || MP-WEIXIN || MP-BAIDU
    options(url, data, options = {}) {
        return this.middleware({
            url,
            data,
            method: 'OPTIONS',
            ...options
        })
    }
    // #endif
    // #ifdef H5 || MP-WEIXIN
    trace(url, data, options = {}) {
        return this.middleware({
            url,
            data,
            method: 'TRACE',
            ...options
        })
    }
    // #endif
    upload(url, config = {}) {
        config.url = url
        config.method = 'UPLOAD'
        return this.middleware(config)
    }
    download(url, config = {}) {
        config.url = url
        config.method = 'DOWNLOAD'
        return this.middleware(config)
    }
}
/**
 * setConfig回调
 * @return {Object} - 返回操作后的config
 * @callback Request~setConfigCallback
 * @param {Object} config - 全局默认config
 */
uni_modules/uv-ui-tools/libs/luch-request/core/buildFullPath.js
New file
@@ -0,0 +1,20 @@
'use strict'
import isAbsoluteURL from '../helpers/isAbsoluteURL'
import combineURLs from '../helpers/combineURLs'
/**
 * Creates a new URL by combining the baseURL with the requestedURL,
 * only when the requestedURL is not already an absolute URL.
 * If the requestURL is absolute, this function returns the requestedURL untouched.
 *
 * @param {string} baseURL The base URL
 * @param {string} requestedURL Absolute or relative URL to combine
 * @returns {string} The combined full path
 */
export default function buildFullPath(baseURL, requestedURL) {
    if (baseURL && !isAbsoluteURL(requestedURL)) {
        return combineURLs(baseURL, requestedURL)
    }
    return requestedURL
}
uni_modules/uv-ui-tools/libs/luch-request/core/defaults.js
New file
@@ -0,0 +1,29 @@
/**
 * 默认的全局配置
 */
export default {
    baseURL: '',
    header: {},
    method: 'GET',
    dataType: 'json',
    // #ifndef MP-ALIPAY
    responseType: 'text',
    // #endif
    custom: {},
    // #ifdef H5 || APP-PLUS || MP-ALIPAY || MP-WEIXIN
    timeout: 60000,
    // #endif
    // #ifdef APP-PLUS
    sslVerify: true,
    // #endif
    // #ifdef H5
    withCredentials: false,
    // #endif
    // #ifdef APP-PLUS
    firstIpv4: false,
    // #endif
    validateStatus: function validateStatus(status) {
        return status >= 200 && status < 300
    }
}
uni_modules/uv-ui-tools/libs/luch-request/core/dispatchRequest.js
New file
@@ -0,0 +1,3 @@
import adapter from '../adapters/index'
export default (config) => adapter(config)
uni_modules/uv-ui-tools/libs/luch-request/core/mergeConfig.js
New file
@@ -0,0 +1,103 @@
import { deepMerge, isUndefined } from '../utils'
/**
 * 合并局部配置优先的配置,如果局部有该配置项则用局部,如果全局有该配置项则用全局
 * @param {Array} keys - 配置项
 * @param {Object} globalsConfig - 当前的全局配置
 * @param {Object} config2 - 局部配置
 * @return {{}}
 */
const mergeKeys = (keys, globalsConfig, config2) => {
    const config = {}
    keys.forEach((prop) => {
        if (!isUndefined(config2[prop])) {
            config[prop] = config2[prop]
        } else if (!isUndefined(globalsConfig[prop])) {
            config[prop] = globalsConfig[prop]
        }
    })
    return config
}
/**
 *
 * @param globalsConfig - 当前实例的全局配置
 * @param config2 - 当前的局部配置
 * @return - 合并后的配置
 */
export default (globalsConfig, config2 = {}) => {
    const method = config2.method || globalsConfig.method || 'GET'
    let config = {
        baseURL: globalsConfig.baseURL || '',
        method,
        url: config2.url || '',
        params: config2.params || {},
        custom: { ...(globalsConfig.custom || {}), ...(config2.custom || {}) },
        header: deepMerge(globalsConfig.header || {}, config2.header || {})
    }
    const defaultToConfig2Keys = ['getTask', 'validateStatus']
    config = { ...config, ...mergeKeys(defaultToConfig2Keys, globalsConfig, config2) }
    // eslint-disable-next-line no-empty
    if (method === 'DOWNLOAD') {
    // #ifdef H5 || APP-PLUS
        if (!isUndefined(config2.timeout)) {
            config.timeout = config2.timeout
        } else if (!isUndefined(globalsConfig.timeout)) {
            config.timeout = globalsConfig.timeout
        }
    // #endif
    } else if (method === 'UPLOAD') {
        delete config.header['content-type']
        delete config.header['Content-Type']
        const uploadKeys = [
            // #ifdef APP-PLUS || H5
            'files',
            // #endif
            // #ifdef MP-ALIPAY
            'fileType',
            // #endif
            // #ifdef H5
            'file',
            // #endif
            'filePath',
            'name',
            // #ifdef H5 || APP-PLUS
            'timeout',
            // #endif
            'formData'
        ]
        uploadKeys.forEach((prop) => {
            if (!isUndefined(config2[prop])) {
                config[prop] = config2[prop]
            }
        })
        // #ifdef H5 || APP-PLUS
        if (isUndefined(config.timeout) && !isUndefined(globalsConfig.timeout)) {
            config.timeout = globalsConfig.timeout
        }
    // #endif
    } else {
        const defaultsKeys = [
            'data',
            // #ifdef H5 || APP-PLUS || MP-ALIPAY || MP-WEIXIN
            'timeout',
            // #endif
            'dataType',
            // #ifndef MP-ALIPAY
            'responseType',
            // #endif
            // #ifdef APP-PLUS
            'sslVerify',
            // #endif
            // #ifdef H5
            'withCredentials',
            // #endif
            // #ifdef APP-PLUS
            'firstIpv4'
            // #endif
        ]
        config = { ...config, ...mergeKeys(defaultsKeys, globalsConfig, config2) }
    }
    return config
}
uni_modules/uv-ui-tools/libs/luch-request/core/settle.js
New file
@@ -0,0 +1,16 @@
/**
 * Resolve or reject a Promise based on response status.
 *
 * @param {Function} resolve A function that resolves the promise.
 * @param {Function} reject A function that rejects the promise.
 * @param {object} response The response.
 */
export default function settle(resolve, reject, response) {
    const { validateStatus } = response.config
    const status = response.statusCode
    if (status && (!validateStatus || validateStatus(status))) {
        resolve(response)
    } else {
        reject(response)
    }
}
uni_modules/uv-ui-tools/libs/luch-request/helpers/buildURL.js
New file
@@ -0,0 +1,69 @@
'use strict'
import * as utils from '../utils'
function encode(val) {
    return encodeURIComponent(val)
        .replace(/%40/gi, '@')
        .replace(/%3A/gi, ':')
        .replace(/%24/g, '$')
        .replace(/%2C/gi, ',')
        .replace(/%20/g, '+')
        .replace(/%5B/gi, '[')
        .replace(/%5D/gi, ']')
}
/**
 * Build a URL by appending params to the end
 *
 * @param {string} url The base of the url (e.g., http://www.google.com)
 * @param {object} [params] The params to be appended
 * @returns {string} The formatted url
 */
export default function buildURL(url, params) {
    /* eslint no-param-reassign:0 */
    if (!params) {
        return url
    }
    let serializedParams
    if (utils.isURLSearchParams(params)) {
        serializedParams = params.toString()
    } else {
        const parts = []
        utils.forEach(params, (val, key) => {
            if (val === null || typeof val === 'undefined') {
                return
            }
            if (utils.isArray(val)) {
                key = `${key}[]`
            } else {
                val = [val]
            }
            utils.forEach(val, (v) => {
                if (utils.isDate(v)) {
                    v = v.toISOString()
                } else if (utils.isObject(v)) {
                    v = JSON.stringify(v)
                }
                parts.push(`${encode(key)}=${encode(v)}`)
            })
        })
        serializedParams = parts.join('&')
    }
    if (serializedParams) {
        const hashmarkIndex = url.indexOf('#')
        if (hashmarkIndex !== -1) {
            url = url.slice(0, hashmarkIndex)
        }
        url += (url.indexOf('?') === -1 ? '?' : '&') + serializedParams
    }
    return url
}
uni_modules/uv-ui-tools/libs/luch-request/helpers/combineURLs.js
New file
@@ -0,0 +1,14 @@
'use strict'
/**
 * Creates a new URL by combining the specified URLs
 *
 * @param {string} baseURL The base URL
 * @param {string} relativeURL The relative URL
 * @returns {string} The combined URL
 */
export default function combineURLs(baseURL, relativeURL) {
    return relativeURL
        ? `${baseURL.replace(/\/+$/, '')}/${relativeURL.replace(/^\/+/, '')}`
        : baseURL
}
uni_modules/uv-ui-tools/libs/luch-request/helpers/isAbsoluteURL.js
New file
@@ -0,0 +1,14 @@
'use strict'
/**
 * Determines whether the specified URL is absolute
 *
 * @param {string} url The URL to test
 * @returns {boolean} True if the specified URL is absolute, otherwise false
 */
export default function isAbsoluteURL(url) {
    // A URL is considered absolute if it begins with "<scheme>://" or "//" (protocol-relative URL).
    // RFC 3986 defines scheme name as a sequence of characters beginning with a letter and followed
    // by any combination of letters, digits, plus, period, or hyphen.
    return /^([a-z][a-z\d+\-.]*:)?\/\//i.test(url)
}
uni_modules/uv-ui-tools/libs/luch-request/index.d.ts
New file
@@ -0,0 +1,116 @@
type AnyObject = Record<string | number | symbol, any>
type HttpPromise<T> = Promise<HttpResponse<T>>;
type Tasks = UniApp.RequestTask | UniApp.UploadTask | UniApp.DownloadTask
export interface RequestTask {
  abort: () => void;
  offHeadersReceived: () => void;
  onHeadersReceived: () => void;
}
export interface HttpRequestConfig<T = Tasks> {
  /** 请求基地址 */
  baseURL?: string;
  /** 请求服务器接口地址 */
  url?: string;
  /** 请求查询参数,自动拼接为查询字符串 */
  params?: AnyObject;
  /** 请求体参数 */
  data?: AnyObject;
  /** 文件对应的 key */
  name?: string;
  /** HTTP 请求中其他额外的 form data */
  formData?: AnyObject;
  /** 要上传文件资源的路径。 */
  filePath?: string;
  /** 需要上传的文件列表。使用 files 时,filePath 和 name 不生效,App、H5( 2.6.15+) */
  files?: Array<{
    name?: string;
    file?: File;
    uri: string;
  }>;
  /** 要上传的文件对象,仅H5(2.6.15+)支持 */
  file?: File;
  /** 请求头信息 */
  header?: AnyObject;
  /** 请求方式 */
  method?: "GET" | "POST" | "PUT" | "DELETE" | "CONNECT" | "HEAD" | "OPTIONS" | "TRACE" | "UPLOAD" | "DOWNLOAD";
  /** 如果设为 json,会尝试对返回的数据做一次 JSON.parse */
  dataType?: string;
  /** 设置响应的数据类型,支付宝小程序不支持 */
  responseType?: "text" | "arraybuffer";
  /** 自定义参数 */
  custom?: AnyObject;
  /** 超时时间,仅微信小程序(2.10.0)、支付宝小程序支持 */
  timeout?: number;
  /** DNS解析时优先使用ipv4,仅 App-Android 支持 (HBuilderX 2.8.0+) */
  firstIpv4?: boolean;
  /** 验证 ssl 证书 仅5+App安卓端支持(HBuilderX 2.3.3+) */
  sslVerify?: boolean;
  /** 跨域请求时是否携带凭证(cookies)仅H5支持(HBuilderX 2.6.15+) */
  withCredentials?: boolean;
  /** 返回当前请求的task, options。请勿在此处修改options。 */
  getTask?: (task: T, options: HttpRequestConfig<T>) => void;
  /**  全局自定义验证器 */
  validateStatus?: (statusCode: number) => boolean | void;
}
export interface HttpResponse<T = any> {
  config: HttpRequestConfig;
  statusCode: number;
  cookies: Array<string>;
  data: T;
  errMsg: string;
  header: AnyObject;
}
export interface HttpUploadResponse<T = any> {
  config: HttpRequestConfig;
  statusCode: number;
  data: T;
  errMsg: string;
}
export interface HttpDownloadResponse extends HttpResponse {
  tempFilePath: string;
}
export interface HttpError {
  config: HttpRequestConfig;
  statusCode?: number;
  cookies?: Array<string>;
  data?: any;
  errMsg: string;
  header?: AnyObject;
}
export interface HttpInterceptorManager<V, E = V> {
  use(
    onFulfilled?: (config: V) => Promise<V> | V,
    onRejected?: (config: E) => Promise<E> | E
  ): void;
  eject(id: number): void;
}
export abstract class HttpRequestAbstract {
  constructor(config?: HttpRequestConfig);
  config: HttpRequestConfig;
  interceptors: {
    request: HttpInterceptorManager<HttpRequestConfig, HttpRequestConfig>;
    response: HttpInterceptorManager<HttpResponse, HttpError>;
  }
  middleware<T = any>(config: HttpRequestConfig): HttpPromise<T>;
  request<T = any>(config: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  get<T = any>(url: string, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  upload<T = any>(url: string, config?: HttpRequestConfig<UniApp.UploadTask>): HttpPromise<T>;
  delete<T = any>(url: string, data?: AnyObject, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  head<T = any>(url: string, data?: AnyObject, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  post<T = any>(url: string, data?: AnyObject, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  put<T = any>(url: string, data?: AnyObject, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  connect<T = any>(url: string, data?: AnyObject, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  options<T = any>(url: string, data?: AnyObject, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  trace<T = any>(url: string, data?: AnyObject, config?: HttpRequestConfig<UniApp.RequestTask>): HttpPromise<T>;
  download(url: string, config?: HttpRequestConfig<UniApp.DownloadTask>): Promise<HttpDownloadResponse>;
  setConfig(onSend: (config: HttpRequestConfig) => HttpRequestConfig): void;
}
declare class HttpRequest extends HttpRequestAbstract { }
export default HttpRequest;
uni_modules/uv-ui-tools/libs/luch-request/index.js
New file
@@ -0,0 +1,3 @@
import Request from './core/Request'
export default Request
uni_modules/uv-ui-tools/libs/luch-request/utils.js
New file
@@ -0,0 +1,131 @@
'use strict'
// utils is a library of generic helper functions non-specific to axios
const { toString } = Object.prototype
/**
 * Determine if a value is an Array
 *
 * @param {Object} val The value to test
 * @returns {boolean} True if value is an Array, otherwise false
 */
export function isArray(val) {
    return toString.call(val) === '[object Array]'
}
/**
 * Determine if a value is an Object
 *
 * @param {Object} val The value to test
 * @returns {boolean} True if value is an Object, otherwise false
 */
export function isObject(val) {
    return val !== null && typeof val === 'object'
}
/**
 * Determine if a value is a Date
 *
 * @param {Object} val The value to test
 * @returns {boolean} True if value is a Date, otherwise false
 */
export function isDate(val) {
    return toString.call(val) === '[object Date]'
}
/**
 * Determine if a value is a URLSearchParams object
 *
 * @param {Object} val The value to test
 * @returns {boolean} True if value is a URLSearchParams object, otherwise false
 */
export function isURLSearchParams(val) {
    return typeof URLSearchParams !== 'undefined' && val instanceof URLSearchParams
}
/**
 * Iterate over an Array or an Object invoking a function for each item.
 *
 * If `obj` is an Array callback will be called passing
 * the value, index, and complete array for each item.
 *
 * If 'obj' is an Object callback will be called passing
 * the value, key, and complete object for each property.
 *
 * @param {Object|Array} obj The object to iterate
 * @param {Function} fn The callback to invoke for each item
 */
export function forEach(obj, fn) {
    // Don't bother if no value provided
    if (obj === null || typeof obj === 'undefined') {
        return
    }
    // Force an array if not already something iterable
    if (typeof obj !== 'object') {
    /* eslint no-param-reassign:0 */
        obj = [obj]
    }
    if (isArray(obj)) {
    // Iterate over array values
        for (let i = 0, l = obj.length; i < l; i++) {
            fn.call(null, obj[i], i, obj)
        }
    } else {
    // Iterate over object keys
        for (const key in obj) {
            if (Object.prototype.hasOwnProperty.call(obj, key)) {
                fn.call(null, obj[key], key, obj)
            }
        }
    }
}
/**
 * 是否为boolean 值
 * @param val
 * @returns {boolean}
 */
export function isBoolean(val) {
    return typeof val === 'boolean'
}
/**
 * 是否为真正的对象{} new Object
 * @param {any} obj - 检测的对象
 * @returns {boolean}
 */
export function isPlainObject(obj) {
    return Object.prototype.toString.call(obj) === '[object Object]'
}
/**
 * Function equal to merge with the difference being that no reference
 * to original objects is kept.
 *
 * @see merge
 * @param {Object} obj1 Object to merge
 * @returns {Object} Result of all merge properties
 */
export function deepMerge(/* obj1, obj2, obj3, ... */) {
    const result = {}
    function assignValue(val, key) {
        if (typeof result[key] === 'object' && typeof val === 'object') {
            result[key] = deepMerge(result[key], val)
        } else if (typeof val === 'object') {
            result[key] = deepMerge({}, val)
        } else {
            result[key] = val
        }
    }
    for (let i = 0, l = arguments.length; i < l; i++) {
        forEach(arguments[i], assignValue)
    }
    return result
}
export function isUndefined(val) {
    return typeof val === 'undefined'
}
uni_modules/uv-ui-tools/libs/luch-request/utils/clone.js
New file
@@ -0,0 +1,264 @@
/* eslint-disable */
var clone = (function() {
  'use strict';
  function _instanceof(obj, type) {
    return type != null && obj instanceof type;
  }
  var nativeMap;
  try {
    nativeMap = Map;
  } catch(_) {
    // maybe a reference error because no `Map`. Give it a dummy value that no
    // value will ever be an instanceof.
    nativeMap = function() {};
  }
  var nativeSet;
  try {
    nativeSet = Set;
  } catch(_) {
    nativeSet = function() {};
  }
  var nativePromise;
  try {
    nativePromise = Promise;
  } catch(_) {
    nativePromise = function() {};
  }
  /**
   * Clones (copies) an Object using deep copying.
   *
   * This function supports circular references by default, but if you are certain
   * there are no circular references in your object, you can save some CPU time
   * by calling clone(obj, false).
   *
   * Caution: if `circular` is false and `parent` contains circular references,
   * your program may enter an infinite loop and crash.
   *
   * @param `parent` - the object to be cloned
   * @param `circular` - set to true if the object to be cloned may contain
   *    circular references. (optional - true by default)
   * @param `depth` - set to a number if the object is only to be cloned to
   *    a particular depth. (optional - defaults to Infinity)
   * @param `prototype` - sets the prototype to be used when cloning an object.
   *    (optional - defaults to parent prototype).
   * @param `includeNonEnumerable` - set to true if the non-enumerable properties
   *    should be cloned as well. Non-enumerable properties on the prototype
   *    chain will be ignored. (optional - false by default)
   */
  function clone(parent, circular, depth, prototype, includeNonEnumerable) {
    if (typeof circular === 'object') {
      depth = circular.depth;
      prototype = circular.prototype;
      includeNonEnumerable = circular.includeNonEnumerable;
      circular = circular.circular;
    }
    // maintain two arrays for circular references, where corresponding parents
    // and children have the same index
    var allParents = [];
    var allChildren = [];
    var useBuffer = typeof Buffer != 'undefined';
    if (typeof circular == 'undefined')
      circular = true;
    if (typeof depth == 'undefined')
      depth = Infinity;
    // recurse this function so we don't reset allParents and allChildren
    function _clone(parent, depth) {
      // cloning null always returns null
      if (parent === null)
        return null;
      if (depth === 0)
        return parent;
      var child;
      var proto;
      if (typeof parent != 'object') {
        return parent;
      }
      if (_instanceof(parent, nativeMap)) {
        child = new nativeMap();
      } else if (_instanceof(parent, nativeSet)) {
        child = new nativeSet();
      } else if (_instanceof(parent, nativePromise)) {
        child = new nativePromise(function (resolve, reject) {
          parent.then(function(value) {
            resolve(_clone(value, depth - 1));
          }, function(err) {
            reject(_clone(err, depth - 1));
          });
        });
      } else if (clone.__isArray(parent)) {
        child = [];
      } else if (clone.__isRegExp(parent)) {
        child = new RegExp(parent.source, __getRegExpFlags(parent));
        if (parent.lastIndex) child.lastIndex = parent.lastIndex;
      } else if (clone.__isDate(parent)) {
        child = new Date(parent.getTime());
      } else if (useBuffer && Buffer.isBuffer(parent)) {
        if (Buffer.from) {
          // Node.js >= 5.10.0
          child = Buffer.from(parent);
        } else {
          // Older Node.js versions
          child = new Buffer(parent.length);
          parent.copy(child);
        }
        return child;
      } else if (_instanceof(parent, Error)) {
        child = Object.create(parent);
      } else {
        if (typeof prototype == 'undefined') {
          proto = Object.getPrototypeOf(parent);
          child = Object.create(proto);
        }
        else {
          child = Object.create(prototype);
          proto = prototype;
        }
      }
      if (circular) {
        var index = allParents.indexOf(parent);
        if (index != -1) {
          return allChildren[index];
        }
        allParents.push(parent);
        allChildren.push(child);
      }
      if (_instanceof(parent, nativeMap)) {
        parent.forEach(function(value, key) {
          var keyChild = _clone(key, depth - 1);
          var valueChild = _clone(value, depth - 1);
          child.set(keyChild, valueChild);
        });
      }
      if (_instanceof(parent, nativeSet)) {
        parent.forEach(function(value) {
          var entryChild = _clone(value, depth - 1);
          child.add(entryChild);
        });
      }
      for (var i in parent) {
        var attrs = Object.getOwnPropertyDescriptor(parent, i);
        if (attrs) {
          child[i] = _clone(parent[i], depth - 1);
        }
        try {
          var objProperty = Object.getOwnPropertyDescriptor(parent, i);
          if (objProperty.set === 'undefined') {
            // no setter defined. Skip cloning this property
            continue;
          }
          child[i] = _clone(parent[i], depth - 1);
        } catch(e){
          if (e instanceof TypeError) {
            // when in strict mode, TypeError will be thrown if child[i] property only has a getter
            // we can't do anything about this, other than inform the user that this property cannot be set.
            continue
          } else if (e instanceof ReferenceError) {
            //this may happen in non strict mode
            continue
          }
        }
      }
      if (Object.getOwnPropertySymbols) {
        var symbols = Object.getOwnPropertySymbols(parent);
        for (var i = 0; i < symbols.length; i++) {
          // Don't need to worry about cloning a symbol because it is a primitive,
          // like a number or string.
          var symbol = symbols[i];
          var descriptor = Object.getOwnPropertyDescriptor(parent, symbol);
          if (descriptor && !descriptor.enumerable && !includeNonEnumerable) {
            continue;
          }
          child[symbol] = _clone(parent[symbol], depth - 1);
          Object.defineProperty(child, symbol, descriptor);
        }
      }
      if (includeNonEnumerable) {
        var allPropertyNames = Object.getOwnPropertyNames(parent);
        for (var i = 0; i < allPropertyNames.length; i++) {
          var propertyName = allPropertyNames[i];
          var descriptor = Object.getOwnPropertyDescriptor(parent, propertyName);
          if (descriptor && descriptor.enumerable) {
            continue;
          }
          child[propertyName] = _clone(parent[propertyName], depth - 1);
          Object.defineProperty(child, propertyName, descriptor);
        }
      }
      return child;
    }
    return _clone(parent, depth);
  }
  /**
   * Simple flat clone using prototype, accepts only objects, usefull for property
   * override on FLAT configuration object (no nested props).
   *
   * USE WITH CAUTION! This may not behave as you wish if you do not know how this
   * works.
   */
  clone.clonePrototype = function clonePrototype(parent) {
    if (parent === null)
      return null;
    var c = function () {};
    c.prototype = parent;
    return new c();
  };
// private utility functions
  function __objToStr(o) {
    return Object.prototype.toString.call(o);
  }
  clone.__objToStr = __objToStr;
  function __isDate(o) {
    return typeof o === 'object' && __objToStr(o) === '[object Date]';
  }
  clone.__isDate = __isDate;
  function __isArray(o) {
    return typeof o === 'object' && __objToStr(o) === '[object Array]';
  }
  clone.__isArray = __isArray;
  function __isRegExp(o) {
    return typeof o === 'object' && __objToStr(o) === '[object RegExp]';
  }
  clone.__isRegExp = __isRegExp;
  function __getRegExpFlags(re) {
    var flags = '';
    if (re.global) flags += 'g';
    if (re.ignoreCase) flags += 'i';
    if (re.multiline) flags += 'm';
    return flags;
  }
  clone.__getRegExpFlags = __getRegExpFlags;
  return clone;
})();
export default clone
uni_modules/uv-ui-tools/libs/mixin/button.js
New file
@@ -0,0 +1,13 @@
export default {
    props: {
        lang: String,
        sessionFrom: String,
        sendMessageTitle: String,
        sendMessagePath: String,
        sendMessageImg: String,
        showMessageCard: Boolean,
        appParameter: String,
        formType: String,
        openType: String
    }
}
uni_modules/uv-ui-tools/libs/mixin/mixin.js
New file
@@ -0,0 +1,166 @@
import * as index from '../function/index.js';
import * as test from '../function/test.js';
export default {
    // 定义每个组件都可能需要用到的外部样式以及类名
    props: {
        // 每个组件都有的父组件传递的样式,可以为字符串或者对象形式
        customStyle: {
            type: [Object, String],
            default: () => ({})
        },
        customClass: {
            type: String,
            default: ''
        },
        // 跳转的页面路径
        url: {
            type: String,
            default: ''
        },
        // 页面跳转的类型
        linkType: {
            type: String,
            default: 'navigateTo'
        }
    },
    data() {
        return {}
    },
    onLoad() {
        // getRect挂载到$uv上,因为这方法需要使用in(this),所以无法把它独立成一个单独的文件导出
        this.$uv.getRect = this.$uvGetRect
    },
    created() {
        // 组件当中,只有created声明周期,为了能在组件使用,故也在created中将方法挂载到$uv
        this.$uv.getRect = this.$uvGetRect
    },
    computed: {
        $uv() {
            return {
                ...index,
                test,
                unit: uni?.$uv?.config?.unit
            }
        },
        /**
         * 生成bem规则类名
         * 由于微信小程序,H5,nvue之间绑定class的差异,无法通过:class="[bem()]"的形式进行同用
         * 故采用如下折中做法,最后返回的是数组(一般平台)或字符串(支付宝和字节跳动平台),类似['a', 'b', 'c']或'a b c'的形式
         * @param {String} name 组件名称
         * @param {Array} fixed 一直会存在的类名
         * @param {Array} change 会根据变量值为true或者false而出现或者隐藏的类名
         * @returns {Array|string}
         */
        bem() {
            return function(name, fixed, change) {
                // 类名前缀
                const prefix = `uv-${name}--`
                const classes = {}
                if (fixed) {
                    fixed.map((item) => {
                        // 这里的类名,会一直存在
                        classes[prefix + this[item]] = true
                    })
                }
                if (change) {
                    change.map((item) => {
                        // 这里的类名,会根据this[item]的值为true或者false,而进行添加或者移除某一个类
                        this[item] ? (classes[prefix + item] = this[item]) : (delete classes[prefix + item])
                    })
                }
                return Object.keys(classes)
                    // 支付宝,头条小程序无法动态绑定一个数组类名,否则解析出来的结果会带有",",而导致失效
                    // #ifdef MP-ALIPAY || MP-TOUTIAO || MP-LARK || MP-BAIDU
                    .join(' ')
                // #endif
            }
        }
    },
    methods: {
        // 跳转某一个页面
        openPage(urlKey = 'url') {
            const url = this[urlKey]
            if (url) {
                // 执行类似uni.navigateTo的方法
                uni[this.linkType]({
                    url
                })
            }
        },
        // 查询节点信息
        // 目前此方法在支付宝小程序中无法获取组件跟接点的尺寸,为支付宝的bug(2020-07-21)
        // 解决办法为在组件根部再套一个没有任何作用的view元素
        $uvGetRect(selector, all) {
            return new Promise((resolve) => {
                uni.createSelectorQuery()
                    .in(this)[all ? 'selectAll' : 'select'](selector)
                    .boundingClientRect((rect) => {
                        if (all && Array.isArray(rect) && rect.length) {
                            resolve(rect)
                        }
                        if (!all && rect) {
                            resolve(rect)
                        }
                    })
                    .exec()
            })
        },
        getParentData(parentName = '') {
            // 避免在created中去定义parent变量
            if (!this.parent) this.parent = {}
            // 这里的本质原理是,通过获取父组件实例(也即类似uv-radio的父组件uv-radio-group的this)
            // 将父组件this中对应的参数,赋值给本组件(uv-radio的this)的parentData对象中对应的属性
            // 之所以需要这么做,是因为所有端中,头条小程序不支持通过this.parent.xxx去监听父组件参数的变化
            // 此处并不会自动更新子组件的数据,而是依赖父组件uv-radio-group去监听data的变化,手动调用更新子组件的方法去重新获取
            this.parent = this.$uv.$parent.call(this, parentName)
            if (this.parent.children) {
                // 如果父组件的children不存在本组件的实例,才将本实例添加到父组件的children中
                this.parent.children.indexOf(this) === -1 && this.parent.children.push(this)
            }
            if (this.parent && this.parentData) {
                // 历遍parentData中的属性,将parent中的同名属性赋值给parentData
                Object.keys(this.parentData).map((key) => {
                    this.parentData[key] = this.parent[key]
                })
            }
        },
        // 阻止事件冒泡
        preventEvent(e) {
            e && typeof(e.stopPropagation) === 'function' && e.stopPropagation()
        },
        // 空操作
        noop(e) {
            this.preventEvent(e)
        }
    },
    onReachBottom() {
        uni.$emit('uvOnReachBottom')
    },
    beforeDestroy() {
        // 判断当前页面是否存在parent和chldren,一般在checkbox和checkbox-group父子联动的场景会有此情况
        // 组件销毁时,移除子组件在父组件children数组中的实例,释放资源,避免数据混乱
        if (this.parent && test.array(this.parent.children)) {
            // 组件销毁时,移除父组件中的children数组中对应的实例
            const childrenList = this.parent.children
            childrenList.map((child, index) => {
                // 如果相等,则移除
                if (child === this) {
                    childrenList.splice(index, 1)
                }
            })
        }
    },
    // 兼容vue3
    unmounted() {
        if (this.parent && test.array(this.parent.children)) {
            // 组件销毁时,移除父组件中的children数组中对应的实例
            const childrenList = this.parent.children
            childrenList.map((child, index) => {
                // 如果相等,则移除
                if (child === this) {
                    childrenList.splice(index, 1)
                }
            })
        }
    }
}
uni_modules/uv-ui-tools/libs/mixin/mpMixin.js
New file
@@ -0,0 +1,8 @@
export default {
    // #ifdef MP-WEIXIN
    // 将自定义节点设置成虚拟的(去掉自定义组件包裹层),更加接近Vue组件的表现,能更好的使用flex属性
    options: {
        virtualHost: true
    }
    // #endif
}
uni_modules/uv-ui-tools/libs/mixin/mpShare.js
New file
@@ -0,0 +1,13 @@
export default {
    onLoad() {
        // 设置默认的转发参数
        uni.$uv.mpShare = {
            title: '', // 默认为小程序名称
            path: '', // 默认为当前页面路径
            imageUrl: '' // 默认为当前页面的截图
        }
    },
    onShareAppMessage() {
        return uni.$uv.mpShare
    }
}
uni_modules/uv-ui-tools/libs/mixin/openType.js
New file
@@ -0,0 +1,47 @@
export default {
    props: {
        openType: String
    },
        emits: ['getphonenumber','getuserinfo','error','opensetting','launchapp','contact','chooseavatar','addgroupapp','chooseaddress','subscribe','login','im'],
    methods: {
        onGetPhoneNumber(event) {
            this.$emit('getphonenumber', event.detail)
        },
        onGetUserInfo(event) {
            this.$emit('getuserinfo', event.detail)
        },
        onError(event) {
            this.$emit('error', event.detail)
        },
        onOpenSetting(event) {
            this.$emit('opensetting', event.detail)
        },
        onLaunchApp(event) {
            this.$emit('launchapp', event.detail)
        },
        onContact(event) {
            this.$emit('contact', event.detail)
        },
        onChooseavatar(event) {
            this.$emit('chooseavatar', event.detail)
        },
        onAgreeprivacyauthorization(event) {
            this.$emit('agreeprivacyauthorization', event.detail)
        },
        onAddgroupapp(event) {
            this.$emit('addgroupapp', event.detail)
        },
        onChooseaddress(event) {
            this.$emit('chooseaddress', event.detail)
        },
        onSubscribe(event) {
            this.$emit('subscribe', event.detail)
        },
        onLogin(event) {
            this.$emit('login', event.detail)
        },
        onIm(event) {
            this.$emit('im', event.detail)
        }
    }
}
uni_modules/uv-ui-tools/libs/mixin/touch.js
New file
@@ -0,0 +1,59 @@
const MIN_DISTANCE = 10
function getDirection(x, y) {
    if (x > y && x > MIN_DISTANCE) {
        return 'horizontal'
    }
    if (y > x && y > MIN_DISTANCE) {
        return 'vertical'
    }
    return ''
}
export default {
    methods: {
        getTouchPoint(e) {
            if (!e) {
                return {
                    x: 0,
                    y: 0
                }
            } if (e.touches && e.touches[0]) {
                return {
                    x: e.touches[0].pageX,
                    y: e.touches[0].pageY
                }
            } if (e.changedTouches && e.changedTouches[0]) {
                return {
                    x: e.changedTouches[0].pageX,
                    y: e.changedTouches[0].pageY
                }
            }
            return {
                x: e.clientX || 0,
                y: e.clientY || 0
            }
        },
        resetTouchStatus() {
            this.direction = ''
            this.deltaX = 0
            this.deltaY = 0
            this.offsetX = 0
            this.offsetY = 0
        },
        touchStart(event) {
            this.resetTouchStatus()
            const touch = this.getTouchPoint(event)
            this.startX = touch.x
            this.startY = touch.y
        },
        touchMove(event) {
            const touch = this.getTouchPoint(event)
            this.deltaX = touch.x - this.startX
            this.deltaY = touch.y - this.startY
            this.offsetX = Math.abs(this.deltaX)
            this.offsetY = Math.abs(this.deltaY)
            this.direction =                this.direction || getDirection(this.offsetX, this.offsetY)
        }
    }
}
uni_modules/uv-ui-tools/libs/util/dayjs.js
New file
@@ -0,0 +1,216 @@
var __getOwnPropNames = Object.getOwnPropertyNames;
var __commonJS = (cb, mod) => function __require() {
  return mod || (0, cb[__getOwnPropNames(cb)[0]])((mod = { exports: {} }).exports, mod), mod.exports;
};
var require_dayjs_min = __commonJS({
  "uvuidayjs"(exports, module) {
    !function(t, e) {
      "object" == typeof exports && "undefined" != typeof module ? module.exports = e() : "function" == typeof define && define.amd ? define(e) : (t = "undefined" != typeof globalThis ? globalThis : t || self).dayjs = e();
    }(exports, function() {
      "use strict";
      var t = 1e3, e = 6e4, n = 36e5, r = "millisecond", i = "second", s = "minute", u = "hour", a = "day", o = "week", f = "month", h = "quarter", c = "year", d = "date", l = "Invalid Date", $ = /^(\d{4})[-/]?(\d{1,2})?[-/]?(\d{0,2})[Tt\s]*(\d{1,2})?:?(\d{1,2})?:?(\d{1,2})?[.:]?(\d+)?$/, y = /\[([^\]]+)]|Y{1,4}|M{1,4}|D{1,2}|d{1,4}|H{1,2}|h{1,2}|a|A|m{1,2}|s{1,2}|Z{1,2}|SSS/g, M = { name: "en", weekdays: "Sunday_Monday_Tuesday_Wednesday_Thursday_Friday_Saturday".split("_"), months: "January_February_March_April_May_June_July_August_September_October_November_December".split("_"), ordinal: function(t2) {
        var e2 = ["th", "st", "nd", "rd"], n2 = t2 % 100;
        return "[" + t2 + (e2[(n2 - 20) % 10] || e2[n2] || e2[0]) + "]";
      } }, m = function(t2, e2, n2) {
        var r2 = String(t2);
        return !r2 || r2.length >= e2 ? t2 : "" + Array(e2 + 1 - r2.length).join(n2) + t2;
      }, v = { s: m, z: function(t2) {
        var e2 = -t2.utcOffset(), n2 = Math.abs(e2), r2 = Math.floor(n2 / 60), i2 = n2 % 60;
        return (e2 <= 0 ? "+" : "-") + m(r2, 2, "0") + ":" + m(i2, 2, "0");
      }, m: function t2(e2, n2) {
        if (e2.date() < n2.date())
          return -t2(n2, e2);
        var r2 = 12 * (n2.year() - e2.year()) + (n2.month() - e2.month()), i2 = e2.clone().add(r2, f), s2 = n2 - i2 < 0, u2 = e2.clone().add(r2 + (s2 ? -1 : 1), f);
        return +(-(r2 + (n2 - i2) / (s2 ? i2 - u2 : u2 - i2)) || 0);
      }, a: function(t2) {
        return t2 < 0 ? Math.ceil(t2) || 0 : Math.floor(t2);
      }, p: function(t2) {
        return { M: f, y: c, w: o, d: a, D: d, h: u, m: s, s: i, ms: r, Q: h }[t2] || String(t2 || "").toLowerCase().replace(/s$/, "");
      }, u: function(t2) {
        return void 0 === t2;
      } }, g = "en", D = {};
      D[g] = M;
      var p = function(t2) {
        return t2 instanceof _;
      }, S = function t2(e2, n2, r2) {
        var i2;
        if (!e2)
          return g;
        if ("string" == typeof e2) {
          var s2 = e2.toLowerCase();
          D[s2] && (i2 = s2), n2 && (D[s2] = n2, i2 = s2);
          var u2 = e2.split("-");
          if (!i2 && u2.length > 1)
            return t2(u2[0]);
        } else {
          var a2 = e2.name;
          D[a2] = e2, i2 = a2;
        }
        return !r2 && i2 && (g = i2), i2 || !r2 && g;
      }, w = function(t2, e2) {
        if (p(t2))
          return t2.clone();
        var n2 = "object" == typeof e2 ? e2 : {};
        return n2.date = t2, n2.args = arguments, new _(n2);
      }, O = v;
      O.l = S, O.i = p, O.w = function(t2, e2) {
        return w(t2, { locale: e2.$L, utc: e2.$u, x: e2.$x, $offset: e2.$offset });
      };
      var _ = function() {
        function M2(t2) {
          this.$L = S(t2.locale, null, true), this.parse(t2);
        }
        var m2 = M2.prototype;
        return m2.parse = function(t2) {
          this.$d = function(t3) {
            var e2 = t3.date, n2 = t3.utc;
            if (null === e2)
              return new Date(NaN);
            if (O.u(e2))
              return new Date();
            if (e2 instanceof Date)
              return new Date(e2);
            if ("string" == typeof e2 && !/Z$/i.test(e2)) {
              var r2 = e2.match($);
              if (r2) {
                var i2 = r2[2] - 1 || 0, s2 = (r2[7] || "0").substring(0, 3);
                return n2 ? new Date(Date.UTC(r2[1], i2, r2[3] || 1, r2[4] || 0, r2[5] || 0, r2[6] || 0, s2)) : new Date(r2[1], i2, r2[3] || 1, r2[4] || 0, r2[5] || 0, r2[6] || 0, s2);
              }
            }
            return new Date(e2);
          }(t2), this.$x = t2.x || {}, this.init();
        }, m2.init = function() {
          var t2 = this.$d;
          this.$y = t2.getFullYear(), this.$M = t2.getMonth(), this.$D = t2.getDate(), this.$W = t2.getDay(), this.$H = t2.getHours(), this.$m = t2.getMinutes(), this.$s = t2.getSeconds(), this.$ms = t2.getMilliseconds();
        }, m2.$utils = function() {
          return O;
        }, m2.isValid = function() {
          return !(this.$d.toString() === l);
        }, m2.isSame = function(t2, e2) {
          var n2 = w(t2);
          return this.startOf(e2) <= n2 && n2 <= this.endOf(e2);
        }, m2.isAfter = function(t2, e2) {
          return w(t2) < this.startOf(e2);
        }, m2.isBefore = function(t2, e2) {
          return this.endOf(e2) < w(t2);
        }, m2.$g = function(t2, e2, n2) {
          return O.u(t2) ? this[e2] : this.set(n2, t2);
        }, m2.unix = function() {
          return Math.floor(this.valueOf() / 1e3);
        }, m2.valueOf = function() {
          return this.$d.getTime();
        }, m2.startOf = function(t2, e2) {
          var n2 = this, r2 = !!O.u(e2) || e2, h2 = O.p(t2), l2 = function(t3, e3) {
            var i2 = O.w(n2.$u ? Date.UTC(n2.$y, e3, t3) : new Date(n2.$y, e3, t3), n2);
            return r2 ? i2 : i2.endOf(a);
          }, $2 = function(t3, e3) {
            return O.w(n2.toDate()[t3].apply(n2.toDate("s"), (r2 ? [0, 0, 0, 0] : [23, 59, 59, 999]).slice(e3)), n2);
          }, y2 = this.$W, M3 = this.$M, m3 = this.$D, v2 = "set" + (this.$u ? "UTC" : "");
          switch (h2) {
            case c:
              return r2 ? l2(1, 0) : l2(31, 11);
            case f:
              return r2 ? l2(1, M3) : l2(0, M3 + 1);
            case o:
              var g2 = this.$locale().weekStart || 0, D2 = (y2 < g2 ? y2 + 7 : y2) - g2;
              return l2(r2 ? m3 - D2 : m3 + (6 - D2), M3);
            case a:
            case d:
              return $2(v2 + "Hours", 0);
            case u:
              return $2(v2 + "Minutes", 1);
            case s:
              return $2(v2 + "Seconds", 2);
            case i:
              return $2(v2 + "Milliseconds", 3);
            default:
              return this.clone();
          }
        }, m2.endOf = function(t2) {
          return this.startOf(t2, false);
        }, m2.$set = function(t2, e2) {
          var n2, o2 = O.p(t2), h2 = "set" + (this.$u ? "UTC" : ""), l2 = (n2 = {}, n2[a] = h2 + "Date", n2[d] = h2 + "Date", n2[f] = h2 + "Month", n2[c] = h2 + "FullYear", n2[u] = h2 + "Hours", n2[s] = h2 + "Minutes", n2[i] = h2 + "Seconds", n2[r] = h2 + "Milliseconds", n2)[o2], $2 = o2 === a ? this.$D + (e2 - this.$W) : e2;
          if (o2 === f || o2 === c) {
            var y2 = this.clone().set(d, 1);
            y2.$d[l2]($2), y2.init(), this.$d = y2.set(d, Math.min(this.$D, y2.daysInMonth())).$d;
          } else
            l2 && this.$d[l2]($2);
          return this.init(), this;
        }, m2.set = function(t2, e2) {
          return this.clone().$set(t2, e2);
        }, m2.get = function(t2) {
          return this[O.p(t2)]();
        }, m2.add = function(r2, h2) {
          var d2, l2 = this;
          r2 = Number(r2);
          var $2 = O.p(h2), y2 = function(t2) {
            var e2 = w(l2);
            return O.w(e2.date(e2.date() + Math.round(t2 * r2)), l2);
          };
          if ($2 === f)
            return this.set(f, this.$M + r2);
          if ($2 === c)
            return this.set(c, this.$y + r2);
          if ($2 === a)
            return y2(1);
          if ($2 === o)
            return y2(7);
          var M3 = (d2 = {}, d2[s] = e, d2[u] = n, d2[i] = t, d2)[$2] || 1, m3 = this.$d.getTime() + r2 * M3;
          return O.w(m3, this);
        }, m2.subtract = function(t2, e2) {
          return this.add(-1 * t2, e2);
        }, m2.format = function(t2) {
          var e2 = this, n2 = this.$locale();
          if (!this.isValid())
            return n2.invalidDate || l;
          var r2 = t2 || "YYYY-MM-DDTHH:mm:ssZ", i2 = O.z(this), s2 = this.$H, u2 = this.$m, a2 = this.$M, o2 = n2.weekdays, f2 = n2.months, h2 = function(t3, n3, i3, s3) {
            return t3 && (t3[n3] || t3(e2, r2)) || i3[n3].slice(0, s3);
          }, c2 = function(t3) {
            return O.s(s2 % 12 || 12, t3, "0");
          }, d2 = n2.meridiem || function(t3, e3, n3) {
            var r3 = t3 < 12 ? "AM" : "PM";
            return n3 ? r3.toLowerCase() : r3;
          }, $2 = { YY: String(this.$y).slice(-2), YYYY: this.$y, M: a2 + 1, MM: O.s(a2 + 1, 2, "0"), MMM: h2(n2.monthsShort, a2, f2, 3), MMMM: h2(f2, a2), D: this.$D, DD: O.s(this.$D, 2, "0"), d: String(this.$W), dd: h2(n2.weekdaysMin, this.$W, o2, 2), ddd: h2(n2.weekdaysShort, this.$W, o2, 3), dddd: o2[this.$W], H: String(s2), HH: O.s(s2, 2, "0"), h: c2(1), hh: c2(2), a: d2(s2, u2, true), A: d2(s2, u2, false), m: String(u2), mm: O.s(u2, 2, "0"), s: String(this.$s), ss: O.s(this.$s, 2, "0"), SSS: O.s(this.$ms, 3, "0"), Z: i2 };
          return r2.replace(y, function(t3, e3) {
            return e3 || $2[t3] || i2.replace(":", "");
          });
        }, m2.utcOffset = function() {
          return 15 * -Math.round(this.$d.getTimezoneOffset() / 15);
        }, m2.diff = function(r2, d2, l2) {
          var $2, y2 = O.p(d2), M3 = w(r2), m3 = (M3.utcOffset() - this.utcOffset()) * e, v2 = this - M3, g2 = O.m(this, M3);
          return g2 = ($2 = {}, $2[c] = g2 / 12, $2[f] = g2, $2[h] = g2 / 3, $2[o] = (v2 - m3) / 6048e5, $2[a] = (v2 - m3) / 864e5, $2[u] = v2 / n, $2[s] = v2 / e, $2[i] = v2 / t, $2)[y2] || v2, l2 ? g2 : O.a(g2);
        }, m2.daysInMonth = function() {
          return this.endOf(f).$D;
        }, m2.$locale = function() {
          return D[this.$L];
        }, m2.locale = function(t2, e2) {
          if (!t2)
            return this.$L;
          var n2 = this.clone(), r2 = S(t2, e2, true);
          return r2 && (n2.$L = r2), n2;
        }, m2.clone = function() {
          return O.w(this.$d, this);
        }, m2.toDate = function() {
          return new Date(this.valueOf());
        }, m2.toJSON = function() {
          return this.isValid() ? this.toISOString() : null;
        }, m2.toISOString = function() {
          return this.$d.toISOString();
        }, m2.toString = function() {
          return this.$d.toUTCString();
        }, M2;
      }(), T = _.prototype;
      return w.prototype = T, [["$ms", r], ["$s", i], ["$m", s], ["$H", u], ["$W", a], ["$M", f], ["$y", c], ["$D", d]].forEach(function(t2) {
        T[t2[1]] = function(e2) {
          return this.$g(e2, t2[0], t2[1]);
        };
      }), w.extend = function(t2, e2) {
        return t2.$i || (t2(e2, _, w), t2.$i = true), w;
      }, w.locale = S, w.isDayjs = p, w.unix = function(t2) {
        return w(1e3 * t2);
      }, w.en = D[g], w.Ls = D, w.p = {}, w;
    });
  }
});
export default require_dayjs_min();
uni_modules/uv-ui-tools/libs/util/route.js
New file
@@ -0,0 +1,126 @@
/**
 * 路由跳转方法,该方法相对于直接使用uni.xxx的好处是使用更加简单快捷
 * 并且带有路由拦截功能
 */
import { queryParams, deepMerge, page } from '@/uni_modules/uv-ui-tools/libs/function/index.js'
class Router {
    constructor() {
        // 原始属性定义
        this.config = {
            type: 'navigateTo',
            url: '',
            delta: 1, // navigateBack页面后退时,回退的层数
            params: {}, // 传递的参数
            animationType: 'pop-in', // 窗口动画,只在APP有效
            animationDuration: 300, // 窗口动画持续时间,单位毫秒,只在APP有效
            intercept: false ,// 是否需要拦截
            events: {} // 页面间通信接口,用于监听被打开页面发送到当前页面的数据。hbuilderx 2.8.9+ 开始支持。
        }
        // 因为route方法是需要对外赋值给另外的对象使用,同时route内部有使用this,会导致route失去上下文
        // 这里在构造函数中进行this绑定
        this.route = this.route.bind(this)
    }
    // 判断url前面是否有"/",如果没有则加上,否则无法跳转
    addRootPath(url) {
        return url[0] === '/' ? url : `/${url}`
    }
    // 整合路由参数
    mixinParam(url, params) {
        url = url && this.addRootPath(url)
        // 使用正则匹配,主要依据是判断是否有"/","?","="等,如“/page/index/index?name=mary"
        // 如果有url中有get参数,转换后无需带上"?"
        let query = ''
        if (/.*\/.*\?.*=.*/.test(url)) {
            // object对象转为get类型的参数
            query = queryParams(params, false)
            // 因为已有get参数,所以后面拼接的参数需要带上"&"隔开
            return url += `&${query}`
        }
        // 直接拼接参数,因为此处url中没有后面的query参数,也就没有"?/&"之类的符号
        query = queryParams(params)
        return url += query
    }
    // 对外的方法名称
    async route(options = {}, params = {}) {
        // 合并用户的配置和内部的默认配置
        let mergeConfig = {}
        if (typeof options === 'string') {
            // 如果options为字符串,则为route(url, params)的形式
            mergeConfig.url = this.mixinParam(options, params)
            mergeConfig.type = 'navigateTo'
        } else {
            mergeConfig = deepMerge(this.config, options)
            // 否则正常使用mergeConfig中的url和params进行拼接
            mergeConfig.url = this.mixinParam(options.url, options.params)
        }
        // 如果本次跳转的路径和本页面路径一致,不执行跳转,防止用户快速点击跳转按钮,造成多次跳转同一个页面的问题
        if (mergeConfig.url === page()) return
        if (params.intercept) {
            mergeConfig.intercept = params.intercept
        }
        // params参数也带给拦截器
        mergeConfig.params = params
        // 合并内外部参数
        mergeConfig = deepMerge(this.config, mergeConfig)
        // 判断用户是否定义了拦截器
        if (typeof mergeConfig.intercept === 'function') {
            // 定一个promise,根据用户执行resolve(true)或者resolve(false)来决定是否进行路由跳转
            const isNext = await new Promise((resolve, reject) => {
                mergeConfig.intercept(mergeConfig, resolve)
            })
            // 如果isNext为true,则执行路由跳转
            isNext && this.openPage(mergeConfig)
        } else {
            this.openPage(mergeConfig)
        }
    }
    // 执行路由跳转
    openPage(config) {
        // 解构参数
        const {
            url,
            type,
            delta,
            animationType,
            animationDuration,
            events
        } = config
        if (config.type == 'navigateTo' || config.type == 'to') {
            uni.navigateTo({
                url,
                animationType,
                animationDuration,
                events
            })
        }
        if (config.type == 'redirectTo' || config.type == 'redirect') {
            uni.redirectTo({
                url
            })
        }
        if (config.type == 'switchTab' || config.type == 'tab') {
            uni.switchTab({
                url
            })
        }
        if (config.type == 'reLaunch' || config.type == 'launch') {
            uni.reLaunch({
                url
            })
        }
        if (config.type == 'navigateBack' || config.type == 'back') {
            uni.navigateBack({
                delta
            })
        }
    }
}
export default (new Router()).route
uni_modules/uv-ui-tools/package.json
New file
@@ -0,0 +1,81 @@
{
  "id": "uv-ui-tools",
  "displayName": "uv-ui-tools 工具集 全面兼容vue3+2、app、h5、小程序等多端",
  "version": "1.1.21",
  "description": "uv-ui-tools,集成工具库,强大的Http请求封装,清晰的文档说明,开箱即用。方便使用,可以全局使用",
  "keywords": [
    "uv-ui-tools,uv-ui组件库,工具集,uvui,uView2.x"
],
  "repository": "",
  "engines": {
    "HBuilderX": "^3.1.0"
  },
  "dcloudext": {
    "type": "component-vue",
    "sale": {
      "regular": {
        "price": "0.00"
      },
      "sourcecode": {
        "price": "0.00"
      }
    },
    "contact": {
      "qq": ""
    },
    "declaration": {
      "ads": "无",
      "data": "插件不采集任何数据",
      "permissions": "无"
    },
    "npmurl": ""
  },
  "uni_modules": {
    "dependencies": [],
    "encrypt": [],
    "platforms": {
      "cloud": {
        "tcb": "y",
        "aliyun": "y"
      },
      "client": {
        "Vue": {
          "vue2": "y",
          "vue3": "y"
        },
        "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",
          "钉钉": "y",
          "快手": "y",
          "飞书": "y",
          "京东": "y"
        },
        "快应用": {
          "华为": "y",
          "联盟": "y"
        }
      }
    }
  }
}
uni_modules/uv-ui-tools/readme.md
New file
@@ -0,0 +1,23 @@
## uv-ui-tools 工具集
> **组件名:uv-ui-tools**
uv-ui工具集成,包括网络Http请求、便捷工具、节流防抖、对象操作、时间格式化、路由跳转、全局唯一标识符、规则校验等等。
该组件推荐配合[uv-ui组件库](https://www.uvui.cn/components/intro.html)使用,单独下载也可以在自己项目中使用,需要做相应的配置,可查看文档。强烈推荐使用[uv-ui组件库](https://www.uvui.cn/components/intro.html),导入组件都会自动导入`uv-ui-tools`。需要在自己的项目中使用请参考[扩展配置](https://www.uvui.cn/components/setting.html)。
uv-ui破釜沉舟之兼容vue3+2、app、h5、多端小程序的uni-app生态框架,大部分组件基于uView2.x,在经过改进后全面支持vue3,部分组件做了进一步的优化,修复大量BUG,支持单独导入,方便开发者选择导入需要的组件。开箱即用,灵活配置。
# <a href="https://www.uvui.cn/js/intro.html" target="_blank">查看文档</a>
## [下载完整示例项目](https://ext.dcloud.net.cn/plugin?name=uv-ui) <small>(请不要 下载插件ZIP)</small>
### [更多插件,请关注uv-ui组件库](https://ext.dcloud.net.cn/plugin?name=uv-ui)
<a href="https://ext.dcloud.net.cn/plugin?name=uv-ui" target="_blank">
![image](https://mp-a667b617-c5f1-4a2d-9a54-683a67cff588.cdn.bspapp.com/uv-ui/banner.png)
</a>
#### 如使用过程中有任何问题反馈,或者您对uv-ui有一些好的建议,欢迎加入uv-ui官方交流群:<a href="https://www.uvui.cn/components/addQQGroup.html" target="_blank">官方QQ群</a>
uni_modules/uv-ui-tools/theme.scss
New file
@@ -0,0 +1,43 @@
// 此文件为uvUI的主题变量,这些变量目前只能通过uni.scss引入才有效,另外由于
// uni.scss中引入的样式会同时混入到全局样式文件和单独每一个页面的样式中,造成微信程序包太大,
// 故uni.scss只建议放scss变量名相关样式,其他的样式可以通过main.js或者App.vue引入
$uv-main-color: #303133;
$uv-content-color: #606266;
$uv-tips-color: #909193;
$uv-light-color: #c0c4cc;
$uv-border-color: #dadbde;
$uv-bg-color: #f3f4f6;
$uv-disabled-color: #c8c9cc;
$uv-primary: #3c9cff;
$uv-primary-dark: #398ade;
$uv-primary-disabled: #9acafc;
$uv-primary-light: #ecf5ff;
$uv-warning: #f9ae3d;
$uv-warning-dark: #f1a532;
$uv-warning-disabled: #f9d39b;
$uv-warning-light: #fdf6ec;
$uv-success: #5ac725;
$uv-success-dark: #53c21d;
$uv-success-disabled: #a9e08f;
$uv-success-light: #f5fff0;
$uv-error: #f56c6c;
$uv-error-dark: #e45656;
$uv-error-disabled: #f7b2b2;
$uv-error-light: #fef0f0;
$uv-info: #909399;
$uv-info-dark: #767a82;
$uv-info-disabled: #c4c6c9;
$uv-info-light: #f4f4f5;
@mixin flex($direction: row) {
    /* #ifndef APP-NVUE */
    display: flex;
    /* #endif */
    flex-direction: $direction;
}