एस्प्रेसो टेस्टिंग फ्रेमवर्क - इंटेंट्स

एंड्रॉइड इंटेंट का उपयोग नई गतिविधि को खोलने के लिए किया जाता है, या तो आंतरिक (उत्पाद सूची स्क्रीन से उत्पाद विवरण स्क्रीन खोलना) या बाहरी (जैसे कॉल करने के लिए डायलर खोलना)। आंतरिक आशय गतिविधि को एस्प्रेसो परीक्षण ढांचे द्वारा पारदर्शी रूप से नियंत्रित किया जाता है और इसे उपयोगकर्ता की ओर से किसी विशिष्ट कार्य की आवश्यकता नहीं होती है। हालांकि, बाहरी गतिविधि को लागू करना वास्तव में एक चुनौती है क्योंकि यह हमारे दायरे से बाहर है, परीक्षण के तहत आवेदन। एक बार जब उपयोगकर्ता किसी बाहरी एप्लिकेशन को इनवॉइस करता है और परीक्षण के तहत एप्लिकेशन से बाहर चला जाता है, तो उपयोगकर्ता द्वारा कार्रवाई के पूर्वनिर्धारित अनुक्रम के साथ एप्लिकेशन पर वापस आने की संभावना कम होती है। इसलिए, हमें एप्लिकेशन का परीक्षण करने से पहले उपयोगकर्ता कार्रवाई को मानने की आवश्यकता है। एस्प्रेसो इस स्थिति को संभालने के लिए दो विकल्प प्रदान करता है। वे इस प्रकार हैं,

इरादा

यह उपयोगकर्ता को यह सुनिश्चित करने की अनुमति देता है कि परीक्षण के तहत आवेदन से सही इरादा खोला गया है।

इच्छुक

यह उपयोगकर्ता को कैमरे से एक तस्वीर लेने, संपर्क सूची से एक नंबर डायल करने, इत्यादि जैसी बाहरी गतिविधि का मजाक बनाने की अनुमति देता है, और मूल्यों के पूर्वनिर्धारित सेट के साथ आवेदन पर वापस लौटता है (जैसे वास्तविक छवि के बजाय कैमरे से पूर्वनिर्धारित छवि) ।

सेट अप

एस्प्रेसो एक प्लगइन लाइब्रेरी के माध्यम से इरादे के विकल्प का समर्थन करता है और लाइब्रेरी को एप्लिकेशन के ग्रेड फ़ाइल में कॉन्फ़िगर करने की आवश्यकता होती है। कॉन्फ़िगरेशन विकल्प निम्नानुसार है,

dependencies {
   // ...
   androidTestImplementation 'androidx.test.espresso:espresso-intents:3.1.1'
}

इरादा ()

एस्प्रेसो इरादे प्लगइन विशेष मिलानकर्ताओं को यह जांचने के लिए प्रदान करता है कि आह्वान किया गया इरादा अपेक्षित इरादा है या नहीं। प्रदान किए गए मैचर्स और मैचर्स का उद्देश्य इस प्रकार है,

hasAction

यह इरादे की कार्रवाई को स्वीकार करता है और एक मिलानकर्ता को लौटाता है, जो निर्दिष्ट इरादे से मेल खाता है।

hasData

यह डेटा को स्वीकार करता है और एक मिलानकर्ता को लौटाता है, जो इसे प्रदान करते समय आशय को प्रदान किए गए डेटा से मेल खाता है।

पैकेज के लिए

यह आशय पैकेज नाम को स्वीकार करता है और एक मिलानकर्ता को लौटाता है, जो कि आशयित इरादे के पैकेज के नाम से मेल खाता है।

अब, एक नया एप्लिकेशन बनाते हैं और अवधारणा को समझने के लिए () का उपयोग करके बाहरी गतिविधि के लिए एप्लिकेशन का परीक्षण करते हैं ।

  • Android स्टूडियो प्रारंभ करें।

  • पहले से चर्चा के अनुसार एक नया प्रोजेक्ट बनाएं और इसे नाम दें, IntentSampleApp।

  • का उपयोग कर AndroidX ढांचे के लिए ऐप्लिकेशन माइग्रेट Refactor → माइग्रेट करने के लिए AndroidX विकल्प मेनू।

  • एक पाठ बॉक्स, संपर्क सूची खोलने के लिए एक बटन और एक और एक गतिविधि डायल करने के लिए कॉल करके डायल करें_मैं. xml नीचे दिखाए अनुसार,

