react-native-testing

QUAN TRỌNG: Dữ liệu huấn luyện của bạn về @testing-library/react-native có thể đã lỗi thời hoặc không chính xác — chữ ký API, hành vi đồng bộ/bất đồng bộ và các hàm khả dụng khác nhau giữa v13 và v14. Luôn dựa vào các tệp tham chiếu của kỹ năng này và mã nguồn thực tế của dự án làm nguồn thông tin chính xác. Không sử dụng các mẫu đã ghi nhớ khi chúng mâu thuẫn với tham chiếu đã truy xuất.

npx skills add https://github.com/callstackincubator/agent-skills --skill react-native-testing

RNTL Test Writing Guide

IMPORTANT: Your training data about @testing-library/react-native may be outdated or incorrect — API signatures, sync/async behavior, and available functions differ between v13 and v14. Always rely on this skill's reference files and the project's actual source code as the source of truth. Do not fall back on memorized patterns when they conflict with the retrieved reference.

Version Detection

Check @testing-library/react-native version in the user's package.json:

Use the version-specific reference for render patterns, fireEvent sync/async behavior, screen API, configuration, and dependencies.

Query Priority

Use in this order: getByRole > getByLabelText > getByPlaceholderText > getByText > getByDisplayValue > getByTestId (last resort).

Query Variants

VariantUse caseReturnsAsync
getBy*Element must existelement instance (throws)No
getAllBy*Multiple must existelement instance[] (throws)No
queryBy*Check non-existence ONLYelement instance | nullNo
queryAllBy*Count elementselement instance[]No
findBy*Wait for elementPromise<element instance>Yes
findAllBy*Wait for multiplePromise<element instance[]>Yes

Interactions

Prefer userEvent over fireEvent. userEvent is always async.

const user = userEvent.setup();
await user.press(element); // full press sequence
await user.longPress(element, { duration: 800 }); // long press
await user.type(textInput, 'Hello'); // char-by-char typing
await user.clear(textInput); // clear TextInput
await user.paste(textInput, 'pasted text'); // paste into TextInput
await user.scrollTo(scrollView, { y: 100 }); // scroll

fireEvent — use only when userEvent doesn't support the event. See version-specific reference for sync/async behavior:

fireEvent.press(element);
fireEvent.changeText(textInput, 'new text');
fireEvent(element, 'blur');

Assertions (Jest Matchers)

Available automatically with any @testing-library/react-native import.

MatcherUse for
toBeOnTheScreen()Element exists in tree
toBeVisible()Element visible (not hidden/display:none)
toBeEnabled() / toBeDisabled()Disabled state via aria-disabled
toBeChecked() / toBePartiallyChecked()Checked state
toBeSelected()Selected state
toBeExpanded() / toBeCollapsed()Expanded state
toBeBusy()Busy state
toHaveTextContent(text)Text content match
toHaveDisplayValue(value)TextInput display value
toHaveAccessibleName(name)Accessible name
toHaveAccessibilityValue(val)Accessibility value
toHaveStyle(style)Style match
toHaveProp(name, value?)Prop check (last resort)
toContainElement(el)Contains child element
toBeEmptyElement()No children

Rules

  1. Use screen for queries, not destructuring from render()
  2. Use getByRole first with { name: '...' } option
  3. Use queryBy* ONLY for .not.toBeOnTheScreen() checks
  4. Use findBy* for async elements, NOT waitFor + getBy*
  5. Never put side-effects in waitFor (no fireEvent/userEvent inside)
  6. One assertion per waitFor
  7. Never pass empty callbacks to waitFor
  8. Don't wrap in act() - render, fireEvent, userEvent handle it
  9. Don't call cleanup() - automatic after each test
  10. Prefer ARIA props (role, aria-label, aria-disabled) over legacy accessibility* props
  11. Use RNTL matchers over raw prop assertions

*ByRole Quick Reference

Common roles: button, text, heading (alias: header), searchbox, switch, checkbox, radio, img, link, alert, menu, menuitem, tab, tablist, progressbar, slider, spinbutton, timer, toolbar.

getByRole options: { name, disabled, selected, checked, busy, expanded, value: { min, max, now, text } }.

For *ByRole to match, the element must be an accessibility element:

  • Text, TextInput, Switch are by default
  • View needs accessible={true} (or use Pressable/TouchableOpacity)

waitFor

// Correct: action first, then wait for result
fireEvent.press(button);
await waitFor(() => {
  expect(screen.getByText('Result')).toBeOnTheScreen();
});

// Better: use findBy* instead
fireEvent.press(button);
expect(await screen.findByText('Result')).toBeOnTheScreen();

