My last question about the same topic was not clear enough and was put on hold by community and later it was automatically deleted. So, I am explaining that Question in deta
I had tried to summarize how i was able to get thru the problem in a blog post. Anyone who needs some help can take a look into it.
Accessibility Service to the next level
My Previous answer was deleted by moderators may due to i just posted a link to blog. So i am posting my answer again with details here too.
The easiest way is provided by android itself. If you are building something on your own app this is the best and easiest possible way of doing it. You have to use “findAccessibilityNodeInfosByViewId ()” as explained below
@Override
public void onAccessibilityEvent(AccessibilityEvent event) {
AccessibilityNodeInfo source = event.getSource();
if (source == null) {
return;
}
List<AccessibilityNodeInfo> findAccessibilityNodeInfosByViewId = source.findAccessibilityNodeInfosByViewId("YOUR PACKAGE NAME:id/RESOURCE ID FROM WHERE YOU WANT DATA");
if (findAccessibilityNodeInfosByViewId.size() > 0) {
AccessibilityNodeInfo parent = (AccessibilityNodeInfo) findAccessibilityNodeInfosByViewId.get(0);
// You can also traverse the list if required data is deep in view hierarchy.
String requiredText = parent.getText().toString();
Log.i("Required Text", requiredText);
}
}
If you are building something on other apps and don’t know the res id. You have to build a logic with the combination of parent, child count, level on which your data is found, type of event. You have to look for a pattern in event or source or the combination of above said values to get the required data preciously. Further, you have to setup some tweak in your code to get the accurate data according to your task. Like in our task we have some regex to get precious data.
You have to take care that if the view or UI is changed or res ids are changed your current logic might fail. So, you have to keep a close look at the app on which you are building your service. Following are some example code in which we are getting data without res id that might be useful for you to understand how it will work.
To print source and event
@Override
public void onAccessibilityEvent(AccessibilityEvent event) {
AccessibilityNodeInfo source = event.getSource();
if (source == null) {
return;
}
Log.i("Event", event.toString() + "");
Log.i("Source", source.toString() + "");
}
To get the childcount
source.getChildCount();
If child count is >0 you might have to look into it.
Example Code 1
if (level == 0 && source.getClassName().equals("android.widget.TextView") && source.getText()!=null && !source.getText().toString().isEmpty()){
// here level is iteration of for loop
String recivedText = source.getText().toString();
if(source.getClassName().equals("android.widget.TextView") && source.getParent()!=null && source.getParent().getClassName().equals("android.widget.FrameLayout") && source.getParent().getParent()==null){
return recivedText;
}
}
Example Code 2
if (source.getPackageName().equals("PACKAGE NAME OF APP FOR WHICH YOUR EXPECTING EVENT")) {
if(event.getEventType()==AccessibilityEvent.TYPE_WINDOW_STATE_CHANGED && "COMPLETE NAME OF ACTIVITY CLASS WITH PACKAGE NAME (if you want it for some specific screen)".equals(event.getClassName())){
if(source.getText()!=null && source.getClassName().equals("android.widget.TextView") && source.getParent()!=null && source.getParent().getClassName().equals("android.widget.RelativeLayout")&& source.getParent().getParent()!=null && source.getParent().getParent().getClassName().equals("android.widget.ScrollView") && source.getParent().getParent().getParent()!=null && source.getParent().getParent().getParent().getClassName().equals("android.support.v4.view.ViewPager")&& source.getParent().getParent().getParent().getParent()!=null && source.getParent().getParent().getParent().getParent().getClassName().equals("android.widget.FrameLayout")&& source.getParent().getParent().getParent().getParent().getParent()==null){
return source.getText().toString();
}
}
It is bit clumsy but to get it working you have to look deep into logs to find a pattern which will take you to your required data.
Update 20/11/2017 Google is suspending all apps using accessibility services for things other than accessibility. I had received a email regarding the same for one of my app and other developers are also receiving same email from google (Reddit). So, I think we should look for some other way of doing things and I request you to please update it here also, if some one comes up with other implementation to achieve same behavior it would help others also.
Thanks Umesh Chauhan
I have made successfully this type of application like voodoo.and get Flipkart product page title, price etc successfully. the following method used to get user open product screen.
private boolean isOpenFlipkartProductScreen(AccessibilityNodeInfo nodeInfo){
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {
if(nodeInfo.getViewIdResourceName()!=null){
if(nodeInfo.getViewIdResourceName().equals("com.flipkart.android:id/callout_last_linearlayout")){
Utils.sPrint("You just open product page : " + nodeInfo.getClass());
isOpenProductPage = true;
return true;
}
}
}
return false;
}
and following code to used get product title name and price.
private String getObtainTitle(AccessibilityNodeInfo sourceInfo){
if(sourceInfo == null)
return "";
AccessibilityNodeInfo nodeInfo = sourceInfo.getParent().getParent();
if(nodeInfo!=null){
int i = 0;
int i2 = 0;
for(int i3=0;i3<nodeInfo.getChildCount();i3++){
try{
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {
if(nodeInfo.getChild(i3).getViewIdResourceName().equals("com.flipkart.android:id/title_layout")) {
i = 0;
// while ( i < nodeInfo.getChild(i).getChildCount()){
AccessibilityNodeInfo child = nodeInfo.getChild(i3).getChild(i);
String charSequence = child.getText().toString();
// Utils.sPrint("Title is : " + charSequence);
if(charSequence!=null){
return charSequence;
}
// }
}
}
/*
*
* Below code to used for getProduct prise in flipkart app product
*
* */
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR2) {
if(nodeInfo.getChild(i3).getViewIdResourceName().equals("com.flipkart.android:id/price_layout")){
AccessibilityNodeInfo child = nodeInfo.getChild(i3).getChild(i);
String charSequence = child.getText().toString();
if (child.getText().toString().toUpperCase().startsWith("RS.") ||
child.getText().toString().toUpperCase().startsWith("\u20b9")) {
String replace = charSequence.replace("Rs.",
"").replace("\u20b9", "");
//return replace
// Utils.sPrint("Prise is : " + replace);
}
}
}
}catch (Exception ex){
// Utils.sPrint("Ex : " + ex.getMessage());
return "";
}
}
}
return "";
}