<?xml version = "1.0" encoding = "utf-8"?>
<RelativeLayout xmlns:android = "http://schemas.android.com/apk/res/android"
   xmlns:app = "http://schemas.android.com/apk/res-auto"
   xmlns:tools = "http://schemas.android.com/tools"
   android:layout_width = "match_parent"
   android:layout_height = "match_parent"
   tools:context = ".MainActivity">
   <EditText
      android:id = "@+id/edit_text_phone_number"
      android:layout_width = "wrap_content"
      android:layout_height = "wrap_content"
      android:layout_centerHorizontal = "true"
      android:text = ""
      android:autofillHints = "@string/phone_number"/>
   <Button
      android:id = "@+id/call_contact_button"
      android:layout_width = "wrap_content"
      android:layout_height = "wrap_content"
      android:layout_centerHorizontal = "true"
      android:layout_below = "@id/edit_text_phone_number"
      android:text = "@string/call_contact"/>
   <Button
      android:id = "@+id/button"
      android:layout_width = "wrap_content"
      android:layout_height = "wrap_content"
      android:layout_centerHorizontal = "true"
      android:layout_below = "@id/call_contact_button"
      android:text = "@string/call"/>
</RelativeLayout>
  • इसके अलावा, स्ट्रिंग्स . xml संसाधन फ़ाइल में निम्न आइटम जोड़ें ,

<string name = "phone_number">Phone number</string>
<string name = "call">Call</string>
<string name = "call_contact">Select from contact list</string>
  • अब, मुख्य गतिविधि (में नीचे दिए गए कोड को जोड़ने MainActivity.java के तहत) onCreate विधि।

public class MainActivity extends AppCompatActivity {
   @Override
   protected void onCreate(Bundle savedInstanceState) {
      // ... code
      // Find call from contact button
      Button contactButton = (Button) findViewById(R.id.call_contact_button);
      contactButton.setOnClickListener(new View.OnClickListener() {
         @Override
         public void onClick(View view) {
            // Uri uri = Uri.parse("content://contacts");
            Intent contactIntent = new Intent(Intent.ACTION_PICK,
               ContactsContract.Contacts.CONTENT_URI);
            contactIntent.setType(ContactsContract.CommonDataKinds.Phone.CONTENT_TYPE);
            startActivityForResult(contactIntent, REQUEST_CODE);
         }
      });
      // Find edit view
      final EditText phoneNumberEditView = (EditText)
         findViewById(R.id.edit_text_phone_number);
      // Find call button
      Button button = (Button) findViewById(R.id.button);
      button.setOnClickListener(new View.OnClickListener() {
         @Override
         public void onClick(View view) {
            if(phoneNumberEditView.getText() != null) {
               Uri number = Uri.parse("tel:" + phoneNumberEditView.getText());
               Intent callIntent = new Intent(Intent.ACTION_DIAL, number);
               startActivity(callIntent);
            }
         }
      });
   }
   // ... code
}

यहाँ, हम आईडी के साथ बटन, प्रोग्राम किया है call_contact_button संपर्क सूची और आईडी, साथ बटन खोलने के लिए बटन कॉल डायल करने के लिए।

  • नीचे दिखाए गए अनुसार MainActivity क्लास में एक स्थिर चर REQUEST_CODE जोड़ें ,

public class MainActivity extends AppCompatActivity {
   // ...
   private static final int REQUEST_CODE = 1;
   // ...
}
  • अब, नीचे के रूप में MainActivity वर्ग में onActivityResult विधि जोड़ें ,

public class MainActivity extends AppCompatActivity {
   // ...
   @Override
   protected void onActivityResult(int requestCode, int resultCode, Intent data) {
      if (requestCode == REQUEST_CODE) {
         if (resultCode == RESULT_OK) {
            // Bundle extras = data.getExtras();
            // String phoneNumber = extras.get("data").toString();
            Uri uri = data.getData();
            Log.e("ACT_RES", uri.toString());
            String[] projection = {
               ContactsContract.CommonDataKinds.Phone.NUMBER, 
               ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME };
            Cursor cursor = getContentResolver().query(uri, projection, null, null, null);
            cursor.moveToFirst();
            
            int numberColumnIndex =
               cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER);
            String number = cursor.getString(numberColumnIndex);
            
            int nameColumnIndex = cursor.getColumnIndex(
               ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME);
            String name = cursor.getString(nameColumnIndex);
            Log.d("MAIN_ACTIVITY", "Selected number : " + number +" , name : "+name);
            
            // Find edit view
            final EditText phoneNumberEditView = (EditText)
               findViewById(R.id.edit_text_phone_number);
            phoneNumberEditView.setText(number);
         }
      }
   };
   // ...
}

यहां, onActivityResult को तब कॉल किया जाएगा जब कोई उपयोगकर्ता call_contact_button बटन का उपयोग करके संपर्क सूची खोलने और संपर्क का चयन करने के बाद आवेदन पर लौटता है । एक बार onActivityResult विधि लागू होने के बाद, यह उपयोगकर्ता द्वारा चयनित संपर्क प्राप्त करता है, संपर्क नंबर ढूंढता है और इसे पाठ बॉक्स में सेट करता है।

  • एप्लिकेशन चलाएँ और सुनिश्चित करें कि सब कुछ ठीक है। आशय नमूना आवेदन का अंतिम रूप नीचे दिखाया गया है,

  • अब, नीचे दिखाए गए के रूप में आवेदन की ग्रेड फ़ाइल में एस्प्रेसो इरादे कॉन्फ़िगर करें,

