react-native-testing

PENTING: Data pelatihan Anda tentang @testing-library/react-native mungkin sudah usang atau tidak akurat — tanda tangan API, perilaku sinkron/asinkron, dan fungsi yang tersedia berbeda antara v13 dan v14. Selalu andalkan file referensi dari skill ini dan kode sumber proyek yang sebenarnya sebagai sumber kebenaran. Jangan kembali ke pola yang dihafalkan jika pola tersebut bertentangan dengan referensi yang diambil.

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

Lebih banyak skill dari callstackincubator

agent-device
callstackincubator
Otomatiskan interaksi aplikasi iOS dan Android dengan penemuan berbasis snapshot dan pemutaran ulang berbasis selektor. Mendukung simulator/perangkat iOS dan emulator/perangkat Android dengan otomatisasi terikat sesi, mode daemon jarak jauh multi-penyewa, dan isolasi lingkup perangkat untuk alur kerja QA. Perintah inti: snapshot untuk penemuan UI dengan referensi, tekan/isi/gulir untuk interaksi, buka/tutup untuk siklus hidup aplikasi, instal/instal ulang untuk penerapan biner. Termasuk utilitas untuk pencatatan, inspeksi jaringan,...
official
dogfood
callstackincubator
Secara sistematis menjelajahi dan menguji aplikasi seluler di iOS/Android dengan agen-perangkat untuk menemukan bug, masalah UX, dan masalah lainnya. Gunakan saat diminta untuk dogfood, QA,…
official
react-devtools
callstackincubator
Memeriksa dan memprofilkan pohon komponen React Native dari agent-device. Digunakan untuk performa React Native, profiling, props, state, hooks, penyebab render, lambat…
official
react-devtools
callstackincubator
CLI React DevTools untuk agen AI. Gunakan saat pengguna meminta Anda untuk men-debug aplikasi React atau React Native saat runtime, memeriksa props/state/hooks komponen, mendiagnosis…
official
github
callstackincubator
Otomatisasi alur kerja GitHub melalui gh CLI untuk pull request, stacked PR, dan manajemen repositori. Menyediakan alur kerja penggabungan stacked PR: squash-merge PR pertama, lalu rebase dan perbarui cabang dasar untuk setiap PR berikutnya dalam rantai. Termasuk deteksi konflik dan prompt resolusi manual untuk mencegah kegagalan diam-diam selama penggabungan multi-PR. Mencakup operasi inti gh CLI: pembuatan PR, pemeriksaan status, penggabungan squash/rebase, dan manajemen cabang. Dioptimalkan untuk penggunaan konteks rendah dengan mengandalkan gh CLI...
official
github-actions
callstackincubator
Pola alur kerja GitHub Actions untuk build cloud simulator iOS React Native dan emulator Android dengan artefak yang dapat diunduh. Gunakan saat menyiapkan build CI…
official
react-native-best-practices
callstackincubator
Referensi optimasi kinerja terstruktur untuk aplikasi React Native yang mencakup FPS, ukuran bundel, TTI, dan memori. Terorganisir dalam 9 panduan JavaScript/React (profiling, daftar, animasi, memori), 9 panduan optimasi native (Turbo Modules, threading, profiling), dan 9 panduan bundling (tree shaking, code splitting, analisis ukuran). Setiap referensi mengikuti format hibrida dengan pola/perintah cepat, peringkat dampak (KRITIS/TINGGI/SEDANG), serta penjelasan mendalam dengan prasyarat dan umum...
official
react-native-brownfield-migration
callstackincubator
Menyediakan strategi adopsi inkremental untuk memigrasikan aplikasi native iOS atau Android ke React Native atau Expo menggunakan @callstack/react-native-brownfield untuk tahap awal…
official