helongyang
2025-10-13 c5afc23437b37d717e892b16b23923907825d2cd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
<template>
    <view class="u-code-wrap">
        <!-- 此组件功能由js完成,无需写html逻辑 -->
    </view>
</template>
 
<script>
    /**
     * verificationCode 验证码输入框
     * @description 考虑到用户实际发送验证码的场景,可能是一个按钮,也可能是一段文字,提示语各有不同,所以本组件 不提供界面显示,只提供提示语,由用户将提示语嵌入到具体的场景
     * @tutorial https://www.uviewui.com/components/verificationCode.html
     * @property {Number String} seconds 倒计时所需的秒数(默认60)
     * @property {String} start-text 开始前的提示语,见官网说明(默认获取验证码)
     * @property {String} change-text 倒计时期间的提示语,必须带有字母"x",见官网说明(默认X秒重新获取)
     * @property {String} end-text 倒计结束的提示语,见官网说明(默认重新获取)
     * @property {Boolean} keep-running 是否在H5刷新或各端返回再进入时继续倒计时(默认false)
     * @event {Function} change 倒计时期间,每秒触发一次
     * @event {Function} start 开始倒计时触发
     * @event {Function} end 结束倒计时触发
     * @example <u-verification-code :seconds="seconds" @end="end" @start="start" ref="uCode" 
     */
    export default {
        name: "u-verification-code",
        props: {
            // 倒计时总秒数
            seconds: {
                type: [String, Number],
                default: 60
            },
            // 尚未开始时提示
            startText: {
                type: String,
                default: '获取验证码'
            },
            // 正在倒计时中的提示
            changeText: {
                type: String,
                default: 'X秒重新获取'
            },
            // 倒计时结束时的提示
            endText: {
                type: String,
                default: '重新获取'
            },
            // 是否在H5刷新或各端返回再进入时继续倒计时
            keepRunning: {
                type: Boolean,
                default: false
            },
            // 为了区分多个页面,或者一个页面多个倒计时组件本地存储的继续倒计时变了
            uniqueKey: {
                type: String,
                default: ''
            }
        },
        data() {
            return {
                secNum: this.seconds,
                timer: null,
                canGetCode: true, // 是否可以执行验证码操作
            }
        },
        mounted() {
            this.checkKeepRunning();
        },
        watch: {
            seconds: {
                immediate: true,
                handler(n) {
                    this.secNum = n;
                }
            }
        },
        methods: {
            checkKeepRunning() {
                // 获取上一次退出页面(H5还包括刷新)时的时间戳,如果没有上次的保存,此值可能为空
                let lastTimestamp = Number(uni.getStorageSync(this.uniqueKey + '_$uCountDownTimestamp'));
                if(!lastTimestamp) return this.changeEvent(this.startText);
                // 当前秒的时间戳
                let nowTimestamp = Math.floor((+ new Date()) / 1000);
                // 判断当前的时间戳,是否小于上一次的本该按设定结束,却提前结束的时间戳
                if(this.keepRunning && lastTimestamp && lastTimestamp > nowTimestamp) {
                    // 剩余尚未执行完的倒计秒数
                    this.secNum = lastTimestamp - nowTimestamp;
                    // 清除本地保存的变量
                    uni.removeStorageSync(this.uniqueKey + '_$uCountDownTimestamp');
                    // 开始倒计时
                    this.start();
                } else {
                    // 如果不存在需要继续上一次的倒计时,执行正常的逻辑
                    this.changeEvent(this.startText);
                }
            },
            // 开始倒计时
            start() {
                // 防止快速点击获取验证码的按钮而导致内部产生多个定时器导致混乱
                if(this.timer) {
                    clearInterval(this.timer);
                    this.timer = null;
                }
                this.$emit('start');
                this.canGetCode = false;
                // 这里放这句,是为了一开始时就提示,否则要等setInterval的1秒后才会有提示
                this.changeEvent(this.changeText.replace(/x|X/, this.secNum));
                this.setTimeToStorage();
                this.timer = setInterval(() => {
                    if (--this.secNum) {
                        // 用当前倒计时的秒数替换提示字符串中的"x"字母
                        this.changeEvent(this.changeText.replace(/x|X/, this.secNum));
                    } else {
                        clearInterval(this.timer);
                        this.timer = null;
                        this.changeEvent(this.endText);
                        this.secNum = this.seconds;
                        this.$emit('end');
                        this.canGetCode = true;
                    }
                }, 1000);
            },
            // 重置,可以让用户再次获取验证码
            reset() {
                this.canGetCode = true;
                clearInterval(this.timer);
                this.secNum = this.seconds;
                this.changeEvent(this.endText);
            },
            changeEvent(text) {
                this.$emit('change', text);
            },
            // 保存时间戳,为了防止倒计时尚未结束,H5刷新或者各端的右上角返回上一页再进来
            setTimeToStorage() {
                if(!this.keepRunning || !this.timer) return;
                // 记录当前的时间戳,为了下次进入页面,如果还在倒计时内的话,继续倒计时
                // 倒计时尚未结束,结果大于0;倒计时已经开始,就会小于初始值,如果等于初始值,说明没有开始倒计时,无需处理
                if(this.secNum > 0 && this.secNum <= this.seconds) {
                    // 获取当前时间戳(+ new Date()为特殊写法),除以1000变成秒,再去除小数部分
                    let nowTimestamp = Math.floor((+ new Date()) / 1000);
                    // 将本该结束时候的时间戳保存起来 => 当前时间戳 + 剩余的秒数
                    uni.setStorage({
                        key: this.uniqueKey + '_$uCountDownTimestamp',
                        data: nowTimestamp + Number(this.secNum)
                    })
                }
            }
        },
        // 组件销毁的时候,清除定时器,否则定时器会继续存在,系统不会自动清除
        beforeDestroy() {
            this.setTimeToStorage();
            clearTimeout(this.timer);
            this.timer = null;
        }
    }
</script>
 
<style lang="scss" scoped>
    @import "../../libs/css/style.components.scss";
    
    .u-code-wrap {
        width: 0;
        height: 0;
        position: fixed;
        z-index: -1;
    }
</style>