fix(ui): wrong form initial values

This commit is contained in:
Fu Diwei 2025-01-05 00:47:27 +08:00
parent 61843a4997
commit ddb6a88392
12 changed files with 38 additions and 29 deletions

View File

@ -1,6 +1,5 @@
import { forwardRef, useEffect, useImperativeHandle, useMemo, useState } from "react"; import { forwardRef, useEffect, useImperativeHandle, useMemo, useState } from "react";
import { useTranslation } from "react-i18next"; import { useTranslation } from "react-i18next";
import { useCreation } from "ahooks";
import { Form, type FormInstance, Input } from "antd"; import { Form, type FormInstance, Input } from "antd";
import { createSchemaFieldRule } from "antd-zod"; import { createSchemaFieldRule } from "antd-zod";
import { z } from "zod"; import { z } from "zod";
@ -8,7 +7,7 @@ import { z } from "zod";
import AccessProviderSelect from "@/components/provider/AccessProviderSelect"; import AccessProviderSelect from "@/components/provider/AccessProviderSelect";
import { type AccessModel } from "@/domain/access"; import { type AccessModel } from "@/domain/access";
import { ACCESS_PROVIDERS } from "@/domain/provider"; import { ACCESS_PROVIDERS } from "@/domain/provider";
import { useAntdForm } from "@/hooks"; import { useAntdForm, useAntdFormName } from "@/hooks";
import AccessEditFormACMEHttpReqConfig from "./AccessEditFormACMEHttpReqConfig"; import AccessEditFormACMEHttpReqConfig from "./AccessEditFormACMEHttpReqConfig";
import AccessEditFormAWSConfig from "./AccessEditFormAWSConfig"; import AccessEditFormAWSConfig from "./AccessEditFormAWSConfig";
@ -71,7 +70,7 @@ const AccessEditForm = forwardRef<AccessEditFormInstance, AccessEditFormProps>((
}, [initialValues?.provider]); }, [initialValues?.provider]);
const [configFormInst] = Form.useForm(); const [configFormInst] = Form.useForm();
const configFormName = useCreation(() => `accessEditForm_config${Math.random().toString(36).substring(2, 10)}${new Date().getTime()}`, []); const configFormName = useAntdFormName({ form: configFormInst, name: "accessEditConfigForm" });
const configFormComponent = useMemo(() => { const configFormComponent = useMemo(() => {
/* /*
ASCII ASCII

View File

@ -1,9 +1,8 @@
import { forwardRef, useImperativeHandle, useMemo } from "react"; import { forwardRef, useImperativeHandle, useMemo } from "react";
import { useCreation } from "ahooks";
import { Form, type FormInstance } from "antd"; import { Form, type FormInstance } from "antd";
import { NOTIFY_CHANNELS, type NotifyChannelsSettingsContent } from "@/domain/settings"; import { NOTIFY_CHANNELS, type NotifyChannelsSettingsContent } from "@/domain/settings";
import { useAntdForm } from "@/hooks"; import { useAntdForm, useAntdFormName } from "@/hooks";
import NotifyChannelEditFormBarkFields from "./NotifyChannelEditFormBarkFields"; import NotifyChannelEditFormBarkFields from "./NotifyChannelEditFormBarkFields";
import NotifyChannelEditFormDingTalkFields from "./NotifyChannelEditFormDingTalkFields"; import NotifyChannelEditFormDingTalkFields from "./NotifyChannelEditFormDingTalkFields";
@ -35,8 +34,8 @@ const NotifyChannelEditForm = forwardRef<NotifyChannelEditFormInstance, NotifyCh
({ className, style, channel, disabled, initialValues, onValuesChange }, ref) => { ({ className, style, channel, disabled, initialValues, onValuesChange }, ref) => {
const { form: formInst, formProps } = useAntdForm({ const { form: formInst, formProps } = useAntdForm({
initialValues: initialValues, initialValues: initialValues,
name: "notifyChannelEditForm",
}); });
const formName = useCreation(() => `notifyChannelEditForm_${Math.random().toString(36).substring(2, 10)}${new Date().getTime()}`, []);
const formFieldsComponent = useMemo(() => { const formFieldsComponent = useMemo(() => {
/* /*
ASCII ASCII
@ -81,16 +80,7 @@ const NotifyChannelEditForm = forwardRef<NotifyChannelEditFormInstance, NotifyCh
}); });
return ( return (
<Form <Form {...formProps} className={className} style={style} form={formInst} disabled={disabled} layout="vertical" onValuesChange={handleFormChange}>
{...formProps}
className={className}
style={style}
form={formInst}
disabled={disabled}
name={formName}
layout="vertical"
onValuesChange={handleFormChange}
>
{formFieldsComponent} {formFieldsComponent}
</Form> </Form>
); );

View File

@ -52,7 +52,9 @@ const ApplyNodeForm = ({ node }: ApplyNodeFormProps) => {
providerAccessId: z providerAccessId: z
.string({ message: t("workflow_node.apply.form.provider_access.placeholder") }) .string({ message: t("workflow_node.apply.form.provider_access.placeholder") })
.min(1, t("workflow_node.apply.form.provider_access.placeholder")), .min(1, t("workflow_node.apply.form.provider_access.placeholder")),
keyAlgorithm: z.string().nullish(), keyAlgorithm: z
.string({ message: t("workflow_node.apply.form.key_algorithm.placeholder") })
.nonempty(t("workflow_node.apply.form.key_algorithm.placeholder")),
nameservers: z nameservers: z
.string() .string()
.nullish() .nullish()
@ -76,6 +78,7 @@ const ApplyNodeForm = ({ node }: ApplyNodeFormProps) => {
formPending, formPending,
formProps, formProps,
} = useAntdForm<z.infer<typeof formSchema>>({ } = useAntdForm<z.infer<typeof formSchema>>({
name: "workflowApplyNodeForm",
initialValues: (node?.config as WorkflowNodeConfigForApply) ?? initFormModel(), initialValues: (node?.config as WorkflowNodeConfigForApply) ?? initFormModel(),
onSubmit: async (values) => { onSubmit: async (values) => {
await formInst.validateFields(); await formInst.validateFields();
@ -305,7 +308,7 @@ const FormFieldEmailSelect = ({
const handleSearch = (text: string) => { const handleSearch = (text: string) => {
const temp = emailsToOptions(); const temp = emailsToOptions();
if (text) { if (text?.trim()) {
if (temp.every((option) => option.label !== text)) { if (temp.every((option) => option.label !== text)) {
temp.unshift({ label: text, value: text }); temp.unshift({ label: text, value: text });
} }

View File

@ -67,6 +67,7 @@ const DeployNodeForm = ({ node }: DeployFormProps) => {
formPending, formPending,
formProps, formProps,
} = useAntdForm<z.infer<typeof formSchema>>({ } = useAntdForm<z.infer<typeof formSchema>>({
name: "workflowDeployNodeForm",
initialValues: (node?.config as WorkflowNodeConfigForDeploy) ?? initFormModel(), initialValues: (node?.config as WorkflowNodeConfigForDeploy) ?? initFormModel(),
onSubmit: async (values) => { onSubmit: async (values) => {
await formInst.validateFields(); await formInst.validateFields();

View File

@ -57,6 +57,7 @@ const NotifyNodeForm = ({ node }: NotifyNodeFormProps) => {
formPending, formPending,
formProps, formProps,
} = useAntdForm<z.infer<typeof formSchema>>({ } = useAntdForm<z.infer<typeof formSchema>>({
name: "workflowNotifyNodeForm",
initialValues: (node?.config as WorkflowNodeConfigForNotify) ?? initFormModel(), initialValues: (node?.config as WorkflowNodeConfigForNotify) ?? initFormModel(),
onSubmit: async (values) => { onSubmit: async (values) => {
await formInst.validateFields(); await formInst.validateFields();

View File

@ -54,6 +54,7 @@ const StartNodeForm = ({ node }: StartNodeFormProps) => {
formPending, formPending,
formProps, formProps,
} = useAntdForm<z.infer<typeof formSchema>>({ } = useAntdForm<z.infer<typeof formSchema>>({
name: "workflowStartNodeForm",
initialValues: (node?.config as WorkflowNodeConfigForStart) ?? initFormModel(), initialValues: (node?.config as WorkflowNodeConfigForStart) ?? initFormModel(),
onSubmit: async (values) => { onSubmit: async (values) => {
await formInst.validateFields(); await formInst.validateFields();

View File

@ -193,7 +193,6 @@ export const newNode = (nodeType: WorkflowNodeType, options: NewNodeOptions = {}
case WorkflowNodeType.Apply: case WorkflowNodeType.Apply:
case WorkflowNodeType.Deploy: case WorkflowNodeType.Deploy:
{ {
node.config = {} as Record<string, unknown>;
node.inputs = workflowNodeTypeDefaultInputs.get(nodeType); node.inputs = workflowNodeTypeDefaultInputs.get(nodeType);
node.outputs = workflowNodeTypeDefaultOutputs.get(nodeType); node.outputs = workflowNodeTypeDefaultOutputs.get(nodeType);
} }

View File

@ -1,6 +1,7 @@
import useAntdForm from "./useAntdForm"; import useAntdForm from "./useAntdForm";
import useAntdFormName from "./useAntdFormName";
import useBrowserTheme from "./useBrowserTheme"; import useBrowserTheme from "./useBrowserTheme";
import useTriggerElement from "./useTriggerElement"; import useTriggerElement from "./useTriggerElement";
import useZustandShallowSelector from "./useZustandShallowSelector"; import useZustandShallowSelector from "./useZustandShallowSelector";
export { useAntdForm, useBrowserTheme, useTriggerElement, useZustandShallowSelector }; export { useAntdForm, useAntdFormName, useBrowserTheme, useTriggerElement, useZustandShallowSelector };

View File

@ -2,9 +2,12 @@
import { useDeepCompareEffect } from "ahooks"; import { useDeepCompareEffect } from "ahooks";
import { Form, type FormInstance, type FormProps } from "antd"; import { Form, type FormInstance, type FormProps } from "antd";
import useAntdFormName from "./useAntdFormName";
export interface UseAntdFormOptions<T extends NonNullable<unknown> = any> { export interface UseAntdFormOptions<T extends NonNullable<unknown> = any> {
form?: FormInstance<T>; form?: FormInstance<T>;
initialValues?: Partial<T> | (() => Partial<T> | Promise<Partial<T>>); initialValues?: Partial<T> | (() => Partial<T> | Promise<Partial<T>>);
name?: string;
onSubmit?: (values: T) => unknown; onSubmit?: (values: T) => unknown;
} }
@ -20,8 +23,9 @@ export interface UseAntdFormReturns<T extends NonNullable<unknown> = any> {
* @param {UseAntdFormOptions} options * @param {UseAntdFormOptions} options
* @returns {UseAntdFormReturns} * @returns {UseAntdFormReturns}
*/ */
const useAntdForm = <T extends NonNullable<unknown> = any>({ initialValues, form, onSubmit }: UseAntdFormOptions<T>): UseAntdFormReturns<T> => { const useAntdForm = <T extends NonNullable<unknown> = any>({ form, initialValues, onSubmit, ...options }: UseAntdFormOptions<T>): UseAntdFormReturns<T> => {
const formInst = form ?? Form["useForm"]()[0]; const formInst = form ?? Form["useForm"]()[0];
const formName = useAntdFormName({ form: formInst, name: options.name });
const [formInitialValues, setFormInitialValues] = useState<Partial<T>>(); const [formInitialValues, setFormInitialValues] = useState<Partial<T>>();
const [formPending, setFormPending] = useState(false); const [formPending, setFormPending] = useState(false);
@ -90,6 +94,7 @@ const useAntdForm = <T extends NonNullable<unknown> = any>({ initialValues, form
const formProps: FormProps = { const formProps: FormProps = {
form: formInst, form: formInst,
initialValues: formInitialValues, initialValues: formInitialValues,
name: options.name ? formName : undefined,
onFinish, onFinish,
}; };

View File

@ -0,0 +1,14 @@
import { useCreation } from "ahooks";
import { type FormInstance } from "antd";
export interface UseAntdFormNameOptions<T extends NonNullable<unknown> = any> {
form: FormInstance<T>;
name?: string;
}
const useAntdFormName = <T extends NonNullable<unknown> = any>(options: UseAntdFormNameOptions<T>) => {
const formName = useCreation(() => `${options.name}_${Math.random().toString(36).substring(2, 10)}${new Date().getTime()}`, [options.name, options.form]);
return formName;
};
export default useAntdFormName;

View File

@ -43,7 +43,7 @@ const WorkflowDetail = () => {
useZustandShallowSelector(["workflow", "initialized", "init", "destroy", "save", "setBaseInfo", "switchEnable"]) useZustandShallowSelector(["workflow", "initialized", "init", "destroy", "save", "setBaseInfo", "switchEnable"])
); );
useEffect(() => { useEffect(() => {
// TODO: loading // TODO: loading & error
init(workflowId!); init(workflowId!);
return () => { return () => {
@ -64,7 +64,7 @@ const WorkflowDetail = () => {
setAllowDiscard(!isRunning && hasReleased && hasChanges); setAllowDiscard(!isRunning && hasReleased && hasChanges);
setAllowRelease(!isRunning && hasChanges); setAllowRelease(!isRunning && hasChanges);
setAllowRun(hasReleased); setAllowRun(hasReleased);
}, [workflow, isRunning]); }, [workflow.content, workflow.draft, workflow.hasDraft, isRunning]);
const handleBaseInfoFormFinish = async (values: Pick<WorkflowModel, "name" | "description">) => { const handleBaseInfoFormFinish = async (values: Pick<WorkflowModel, "name" | "description">) => {
try { try {
@ -141,11 +141,6 @@ const WorkflowDetail = () => {
}; };
const handleRunClick = () => { const handleRunClick = () => {
if (!workflow.enabled) {
alert("TODO: 暂时只支持执行已启用的工作流");
return;
}
const { promise, resolve, reject } = Promise.withResolvers(); const { promise, resolve, reject } = Promise.withResolvers();
if (workflow.hasDraft) { if (workflow.hasDraft) {
modalApi.confirm({ modalApi.confirm({

View File

@ -30,7 +30,7 @@ export const useContactEmailsStore = create<ContactEmailsState>((set, get) => {
try { try {
set({ loading: true }); set({ loading: true });
settings = await fetcher; settings = await fetcher;
set({ emails: settings.content.emails?.sort() ?? [], loadedAtOnce: true }); set({ emails: settings.content.emails?.filter((s) => !!s)?.sort() ?? [], loadedAtOnce: true });
} finally { } finally {
fetcher = null; fetcher = null;
set({ loading: false }); set({ loading: false });