Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Allow Appium to Scroll To a particular element on screen #109

Open
wants to merge 3 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions AppiumForMac/Server/Controller/AfMSessionController.h
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,7 @@ extern NSString * const kCookieDiagnosticsDirectory;
-(GDataXMLDocument*)xmlPageSource;
-(GDataXMLDocument*)xmlPageSourceFromRootUIElement:(PFUIElement*)rootUIElement pathMap:(NSMutableDictionary*)pathMap xPath:(NSString*)xPath;
- (NSArray *)findAllUsingAbsoluteAXPath:(NSString *)path;
-(BOOL) scrollPage : (id)element;

- (AppiumMacHTTPJSONResponse *)executeWebDriverCommandWithPath:(NSString*)path data:(NSData*)postData onMainThread:(BOOL)onMainThread commandBlock:(AppiumMacHTTPJSONResponse *(^)(AfMSessionController *session, NSDictionary *commandParams, int *statusCode))commandBlock;

Expand Down
49 changes: 49 additions & 0 deletions AppiumForMac/Server/Controller/AfMSessionController.m
Original file line number Diff line number Diff line change
Expand Up @@ -891,6 +891,55 @@ -(PFUIElement*) windowForHandle:(NSString*)windowHandle
return [windows objectAtIndex:windowIndex];
}

-(BOOL) scrollPage:(id)element
{
float increment_by = 0.25;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

could you please use camelCase naming to match with the current project coding style?

bool scroll_found = false;
Copy link
Contributor

@mykola-mokhnach mykola-mokhnach Nov 23, 2020

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Please use YES/NO constants with BOOL type

PFUIElement* scroll;
float axValue = 0,adder = 0;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please define variables closer to the block where they used, also each variable should be defined in its own row


id parent = [(PFUIElement*)element AXParent];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

can the parent's type be more specific?

NSArray* siblings = [parent AXChildren];

while(scroll == nil && parent != nil)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

please add space after while, if and other similar keywords to match with the common project style

{
for(id child in siblings)
{
if([[(PFUIElement*)child AXRole] isEqualToString:@"AXScrollBar"])
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

is there any reason to define the child element type as id in the loop and cast it to PFUIElement* here?

{
scroll = child;
scroll_found = true;
break;
}
}

parent = [(PFUIElement*)parent AXParent];
siblings = [parent AXChildren];
}

NSPoint middlePoint = [[element AXPosition] pointValue];
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

what middlePoint is used for?

if(middlePoint.y < 0)
{

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

should this block be empty?

}

if(scroll_found)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

we could immediately return NO here if the var is false

{
while(adder<=1)
{
axValue = [scroll.AXValue floatValue];
adder = axValue + increment_by;
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

would it be possible to have more descriptive name for the adder variable?

scroll.AXValue = [NSNumber numberWithFloat: adder];

if([self isElementDisplayed:element])
{
return YES;
}
}
}
return NO;
}

-(GDataXMLDocument*)xmlPageSource
{
return [self xmlPageSourceFromRootUIElement:nil pathMap:nil xPath:nil];
Expand Down
3 changes: 3 additions & 0 deletions AppiumForMac/Server/Handlers/AfMHandlers.h
Original file line number Diff line number Diff line change
Expand Up @@ -179,6 +179,9 @@
// POST /session/:sessionId/doubleclick
- (AppiumMacHTTPJSONResponse *)post_doubleclick:(NSString*)path data:(NSData *)postData;

// POST /session/:sessionId/element/:id/scrollTo
- (AppiumMacHTTPJSONResponse *)post_element_scrollTo:(NSString*)path data:(NSData*)postData;

// POST /session/:sessionId/touch/click
// POST /session/:sessionId/touch/down
// POST /session/:sessionId/touch/up
Expand Down
22 changes: 22 additions & 0 deletions AppiumForMac/Server/Handlers/AfMHandlers.m
Original file line number Diff line number Diff line change
Expand Up @@ -963,6 +963,28 @@ - (AppiumMacHTTPJSONResponse *)post_doubleclick:(NSString*)path data:(NSData *)p
}];
}


// POST /session/:sessionId/element/:id/scrollTo
// Scroll to the particular element passed to Appium. Scrolls till the bottom of the screen till the element becomes visible
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

is it always the case we need to scroll to the bottom? what if the destination element is located in the other direction, for example upper or at the right side?

// If the element is still not visible returns element not found
- (AppiumMacHTTPJSONResponse *)post_element_scrollTo:(NSString *)path data:(NSData *)postData
{
return [self executeWebDriverCommandWithPath:path data:nil onMainThread:YES commandBlock:^(AfMSessionController *session, NSDictionary *commandParams, int *statusCode)
{
id element = [commandParams objectForKey:@"elementObject"];

if ([session isElementDisplayed:element]) {
return [AppiumMacHTTPJSONResponse responseWithJson:nil status:kAfMStatusCodeSuccess session:session.sessionId];
}

BOOL result = [session scrollPage:element];

return result ? [AppiumMacHTTPJSONResponse responseWithJson:nil status:kAfMStatusCodeSuccess session:session.sessionId] :
[AppiumMacHTTPJSONResponse responseWithJsonError:kAfMStatusCodeNoSuchElement session:session.sessionId];
}];
}


// POST /session/:sessionId/touch/click
// Single tap on the touch enabled device.

Expand Down