ความสามารถในการแลกเปลี่ยนข้อมูล

Compose ทำงานร่วมกับเฟรมเวิร์กการทดสอบทั่วไป

ความสามารถในการทำงานร่วมกับ Espresso

ในแอปแบบไฮบริด คุณจะเห็นคอมโพเนนต์ Compose ภายในลำดับชั้นของ View และ View ภายใน Composable ของ Compose (ผ่าน Composable ของ AndroidView)

คุณไม่จำเป็นต้องทำตามขั้นตอนพิเศษใดๆ เพื่อจับคู่หูฟังทั้ง 2 ประเภท คุณจับคู่ View กับ onView ของ Espresso และจับคู่องค์ประกอบ Compose กับ ComposeTestRule

@Test
fun androidViewInteropTest() {
    // Check the initial state of a TextView that depends on a Compose state.
    Espresso.onView(withText("Hello Views")).check(matches(isDisplayed()))
    // Click on the Compose button that changes the state.
    composeTestRule.onNodeWithText("Click here").performClick()
    // Check the new value.
    Espresso.onView(withText("Hello Compose")).check(matches(isDisplayed()))
}

เพิ่มความหมายที่กำหนดขอบเขตระดับ View สำหรับการทดสอบการทำงานร่วมกันของ Compose

จำกัดขอบเขตการค้นหาใน Compose ให้เฉพาะมุมมองที่ต้องการ

เมื่อย้ายข้อมูล UI ที่ซับซ้อนไปยัง Compose คุณอาจพบองค์ประกอบ Compose ที่เหมือนกันซึ่งซ้อนอยู่ภายใน Android View แบบเดิมหลายรายการ เช่น ภายใน RecyclerView หรือ ViewPager ในสถานการณ์เหล่านี้ การค้นหาการเขียนมาตรฐาน เช่น onNodeWithText("Save") อาจล้มเหลวและแสดงข้อผิดพลาด "พบหลายโหนด"

คุณสามารถกำหนดขอบเขตการทดสอบ Compose ไปยัง Android View ที่เฉพาะเจาะจงได้โดยตรงแทนที่จะแก้ไขโค้ดเวอร์ชันที่ใช้งานจริงเพื่อแท็กทดสอบแบบไดนามิกเพื่อแยกความแตกต่างขององค์ประกอบเหล่านี้

ใช้ onRootWithViewInteraction API ในกฎการทดสอบ ฟังก์ชันนี้รับ ViewInteraction ของ Espresso ซึ่งช่วยให้คุณใช้ประโยชน์จาก Espresso เพื่อแยก View คอนเทนเนอร์ที่เฉพาะเจาะจง และดำเนินการโต้ตอบ Compose ภายในลำดับชั้นที่กำหนดขอบเขตนั้นโดยเฉพาะ

โต้ตอบกับรายการในลิสต์

หากต้องการโต้ตอบกับองค์ประกอบ Compose ภายใน RecyclerView row ที่เฉพาะเจาะจง ให้ใช้ Espresso เพื่อค้นหาแถว จากนั้นกำหนดขอบเขตการโต้ตอบ Compose ของคุณกับแถวนั้น ซึ่งจะไม่สนใจองค์ประกอบ Compose ที่เหมือนกันในแถวอื่นๆ ทั้งหมด

@Test
fun testComposeButtonInsideRecyclerViewItem() = runComposeUiTest {
    // Scroll to the desired position using Espresso
    Espresso.onView(withId(recyclerViewId))
        .perform(RecyclerViewActions.scrollToPosition<MyViewHolder>(3))

    // Define an Espresso ViewInteraction that uniquely identifies the row
    val rowView = Espresso.onView(
        allOf(
            withId(rootViewId),
            hasDescendant(withText("Item #3"))
        )
    )

    // Scope the Compose search strictly to that specific row View
    onRootWithViewInteraction(rowView)
        .onNode(hasText("Like"))
        .performClick()
}

แก้ไขความคลุมเครือใน ViewPager

เมื่อมี Fragment หลายรายการที่มีเลย์เอาต์ Compose เหมือนกันอยู่ในหน่วยความจำ พร้อมกัน คุณสามารถกำหนดขอบเขตการค้นหาไปยังรหัส View รูทของ Fragment ที่เฉพาะเจาะจง เพื่อป้องกันความคลุมเครือในการจับคู่