dependencies {
   // ...
   androidTestImplementation 'androidx.test.espresso:espresso-intents:3.1.1'
}
  • एंड्रॉइड स्टूडियो द्वारा प्रदान किए गए सिंक नाउ मेनू विकल्प पर क्लिक करें । यह आशय परीक्षण पुस्तकालय डाउनलोड करेगा और इसे ठीक से कॉन्फ़िगर करेगा।

  • ExampleInstrumentedTest.java फ़ाइल खोलें और आम तौर पर उपयोग किए जाने वाले AndroidTestRule के बजाय IntentsTestRule जोड़ें । IntentTestRule इरादे परीक्षण को संभालने के लिए एक विशेष नियम है।

public class ExampleInstrumentedTest {
   // ... code
   @Rule
   public IntentsTestRule<MainActivity> mActivityRule =
   new IntentsTestRule<>(MainActivity.class);
   // ... code
}
  • नीचे दिए गए परीक्षण फोन नंबर और डायलर पैकेज का नाम सेट करने के लिए दो स्थानीय चर जोड़ें,

public class ExampleInstrumentedTest {
   // ... code
   private static final String PHONE_NUMBER = "1 234-567-890";
   private static final String DIALER_PACKAGE_NAME = "com.google.android.dialer";
   // ... code
}
  • Android स्टूडियो द्वारा दिए गए Alt + Enter विकल्प का उपयोग करके आयात के मुद्दों को ठीक करें या फिर नीचे दिए गए आयात विवरणों को शामिल करें,

import android.content.Context;
import android.content.Intent;

import androidx.test.InstrumentationRegistry;
import androidx.test.espresso.intent.rule.IntentsTestRule;
import androidx.test.runner.AndroidJUnit4;

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

import static androidx.test.espresso.Espresso.onView;
import static androidx.test.espresso.action.ViewActions.click;
import static androidx.test.espresso.action.ViewActions.closeSoftKeyboard;
import static androidx.test.espresso.action.ViewActions.typeText;
import static androidx.test.espresso.intent.Intents.intended;
import static androidx.test.espresso.intent.matcher.IntentMatchers.hasAction;
import static androidx.test.espresso.intent.matcher.IntentMatchers.hasData;
import static androidx.test.espresso.intent.matcher.IntentMatchers.toPackage;
import static androidx.test.espresso.matcher.ViewMatchers.withId;
import static org.hamcrest.core.AllOf.allOf;
import static org.junit.Assert.*;
  • डायलर को ठीक से बुलाया जाए या नहीं, यह जांचने के लिए नीचे का परीक्षण मामला जोड़ें

public class ExampleInstrumentedTest {
   // ... code
   @Test
   public void validateIntentTest() {
      onView(withId(R.id.edit_text_phone_number))
         .perform(typeText(PHONE_NUMBER), closeSoftKeyboard());
      onView(withId(R.id.button)) .perform(click());
      intended(allOf(
         hasAction(Intent.ACTION_DIAL),
         hasData("tel:" + PHONE_NUMBER),
         toPackage(DIALER_PACKAGE_NAME)));
   }
   // ... code
}

यहाँ, हैवेशन , हैडटा और टापेज मैचर्स का उपयोग ऑलऑफ मैचर के साथ किया जाता है, तभी सफल होगा जब सभी मैचर्स पास हो जाएंगे

  • अब, Android स्टूडियो में सामग्री मेनू के माध्यम से ExampleInstrumentedTest चलाएं

इच्छुक ()

एस्प्रेसो एक विशेष विधि प्रदान करता है - इरादा () बाहरी इरादे की कार्रवाई का मजाक उड़ाना। इरादा () मंज़ूर किए जाने के इरादे के पैकेज के नाम को स्वीकार करते हैं और एक विधि का जवाब प्रदान करते हैं। यह निर्धारित करने के लिए कि नीचे दिए गए इरादे के साथ किस तरह मॉक किए गए इरादे का जवाब देना है

intending(toPackage("com.android.contacts")).respondWith(result);

यहाँ, responseWith () टाइप इंस्ट्रूमेंटेशन के अभिप्राय परिणाम को स्वीकार करता है। ActivityResult । हम नए स्टब आशय बना सकते हैं और मैन्युअल रूप से परिणाम को नीचे बताए अनुसार सेट कर सकते हैं,

// Stub intent
Intent intent = new Intent();
intent.setData(Uri.parse("content://com.android.contacts/data/1"));
Instrumentation.ActivityResult result =
   new Instrumentation.ActivityResult(Activity.RESULT_OK, intent);

