TanStack Form 支援將陣列 (array) 作為表單值,包含陣列中的子物件值。
要使用陣列,您可以對陣列值使用 field.state.value,並搭配 solid-js 的 Index:
<script setup lang="ts">
import { useForm } from '@tanstack/vue-form'
const form = useForm({
defaultValues: {
people: [] as Array<{ age: number; name: string }>,
},
onSubmit: ({ value }) => alert(JSON.stringify(value)),
})
</script>
<template>
<form.Field name="people">
<template v-slot="{ field, state }">
<div>
<form.Field
v-for="(_, i) of field.state.value"
:key="i"
:name="`people[${i}].name`"
>
<template v-slot="{ field: subField, state }">
<!-- ... -->
</template>
</form.Field>
</div>
</template>
</form.Field>
</template>
<script setup lang="ts">
import { useForm } from '@tanstack/vue-form'
const form = useForm({
defaultValues: {
people: [] as Array<{ age: number; name: string }>,
},
onSubmit: ({ value }) => alert(JSON.stringify(value)),
})
</script>
<template>
<form.Field name="people">
<template v-slot="{ field, state }">
<div>
<form.Field
v-for="(_, i) of field.state.value"
:key="i"
:name="`people[${i}].name`"
>
<template v-slot="{ field: subField, state }">
<!-- ... -->
</template>
</form.Field>
</div>
</template>
</form.Field>
</template>
這會在每次對 field 執行 pushValue 時生成對應的插槽 (slot):
<button @click="field.pushValue({ name: '', age: 0 })" type="button">
新增人員
</button>
<button @click="field.pushValue({ name: '', age: 0 })" type="button">
新增人員
</button>
最後,您可以像這樣使用子欄位 (subfield):
<form.Field
v-for="(_, i) of field.state.value"
:key="i"
:name="`people[${i}].name`"
>
<template v-slot="{ field: subField, state }">
<div>
<label>
<div>人員 {{ i }} 的姓名</div>
<input
:value="subField.state.value"
@input="
(e) =>
subField.handleChange(
(e.target as HTMLInputElement).value,
)
"
/>
</label>
</div>
</template>
</form.Field>
<form.Field
v-for="(_, i) of field.state.value"
:key="i"
:name="`people[${i}].name`"
>
<template v-slot="{ field: subField, state }">
<div>
<label>
<div>人員 {{ i }} 的姓名</div>
<input
:value="subField.state.value"
@input="
(e) =>
subField.handleChange(
(e.target as HTMLInputElement).value,
)
"
/>
</label>
</div>
</template>
</form.Field>
<script setup lang="ts">
import { useForm } from '@tanstack/vue-form'
const form = useForm({
defaultValues: {
people: [] as Array<{ age: number; name: string }>,
},
onSubmit: ({ value }) => alert(JSON.stringify(value)),
})
</script>
<template>
<form
@submit="
(e) => {
e.preventDefault()
e.stopPropagation()
form.handleSubmit()
}
"
>
<div>
<form.Field name="people">
<template v-slot="{ field, state }">
<div>
<form.Field
v-for="(_, i) of field.state.value"
:key="i"
:name="`people[${i}].name`"
>
<template v-slot="{ field: subField, state }">
<div>
<label>
<div>人員 {{ i }} 的姓名</div>
<input
:value="subField.state.value"
@input="
(e) =>
subField.handleChange(
(e.target as HTMLInputElement).value,
)
"
/>
</label>
</div>
</template>
</form.Field>
<button
@click="field.pushValue({ name: '', age: 0 })"
type="button"
>
新增人員
</button>
</div>
</template>
</form.Field>
</div>
<form.Subscribe>
<template v-slot="{ canSubmit, isSubmitting }">
<button type="submit" :disabled="!canSubmit">
{{ isSubmitting ? '...' : '提交' }}
</button>
</template>
</form.Subscribe>
</form>
</template>
<script setup lang="ts">
import { useForm } from '@tanstack/vue-form'
const form = useForm({
defaultValues: {
people: [] as Array<{ age: number; name: string }>,
},
onSubmit: ({ value }) => alert(JSON.stringify(value)),
})
</script>
<template>
<form
@submit="
(e) => {
e.preventDefault()
e.stopPropagation()
form.handleSubmit()
}
"
>
<div>
<form.Field name="people">
<template v-slot="{ field, state }">
<div>
<form.Field
v-for="(_, i) of field.state.value"
:key="i"
:name="`people[${i}].name`"
>
<template v-slot="{ field: subField, state }">
<div>
<label>
<div>人員 {{ i }} 的姓名</div>
<input
:value="subField.state.value"
@input="
(e) =>
subField.handleChange(
(e.target as HTMLInputElement).value,
)
"
/>
</label>
</div>
</template>
</form.Field>
<button
@click="field.pushValue({ name: '', age: 0 })"
type="button"
>
新增人員
</button>
</div>
</template>
</form.Field>
</div>
<form.Subscribe>
<template v-slot="{ canSubmit, isSubmitting }">
<button type="submit" :disabled="!canSubmit">
{{ isSubmitting ? '...' : '提交' }}
</button>
</template>
</form.Subscribe>
</form>
</template>
Your weekly dose of JavaScript news. Delivered every Monday to over 100,000 devs, for free.