@Test
fun testComposeButtonInsideViewPagerItem() = runComposeUiTest {
    // Swipe to the desired page using Espresso
    Espresso.onView(withId(viewPagerViewId)).perform(swipeLeft())

    // Identify the specific container view using Espresso
    val fragmentB = Espresso.onView(withId(fragmentRootViewId))

    // The generic text "Save" is now unique within this view scope
    onRootWithViewInteraction(fragmentB)
        .onNode(hasText("Save"))
        .assertIsDisplayed()
}

ความสามารถในการทำงานร่วมกับ UiAutomator

โดยค่าเริ่มต้น องค์ประกอบที่ใช้ได้จะเข้าถึงได้จาก UiAutomator โดยใช้ตัวอธิบายที่สะดวก (ข้อความที่แสดง คำอธิบายเนื้อหา ฯลฯ) เท่านั้น หากต้องการเข้าถึง Composable ใดก็ตามที่ใช้ Modifier.testTag คุณต้องเปิดใช้ พร็อพเพอร์ตี้เชิงความหมาย testTagsAsResourceId สำหรับ Subtree ของ Composable นั้นๆ การเปิดใช้ลักษณะการทำงานนี้มีประโยชน์สำหรับ Composable ที่ไม่มีแฮนเดิลอื่นๆ ที่ไม่ซ้ำกัน เช่น Composable ที่เลื่อนได้ (เช่น LazyColumn)

เปิดใช้พร็อพเพอร์ตี้เชิงความหมายเพียงครั้งเดียวที่ระดับสูงในลำดับชั้นของ Composable เพื่อ ให้มั่นใจว่า Composable ที่ซ้อนกันทั้งหมดที่มี Modifier.testTag จะเข้าถึงได้จาก UiAutomator

Scaffold(
    // Enables for all composables in the hierarchy.
    modifier = Modifier.semantics {
        testTagsAsResourceId = true
    }
){
    // Modifier.testTag is accessible from UiAutomator for composables nested here.
    LazyColumn(
        modifier = Modifier.testTag("myLazyColumn")
    ){
        // Content
    }
}

Composable ใดๆ ที่มี Modifier.testTag(tag) จะเข้าถึงได้โดยใช้ By.res(resourceName) โดยใช้ tag เดียวกันกับ resourceName

val device = UiDevice.getInstance(getInstrumentation())

val lazyColumn: UiObject2 = device.findObject(By.res("myLazyColumn"))
// Some interaction with the lazyColumn.

แหล่งข้อมูลเพิ่มเติม

  • ทดสอบแอปใน Android: หน้า Landing Page หลักของการทดสอบ Android จะแสดงมุมมองที่กว้างขึ้นเกี่ยวกับพื้นฐานและเทคนิคการทดสอบ
  • หลักพื้นฐานของการทดสอบ: ดูข้อมูลเพิ่มเติม เกี่ยวกับแนวคิดหลักเบื้องหลังการทดสอบแอป Android
  • การทดสอบในเครื่อง: คุณสามารถเรียกใช้การทดสอบบางอย่าง ในเครื่องบนเวิร์กสเตชันของคุณเองได้
  • การทดสอบที่มีการวัดคุม: คุณควรเรียกใช้การทดสอบที่มีการวัดคุมด้วย กล่าวคือ การทดสอบที่ทำงานโดยตรงในอุปกรณ์
  • การรวมอย่างต่อเนื่อง: การรวมอย่างต่อเนื่องช่วยให้คุณผสานรวมการทดสอบเข้ากับไปป์ไลน์การติดตั้งใช้งานได้
  • ทดสอบขนาดหน้าจอต่างๆ: เนื่องจากผู้ใช้มีอุปกรณ์ให้เลือกมากมาย คุณจึงควรทดสอบขนาดหน้าจอต่างๆ
  • Espresso: แม้ว่าจะมีไว้สำหรับ UI ที่อิงตาม View แต่ความรู้เกี่ยวกับ Espresso ก็ยังเป็นประโยชน์สำหรับบางแง่มุมของการทดสอบ Compose