Как Android ActivityUnitTestCase включает в себя AndroidManifest.xml целевого приложения? - PullRequest
3 голосов
/ 13 января 2011

У меня есть активность Допустим, A

это определено в данном приложении и это соответствующий манифест. Это действие загружает contentView, который он просто загружает через статический индекс R. Допустим, R.layout.foo. В этом макете есть компонент, который смотрит на что-то, что не является базовым атрибутом Android. Я вижу, что когда тестовое приложение запускает это действие, тема и стили внутри темы не заполняются ничем.

Образец манифеста

   <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
        package="com.foo.bar"
        android:versionCode="1"
        android:versionName="Test"
        android:installLocation="auto">

       <uses-sdk android:minSdkVersion="3" android:targetSdkVersion="8" />
       <uses-permission android:name="android.permission.INTERNET" />

       <application 
                  android:icon="@drawable/app_icon"
              android:label="@string/app_name"
              android:description="@string/description"
              android:theme="@style/Theme.Custom"
              android:name=".MyApplicationObject">

          <activity android:name=".activity.A"/>

          <supports-screens
              android:smallScreens="true"
                  android:normalScreens="true"
                  android:largeScreens="true"
                  android:anyDensity="true" />
    </manifest>

Задание A

public class A extends Activity {
   public void onCreate(Bundle a) {
     setContentView(R.layout.foo);
   }
}

Макет, foo.xml

<?xml version="1.0" encoding="utf-8"?>
<LinearLayout 
    xmlns:android="http://schemas.android.com/apk/res/android"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content">
   <com.foo.bar.CustomView
      android:layout_width="wrap_content"
      android:layout_height="wrap_content"
    />

</LinearLayout>

CustomView

public class CustomView extends RelativeLayout {

  public CustomView(Context context, AttributeSet set) {
     this(context, set, R.attr.CustomViewStyle);
  }

  public CustomView(Context c, AttributeSet set, int defStyle) {
    super(c, set, defStyle);

    TypedArray array = c.obtainStyledAttributes(set, R.styleable.CustomViewAttrs, defStyle, defStyle);
    final int layout = array.getResourceId(R.styleable.CustomViewAttrs_layout, 0);
    final Drawable icon = array.getDrawable(R.styleable.CustomViewAttrs_icon);
    array.recycle();
    if (layout == null) {
      throw new IllegalStateException("WTF");
    }
    if (icon == null) {
      throw new IllegalStateException("For real, WTF");
    }
  }
}

Некоторые ресурсы в файле значений

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <!-- Used to define a namespace for our special types -->
    <declare-styleable name="CustomTypes">
        <attr name="CustomViewStyle" format="reference"/>
    </declare-styleable>

    <!-- Used to define Attributes specific to our new View, "CustomView" -->
    <declare-styleable name="CustomViewAttrs">
        <attr name="layout" format="reference"/>
        <attr name="anotherOne" format="reference"/>
    </declare-styleable>

    <!-- A usable style that we can reference later to pass to an instance of CustumView -->
    <style name="CustomView">
         <item name="layout">@layout/foo</item>
         <item name="AnotherOne">@drawable/icon</item>
    </style>

    <!-- A Style to act as our Theme, referenced in our Manifest as the Theme for all activities -->
    <style name="Theme.Custom" parent="android:Theme">
        <item name="CustomViewStyle">@style/CustomView</item>
    </style>
<resources>

Это работает нормально, но когда я использую ActivityUnitTest для загрузки экземпляра A, значения внутри TypedArray пусты.

Некоторые тестовые классы

public class ActivityTester extends ActivityUnitTestCase<A> {
   public ActivityTester() {
     super(A.class);
   }

   public void testOne() {
      Intent intent = new Intent(getInstrumentation().getTargetContext(), A.class);
      // This fails with my IllegalStateException 
      startActivity(intent, null, null);
   }
}

Есть идеи, как / если целевое приложение получило его разбор? Кажется, тема даже не загружается. Документация для startActivity () гласит, что она начнет действие так же, как и context.startActivity (). Я не вижу, чтобы это происходило, так как это не соответствует манифестным данным о деятельности.

Ответы [ 2 ]

2 голосов
/ 14 октября 2012

Итак, после большого количества времени, потраченного на попытки сделать это, я обнаружил, что лично для меня лучший способ сделать это - смоделировать саму тему.

   public void setUp() {
      ContextThemeWrapper context = new ContextThemeWrapper(getInstrumentation().getTargetContext(), R.style.Theme_MyTheme);
      setActivityContext(context);
   }

   public void testOne() {
      Intent intent = new Intent(getInstrumentation().getTargetContext(), A.class);
      startActivity(intent, null, null);
   }

Лично мне это не нравится, потому что оно связывает реализацию модульного теста со стилистическими нюансами, которые я хотел бы возложить на другую роль (например, визуальный дизайнер или художник-график) и добавляет возможность теста быть полностью не синхронизированным с реализацией, которая будет отправлена. то есть кто-то изменяет файлы манифеста так, чтобы ресурс Theme, связанный с Activity, отличался от того, что тестируется. Этот тип TestCase кажется просто недальновидным, поскольку он может привести к куче плохих вещей, подобных этой. Я имею в виду, что в конечном итоге вы получите кучу ActivityUnitTestCase экземпляров, которые тестируют базовую функциональность, а затем несколько тестовых примеров для того же самого Activity, которые просто запускают его или проходят через переходы, которые могут вызвать наложение макета. В настоящий момент не тестируется настоящее приложение.

0 голосов
/ 13 января 2011

Вместо этого вы можете использовать ActivityInstrumentationTestCase2 , который предназначен для тестирования действий, созданных с использованием инфраструктуры системы:

import android.test.ActivityInstrumentationTestCase2;

public class ActivityTester  extends ActivityInstrumentationTestCase2<ActivityTester> {
    public ActivityTester() {
        super(ActivityTester.class);
    }

    public void testOne() {
        //Intent intent = new Intent(getInstrumentation().getTargetContext(), ActivityTester.class);
        // This fails with my IllegalStateException 
        //startActivity(intent, null, null);
        getActivity();
    }
}

ActivityUnitTestCase предназначен для модульного тестирования операций в изоляции.

В вашем коде есть и другие ошибки, но я думаю, что они являются результатом копирования, исключающего некоторые части.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...