测试用打字稿编写的自定义反应挂钩
Testing a custom react hook written in typescript
我有一个用 TS 编写的自定义挂钩 useForm
,它工作得很好。用 @testing-library/react-hooks
测试它。测试通过。但是打字稿在一个特定的地方抱怨 - 见下文:
这是自定义挂钩:
import { useState } from 'react';
export function useForm<T>(initialValues: T) {
const [values, setValues] = useState(initialValues);
return {
values: values,
handleChange: (e: React.ChangeEvent<HTMLTextAreaElement | HTMLInputElement>) => {
setValues({
...values,
[e.target.name]: e.target.value,
});
},
reset: () => setValues(initialValues),
};
}
这是测试:
import { renderHook, act } from '@testing-library/react-hooks';
import { useForm } from './useForm';
test('should render useForm hook', () => {
const { result } = renderHook(() =>
useForm({
text: 'Note One',
}),
);
expect(result.current).toBeDefined();
expect(result.current.values.text).toBe('Note One');
act(() => {
result.current.handleChange({
target: { //complains here - Type '{ name: string; value: string; }' is not assignable to type 'EventTarget & (HTMLTextAreaElement | HTMLInputElement)'.
name: 'text',
value: 'Note Two',
},
});
});
expect(result.current.values.text).toBe('Note Two');
act(() => {
result.current.reset();
});
expect(result.current.values.text).toBe('Note One');
});
这是一个工作代码沙箱:https://codesandbox.io/s/flamboyant-curie-6i5moy?file=/src/useForm.test.tsx
我将使用 Event() constructor. And set the event target for this event by using Object.defineProperty(event, 'target', {value: {}, writable: false})
, also see this 创建一个 change
事件。然后使用类型转换 event as unknown as React.ChangeEvent<HTMLInputElement>
转换事件。
例如
import "@testing-library/react-hooks/lib/dom/pure";
import { renderHook, act } from "@testing-library/react-hooks";
import { useForm } from "./useForm";
test("should render useForm hook", () => {
const { result } = renderHook(() =>
useForm({
text: "Note One"
})
);
expect(result.current).toBeDefined();
expect(result.current.values.text).toBe("Note One");
const event = new Event("change");
Object.defineProperty(event, "target", {
value: {
name: "text",
value: "Note Two"
},
writable: false
});
act(() => {
result.current.handleChange(event as unknown as React.ChangeEvent<HTMLInputElement>);
});
expect(result.current.values.text).toBe("Note Two");
act(() => {
result.current.reset();
});
expect(result.current.values.text).toBe("Note One");
});
你可以简单地施放它:
act(() => {
result.current.handleChange({
target: {
name: 'text',
value: 'Note Two',
},
}) as React.ChangeEvent<HTMLTextAreaElement | HTMLInputElement>;
});
我有一个用 TS 编写的自定义挂钩 useForm
,它工作得很好。用 @testing-library/react-hooks
测试它。测试通过。但是打字稿在一个特定的地方抱怨 - 见下文:
这是自定义挂钩:
import { useState } from 'react';
export function useForm<T>(initialValues: T) {
const [values, setValues] = useState(initialValues);
return {
values: values,
handleChange: (e: React.ChangeEvent<HTMLTextAreaElement | HTMLInputElement>) => {
setValues({
...values,
[e.target.name]: e.target.value,
});
},
reset: () => setValues(initialValues),
};
}
这是测试:
import { renderHook, act } from '@testing-library/react-hooks';
import { useForm } from './useForm';
test('should render useForm hook', () => {
const { result } = renderHook(() =>
useForm({
text: 'Note One',
}),
);
expect(result.current).toBeDefined();
expect(result.current.values.text).toBe('Note One');
act(() => {
result.current.handleChange({
target: { //complains here - Type '{ name: string; value: string; }' is not assignable to type 'EventTarget & (HTMLTextAreaElement | HTMLInputElement)'.
name: 'text',
value: 'Note Two',
},
});
});
expect(result.current.values.text).toBe('Note Two');
act(() => {
result.current.reset();
});
expect(result.current.values.text).toBe('Note One');
});
这是一个工作代码沙箱:https://codesandbox.io/s/flamboyant-curie-6i5moy?file=/src/useForm.test.tsx
我将使用 Event() constructor. And set the event target for this event by using Object.defineProperty(event, 'target', {value: {}, writable: false})
, also see this change
事件。然后使用类型转换 event as unknown as React.ChangeEvent<HTMLInputElement>
转换事件。
例如
import "@testing-library/react-hooks/lib/dom/pure";
import { renderHook, act } from "@testing-library/react-hooks";
import { useForm } from "./useForm";
test("should render useForm hook", () => {
const { result } = renderHook(() =>
useForm({
text: "Note One"
})
);
expect(result.current).toBeDefined();
expect(result.current.values.text).toBe("Note One");
const event = new Event("change");
Object.defineProperty(event, "target", {
value: {
name: "text",
value: "Note Two"
},
writable: false
});
act(() => {
result.current.handleChange(event as unknown as React.ChangeEvent<HTMLInputElement>);
});
expect(result.current.values.text).toBe("Note Two");
act(() => {
result.current.reset();
});
expect(result.current.values.text).toBe("Note One");
});
你可以简单地施放它:
act(() => {
result.current.handleChange({
target: {
name: 'text',
value: 'Note Two',
},
}) as React.ChangeEvent<HTMLTextAreaElement | HTMLInputElement>;
});