@assistant-ui/react-hook-form 0.10.0 → 0.10.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,21 @@
1
+ import { z } from "zod";
2
+ export declare const formTools: {
3
+ set_form_field: {
4
+ description: string;
5
+ parameters: z.ZodObject<{
6
+ name: z.ZodString;
7
+ value: z.ZodString;
8
+ }, "strip", z.ZodTypeAny, {
9
+ name: string;
10
+ value: string;
11
+ }, {
12
+ name: string;
13
+ value: string;
14
+ }>;
15
+ };
16
+ submit_form: {
17
+ description: string;
18
+ parameters: z.ZodObject<{}, "strip", z.ZodTypeAny, {}, {}>;
19
+ };
20
+ };
21
+ //# sourceMappingURL=formTools.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"formTools.d.ts","sourceRoot":"","sources":["../src/formTools.tsx"],"names":[],"mappings":"AAAA,OAAO,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAExB,eAAO,MAAM,SAAS;;;;;;;;;;;;;;;;;;CAarB,CAAC"}
package/dist/formTools.js CHANGED
@@ -1,5 +1,6 @@
1
+ // src/formTools.tsx
1
2
  import { z } from "zod";
2
- const formTools = {
3
+ var formTools = {
3
4
  set_form_field: {
4
5
  description: "Sets a form field. Call this function as soon as the user provides the data for each field.",
5
6
  parameters: z.object({
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/formTools.tsx"],"sourcesContent":["import { z } from \"zod\";\n\nexport const formTools = {\n set_form_field: {\n description:\n \"Sets a form field. Call this function as soon as the user provides the data for each field.\",\n parameters: z.object({\n name: z.string(),\n value: z.string(),\n }),\n },\n submit_form: {\n description: \"Submits the form. Confirm with user before submitting.\",\n parameters: z.object({}),\n },\n};\n"],"mappings":"AAAA,SAAS,SAAS;AAEX,MAAM,YAAY;AAAA,EACvB,gBAAgB;AAAA,IACd,aACE;AAAA,IACF,YAAY,EAAE,OAAO;AAAA,MACnB,MAAM,EAAE,OAAO;AAAA,MACf,OAAO,EAAE,OAAO;AAAA,IAClB,CAAC;AAAA,EACH;AAAA,EACA,aAAa;AAAA,IACX,aAAa;AAAA,IACb,YAAY,EAAE,OAAO,CAAC,CAAC;AAAA,EACzB;AACF;","names":[]}
1
+ {"version":3,"sources":["../src/formTools.tsx"],"sourcesContent":["import { z } from \"zod\";\n\nexport const formTools = {\n set_form_field: {\n description:\n \"Sets a form field. Call this function as soon as the user provides the data for each field.\",\n parameters: z.object({\n name: z.string(),\n value: z.string(),\n }),\n },\n submit_form: {\n description: \"Submits the form. Confirm with user before submitting.\",\n parameters: z.object({}),\n },\n};\n"],"mappings":";AAAA,SAAS,SAAS;AAEX,IAAM,YAAY;AAAA,EACvB,gBAAgB;AAAA,IACd,aACE;AAAA,IACF,YAAY,EAAE,OAAO;AAAA,MACnB,MAAM,EAAE,OAAO;AAAA,MACf,OAAO,EAAE,OAAO;AAAA,IAClB,CAAC;AAAA,EACH;AAAA,EACA,aAAa;AAAA,IACX,aAAa;AAAA,IACb,YAAY,EAAE,OAAO,CAAC,CAAC;AAAA,EACzB;AACF;","names":[]}
@@ -0,0 +1,3 @@
1
+ export { useAssistantForm, type UseAssistantFormProps, } from "./useAssistantForm";
2
+ export { formTools } from "./formTools";
3
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,gBAAgB,EAChB,KAAK,qBAAqB,GAC3B,MAAM,oBAAoB,CAAC;AAC5B,OAAO,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC"}
package/dist/index.js CHANGED
@@ -1,7 +1,8 @@
1
+ // src/index.ts
1
2
  import {
2
3
  useAssistantForm
3
- } from "./useAssistantForm";
4
- import { formTools } from "./formTools";
4
+ } from "./useAssistantForm.js";
5
+ import { formTools } from "./formTools.js";
5
6
  export {
6
7
  formTools,
7
8
  useAssistantForm
package/dist/index.js.map CHANGED
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/index.ts"],"sourcesContent":["export {\n useAssistantForm,\n type UseAssistantFormProps,\n} from \"./useAssistantForm\";\nexport { formTools } from \"./formTools\";\n"],"mappings":"AAAA;AAAA,EACE;AAAA,OAEK;AACP,SAAS,iBAAiB;","names":[]}
1
+ {"version":3,"sources":["../src/index.ts"],"sourcesContent":["export {\n useAssistantForm,\n type UseAssistantFormProps,\n} from \"./useAssistantForm\";\nexport { formTools } from \"./formTools\";\n"],"mappings":";AAAA;AAAA,EACE;AAAA,OAEK;AACP,SAAS,iBAAiB;","names":[]}
@@ -0,0 +1,18 @@
1
+ import { type ToolCallContentPartComponent } from "@assistant-ui/react";
2
+ import { type FieldValues, type UseFormProps, type UseFormReturn } from "react-hook-form";
3
+ import type { z } from "zod";
4
+ import { formTools } from "./formTools";
5
+ export type UseAssistantFormProps<TFieldValues extends FieldValues, TContext, TTransformedValues> = UseFormProps<TFieldValues, TContext, TTransformedValues> & {
6
+ assistant?: {
7
+ tools?: {
8
+ set_form_field?: {
9
+ render?: ToolCallContentPartComponent<z.infer<(typeof formTools.set_form_field)["parameters"]>, unknown> | undefined;
10
+ } | undefined;
11
+ submit_form?: {
12
+ render?: ToolCallContentPartComponent<z.infer<(typeof formTools.submit_form)["parameters"]>, unknown> | undefined;
13
+ } | undefined;
14
+ } | undefined;
15
+ } | undefined;
16
+ };
17
+ export declare const useAssistantForm: <TFieldValues extends FieldValues = FieldValues, TContext = any, TTransformedValues = TFieldValues>(props?: UseAssistantFormProps<TFieldValues, TContext, TTransformedValues>) => UseFormReturn<TFieldValues, TContext, TTransformedValues>;
18
+ //# sourceMappingURL=useAssistantForm.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useAssistantForm.d.ts","sourceRoot":"","sources":["../src/useAssistantForm.tsx"],"names":[],"mappings":"AAEA,OAAO,EAGL,KAAK,4BAA4B,EAGlC,MAAM,qBAAqB,CAAC;AAE7B,OAAO,EACL,KAAK,WAAW,EAGhB,KAAK,YAAY,EACjB,KAAK,aAAa,EAEnB,MAAM,iBAAiB,CAAC;AACzB,OAAO,KAAK,EAAE,CAAC,EAAE,MAAM,KAAK,CAAC;AAC7B,OAAO,EAAE,SAAS,EAAE,MAAM,aAAa,CAAC;AAExC,MAAM,MAAM,qBAAqB,CAC/B,YAAY,SAAS,WAAW,EAChC,QAAQ,EACR,kBAAkB,IAChB,YAAY,CAAC,YAAY,EAAE,QAAQ,EAAE,kBAAkB,CAAC,GAAG;IAC7D,SAAS,CAAC,EACN;QACE,KAAK,CAAC,EACF;YACE,cAAc,CAAC,EACX;gBACE,MAAM,CAAC,EACH,4BAA4B,CAC1B,CAAC,CAAC,KAAK,CACL,CAAC,OAAO,SAAS,CAAC,cAAc,CAAC,CAAC,YAAY,CAAC,CAChD,EACD,OAAO,CACR,GACD,SAAS,CAAC;aACf,GACD,SAAS,CAAC;YACd,WAAW,CAAC,EACR;gBACE,MAAM,CAAC,EACH,4BAA4B,CAC1B,CAAC,CAAC,KAAK,CAAC,CAAC,OAAO,SAAS,CAAC,WAAW,CAAC,CAAC,YAAY,CAAC,CAAC,EACrD,OAAO,CACR,GACD,SAAS,CAAC;aACf,GACD,SAAS,CAAC;SACf,GACD,SAAS,CAAC;KACf,GACD,SAAS,CAAC;CACf,CAAC;AAEF,eAAO,MAAM,gBAAgB,GAC3B,YAAY,SAAS,WAAW,GAAG,WAAW,EAC9C,QAAQ,GAAG,GAAG,EACd,kBAAkB,GAAG,YAAY,EAEjC,QAAQ,qBAAqB,CAAC,YAAY,EAAE,QAAQ,EAAE,kBAAkB,CAAC,KACxE,aAAa,CAAC,YAAY,EAAE,QAAQ,EAAE,kBAAkB,CA8E1D,CAAC"}
@@ -1,4 +1,6 @@
1
1
  "use client";
2
+
3
+ // src/useAssistantForm.tsx
2
4
  import {
3
5
  tool,
4
6
  useAssistantRuntime,
@@ -8,8 +10,8 @@ import { useEffect } from "react";
8
10
  import {
9
11
  useForm
10
12
  } from "react-hook-form";
11
- import { formTools } from "./formTools";
12
- const useAssistantForm = (props) => {
13
+ import { formTools } from "./formTools.js";
14
+ var useAssistantForm = (props) => {
13
15
  const form = useForm(props);
14
16
  const { control, getValues, setValue } = form;
15
17
  const assistantRuntime = useAssistantRuntime();
@@ -1 +1 @@
1
- {"version":3,"sources":["../src/useAssistantForm.tsx"],"sourcesContent":["\"use client\";\n\nimport {\n type ModelContext,\n tool,\n type ToolCallContentPartComponent,\n useAssistantRuntime,\n useAssistantToolUI,\n} from \"@assistant-ui/react\";\nimport { useEffect } from \"react\";\nimport {\n type FieldValues,\n Path,\n PathValue,\n type UseFormProps,\n type UseFormReturn,\n useForm,\n} from \"react-hook-form\";\nimport type { z } from \"zod\";\nimport { formTools } from \"./formTools\";\n\nexport type UseAssistantFormProps<\n TFieldValues extends FieldValues,\n TContext,\n TTransformedValues,\n> = UseFormProps<TFieldValues, TContext, TTransformedValues> & {\n assistant?:\n | {\n tools?:\n | {\n set_form_field?:\n | {\n render?:\n | ToolCallContentPartComponent<\n z.infer<\n (typeof formTools.set_form_field)[\"parameters\"]\n >,\n unknown\n >\n | undefined;\n }\n | undefined;\n submit_form?:\n | {\n render?:\n | ToolCallContentPartComponent<\n z.infer<(typeof formTools.submit_form)[\"parameters\"]>,\n unknown\n >\n | undefined;\n }\n | undefined;\n }\n | undefined;\n }\n | undefined;\n};\n\nexport const useAssistantForm = <\n TFieldValues extends FieldValues = FieldValues,\n TContext = any,\n TTransformedValues = TFieldValues,\n>(\n props?: UseAssistantFormProps<TFieldValues, TContext, TTransformedValues>,\n): UseFormReturn<TFieldValues, TContext, TTransformedValues> => {\n const form = useForm<TFieldValues, TContext, TTransformedValues>(props);\n const { control, getValues, setValue } = form;\n\n const assistantRuntime = useAssistantRuntime();\n useEffect(() => {\n const value: ModelContext = {\n system: `Form State:\\n${JSON.stringify(getValues())}`,\n\n tools: {\n set_form_field: tool({\n ...formTools.set_form_field,\n execute: async (args) => {\n setValue(\n args.name as Path<TFieldValues>,\n args.value as PathValue<TFieldValues, Path<TFieldValues>>,\n );\n\n return { success: true };\n },\n }),\n submit_form: tool({\n ...formTools.submit_form,\n execute: async () => {\n const { _names, _fields } = control;\n for (const name of _names.mount) {\n const field = _fields[name];\n if (field?._f && \"refs\" in field._f) {\n const fieldReference = Array.isArray(field._f.refs)\n ? field._f.refs[0]\n : field._f.ref;\n\n if (fieldReference instanceof HTMLElement) {\n const form = fieldReference.closest(\"form\");\n if (form) {\n form.requestSubmit();\n\n return { success: true };\n }\n }\n }\n }\n\n return {\n success: false,\n message:\n \"Unable retrieve the form element. This is a coding error.\",\n };\n },\n }),\n },\n };\n return assistantRuntime.registerModelContextProvider({\n getModelContext: () => value,\n });\n }, [control, setValue, getValues, assistantRuntime]);\n\n const renderFormFieldTool = props?.assistant?.tools?.set_form_field?.render;\n useAssistantToolUI(\n renderFormFieldTool\n ? {\n toolName: \"set_form_field\",\n render: renderFormFieldTool,\n }\n : null,\n );\n\n const renderSubmitFormTool = props?.assistant?.tools?.submit_form?.render;\n useAssistantToolUI(\n renderSubmitFormTool\n ? {\n toolName: \"submit_form\",\n render: renderSubmitFormTool,\n }\n : null,\n );\n\n return form;\n};\n"],"mappings":";AAEA;AAAA,EAEE;AAAA,EAEA;AAAA,EACA;AAAA,OACK;AACP,SAAS,iBAAiB;AAC1B;AAAA,EAME;AAAA,OACK;AAEP,SAAS,iBAAiB;AAuCnB,MAAM,mBAAmB,CAK9B,UAC8D;AAC9D,QAAM,OAAO,QAAoD,KAAK;AACtE,QAAM,EAAE,SAAS,WAAW,SAAS,IAAI;AAEzC,QAAM,mBAAmB,oBAAoB;AAC7C,YAAU,MAAM;AACd,UAAM,QAAsB;AAAA,MAC1B,QAAQ;AAAA,EAAgB,KAAK,UAAU,UAAU,CAAC,CAAC;AAAA,MAEnD,OAAO;AAAA,QACL,gBAAgB,KAAK;AAAA,UACnB,GAAG,UAAU;AAAA,UACb,SAAS,OAAO,SAAS;AACvB;AAAA,cACE,KAAK;AAAA,cACL,KAAK;AAAA,YACP;AAEA,mBAAO,EAAE,SAAS,KAAK;AAAA,UACzB;AAAA,QACF,CAAC;AAAA,QACD,aAAa,KAAK;AAAA,UAChB,GAAG,UAAU;AAAA,UACb,SAAS,YAAY;AACnB,kBAAM,EAAE,QAAQ,QAAQ,IAAI;AAC5B,uBAAW,QAAQ,OAAO,OAAO;AAC/B,oBAAM,QAAQ,QAAQ,IAAI;AAC1B,kBAAI,OAAO,MAAM,UAAU,MAAM,IAAI;AACnC,sBAAM,iBAAiB,MAAM,QAAQ,MAAM,GAAG,IAAI,IAC9C,MAAM,GAAG,KAAK,CAAC,IACf,MAAM,GAAG;AAEb,oBAAI,0BAA0B,aAAa;AACzC,wBAAMA,QAAO,eAAe,QAAQ,MAAM;AAC1C,sBAAIA,OAAM;AACR,oBAAAA,MAAK,cAAc;AAEnB,2BAAO,EAAE,SAAS,KAAK;AAAA,kBACzB;AAAA,gBACF;AAAA,cACF;AAAA,YACF;AAEA,mBAAO;AAAA,cACL,SAAS;AAAA,cACT,SACE;AAAA,YACJ;AAAA,UACF;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF;AACA,WAAO,iBAAiB,6BAA6B;AAAA,MACnD,iBAAiB,MAAM;AAAA,IACzB,CAAC;AAAA,EACH,GAAG,CAAC,SAAS,UAAU,WAAW,gBAAgB,CAAC;AAEnD,QAAM,sBAAsB,OAAO,WAAW,OAAO,gBAAgB;AACrE;AAAA,IACE,sBACI;AAAA,MACE,UAAU;AAAA,MACV,QAAQ;AAAA,IACV,IACA;AAAA,EACN;AAEA,QAAM,uBAAuB,OAAO,WAAW,OAAO,aAAa;AACnE;AAAA,IACE,uBACI;AAAA,MACE,UAAU;AAAA,MACV,QAAQ;AAAA,IACV,IACA;AAAA,EACN;AAEA,SAAO;AACT;","names":["form"]}
1
+ {"version":3,"sources":["../src/useAssistantForm.tsx"],"sourcesContent":["\"use client\";\n\nimport {\n type ModelContext,\n tool,\n type ToolCallContentPartComponent,\n useAssistantRuntime,\n useAssistantToolUI,\n} from \"@assistant-ui/react\";\nimport { useEffect } from \"react\";\nimport {\n type FieldValues,\n Path,\n PathValue,\n type UseFormProps,\n type UseFormReturn,\n useForm,\n} from \"react-hook-form\";\nimport type { z } from \"zod\";\nimport { formTools } from \"./formTools\";\n\nexport type UseAssistantFormProps<\n TFieldValues extends FieldValues,\n TContext,\n TTransformedValues,\n> = UseFormProps<TFieldValues, TContext, TTransformedValues> & {\n assistant?:\n | {\n tools?:\n | {\n set_form_field?:\n | {\n render?:\n | ToolCallContentPartComponent<\n z.infer<\n (typeof formTools.set_form_field)[\"parameters\"]\n >,\n unknown\n >\n | undefined;\n }\n | undefined;\n submit_form?:\n | {\n render?:\n | ToolCallContentPartComponent<\n z.infer<(typeof formTools.submit_form)[\"parameters\"]>,\n unknown\n >\n | undefined;\n }\n | undefined;\n }\n | undefined;\n }\n | undefined;\n};\n\nexport const useAssistantForm = <\n TFieldValues extends FieldValues = FieldValues,\n TContext = any,\n TTransformedValues = TFieldValues,\n>(\n props?: UseAssistantFormProps<TFieldValues, TContext, TTransformedValues>,\n): UseFormReturn<TFieldValues, TContext, TTransformedValues> => {\n const form = useForm<TFieldValues, TContext, TTransformedValues>(props);\n const { control, getValues, setValue } = form;\n\n const assistantRuntime = useAssistantRuntime();\n useEffect(() => {\n const value: ModelContext = {\n system: `Form State:\\n${JSON.stringify(getValues())}`,\n\n tools: {\n set_form_field: tool({\n ...formTools.set_form_field,\n execute: async (args) => {\n setValue(\n args.name as Path<TFieldValues>,\n args.value as PathValue<TFieldValues, Path<TFieldValues>>,\n );\n\n return { success: true };\n },\n }),\n submit_form: tool({\n ...formTools.submit_form,\n execute: async () => {\n const { _names, _fields } = control;\n for (const name of _names.mount) {\n const field = _fields[name];\n if (field?._f && \"refs\" in field._f) {\n const fieldReference = Array.isArray(field._f.refs)\n ? field._f.refs[0]\n : field._f.ref;\n\n if (fieldReference instanceof HTMLElement) {\n const form = fieldReference.closest(\"form\");\n if (form) {\n form.requestSubmit();\n\n return { success: true };\n }\n }\n }\n }\n\n return {\n success: false,\n message:\n \"Unable retrieve the form element. This is a coding error.\",\n };\n },\n }),\n },\n };\n return assistantRuntime.registerModelContextProvider({\n getModelContext: () => value,\n });\n }, [control, setValue, getValues, assistantRuntime]);\n\n const renderFormFieldTool = props?.assistant?.tools?.set_form_field?.render;\n useAssistantToolUI(\n renderFormFieldTool\n ? {\n toolName: \"set_form_field\",\n render: renderFormFieldTool,\n }\n : null,\n );\n\n const renderSubmitFormTool = props?.assistant?.tools?.submit_form?.render;\n useAssistantToolUI(\n renderSubmitFormTool\n ? {\n toolName: \"submit_form\",\n render: renderSubmitFormTool,\n }\n : null,\n );\n\n return form;\n};\n"],"mappings":";;;AAEA;AAAA,EAEE;AAAA,EAEA;AAAA,EACA;AAAA,OACK;AACP,SAAS,iBAAiB;AAC1B;AAAA,EAME;AAAA,OACK;AAEP,SAAS,iBAAiB;AAuCnB,IAAM,mBAAmB,CAK9B,UAC8D;AAC9D,QAAM,OAAO,QAAoD,KAAK;AACtE,QAAM,EAAE,SAAS,WAAW,SAAS,IAAI;AAEzC,QAAM,mBAAmB,oBAAoB;AAC7C,YAAU,MAAM;AACd,UAAM,QAAsB;AAAA,MAC1B,QAAQ;AAAA,EAAgB,KAAK,UAAU,UAAU,CAAC,CAAC;AAAA,MAEnD,OAAO;AAAA,QACL,gBAAgB,KAAK;AAAA,UACnB,GAAG,UAAU;AAAA,UACb,SAAS,OAAO,SAAS;AACvB;AAAA,cACE,KAAK;AAAA,cACL,KAAK;AAAA,YACP;AAEA,mBAAO,EAAE,SAAS,KAAK;AAAA,UACzB;AAAA,QACF,CAAC;AAAA,QACD,aAAa,KAAK;AAAA,UAChB,GAAG,UAAU;AAAA,UACb,SAAS,YAAY;AACnB,kBAAM,EAAE,QAAQ,QAAQ,IAAI;AAC5B,uBAAW,QAAQ,OAAO,OAAO;AAC/B,oBAAM,QAAQ,QAAQ,IAAI;AAC1B,kBAAI,OAAO,MAAM,UAAU,MAAM,IAAI;AACnC,sBAAM,iBAAiB,MAAM,QAAQ,MAAM,GAAG,IAAI,IAC9C,MAAM,GAAG,KAAK,CAAC,IACf,MAAM,GAAG;AAEb,oBAAI,0BAA0B,aAAa;AACzC,wBAAMA,QAAO,eAAe,QAAQ,MAAM;AAC1C,sBAAIA,OAAM;AACR,oBAAAA,MAAK,cAAc;AAEnB,2BAAO,EAAE,SAAS,KAAK;AAAA,kBACzB;AAAA,gBACF;AAAA,cACF;AAAA,YACF;AAEA,mBAAO;AAAA,cACL,SAAS;AAAA,cACT,SACE;AAAA,YACJ;AAAA,UACF;AAAA,QACF,CAAC;AAAA,MACH;AAAA,IACF;AACA,WAAO,iBAAiB,6BAA6B;AAAA,MACnD,iBAAiB,MAAM;AAAA,IACzB,CAAC;AAAA,EACH,GAAG,CAAC,SAAS,UAAU,WAAW,gBAAgB,CAAC;AAEnD,QAAM,sBAAsB,OAAO,WAAW,OAAO,gBAAgB;AACrE;AAAA,IACE,sBACI;AAAA,MACE,UAAU;AAAA,MACV,QAAQ;AAAA,IACV,IACA;AAAA,EACN;AAEA,QAAM,uBAAuB,OAAO,WAAW,OAAO,aAAa;AACnE;AAAA,IACE,uBACI;AAAA,MACE,UAAU;AAAA,MACV,QAAQ;AAAA,IACV,IACA;AAAA,EACN;AAEA,SAAO;AACT;","names":["form"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@assistant-ui/react-hook-form",
3
- "version": "0.10.0",
3
+ "version": "0.10.2",
4
4
  "license": "MIT",
5
5
  "type": "module",
6
6
  "exports": {
@@ -13,6 +13,7 @@
13
13
  "types": "./dist/index.d.ts",
14
14
  "files": [
15
15
  "dist",
16
+ "src",
16
17
  "README.md"
17
18
  ],
18
19
  "sideEffects": false,
@@ -20,7 +21,7 @@
20
21
  "zod": "^3.24.3"
21
22
  },
22
23
  "peerDependencies": {
23
- "@assistant-ui/react": "^0.10.0",
24
+ "@assistant-ui/react": "^0.10.2",
24
25
  "@types/react": "*",
25
26
  "react": "^18 || ^19 || ^19.0.0-rc",
26
27
  "react-hook-form": "^7"
@@ -32,11 +33,11 @@
32
33
  },
33
34
  "devDependencies": {
34
35
  "@types/node": "^22.14.1",
35
- "@types/react": "^19.1.0",
36
+ "@types/react": "^19.1.2",
36
37
  "eslint": "^9",
37
38
  "eslint-config-next": "15.3.1",
38
39
  "react": "^19.1.0",
39
- "react-hook-form": "^7.51.1",
40
+ "react-hook-form": "^7.56.0",
40
41
  "tsx": "^4.19.3",
41
42
  "@assistant-ui/tsbuildutils": "0.0.1",
42
43
  "@assistant-ui/tsconfig": "0.0.0"
@@ -0,0 +1,16 @@
1
+ import { z } from "zod";
2
+
3
+ export const formTools = {
4
+ set_form_field: {
5
+ description:
6
+ "Sets a form field. Call this function as soon as the user provides the data for each field.",
7
+ parameters: z.object({
8
+ name: z.string(),
9
+ value: z.string(),
10
+ }),
11
+ },
12
+ submit_form: {
13
+ description: "Submits the form. Confirm with user before submitting.",
14
+ parameters: z.object({}),
15
+ },
16
+ };
package/src/index.ts ADDED
@@ -0,0 +1,5 @@
1
+ export {
2
+ useAssistantForm,
3
+ type UseAssistantFormProps,
4
+ } from "./useAssistantForm";
5
+ export { formTools } from "./formTools";
@@ -0,0 +1,143 @@
1
+ "use client";
2
+
3
+ import {
4
+ type ModelContext,
5
+ tool,
6
+ type ToolCallContentPartComponent,
7
+ useAssistantRuntime,
8
+ useAssistantToolUI,
9
+ } from "@assistant-ui/react";
10
+ import { useEffect } from "react";
11
+ import {
12
+ type FieldValues,
13
+ Path,
14
+ PathValue,
15
+ type UseFormProps,
16
+ type UseFormReturn,
17
+ useForm,
18
+ } from "react-hook-form";
19
+ import type { z } from "zod";
20
+ import { formTools } from "./formTools";
21
+
22
+ export type UseAssistantFormProps<
23
+ TFieldValues extends FieldValues,
24
+ TContext,
25
+ TTransformedValues,
26
+ > = UseFormProps<TFieldValues, TContext, TTransformedValues> & {
27
+ assistant?:
28
+ | {
29
+ tools?:
30
+ | {
31
+ set_form_field?:
32
+ | {
33
+ render?:
34
+ | ToolCallContentPartComponent<
35
+ z.infer<
36
+ (typeof formTools.set_form_field)["parameters"]
37
+ >,
38
+ unknown
39
+ >
40
+ | undefined;
41
+ }
42
+ | undefined;
43
+ submit_form?:
44
+ | {
45
+ render?:
46
+ | ToolCallContentPartComponent<
47
+ z.infer<(typeof formTools.submit_form)["parameters"]>,
48
+ unknown
49
+ >
50
+ | undefined;
51
+ }
52
+ | undefined;
53
+ }
54
+ | undefined;
55
+ }
56
+ | undefined;
57
+ };
58
+
59
+ export const useAssistantForm = <
60
+ TFieldValues extends FieldValues = FieldValues,
61
+ TContext = any,
62
+ TTransformedValues = TFieldValues,
63
+ >(
64
+ props?: UseAssistantFormProps<TFieldValues, TContext, TTransformedValues>,
65
+ ): UseFormReturn<TFieldValues, TContext, TTransformedValues> => {
66
+ const form = useForm<TFieldValues, TContext, TTransformedValues>(props);
67
+ const { control, getValues, setValue } = form;
68
+
69
+ const assistantRuntime = useAssistantRuntime();
70
+ useEffect(() => {
71
+ const value: ModelContext = {
72
+ system: `Form State:\n${JSON.stringify(getValues())}`,
73
+
74
+ tools: {
75
+ set_form_field: tool({
76
+ ...formTools.set_form_field,
77
+ execute: async (args) => {
78
+ setValue(
79
+ args.name as Path<TFieldValues>,
80
+ args.value as PathValue<TFieldValues, Path<TFieldValues>>,
81
+ );
82
+
83
+ return { success: true };
84
+ },
85
+ }),
86
+ submit_form: tool({
87
+ ...formTools.submit_form,
88
+ execute: async () => {
89
+ const { _names, _fields } = control;
90
+ for (const name of _names.mount) {
91
+ const field = _fields[name];
92
+ if (field?._f && "refs" in field._f) {
93
+ const fieldReference = Array.isArray(field._f.refs)
94
+ ? field._f.refs[0]
95
+ : field._f.ref;
96
+
97
+ if (fieldReference instanceof HTMLElement) {
98
+ const form = fieldReference.closest("form");
99
+ if (form) {
100
+ form.requestSubmit();
101
+
102
+ return { success: true };
103
+ }
104
+ }
105
+ }
106
+ }
107
+
108
+ return {
109
+ success: false,
110
+ message:
111
+ "Unable retrieve the form element. This is a coding error.",
112
+ };
113
+ },
114
+ }),
115
+ },
116
+ };
117
+ return assistantRuntime.registerModelContextProvider({
118
+ getModelContext: () => value,
119
+ });
120
+ }, [control, setValue, getValues, assistantRuntime]);
121
+
122
+ const renderFormFieldTool = props?.assistant?.tools?.set_form_field?.render;
123
+ useAssistantToolUI(
124
+ renderFormFieldTool
125
+ ? {
126
+ toolName: "set_form_field",
127
+ render: renderFormFieldTool,
128
+ }
129
+ : null,
130
+ );
131
+
132
+ const renderSubmitFormTool = props?.assistant?.tools?.submit_form?.render;
133
+ useAssistantToolUI(
134
+ renderSubmitFormTool
135
+ ? {
136
+ toolName: "submit_form",
137
+ render: renderSubmitFormTool,
138
+ }
139
+ : null,
140
+ );
141
+
142
+ return form;
143
+ };