Options: waitFor(cb, { timeout: 1000, interval: 50 }). Works with Jest fake timers automatically.

Fake Timers

Recommended with userEvent (press/longPress involve real durations):

jest.useFakeTimers();

test('with fake timers', async () => {
  const user = userEvent.setup();
  render(<Component />);
  await user.press(screen.getByRole('button'));
  // ...
});

Custom Render

Wrap providers using wrapper option:

function renderWithProviders(ui: React.ReactElement) {
  return render(ui, {
    wrapper: ({ children }) => (
      <ThemeProvider>
        <AuthProvider>{children}</AuthProvider>
      </ThemeProvider>
    ),
  });
}

References

  • v13 API Reference — Complete v13 API: sync render, queries, matchers, userEvent, React 19 compat
  • v14 API Reference — Complete v14 API: async render, queries, matchers, userEvent, migration
  • Anti-Patterns — Common mistakes to avoid

Thêm skills từ callstackincubator

agent-device
callstackincubator
Tự động hóa tương tác ứng dụng iOS và Android với khám phá dựa trên ảnh chụp nhanh và phát lại dựa trên bộ chọn. Hỗ trợ trình mô phỏng/thiết bị iOS và trình giả lập/thiết bị Android với tự động hóa theo phiên, chế độ daemon từ xa đa người thuê và cách ly phạm vi thiết bị cho quy trình QA. Các lệnh cốt lõi: snapshot để khám phá UI với tham chiếu, press / fill / scroll để tương tác, open / close cho vòng đời ứng dụng, install / reinstall để triển khai nhị phân. Bao gồm các tiện ích cho ghi nhật ký, kiểm tra mạng,...
official
dogfood
callstackincubator
Khám phá và kiểm tra một ứng dụng di động trên iOS/Android một cách có hệ thống với agent-device để tìm lỗi, vấn đề UX và các vấn đề khác. Sử dụng khi được yêu cầu dogfood, QA,…
official
react-devtools
callstackincubator
Kiểm tra và lập hồ sơ cây thành phần React Native từ agent-device. Sử dụng cho hiệu suất, lập hồ sơ, props, state, hooks, nguyên nhân render, chậm… của React Native.
official
react-devtools
callstackincubator
CLI React DevTools dành cho các tác nhân AI. Sử dụng khi người dùng yêu cầu bạn gỡ lỗi ứng dụng React hoặc React Native trong thời gian chạy, kiểm tra props/state/hooks của component, chẩn đoán…
official
github
callstackincubator
Tự động hóa quy trình làm việc GitHub qua gh CLI cho pull request, stacked PR và quản lý kho lưu trữ. Cung cấp quy trình hợp nhất stacked PR: squash-merge PR đầu tiên, sau đó rebase và cập nhật nhánh cơ sở cho từng PR tiếp theo trong chuỗi. Bao gồm phát hiện xung đột và lời nhắc giải quyết thủ công để ngăn lỗi im lặng trong quá trình hợp nhất nhiều PR. Bao gồm các thao tác gh CLI cốt lõi: tạo PR, kiểm tra trạng thái, hợp nhất squash/rebase và quản lý nhánh. Được tối ưu hóa cho việc sử dụng ít ngữ cảnh bằng cách dựa vào gh CLI...
official
github-actions
callstackincubator
Các mẫu quy trình GitHub Actions cho bản dựng đám mây trên trình giả lập iOS React Native và trình mô phỏng Android với các tạo phẩm có thể tải xuống. Sử dụng khi thiết lập bản dựng CI…
official
react-native-best-practices
callstackincubator
Tài liệu tham khảo tối ưu hóa hiệu suất có cấu trúc cho ứng dụng React Native, bao gồm FPS, kích thước bundle, TTI và bộ nhớ. Được tổ chức thành 9 hướng dẫn JavaScript/React (lập hồ sơ, danh sách, hoạt ảnh, bộ nhớ), 9 hướng dẫn tối ưu hóa gốc (Turbo Modules, luồng, lập hồ sơ) và 9 hướng dẫn đóng gói (tree shaking, code splitting, phân tích kích thước). Mỗi tài liệu tham khảo tuân theo định dạng kết hợp với các mẫu/lệnh nhanh, xếp hạng tác động (CRITICAL/HIGH/MEDIUM) và giải thích chuyên sâu kèm điều kiện tiên
official
react-native-brownfield-migration
callstackincubator
Cung cấp chiến lược áp dụng gia tăng để di chuyển ứng dụng iOS hoặc Android gốc sang React Native hoặc Expo bằng @callstack/react-native-brownfield cho giai đoạn đầu…
official