重要前提
安装AI Skills的关键前提是:必须科学上网,且开启TUN模式,这一点至关重要,直接决定安装能否顺利完成,在此郑重提醒三遍:科学上网,科学上网,科学上网。查看完整安装教程 →
detox-mobile-test by dengineproblem/agents-monorepo
npx skills add https://github.com/dengineproblem/agents-monorepo --skill detox-mobile-test使用 Detox 进行 React Native 应用端到端测试的专家。
waitFor() 进行显式等待toBeVisible() 而非 toExist()beforeEach() 和 afterEach() 进行隔离describe() 进行分组{
"testRunner": {
"args": {
"$0": "jest",
"config": "e2e/jest.config.js"
},
"jest": {
"setupTimeout": 120000
}
},
"apps": {
"ios.debug": {
"type": "ios.app",
"binaryPath": "ios/build/Build/Products/Debug-iphonesimulator/MyApp.app",
"build": "xcodebuild -workspace ios/MyApp.xcworkspace -scheme MyApp -configuration Debug -sdk iphonesimulator -derivedDataPath ios/build"
},
"ios.release": {
"type": "ios.app",
"binaryPath": "ios/build/Build/Products/Release-iphonesimulator/MyApp.app",
"build": "xcodebuild -workspace ios/MyApp.xcworkspace -scheme MyApp -configuration Release -sdk iphonesimulator -derivedDataPath ios/build"
},
"android.debug": {
"type": "android.apk",
"binaryPath": "android/app/build/outputs/apk/debug/app-debug.apk",
"build": "cd android && ./gradlew assembleDebug assembleAndroidTest -DtestBuildType=debug"
},
"android.release": {
"type": "android.apk",
"binaryPath": "android/app/build/outputs/apk/release/app-release.apk",
"build": "cd android && ./gradlew assembleRelease assembleAndroidTest -DtestBuildType=release"
}
},
"devices": {
"simulator": {
"type": "ios.simulator",
"device": { "type": "iPhone 14" }
},
"emulator": {
"type": "android.emulator",
"device": { "avdName": "Pixel_4_API_30" }
}
},
"configurations": {
"ios.sim.debug": {
"device": "simulator",
"app": "ios.debug"
},
"ios.sim.release": {
"device": "simulator",
"app": "ios.release"
},
"android.emu.debug": {
"device": "emulator",
"app": "android.debug"
},
"android.emu.release": {
"device": "emulator",
"app": "android.release"
}
}
}
广告位招租
在这里展示您的产品或服务
触达数万 AI 开发者,精准高效
// e2e/jest.config.js
module.exports = {
rootDir: '..',
testMatch: ['<rootDir>/e2e/**/*.test.js'],
testTimeout: 120000,
maxWorkers: 1,
globalSetup: 'detox/runners/jest/globalSetup',
globalTeardown: 'detox/runners/jest/globalTeardown',
reporters: ['detox/runners/jest/reporter'],
testEnvironment: 'detox/runners/jest/testEnvironment',
verbose: true
};
describe('登录流程', () => {
beforeAll(async () => {
await device.launchApp();
});
beforeEach(async () => {
await device.reloadReactNative();
});
afterAll(async () => {
await device.terminateApp();
});
it('应使用有效凭据登录', async () => {
// 准备
const email = 'test@example.com';
const password = 'password123';
// 执行
await element(by.id('email-input')).typeText(email);
await element(by.id('password-input')).typeText(password);
await element(by.id('login-button')).tap();
// 断言
await expect(element(by.id('home-screen'))).toBeVisible();
});
it('应为无效凭据显示错误', async () => {
// 准备
const email = 'wrong@example.com';
const password = 'wrongpassword';
// 执行
await element(by.id('email-input')).typeText(email);
await element(by.id('password-input')).typeText(password);
await element(by.id('login-button')).tap();
// 断言
await expect(element(by.id('error-message'))).toBeVisible();
await expect(element(by.text('Invalid credentials'))).toBeVisible();
});
});
// 通过 testID
element(by.id('submit-button'))
// 通过文本
element(by.text('Submit'))
// 通过标签(无障碍功能)
element(by.label('Submit form'))
// 通过类型
element(by.type('RCTTextInput'))
// 通过特性(iOS)
element(by.traits(['button']))
// 组合匹配器
element(by.id('item').withAncestor(by.id('list')))
element(by.id('item').withDescendant(by.text('Title')))
// 用于多个匹配项的索引
element(by.id('list-item')).atIndex(0)
// 点击
await element(by.id('button')).tap();
await element(by.id('button')).multiTap(2);
await element(by.id('button')).longPress();
await element(by.id('button')).longPress(2000); // 2 秒
// 文本输入
await element(by.id('input')).typeText('Hello');
await element(by.id('input')).replaceText('New text');
await element(by.id('input')).clearText();
// 滚动
await element(by.id('scrollView')).scroll(200, 'down');
await element(by.id('scrollView')).scroll(200, 'up');
await element(by.id('scrollView')).scrollTo('bottom');
await element(by.id('scrollView')).scrollTo('top');
// 滚动直到可见
await waitFor(element(by.id('item')))
.toBeVisible()
.whileElement(by.id('scrollView'))
.scroll(200, 'down');
// 滑动
await element(by.id('card')).swipe('left');
await element(by.id('card')).swipe('right', 'fast', 0.9);
// 捏合
await element(by.id('map')).pinch(1.5); // 放大
await element(by.id('map')).pinch(0.5); // 缩小
// 可见性
await expect(element(by.id('view'))).toBeVisible();
await expect(element(by.id('view'))).not.toBeVisible();
await expect(element(by.id('view'))).toExist();
await expect(element(by.id('view'))).not.toExist();
// 焦点
await expect(element(by.id('input'))).toBeFocused();
// 文本
await expect(element(by.id('label'))).toHaveText('Hello');
await expect(element(by.id('input'))).toHaveValue('input value');
// 开关状态
await expect(element(by.id('switch'))).toHaveToggleValue(true);
// 滑块
await expect(element(by.id('slider'))).toHaveSliderPosition(0.5);
// ID
await expect(element(by.id('view'))).toHaveId('view');
// 标签
await expect(element(by.id('button'))).toHaveLabel('Submit');
// 等待元素可见
await waitFor(element(by.id('loading')))
.not.toBeVisible()
.withTimeout(10000);
// 等待元素存在
await waitFor(element(by.id('data')))
.toExist()
.withTimeout(5000);
// 边滚动边等待
await waitFor(element(by.id('item-50')))
.toBeVisible()
.whileElement(by.id('list'))
.scroll(100, 'down');
// 自定义轮询
await waitFor(element(by.id('result')))
.toHaveText('Success')
.withTimeout(30000);
// e2e/pages/LoginPage.js
class LoginPage {
get emailInput() {
return element(by.id('email-input'));
}
get passwordInput() {
return element(by.id('password-input'));
}
get loginButton() {
return element(by.id('login-button'));
}
get errorMessage() {
return element(by.id('error-message'));
}
async login(email, password) {
await this.emailInput.typeText(email);
await this.passwordInput.typeText(password);
await this.loginButton.tap();
}
async assertErrorVisible(message) {
await expect(this.errorMessage).toBeVisible();
if (message) {
await expect(element(by.text(message))).toBeVisible();
}
}
}
module.exports = new LoginPage();
// e2e/tests/login.test.js
const LoginPage = require('../pages/LoginPage');
const HomePage = require('../pages/HomePage');
describe('登录', () => {
it('应成功登录', async () => {
await LoginPage.login('user@test.com', 'password123');
await expect(HomePage.welcomeMessage).toBeVisible();
});
});
// 在测试中
await device.launchApp({
launchArgs: {
detoxPrintBusyIdleResources: 'YES'
}
});
// 截图
await device.takeScreenshot('login-screen');
// 失败时(在 jest 设置中)
afterEach(async () => {
if (jasmine.currentTest.failedExpectations.length > 0) {
await device.takeScreenshot(`failed-${jasmine.currentTest.fullName}`);
}
});
// 获取元素属性
const attributes = await element(by.id('button')).getAttributes();
console.log(attributes);
// { text: 'Submit', visible: true, enabled: true, ... }
// 对于非 React Native 屏幕(WebViews 等)
await device.disableSynchronization();
await element(by.id('webview-button')).tap();
await device.enableSynchronization();
// iOS
await device.launchApp({
permissions: {
notifications: 'YES',
camera: 'YES',
photos: 'YES',
location: 'always'
}
});
// Android - 运行时处理
await element(by.text('Allow')).tap();
// 关闭键盘
await element(by.id('input')).typeText('text\n');
// 或者
await device.pressBack(); // Android
// 避免键盘遮挡
await element(by.id('input')).tap();
await element(by.id('input')).typeText('text');
await element(by.id('submit')).tap();
name: E2E Tests
on:
push:
branches: [main]
pull_request:
branches: [main]
jobs:
ios-e2e:
runs-on: macos-latest
steps:
- uses: actions/checkout@v4
- name: Setup Node
uses: actions/setup-node@v4
with:
node-version: '18'
cache: 'npm'
- name: Install dependencies
run: npm ci
- name: Install pods
run: cd ios && pod install
- name: Build app
run: npx detox build --configuration ios.sim.release
- name: Run tests
run: npx detox test --configuration ios.sim.release --cleanup
- name: Upload artifacts
if: failure()
uses: actions/upload-artifact@v3
with:
name: detox-artifacts
path: artifacts/
android-e2e:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Setup Node
uses: actions/setup-node@v4
with:
node-version: '18'
- name: Setup Java
uses: actions/setup-java@v3
with:
distribution: 'zulu'
java-version: '11'
- name: Install dependencies
run: npm ci
- name: Build app
run: npx detox build --configuration android.emu.release
- name: Start emulator
uses: reactivecircus/android-emulator-runner@v2
with:
api-level: 30
target: google_apis
script: npx detox test --configuration android.emu.release --cleanup
// 使用 reloadReactNative 而非 launchApp
beforeEach(async () => {
await device.reloadReactNative(); // 快速
// await device.launchApp({ newInstance: true }); // 慢
});
// 仅在失败时录制视频
// 在 detoxrc.json 中
{
"artifacts": {
"plugins": {
"video": {
"enabled": true,
"keepOnlyFailedTestsArtifacts": true
}
}
}
}
// 测试分片以实现并行执行
// jest.config.js
module.exports = {
maxWorkers: process.env.CI ? 2 : 1,
// ...
};
每周安装数
59
代码仓库
GitHub Stars
3
首次出现
2026年1月29日
安全审计
安装于
github-copilot59
opencode58
codex58
amp58
kimi-cli58
gemini-cli58
Эксперт по E2E тестированию React Native приложений с Detox.
waitFor() для явных ожиданийtoBeVisible() вместо toExist() для стабильностиbeforeEach() и afterEach()describe() для группировки{
"testRunner": {
"args": {
"$0": "jest",
"config": "e2e/jest.config.js"
},
"jest": {
"setupTimeout": 120000
}
},
"apps": {
"ios.debug": {
"type": "ios.app",
"binaryPath": "ios/build/Build/Products/Debug-iphonesimulator/MyApp.app",
"build": "xcodebuild -workspace ios/MyApp.xcworkspace -scheme MyApp -configuration Debug -sdk iphonesimulator -derivedDataPath ios/build"
},
"ios.release": {
"type": "ios.app",
"binaryPath": "ios/build/Build/Products/Release-iphonesimulator/MyApp.app",
"build": "xcodebuild -workspace ios/MyApp.xcworkspace -scheme MyApp -configuration Release -sdk iphonesimulator -derivedDataPath ios/build"
},
"android.debug": {
"type": "android.apk",
"binaryPath": "android/app/build/outputs/apk/debug/app-debug.apk",
"build": "cd android && ./gradlew assembleDebug assembleAndroidTest -DtestBuildType=debug"
},
"android.release": {
"type": "android.apk",
"binaryPath": "android/app/build/outputs/apk/release/app-release.apk",
"build": "cd android && ./gradlew assembleRelease assembleAndroidTest -DtestBuildType=release"
}
},
"devices": {
"simulator": {
"type": "ios.simulator",
"device": { "type": "iPhone 14" }
},
"emulator": {
"type": "android.emulator",
"device": { "avdName": "Pixel_4_API_30" }
}
},
"configurations": {
"ios.sim.debug": {
"device": "simulator",
"app": "ios.debug"
},
"ios.sim.release": {
"device": "simulator",
"app": "ios.release"
},
"android.emu.debug": {
"device": "emulator",
"app": "android.debug"
},
"android.emu.release": {
"device": "emulator",
"app": "android.release"
}
}
}
// e2e/jest.config.js
module.exports = {
rootDir: '..',
testMatch: ['<rootDir>/e2e/**/*.test.js'],
testTimeout: 120000,
maxWorkers: 1,
globalSetup: 'detox/runners/jest/globalSetup',
globalTeardown: 'detox/runners/jest/globalTeardown',
reporters: ['detox/runners/jest/reporter'],
testEnvironment: 'detox/runners/jest/testEnvironment',
verbose: true
};
describe('Login Flow', () => {
beforeAll(async () => {
await device.launchApp();
});
beforeEach(async () => {
await device.reloadReactNative();
});
afterAll(async () => {
await device.terminateApp();
});
it('should login with valid credentials', async () => {
// Arrange
const email = 'test@example.com';
const password = 'password123';
// Act
await element(by.id('email-input')).typeText(email);
await element(by.id('password-input')).typeText(password);
await element(by.id('login-button')).tap();
// Assert
await expect(element(by.id('home-screen'))).toBeVisible();
});
it('should show error for invalid credentials', async () => {
// Arrange
const email = 'wrong@example.com';
const password = 'wrongpassword';
// Act
await element(by.id('email-input')).typeText(email);
await element(by.id('password-input')).typeText(password);
await element(by.id('login-button')).tap();
// Assert
await expect(element(by.id('error-message'))).toBeVisible();
await expect(element(by.text('Invalid credentials'))).toBeVisible();
});
});
// By testID
element(by.id('submit-button'))
// By text
element(by.text('Submit'))
// By label (accessibility)
element(by.label('Submit form'))
// By type
element(by.type('RCTTextInput'))
// By traits (iOS)
element(by.traits(['button']))
// Combining matchers
element(by.id('item').withAncestor(by.id('list')))
element(by.id('item').withDescendant(by.text('Title')))
// Index for multiple matches
element(by.id('list-item')).atIndex(0)
// Tap
await element(by.id('button')).tap();
await element(by.id('button')).multiTap(2);
await element(by.id('button')).longPress();
await element(by.id('button')).longPress(2000); // 2 seconds
// Text input
await element(by.id('input')).typeText('Hello');
await element(by.id('input')).replaceText('New text');
await element(by.id('input')).clearText();
// Scroll
await element(by.id('scrollView')).scroll(200, 'down');
await element(by.id('scrollView')).scroll(200, 'up');
await element(by.id('scrollView')).scrollTo('bottom');
await element(by.id('scrollView')).scrollTo('top');
// Scroll until visible
await waitFor(element(by.id('item')))
.toBeVisible()
.whileElement(by.id('scrollView'))
.scroll(200, 'down');
// Swipe
await element(by.id('card')).swipe('left');
await element(by.id('card')).swipe('right', 'fast', 0.9);
// Pinch
await element(by.id('map')).pinch(1.5); // zoom in
await element(by.id('map')).pinch(0.5); // zoom out
// Visibility
await expect(element(by.id('view'))).toBeVisible();
await expect(element(by.id('view'))).not.toBeVisible();
await expect(element(by.id('view'))).toExist();
await expect(element(by.id('view'))).not.toExist();
// Focus
await expect(element(by.id('input'))).toBeFocused();
// Text
await expect(element(by.id('label'))).toHaveText('Hello');
await expect(element(by.id('input'))).toHaveValue('input value');
// Toggle state
await expect(element(by.id('switch'))).toHaveToggleValue(true);
// Slider
await expect(element(by.id('slider'))).toHaveSliderPosition(0.5);
// ID
await expect(element(by.id('view'))).toHaveId('view');
// Label
await expect(element(by.id('button'))).toHaveLabel('Submit');
// Wait for element to be visible
await waitFor(element(by.id('loading')))
.not.toBeVisible()
.withTimeout(10000);
// Wait for element to exist
await waitFor(element(by.id('data')))
.toExist()
.withTimeout(5000);
// Wait while scrolling
await waitFor(element(by.id('item-50')))
.toBeVisible()
.whileElement(by.id('list'))
.scroll(100, 'down');
// Custom polling
await waitFor(element(by.id('result')))
.toHaveText('Success')
.withTimeout(30000);
// e2e/pages/LoginPage.js
class LoginPage {
get emailInput() {
return element(by.id('email-input'));
}
get passwordInput() {
return element(by.id('password-input'));
}
get loginButton() {
return element(by.id('login-button'));
}
get errorMessage() {
return element(by.id('error-message'));
}
async login(email, password) {
await this.emailInput.typeText(email);
await this.passwordInput.typeText(password);
await this.loginButton.tap();
}
async assertErrorVisible(message) {
await expect(this.errorMessage).toBeVisible();
if (message) {
await expect(element(by.text(message))).toBeVisible();
}
}
}
module.exports = new LoginPage();
// e2e/tests/login.test.js
const LoginPage = require('../pages/LoginPage');
const HomePage = require('../pages/HomePage');
describe('Login', () => {
it('should login successfully', async () => {
await LoginPage.login('user@test.com', 'password123');
await expect(HomePage.welcomeMessage).toBeVisible();
});
});
// In test
await device.launchApp({
launchArgs: {
detoxPrintBusyIdleResources: 'YES'
}
});
// Take screenshot
await device.takeScreenshot('login-screen');
// On failure (in jest setup)
afterEach(async () => {
if (jasmine.currentTest.failedExpectations.length > 0) {
await device.takeScreenshot(`failed-${jasmine.currentTest.fullName}`);
}
});
// Get element attributes
const attributes = await element(by.id('button')).getAttributes();
console.log(attributes);
// { text: 'Submit', visible: true, enabled: true, ... }
// For non-React Native screens (WebViews, etc.)
await device.disableSynchronization();
await element(by.id('webview-button')).tap();
await device.enableSynchronization();
// iOS
await device.launchApp({
permissions: {
notifications: 'YES',
camera: 'YES',
photos: 'YES',
location: 'always'
}
});
// Android - handle at runtime
await element(by.text('Allow')).tap();
// Dismiss keyboard
await element(by.id('input')).typeText('text\n');
// or
await device.pressBack(); // Android
// Avoid keyboard overlap
await element(by.id('input')).tap();
await element(by.id('input')).typeText('text');
await element(by.id('submit')).tap();
name: E2E Tests
on:
push:
branches: [main]
pull_request:
branches: [main]
jobs:
ios-e2e:
runs-on: macos-latest
steps:
- uses: actions/checkout@v4
- name: Setup Node
uses: actions/setup-node@v4
with:
node-version: '18'
cache: 'npm'
- name: Install dependencies
run: npm ci
- name: Install pods
run: cd ios && pod install
- name: Build app
run: npx detox build --configuration ios.sim.release
- name: Run tests
run: npx detox test --configuration ios.sim.release --cleanup
- name: Upload artifacts
if: failure()
uses: actions/upload-artifact@v3
with:
name: detox-artifacts
path: artifacts/
android-e2e:
runs-on: ubuntu-latest
steps:
- uses: actions/checkout@v4
- name: Setup Node
uses: actions/setup-node@v4
with:
node-version: '18'
- name: Setup Java
uses: actions/setup-java@v3
with:
distribution: 'zulu'
java-version: '11'
- name: Install dependencies
run: npm ci
- name: Build app
run: npx detox build --configuration android.emu.release
- name: Start emulator
uses: reactivecircus/android-emulator-runner@v2
with:
api-level: 30
target: google_apis
script: npx detox test --configuration android.emu.release --cleanup
// Use reloadReactNative instead of launchApp
beforeEach(async () => {
await device.reloadReactNative(); // Fast
// await device.launchApp({ newInstance: true }); // Slow
});
// Record videos only on failure
// In detoxrc.json
{
"artifacts": {
"plugins": {
"video": {
"enabled": true,
"keepOnlyFailedTestsArtifacts": true
}
}
}
}
// Test sharding for parallel execution
// jest.config.js
module.exports = {
maxWorkers: process.env.CI ? 2 : 1,
// ...
};
Weekly Installs
59
Repository
GitHub Stars
3
First Seen
Jan 29, 2026
Security Audits
Gen Agent Trust HubPassSocketPassSnykPass
Installed on
github-copilot59
opencode58
codex58
amp58
kimi-cli58
gemini-cli58
Vue 3 调试指南:解决响应式、计算属性与监听器常见错误
12,200 周安装