Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance. Project price only 1 $
You can buy this project and download/modify it how often you want.
package com.qa.automation.android.view;
import android.content.Context;
import android.os.SystemClock;
import android.view.View;
import android.view.ViewGroup;
import android.view.ViewParent;
import android.view.WindowManager;
import android.webkit.WebView;
import java.lang.reflect.Field;
import java.util.ArrayList;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
/**
* Contains view methods. Examples are getViews(),
* getCurrentTextViews(), getCurrentImageViews().
*
* @author Renas Reda, [email protected]
*/
class ViewFetcher {
private static Class> windowManager;
static {
try {
String windowManagerClassName;
if (android.os.Build.VERSION.SDK_INT >= 17) {
windowManagerClassName = "android.view.WindowManagerGlobal";
} else {
windowManagerClassName = "android.view.WindowManagerImpl";
}
windowManager = Class.forName(windowManagerClassName);
} catch (ClassNotFoundException e) {
throw new RuntimeException(e);
} catch (SecurityException e) {
e.printStackTrace();
}
}
private String windowManagerString;
private Context mcontext;
private Sleeper sleeper;
/**
* Constructs this object.
*
* @param context the instance.
* @param sleeper the sleeper
*/
public ViewFetcher(Context context, Sleeper sleeper) {
this.mcontext = context;
this.sleeper = sleeper;
setWindowManagerString();
}
/**
* Returns the absolute top parent {@code View} in for a given {@code View}.
*
* @param view the {@code View} whose top parent is requested
* @return the top parent {@code View}
*/
public View getTopParent(View view) {
final ViewParent viewParent = view.getParent();
if (viewParent != null
&& viewParent instanceof View) {
return getTopParent((View) viewParent);
} else {
return view;
}
}
/**
* Returns the scroll or list parent view
*
* @param view the view who's parent should be returned
* @return the parent scroll view, list view or null
*/
public View getScrollOrListParent(View view) {
if (!(view instanceof android.widget.AbsListView) && !(view instanceof android.widget.ScrollView) && !(view instanceof WebView)) {
try {
return getScrollOrListParent((View) view.getParent());
} catch (Exception e) {
return null;
}
} else {
return view;
}
}
/**
* Returns views from the shown DecorViews.
*
* @param onlySufficientlyVisible if only sufficiently visible views should be returned
* @return all the views contained in the DecorViews
*/
public ArrayList getAllViews(boolean onlySufficientlyVisible) {
final View[] views = getWindowDecorViews();
final ArrayList allViews = new ArrayList();
final View[] nonDecorViews = getNonDecorViews(views);
View view = null;
if (nonDecorViews != null) {
for (View nonDecorView : nonDecorViews) {
view = nonDecorView;
try {
addChildren(allViews, (ViewGroup) view, onlySufficientlyVisible);
} catch (Exception ignored) {
}
if (view != null) allViews.add(view);
}
}
if (views != null && views.length > 0) {
view = getRecentDecorView(views);
try {
addChildren(allViews, (ViewGroup) view, onlySufficientlyVisible);
} catch (Exception ignored) {
}
if (view != null) allViews.add(view);
}
return allViews;
}
/**
* Returns the most recent DecorView
*
* @param views the views to check
* @return the most recent DecorView
*/
public final View getRecentDecorView(View[] views) {
if (views == null)
return null;
final View[] decorViews = new View[views.length];
int i = 0;
View view;
for (View view1 : views) {
view = view1;
if (isDecorView(view)) {
decorViews[i] = view;
i++;
}
}
return getRecentContainer(decorViews);
}
/**
* Returns the most recent view container
*
* @param views the views to check
* @return the most recent view container
*/
private View getRecentContainer(View[] views) {
View container = null;
long drawingTime = 0;
View view;
for (View view1 : views) {
view = view1;
if (view != null && view.isShown() && view.hasWindowFocus() && view.getDrawingTime() > drawingTime) {
container = view;
drawingTime = view.getDrawingTime();
}
}
return container;
}
/**
* Returns all views that are non DecorViews
*
* @param views the views to check
* @return the non DecorViews
*/
private View[] getNonDecorViews(View[] views) {
View[] decorViews = null;
if (views != null) {
decorViews = new View[views.length];
int i = 0;
View view;
for (View view1 : views) {
view = view1;
if (!isDecorView(view)) {
decorViews[i] = view;
i++;
}
}
}
return decorViews;
}
/**
* Returns whether a view is a DecorView
*
* @return true if view is a DecorView, false otherwise
*/
private boolean isDecorView(View view) {
if (view == null) {
return false;
}
final String nameOfClass = view.getClass().getName();
return (nameOfClass.equals("com.android.internal.policy.impl.PhoneWindow$DecorView") ||
nameOfClass.equals("com.android.internal.policy.impl.MultiPhoneWindow$MultiPhoneDecorView") ||
nameOfClass.equals("com.android.internal.policy.PhoneWindow$DecorView"));
}
/**
* Extracts all {@code View}s located in the currently active {@code Activity}, recursively.
*
* @param parent the {@code View} whose children should be returned, or {@code null} for all
* @param onlySufficientlyVisible if only sufficiently visible views should be returned
* @return all {@code View}s located in the currently active {@code Activity}, never {@code null}
*/
public ArrayList getViews(View parent, boolean onlySufficientlyVisible) {
final ArrayList views = new ArrayList();
final View parentToUse;
if (parent == null) {
return getAllViews(onlySufficientlyVisible);
} else {
parentToUse = parent;
views.add(parentToUse);
if (parentToUse instanceof ViewGroup) {
addChildren(views, (ViewGroup) parentToUse, onlySufficientlyVisible);
}
}
return views;
}
/**
* Adds all children of {@code viewGroup} (recursively) into {@code views}.
*
* @param views an {@code ArrayList} of {@code View}s
* @param viewGroup the {@code ViewGroup} to extract children from
* @param onlySufficientlyVisible if only sufficiently visible views should be returned
*/
private void addChildren(ArrayList views, ViewGroup viewGroup, boolean onlySufficientlyVisible) {
if (viewGroup != null) {
for (int i = 0; i < viewGroup.getChildCount(); i++) {
final View child = viewGroup.getChildAt(i);
if (onlySufficientlyVisible && isViewSufficientlyShown(child)) {
views.add(child);
} else if (!onlySufficientlyVisible && child != null) {
views.add(child);
}
if (child instanceof ViewGroup) {
addChildren(views, (ViewGroup) child, onlySufficientlyVisible);
}
}
}
}
/**
* Returns true if the view is sufficiently shown
*
* @param view the view to check
* @return true if the view is sufficiently shown
*/
public final boolean isViewSufficientlyShown(View view) {
final int[] xyView = new int[2];
final int[] xyParent = new int[2];
if (view == null)
return false;
final float viewHeight = view.getHeight();
final View parent = getScrollOrListParent(view);
view.getLocationOnScreen(xyView);
if (parent == null) {
xyParent[1] = 0;
} else {
parent.getLocationOnScreen(xyParent);
}
if (xyView[1] + (viewHeight / 2.0f) > getScrollListWindowHeight(view))
return false;
else if (xyView[1] + (viewHeight / 2.0f) < xyParent[1])
return false;
return true;
}
/**
* Returns the height of the scroll or list view parent
*
* @param view the view who's parents height should be returned
* @return the height of the scroll or list view parent
*/
@SuppressWarnings("deprecation")
public float getScrollListWindowHeight(View view) {
final int[] xyParent = new int[2];
View parent = getScrollOrListParent(view);
final float windowHeight;
if (parent == null) {
WindowManager windowManager = (WindowManager)
mcontext.getSystemService(Context.WINDOW_SERVICE);
windowHeight = windowManager.getDefaultDisplay().getHeight();
} else {
parent.getLocationOnScreen(xyParent);
windowHeight = xyParent[1] + parent.getHeight();
}
parent = null;
return windowHeight;
}
/**
* Returns an {@code ArrayList} of {@code View}s of the specified {@code Class} located in the current
* {@code Activity}.
*
* @param the type parameter
* @param classToFilterBy return all instances of this class, e.g. {@code Button.class} or {@code GridView.class}
* @param includeSubclasses include instances of the subclasses in the {@code ArrayList} that will be returned
* @return an {@code ArrayList} of {@code View}s of the specified {@code Class} located in the current {@code Activity}
*/
public ArrayList getCurrentViews(Class classToFilterBy, boolean includeSubclasses) {
return getCurrentViews(classToFilterBy, includeSubclasses, null);
}
/**
* Returns an {@code ArrayList} of {@code View}s of the specified {@code Class} located under the specified {@code parent}.
*
* @param the type parameter
* @param classToFilterBy return all instances of this class, e.g. {@code Button.class} or {@code GridView.class}
* @param includeSubclasses include instances of subclasses in {@code ArrayList} that will be returned
* @param parent the parent {@code View} for where to start the traversal
* @return an {@code ArrayList} of {@code View}s of the specified {@code Class} located under the specified {@code parent}
*/
public ArrayList getCurrentViews(Class classToFilterBy, boolean includeSubclasses, View parent) {
ArrayList filteredViews = new ArrayList();
List allViews = getViews(parent, true);
for (View view : allViews) {
if (view == null) {
continue;
}
Class extends View> classOfView = view.getClass();
if (includeSubclasses && classToFilterBy.isAssignableFrom(classOfView) || !includeSubclasses && classToFilterBy == classOfView) {
filteredViews.add(classToFilterBy.cast(view));
}
}
allViews = null;
return filteredViews;
}
/**
* Tries to guess which view is the most likely to be interesting. Returns
* the most recently drawn view, which presumably will be the one that the
* user was most recently interacting with.
*
* @param the type parameter
* @param views the views
* @return most recently drawn view, or null if no views were passed
*/
public final T getFreshestView(ArrayList views) {
final int[] locationOnScreen = new int[2];
T viewToReturn = null;
long drawingTime = 0;
if (views == null) {
return null;
}
for (T view : views) {
if (view != null) {
view.getLocationOnScreen(locationOnScreen);
if (locationOnScreen[0] < 0)
continue;
if (view.getDrawingTime() > drawingTime && view.getHeight() > 0) {
drawingTime = view.getDrawingTime();
viewToReturn = view;
}
}
}
views = null;
return viewToReturn;
}
/**
* Waits for a RecyclerView and returns it.
*
* @param the type parameter
* @param recyclerViewIndex the index of the RecyclerView
* @param timeOut the time out
* @return {@code ViewGroup} if RecycleView is displayed
*/
public ViewGroup getRecyclerView(int recyclerViewIndex, int timeOut) {
final long endTime = SystemClock.uptimeMillis() + timeOut;
while (SystemClock.uptimeMillis() < endTime) {
View recyclerView = getRecyclerView(true, recyclerViewIndex);
if (recyclerView != null) {
return (ViewGroup) recyclerView;
}
}
return null;
}
/**
* Returns a RecyclerView or null if none is found
*
* @param shouldSleep the should sleep
* @param recyclerViewIndex the recycler view index
* @return a RecyclerView
*/
public View getRecyclerView(boolean shouldSleep, int recyclerViewIndex) {
Set uniqueViews = new HashSet();
if (shouldSleep) {
sleeper.sleep();
}
@SuppressWarnings("unchecked")
ArrayList views = ViewUtils.filterViewsToSet(new Class[]{ViewGroup.class}, getAllViews(false));
views = ViewUtils.removeInvisibleViews(views);
for (View view : views) {
if (isViewType(view.getClass(), "widget.RecyclerView")) {
uniqueViews.add(view);
}
if (uniqueViews.size() > recyclerViewIndex) {
return (ViewGroup) view;
}
}
return null;
}
/**
* Returns a Set of all RecyclerView or empty Set if none is found
*
* @param shouldSleep the should sleep
* @return a Set of RecyclerViews
*/
public List getAllRecyclerViews(boolean shouldSleep) {
List viewsToReturn = new ArrayList();
if (shouldSleep) {
sleeper.sleep();
}
@SuppressWarnings("unchecked")
ArrayList views = ViewUtils.filterViewsToSet(new Class[]{ViewGroup.class}, getAllViews(true));
views = ViewUtils.removeInvisibleViews(views);
for (View view : views) {
if (isViewType(view.getClass(), "widget.RecyclerView")) {
viewsToReturn.add(view);
}
}
return viewsToReturn;
}
private boolean isViewType(Class> aClass, String typeName) {
return aClass.getName().contains(typeName) || aClass.getSuperclass() != null && isViewType(aClass.getSuperclass(), typeName);
}
/**
* Returns an identical View to the one specified.
*
* @param view the view to find
* @return identical view of the specified view
*/
public View getIdenticalView(View view) {
if (view == null) {
return null;
}
View viewToReturn = null;
List extends View> visibleViews = ViewUtils.removeInvisibleViews(getCurrentViews(view.getClass(), true));
for (View v : visibleViews) {
if (areViewsIdentical(v, view)) {
viewToReturn = v;
break;
}
}
return viewToReturn;
}
/**
* Compares if the specified views are identical. This is used instead of View.compare
* as it always returns false in cases where the View tree is refreshed.
*
* @param firstView the first view
* @param secondView the second view
* @return true if views are equal
*/
private boolean areViewsIdentical(View firstView, View secondView) {
return !(firstView.getId() != secondView.getId() || !firstView.getClass().isAssignableFrom(secondView.getClass())) && (!(firstView.getParent() != null && firstView.getParent() instanceof View && secondView.getParent() != null && secondView.getParent() instanceof View) || areViewsIdentical((View) firstView.getParent(), (View) secondView.getParent()));
}
/**
* Returns the WindorDecorViews shown on the screen.
*
* @return the WindorDecorViews shown on the screen
*/
@SuppressWarnings("unchecked")
public View[] getWindowDecorViews() {
Field viewsField;
Field instanceField;
try {
viewsField = windowManager.getDeclaredField("mViews");
instanceField = windowManager.getDeclaredField(windowManagerString);
viewsField.setAccessible(true);
instanceField.setAccessible(true);
Object instance = instanceField.get(null);
View[] result;
ArrayList viewList = (ArrayList) viewsField.get(instance);
if (android.os.Build.VERSION.SDK_INT >= 19) {
result = (viewList.toArray(new View[viewList.size()]));
} else {
result = (View[]) viewsField.get(instance);
}
return result;
} catch (Exception e) {
e.printStackTrace();
}
return null;
}
/**
* Sets the window manager string.
*/
private void setWindowManagerString() {
if (android.os.Build.VERSION.SDK_INT >= 17) {
windowManagerString = "sDefaultWindowManager";
} else if (android.os.Build.VERSION.SDK_INT >= 13) {
windowManagerString = "sWindowManager";
} else {
windowManagerString = "mWindowManager";
}
}
}