-
-
Notifications
You must be signed in to change notification settings - Fork 264
/
Gestures.ts
215 lines (205 loc) · 11.1 KB
/
Gestures.ts
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
interface XY {
x:number;
y:number;
}
export const DIRECTIONS = {
// Starting Point: You place your finger towards the top of the screen.
// Movement: You slide your finger downwards towards the bottom of the screen.
// Action: This also varies by context:
// - On the home screen or in applications, it typically scrolls the content upwards.
// - From the top edge, it often opens the notifications panel or quick settings.
// - In browsers or reading apps, it can be used to scroll through content.
DOWN: 'down',
// Starting Point: You place your finger on the right side of the screen.
// Movement: You slide your finger horizontally to the left.
// Action: The response to this gesture depends on the application:
// - It can move to the next item in a carousel or a set of images.
// - In a navigation context, it might go back to the previous page or close the current view.
// - On the home screen, it usually switches to the next virtual desktop or screen.
LEFT: 'left',
// Starting Point: You place your finger on the left side of the screen.
// Movement: You slide your finger horizontally to the right.
// Action: Similar to swiping left, but in the opposite direction:
// - It often moves to the previous item in a carousel or gallery.
// - Can be used to open side menus or navigation drawers in apps.
// - On the home screen, it typically switches to the previous virtual desktop.
RIGHT: 'right',
// Starting Point: You place your finger towards the bottom of the screen.
// Movement: You slide your finger upwards towards the top of the screen.
// Action: Depending on the context, different actions can occur:
// - On the home screen or in a list, this usually scrolls the content downwards.
// - In a full-screen app, it might open additional options or the app drawer.
// - On certain interfaces, it could trigger a 'refresh' action or open a search bar.
UP: 'up',
} as const;
type SwipeDirectionType = typeof DIRECTIONS[keyof typeof DIRECTIONS];
class Gestures {
/**
* Check if an element is visible and if not wipe up a portion of the screen to
* check if it visible after x amount of scrolls
*/
static async checkIfDisplayedWithSwipe (
{
scrollContainer,
searchableElement,
maxScrolls,
amount=0,
direction=DIRECTIONS.DOWN,
// Never scroll from the exact top or bottom of the screen, you might trigger the notification bar or other OS/App features
percentage=0.99,
}:
{
scrollContainer:WebdriverIO.Element | ChainablePromiseElement,
searchableElement:WebdriverIO.Element | ChainablePromiseElement,
maxScrolls:number,
amount?: number,
direction?: SwipeDirectionType,
percentage?: number,
}
){
// If the element is not displayed and we haven't scrolled the max amount of scrolls
// then scroll and execute the method again
if (!await searchableElement.isDisplayed() && amount <= maxScrolls) {
// 1. Determine the percentage of the scrollable container to be scrolled
// The scroll percentage is the percentage of the scrollable container that should be scrolled
let scrollPercentage;
if (isNaN(percentage)){
console.log('\nThe percentage to scroll should be a number.\n');
// Never scroll from the exact top or bottom of the screen, you might trigger the notification bar or other OS/App features
scrollPercentage = 0.99;
} else if (percentage > 1) {
console.log('\nThe percentage to scroll should be a number between 0 and 1.\n');
// Never scroll from the exact top or bottom of the screen, you might trigger the notification bar or other OS/App features
scrollPercentage = 0.99;
} else {
scrollPercentage = 1-percentage;
}
// 2. Determine the swipe coordinates
// When we get the element rect we get the position of the element on the screen based on the
// - x (position from the left of the screen)
// - y (position from the top of the screen)
// - width (width of the element)
// - height (height of the element)
// We can use this to calculate the position of the swipe by determining the
// - top
// - right
// - bottom
// - left
// of the element. These positions will contain the x and y coordinates on where to put the finger
const { x, y, width, height } = await driver.getElementRect(scrollContainer.elementId);
// It's always advisable to swipe from the center of the element.
const scrollRectangles = {
// The x is the center of the element,
// The y is the y of the element + the height of the element * the scroll percentage
top: { x: Math.round(x + width / 2), y: Math.round(y + height * scrollPercentage) },
// The x is the x of the element + the width of the element, minus the width of the element * the scroll percentage
// The y is the center of the element,
right: { x: Math.round(x + width - width * scrollPercentage), y: Math.round(y + height / 2) },
// The x is the center of the element,
// The y is the y of the element, plus the height, minus the height of the element * the scroll percentage
bottom: { x: Math.round(x + width / 2), y: Math.round(y + height - height * scrollPercentage) },
// The x is the x of the element, plus the width of the element * the scroll percentage
// The y is the center of the element,
left: { x: Math.round(x + width * scrollPercentage), y: Math.round(y + height / 2) },
};
// 3. Swipe in the given direction
if (direction === DIRECTIONS.DOWN) {
await this.executeGesture({
from: scrollRectangles.top,
to: scrollRectangles.bottom,
});
} else if (direction === DIRECTIONS.LEFT) {
await this.executeGesture({
from: scrollRectangles.right,
to: scrollRectangles.left,
});
} else if (direction === DIRECTIONS.RIGHT) {
await this.executeGesture({
from: scrollRectangles.left,
to: scrollRectangles.right,
});
} else if (direction === DIRECTIONS.UP) {
await this.executeGesture({
from: scrollRectangles.bottom,
to: scrollRectangles.top,
});
} else {
console.log('\nThe direction to scroll should be one of the following: down, left, right or up.\n');
}
// 4. Check if the element is visible or swipe again
await this.checkIfDisplayedWithSwipe({
scrollContainer,
searchableElement,
maxScrolls,
amount:amount + 1,
direction,
percentage,
});
} else if (amount > maxScrolls) {
// If the element is still not visible after the max amount of scroll let it fail
throw new Error(`The element '${searchableElement}' could not be found or is not visible.`);
}
// The element was found, proceed with the next action
}
/**
* Execute a gesture on the screen from coordinates (from) to the new coordinates (to). The given coordinates are in pixels.
*
* There are two ways to execute a gesture:
* 1. The "clean and easy" way
* 2. The "verbose" way
*
* The "clean and easy" way is the recommended way to execute a gesture. It is easier to read and understand.
* The "verbose" way is the way the Appium server expects the gesture to be send to the server and this is also how the "clean" way is translated to.
*/
static async executeGesture ({ from, to }:{from: XY, to: XY}) {
// The "clean" way
await driver
// a. Create the event
.action('pointer')
// b. Move finger into start position
.move(from.x, from.y) // This can also be written as .move({ x:from.x, y:from.y }) which allows you to add more options
// c. Finger comes down into contact with screen
.down() // This can also be written as .down({ button:0 }) which allows you to add more options
// d. Pause for a little bit
.pause(100)
// e. Finger moves to end position
// IMPORTANT. The default duration, if you don't provide it, is 100ms. This means that the movement will be so fast that it:
// - might not be registered
// - might not have the correct result on longer movements.
// Short durations will move elements on the screen over longer move coordinates very fast.
// Play with the duration to make the swipe go slower / faster
.move({ duration: 1000, x: to.x, y: to.y })
// f. Finger gets up, off the screen
.up() // this can also be written as .up({ button:0 }) which allows you to add more options
// g. Perform the action
.perform();
// The "verbose" way. The advantage of this way is that you can add more options to control the action in comparison to the "clean and easy" way.
// More information about this can be found here: https://github.com/jlipps/simple-wd-spec?tab=readme-ov-file#perform-actions
// await driver.performActions([
// {
// // a. Create the event
// type: 'pointer',
// id: 'finger1',
// parameters: { pointerType: 'touch' },
// actions: [
// // b. Move finger into start position
// { type: 'pointerMove', duration: 0, x: from.x, y: from.y },
// // c. Finger comes down into contact with screen
// { type: 'pointerDown', button: 0 },
// // d. Pause for a little bit
// { type: 'pause', duration: 100 },
// // e. Finger moves to end position
// // We move our finger from the center of the element to the
// // starting position of the element.
// // Play with the duration to make the swipe go slower / faster
// { type: 'pointerMove', duration: 1000, x: to.x, y: to.y },
// // f. Finger gets up, off the screen
// { type: 'pointerUp', button: 0 },
// ],
// },
// ]);
// Add a pause, just to make sure the swipe is done
await driver.pause(1000);
}
}
export default Gestures;