woshidan's blog

そんなことよりコードにダイブ。

はじめてのEspresso

EspressoではUIViewの表示、外部からIntentを渡されて(BroadcastReceiver?)起動する時の画面等、外部とのやりとりはなるべく行わないで、この画面でこの値が与えられている時は、 この要素が表示されているよね、ということをテストするみたいです。

とりあえず、最初のテストを書くところはいろいろな本に載っていたので、簡単なActivityの行き来についてテストを書いてみました。

参考:

2015年7月時点でのJUnit4やEspressoを使ったAndroidアプリのテストについて - Qiita

Android改善プログラミング - TechBooster - BOOTH(同人誌通販・ダウンロード)

設定ファイル

apply plugin: 'com.android.application'

android {
    compileSdkVersion 23
    buildToolsVersion "23.0.1"

    defaultConfig {
        applicationId "com.example.woshidan.memoteclient"
        minSdkVersion 16
        targetSdkVersion 23
        versionCode 1
        versionName "1.0"
        testInstrumentationRunner "android.support.test.runner.AndroidJUnitRunner"
    }
    buildTypes {
        release {
            minifyEnabled false
            proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro'
        }
    }
}

dependencies {
    compile fileTree(dir: 'libs', include: ['*.jar'])
    compile 'com.android.support:appcompat-v7:23.0.1'

    compile 'com.android.support:support-annotations:23.0.1'
    // Testing-only dependencies
    androidTestCompile 'com.android.support.test:runner:0.4'
    androidTestCompile 'com.android.support.test:rules:0.4'
    androidTestCompile 'com.android.support.test.espresso:espresso-core:2.2.1'
}

コード

public class MainActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Intent intent = new Intent(MainActivity.this, SecondActivity.class);
                startActivity(intent);
            }
        });
    }

public class SecondActivity extends AppCompatActivity {

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_second);

        Button button = (Button) findViewById(R.id.button);
        button.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                finish();
            }
        });
    }
<!-- MainActivity -->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
    android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin" tools:context=".MainActivity">

    <TextView android:text="@string/hello_world" android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

    <Button
        android:layout_height="wrap_content"
        android:layout_width="wrap_content"
        android:id="@+id/button"
        android:text="to Second"
        />
</LinearLayout>

<!-- SecondActivity -->
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent"
    android:layout_height="match_parent" android:paddingLeft="@dimen/activity_horizontal_margin"
    android:paddingRight="@dimen/activity_horizontal_margin"
    android:paddingTop="@dimen/activity_vertical_margin"
    android:paddingBottom="@dimen/activity_vertical_margin"
    tools:context="com.example.woshidan.memoteclient.SecondActivity">

    <TextView android:text="Second Activity" android:layout_width="wrap_content"
        android:layout_height="wrap_content" />

    <Button
        android:layout_height="wrap_content"
        android:layout_width="wrap_content"
        android:id="@+id/button"
        android:text="Return"
        />
</LinearLayout>

テスト

package com.example.woshidan.memoteclient;

import android.app.Activity;
import android.support.test.rule.ActivityTestRule;
import android.support.test.runner.AndroidJUnit4;
import android.test.suitebuilder.annotation.LargeTest;

import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;

import static android.support.test.espresso.Espresso.onView;
import static android.support.test.espresso.action.ViewActions.click;
import static android.support.test.espresso.assertion.ViewAssertions.matches;
import static android.support.test.espresso.matcher.ViewMatchers.isDisplayed;
import static android.support.test.espresso.matcher.ViewMatchers.withId;
import static android.support.test.espresso.matcher.ViewMatchers.withText;

/**
 * Created by woshidan on 2015/11/16.
 */
@RunWith(AndroidJUnit4.class)
@LargeTest
public class MoveActivityTest {
    @Rule
    public ActivityTestRule<MainActivity> mActivityRule = new ActivityTestRule(MainActivity.class);

    @Test
    public void goToSecondActivity() {
        onView(withId(R.id.button)).perform(click());
        onView(withText("Second Activity")).check(matches(isDisplayed()));
        onView(withId(R.id.button)).perform(click());
        onView(withText("Hello world!")).check(matches(isDisplayed()));
    }
}

メモ

  • Espresso, バージョンアップされて、ライブラリが分かれてる
  • Import時なんでかalt + returnが効かないので手動でクリックしてた
  • Stubにも反応したし、いろいろ外部APIへの通信とかStubできそう
  • Activityの移動とかAndroid内部だったら特に問題ないので、この情報を与えた上で、このメニューが出ていて...とか、このActivityへリダイレクトされて...という画面遷移のテストも簡単にできそう
  • 複数のマッチャを組み合わせたい場合はallOf(マッチャ1, マッチャ2...)という風に書く