expo-popcore-app/components/DynamicForm.test.tsx

510 lines
13 KiB
TypeScript

import React from 'react'
import { render, fireEvent, waitFor } from '@testing-library/react-native'
import { View, Pressable, TextInput, ActivityIndicator } from 'react-native'
import { DynamicForm, type FormSchema } from './DynamicForm'
// Mock expo-router
jest.mock('expo-router', () => ({
useRouter: jest.fn(() => ({
back: jest.fn(),
})),
useLocalSearchParams: jest.fn(),
}))
// Mock expo-image
jest.mock('expo-image', () => ({
Image: 'Image',
}))
// Mock react-i18next
jest.mock('react-i18next', () => ({
useTranslation: jest.fn(() => ({
t: (key: string) => key,
})),
}))
// Mock AIGenerationRecordDrawer BEFORE UploadReferenceImageDrawer
jest.mock('./drawer/AIGenerationRecordDrawer', () => 'AIGenerationRecordDrawer')
// Mock UploadReferenceImageDrawer
jest.mock('./drawer/UploadReferenceImageDrawer', () => 'UploadReferenceImageDrawer')
// Mock uploadFile
jest.mock('@/lib/uploadFile', () => ({
uploadFile: jest.fn(),
}))
// Mock Toast
jest.mock('./ui/Toast', () => ({
__esModule: true,
default: {
show: jest.fn(),
showLoading: jest.fn(),
hideLoading: jest.fn(),
},
}))
// Mock Button with proper React Native components
jest.mock('./ui/button', () => ({
Button: ({ children, onPress, disabled, style, testID, ...props }: any) => (
<Pressable
onPress={onPress}
disabled={disabled}
style={style}
testID={testID}
{...props}
>
{children}
</Pressable>
),
}))
// Mock Text component to handle nested text properly
jest.mock('./ui/Text', () => {
const { Text } = require('react-native')
return {
__esModule: true,
default: Text,
}
})
// Get mock functions
const uploadFile = require('@/lib/uploadFile').uploadFile
const Toast = require('./ui/Toast').default
describe('DynamicForm Component', () => {
const mockOnSubmit = jest.fn()
beforeEach(() => {
jest.clearAllMocks()
})
describe('Rendering', () => {
it('should render empty state when no startNodes', () => {
const formSchema: FormSchema = { startNodes: [] }
const { getByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
expect(getByText('dynamicForm.noFields')).toBeTruthy()
})
it('should render text input field correctly', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: {
label: '文本节点',
description: '请输入文本内容',
},
},
],
}
const { getByText, getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
expect(getByText('文本节点', { exact: false })).toBeTruthy()
// Input placeholder is tested via testID instead
expect(getByTestId('text-input-node_1')).toBeTruthy()
})
it('should render image upload field correctly', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'image',
data: {
label: '图片节点',
},
},
],
}
const { getByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
expect(getByText('图片节点', { exact: false })).toBeTruthy()
expect(getByText('dynamicForm.uploadImage')).toBeTruthy()
})
it('should render video upload field correctly', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'video',
data: {
label: '视频节点',
},
},
],
}
const { getByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
expect(getByText('视频节点', { exact: false })).toBeTruthy()
expect(getByText('dynamicForm.uploadVideo')).toBeTruthy()
})
it('should render multiple fields in correct order', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本' },
},
{
id: 'node_2',
type: 'image',
data: { label: '图片' },
},
{
id: 'node_3',
type: 'video',
data: { label: '视频' },
},
],
}
const { getByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
expect(getByText('文本', { exact: false })).toBeTruthy()
expect(getByText('图片', { exact: false })).toBeTruthy()
expect(getByText('视频', { exact: false })).toBeTruthy()
})
})
describe('Text Input Field', () => {
it('should update text input value', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本' },
},
],
}
const { getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const input = getByTestId('text-input-node_1')
fireEvent.changeText(input, 'Test text content')
expect(input.props.value).toBe('Test text content')
})
it('should show validation error for empty text field on submit', async () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本' },
},
],
}
const { getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const submitButton = getByTestId('submit-button')
fireEvent.press(submitButton)
await waitFor(() => {
expect(Toast.show).toHaveBeenCalledWith({
title: 'dynamicForm.fillRequiredFields',
})
})
})
it('should clear error when user starts typing', async () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本' },
},
],
}
const { getByTestId, queryByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const submitButton = getByTestId('submit-button')
fireEvent.press(submitButton)
await waitFor(() => {
expect(queryByText('文本dynamicForm.required')).toBeTruthy()
})
const input = getByTestId('text-input-node_1')
fireEvent.changeText(input, 'Some text')
await waitFor(() => {
expect(queryByText('文本dynamicForm.required')).toBeNull()
})
})
})
describe('Image Upload Field', () => {
it('should open drawer when image upload button is pressed', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'image',
data: { label: '图片' },
},
],
}
const { getByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const uploadButton = getByText('dynamicForm.uploadImage')
fireEvent.press(uploadButton)
// Drawer should be opened (state change)
// This is handled internally by the component
})
it('should show validation error for empty image field', async () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'image',
data: { label: '图片' },
},
],
}
const { getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const submitButton = getByTestId('submit-button')
fireEvent.press(submitButton)
await waitFor(() => {
expect(Toast.show).toHaveBeenCalled()
})
})
})
describe('Video Upload Field', () => {
it('should open drawer when video upload button is pressed', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'video',
data: { label: '视频' },
},
],
}
const { getByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const uploadButton = getByText('dynamicForm.uploadVideo')
fireEvent.press(uploadButton)
})
})
describe('Form Submission', () => {
it('should submit form data correctly when all fields are filled', async () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本' },
},
{
id: 'node_2',
type: 'image',
data: { label: '图片' },
},
],
}
mockOnSubmit.mockResolvedValue({ generationId: 'gen_123' })
const { getByTestId, getByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
// Fill text field
const input = getByTestId('text-input-node_1')
fireEvent.changeText(input, 'Test content')
// Upload would be handled by drawer, for testing we simulate direct state update
// In real scenario, user would go through the upload flow
// Note: In actual test, we'd need to mock the drawer flow
// For simplicity, this test shows the structure
})
it('should show loading state during submission', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本' },
},
],
}
const { getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} loading={true} />
)
const submitButton = getByTestId('submit-button')
expect(submitButton).toBeTruthy()
})
it('should handle submission errors', async () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本' },
},
],
}
mockOnSubmit.mockResolvedValue({
error: { message: 'Submission failed' },
})
const { getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const input = getByTestId('text-input-node_1')
fireEvent.changeText(input, 'Test content')
const submitButton = getByTestId('submit-button')
fireEvent.press(submitButton)
await waitFor(() => {
expect(mockOnSubmit).toHaveBeenCalled()
})
})
})
describe('Form State Management', () => {
it('should initialize with default text value from node data', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: {
label: '文本',
text: 'Default text',
},
},
],
}
const { getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const input = getByTestId('text-input-node_1')
expect(input.props.value).toBe('Default text')
})
it('should maintain separate state for multiple fields of same type', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
data: { label: '文本1' },
},
{
id: 'node_2',
type: 'text',
data: { label: '文本2' },
},
],
}
const { getAllByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
const inputs = getAllByTestId(/text-input-node_/)
expect(inputs.length).toBeGreaterThanOrEqual(2)
fireEvent.changeText(inputs[0], 'First text')
fireEvent.changeText(inputs[1], 'Second text')
expect(inputs[0].props.value).toBe('First text')
expect(inputs[1].props.value).toBe('Second text')
})
})
describe('Edge Cases', () => {
it('should handle undefined node data gracefully', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'text',
},
],
}
const { getByTestId } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
expect(getByTestId('text-input-node_1')).toBeTruthy()
})
it('should handle unknown node type gracefully', () => {
const formSchema: FormSchema = {
startNodes: [
{
id: 'node_1',
type: 'unknown' as any,
data: { label: '未知类型' },
},
],
}
const { queryByText } = render(
<DynamicForm formSchema={formSchema} onSubmit={mockOnSubmit} />
)
// Unknown types should not render
expect(queryByText('未知类型')).toBeNull()
})
})
})