react-native-testing

作成者: callstackincubator

重要: @testing-library/react-native に関するあなたのトレーニングデータは古いか不正確な可能性があります — v13とv14ではAPIシグネチャ、同期/非同期動作、利用可能な関数が異なります。常にこのスキルの参照ファイルとプロジェクトの実際のソースコードを真実の情報源として信頼してください。取得した参照と矛盾する場合、記憶されたパターンに頼らないでください。

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

callstackincubatorのその他のスキル

agent-device
callstackincubator
iOSおよびAndroidアプリの操作を自動化し、スナップショットベースのUI探索とセレクタ駆動のリプレイを実現。iOSシミュレータ/実機およびAndroidエミュレータ/実機に対応し、セッション単位の自動化、マルチテナント対応のリモートデーモンモード、QAワークフロー向けのデバイススコープ分離を提供。コアコマンド:UI探索用のスナップショット(参照付き)、操作(press/fill/scroll)、アプリライフサイクル(open/close)、バイナリデプロイ(install/reinstall)。ログ、ネットワークインスペクションなどのユーティリティを含む。
official
dogfood
callstackincubator
iOS/Androidのモバイルアプリをエージェントデバイスで体系的に探索・テストし、バグやUXの問題、その他の課題を発見します。dogfoodやQAを依頼された際に使用します。
official
react-devtools
callstackincubator
エージェントデバイスからReact Nativeコンポーネントツリーを検査・プロファイリングします。React Nativeのパフォーマンス、プロファイリング、props、state、フック、レンダリング原因、低速処理などに使用します。
official
react-devtools
callstackincubator
React DevTools CLI for AIエージェント。ユーザーがReactまたはReact Nativeアプリのランタイムデバッグ、コンポーネントのprops/state/hooksの検査、診断を依頼した場合に使用します…
official
github
callstackincubator
gh CLIを使用したGitHubワークフローの自動化:プルリクエスト、スタックPR、リポジトリ管理に対応。スタックPRマージワークフローを提供:最初のPRをスカッシュマージし、チェーン内の後続の各PRに対してリベースとベースブランチの更新を実行。マルチPRマージ中のサイレント障害を防ぐため、競合検出と手動解決プロンプトを含む。gh CLIの主要操作をカバー:PR作成、ステータスチェック、スカッシュ/リベースマージ、ブランチ管理。gh CLIに依存することで低コンテキスト使用に最適化。
official
github-actions
callstackincubator
GitHub Actionsのワークフローパターン。React NativeのiOSシミュレーターおよびAndroidエミュレーターのクラウドビルドに対応し、ダウンロード可能なアーティファクトを生成します。CIビルドを設定する際に使用します。
official
react-native-best-practices
callstackincubator
React Nativeアプリ向けの構造化されたパフォーマンス最適化リファレンス。FPS、バンドルサイズ、TTI、メモリをカバー。9つのJavaScript/Reactガイド(プロファイリング、リスト、アニメーション、メモリ)、9つのネイティブ最適化ガイド(Turbo Modules、スレッド処理、プロファイリング)、9つのバンドリングガイド(ツリーシェイキング、コード分割、サイズ分析)で構成。各リファレンスは、クイックパターン/コマンド、影響度評価(CRITICAL/HIGH/MEDIUM)、前提条件や一般的な...を含む詳細解説を備えたハイブリッド形式に従っています。
official
react-native-brownfield-migration
callstackincubator
ネイティブiOSまたはAndroidアプリをReact NativeやExpoに移行するための段階的導入戦略を提供し、初期段階では@callstack/react-native-brownfieldを使用します…
official