zwl
2025-09-01 aa379b2a1868a2c3ece8c71b3979ff0b3b1958e6
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
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
const MIN_DISTANCE = 10;
export default {
    showWatch(newVal, oldVal, ownerInstance, instance,self) {
        let state = self.state
        this.getDom(instance, ownerInstance,self)
        if (newVal && newVal !== 'none') {
            this.openState(newVal, instance, ownerInstance,self)
            return
        }
 
        if (state.left) {
            this.openState('none', instance, ownerInstance,self)
        }
        this.resetTouchStatus(instance,self)
    },
 
    /**
     * 开始触摸操作
     * @param {Object} e
     * @param {Object} ins
     */
    touchstart(e, ownerInstance, self) {
        let instance = e.instance;
        let disabled = instance.getDataset().disabled
        let state = self.state;
        this.getDom(instance, ownerInstance, self)
        // fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
        disabled = this.getDisabledType(disabled)
        if (disabled) return
        // 开始触摸时移除动画类
        instance.requestAnimationFrame(function() {
            instance.removeClass('ani');
            ownerInstance.callMethod('closeSwipe');
        })
 
        // 记录上次的位置
        state.x = state.left || 0
        // 计算滑动开始位置
        this.stopTouchStart(e, ownerInstance, self)
    },
 
    /**
     * 开始滑动操作
     * @param {Object} e
     * @param {Object} ownerInstance
     */
    touchmove(e, ownerInstance, self) {
        let instance = e.instance;
        let disabled = instance.getDataset().disabled
        let state = self.state
        // fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
        disabled = this.getDisabledType(disabled)
        if (disabled) return
        // 是否可以滑动页面
        this.stopTouchMove(e, self);
        if (state.direction !== 'horizontal') {
            return;
        }
        if (e.preventDefault) {
            // 阻止页面滚动
            e.preventDefault()
        }
        let x = state.x + state.deltaX
        this.move(x, instance, ownerInstance, self)
    },
 
    /**
     * 结束触摸操作
     * @param {Object} e
     * @param {Object} ownerInstance
     */
    touchend(e, ownerInstance, self) {
        let instance = e.instance;
        let disabled = instance.getDataset().disabled
        let state = self.state
        // fix by mehaotian, TODO 兼容 app-vue 获取dataset为字符串 , h5 获取 为 undefined 的问题,待框架修复
        disabled = this.getDisabledType(disabled)
 
        if (disabled) return
        // 滑动过程中触摸结束,通过阙值判断是开启还是关闭
        // fixed by mehaotian 定时器解决点击按钮,touchend 触发比 click 事件时机早的问题 ,主要是 ios13
        this.moveDirection(state.left, instance, ownerInstance, self)
 
    },
 
    /**
     * 设置移动距离
     * @param {Object} value
     * @param {Object} instance
     * @param {Object} ownerInstance
     */
    move(value, instance, ownerInstance, self) {
        value = value || 0
        let state = self.state
        let leftWidth = state.leftWidth
        let rightWidth = state.rightWidth
        // 获取可滑动范围
        state.left = this.range(value, -rightWidth, leftWidth);
        instance.requestAnimationFrame(function() {
            instance.setStyle({
                transform: 'translateX(' + state.left + 'px)',
                '-webkit-transform': 'translateX(' + state.left + 'px)'
            })
        })
 
    },
 
    /**
     * 获取元素信息
     * @param {Object} instance
     * @param {Object} ownerInstance
     */
    getDom(instance, ownerInstance, self) {
        let state = self.state
        var leftDom = ownerInstance.$el.querySelector('.button-group--left')
        var rightDom = ownerInstance.$el.querySelector('.button-group--right')
 
        state.leftWidth = leftDom.offsetWidth || 0
        state.rightWidth = rightDom.offsetWidth || 0
        state.threshold = instance.getDataset().threshold
    },
 
    getDisabledType(value) {
        return (typeof(value) === 'string' ? JSON.parse(value) : value) || false;
    },
 
    /**
     * 获取范围
     * @param {Object} num
     * @param {Object} min
     * @param {Object} max
     */
    range(num, min, max) {
        return Math.min(Math.max(num, min), max);
    },
 
 
    /**
     * 移动方向判断
     * @param {Object} left
     * @param {Object} value
     * @param {Object} ownerInstance
     * @param {Object} ins
     */
    moveDirection(left, ins, ownerInstance, self) {
        var state = self.state
        var threshold = state.threshold
        var position = state.position
        var isopen = state.isopen || 'none'
        var leftWidth = state.leftWidth
        var rightWidth = state.rightWidth
        if (state.deltaX === 0) {
            this.openState('none', ins, ownerInstance, self)
            return
        }
        if ((isopen === 'none' && rightWidth > 0 && -left > threshold) || (isopen !== 'none' && rightWidth > 0 &&
                rightWidth +
                left < threshold)) {
            // right
            this.openState('right', ins, ownerInstance, self)
        } else if ((isopen === 'none' && leftWidth > 0 && left > threshold) || (isopen !== 'none' && leftWidth > 0 &&
                leftWidth - left < threshold)) {
            // left
            this.openState('left', ins, ownerInstance, self)
        } else {
            // default
            this.openState('none', ins, ownerInstance, self)
        }
    },
 
 
    /**
     * 开启状态
     * @param {Boolean} type
     * @param {Object} ins
     * @param {Object} ownerInstance
     */
    openState(type, ins, ownerInstance, self) {
        let state = self.state
        let leftWidth = state.leftWidth
        let rightWidth = state.rightWidth
        let left = ''
        state.isopen = state.isopen ? state.isopen : 'none'
        switch (type) {
            case "left":
                left = leftWidth
                break
            case "right":
                left = -rightWidth
                break
            default:
                left = 0
        }
 
        // && !state.throttle
 
        if (state.isopen !== type) {
            state.throttle = true
            ownerInstance.callMethod('change', {
                open: type
            })
 
        }
 
        state.isopen = type
        // 添加动画类
        ins.requestAnimationFrame(()=> {
            ins.addClass('ani');
            this.move(left, ins, ownerInstance, self)
        })
    },
 
 
    getDirection(x, y) {
        if (x > y && x > MIN_DISTANCE) {
            return 'horizontal';
        }
        if (y > x && y > MIN_DISTANCE) {
            return 'vertical';
        }
        return '';
    },
 
    /**
     * 重置滑动状态
     * @param {Object} event
     */
    resetTouchStatus(instance, self) {
        let state = self.state;
        state.direction = '';
        state.deltaX = 0;
        state.deltaY = 0;
        state.offsetX = 0;
        state.offsetY = 0;
    },
 
    /**
     * 设置滑动开始位置
     * @param {Object} event
     */
    stopTouchStart(event, ownerInstance, self) {
        let instance = event.instance;
        let state = self.state
        this.resetTouchStatus(instance, self);
        var touch = event.touches[0];
        state.startX = touch.clientX;
        state.startY = touch.clientY;
    },
 
    /**
     * 滑动中,是否禁止打开
     * @param {Object} event
     */
    stopTouchMove(event, self) {
        let instance = event.instance;
        let state = self.state;
        let touch = event.touches[0];
 
        state.deltaX = touch.clientX - state.startX;
        state.deltaY = touch.clientY - state.startY;
        state.offsetY = Math.abs(state.deltaY);
        state.offsetX = Math.abs(state.deltaX);
        state.direction = state.direction || this.getDirection(state.offsetX, state.offsetY);
    }
}