यह जानने के लिए कि क्या कोई एप्लिकेशन ठीक से खोला गया है, यह जांचने के लिए पूरा कोड निम्नानुसार है,

@Test
public void stubIntentTest() {
   // Stub intent
   Intent intent = new Intent();
   intent.setData(Uri.parse("content://com.android.contacts/data/1"));
   Instrumentation.ActivityResult result =
      new Instrumentation.ActivityResult(Activity.RESULT_OK, intent);
   intending(toPackage("com.android.contacts")).respondWith(result);
   
   // find the button and perform click action
   onView(withId(R.id.call_contact_button)).perform(click());
   
   // get context
   Context targetContext2 = InstrumentationRegistry.getInstrumentation().getTargetContext();
   
   // get phone number
   String[] projection = { ContactsContract.CommonDataKinds.Phone.NUMBER,
      ContactsContract.CommonDataKinds.Phone.DISPLAY_NAME };
   Cursor cursor =
      targetContext2.getContentResolver().query(Uri.parse("content://com.android.cont
      acts/data/1"), projection, null, null, null);
   
   cursor.moveToFirst();
   int numberColumnIndex =
      cursor.getColumnIndex(ContactsContract.CommonDataKinds.Phone.NUMBER);
   String number = cursor.getString(numberColumnIndex);
   
   // now, check the data
   onView(withId(R.id.edit_text_phone_number))
   .check(matches(withText(number)));
}

यहां, हमने एक नया इरादा बनाया है और संपर्क सूची, सामग्री: //com.android.contacts/data/1 की पहली प्रविष्टि के रूप में रिटर्न वैल्यू (जब इरादे को लागू करते हैं) सेट किया है । फिर हमने संपर्क सूची के स्थान पर नए बनाए गए इरादों का मजाक उड़ाने का इरादा बनाया है। यह पैकेज, com.android.contacts को लागू करने पर हमारी नई बनाई गई मंशा को सेट और कॉल करता है और सूची की डिफ़ॉल्ट पहली प्रविष्टि वापस आ जाती है। फिर, हमने नकली इरादे को शुरू करने के लिए क्लिक () कार्रवाई को निकाल दिया और अंत में जाँच करता है कि क्या नकली इरादे को लागू करने से फोन नंबर और संपर्क सूची में पहली प्रविष्टि की संख्या समान हैं।

यह कोई भी गायब आयात समस्या है, फिर उन आयात मुद्दों को Alt + Enter विकल्प का उपयोग करके ठीक करें जो कि एंड्रॉइड स्टूडियो द्वारा दिए गए हैं या फिर नीचे दिए गए आयात विवरणों को शामिल करें,

import android.app.Activity;
import android.app.Instrumentation;
import android.content.Context;
import android.content.Intent;
import android.database.Cursor;
import android.net.Uri;
import android.provider.ContactsContract;

import androidx.test.InstrumentationRegistry;
import androidx.test.espresso.ViewInteraction;
import androidx.test.espresso.intent.rule.IntentsTestRule;
import androidx.test.runner.AndroidJUnit4;

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

import static androidx.test.espresso.Espresso.onView;
import static androidx.test.espresso.action.ViewActions.click;
import static androidx.test.espresso.action.ViewActions.closeSoftKeyboard;
import static androidx.test.espresso.action.ViewActions.typeText;
import static androidx.test.espresso.assertion.ViewAssertions.matches;
import static androidx.test.espresso.intent.Intents.intended;
import static androidx.test.espresso.intent.Intents.intending;
import static androidx.test.espresso.intent.matcher.IntentMatchers.hasAction;
import static androidx.test.espresso.intent.matcher.IntentMatchers.hasData;
import static androidx.test.espresso.intent.matcher.IntentMatchers.toPackage;
import static androidx.test.espresso.matcher.ViewMatchers.withId;
import static androidx.test.espresso.matcher.ViewMatchers.withText;
import static org.hamcrest.core.AllOf.allOf;
import static org.junit.Assert.*;

संपर्क सूची पढ़ने की अनुमति प्रदान करने के लिए परीक्षण वर्ग में नीचे का नियम जोड़ें -

@Rule
public GrantPermissionRule permissionRule =
GrantPermissionRule.grant(Manifest.permission.READ_CONTACTS);

एप्लिकेशन मैनिफ़ेस्ट फ़ाइल में निम्न विकल्प जोड़ें, AndroidManifest.xml -

<uses-permission android:name = "android.permission.READ_CONTACTS" />

अब, सुनिश्चित करें कि संपर्क सूची में कम से कम एक प्रविष्टि है और फिर एंड्रॉइड स्टूडियो के संदर्भ मेनू का उपयोग करके परीक्षण चलाएं।