跳到主要内容

表单服务器操作

Analog 支持服务器端处理表单提交和验证。

设置表单

要处理表单提交,请使用 @analogjs/router 包中的 FormAction 指令。该指令处理收集 FormData 并发送 POST 请求到服务器。

该指令在处理表单后会触发以下事件:

  • onSuccess:当表单在服务器上处理并返回成功响应时触发。
  • onError:当表单返回错误响应时触发。
  • onStateChange:当表单提交时触发。

下面的示例页面提交一个电子邮件用于新闻通讯注册。

// src/app/pages/newsletter.page.ts
import { Component, signal } from '@angular/core';
import { FormAction } from '@analogjs/router';

type FormErrors =
| {
email?: string;
}
| undefined;

@Component({
selector: 'app-newsletter-page',
standalone: true,
imports: [FormAction],
template: `
<h3>Newsletter Signup</h3>

@if (!signedUp()) {
<form
method="post"
(onSuccess)="onSuccess()"
(onError)="onError($any($event))"
(onStateChange)="errors.set(undefined)"
>
<div>
<label for="email"> Email </label>
<input type="email" name="email" />
</div>

<button class="button" type="submit">Submit</button>
</form>

@if (errors()?.email) {
<p>{{ errors()?.email }}</p>
}
} @else {
<div>Thanks for signing up!</div>
}
`,
})
export default class NewsletterComponent {
signedUp = signal(false);
errors = signal<FormErrors>(undefined);

onSuccess() {
this.signedUp.set(true);
}

onError(result?: FormErrors) {
this.errors.set(result);
}
}

FormAction 指令将表单数据提交到服务器,由其处理程序处理。

处理表单操作

要处理表单操作,请在包含异步 action 函数的 .page.ts 文件旁边定义 .server.ts 文件,以处理表单提交。

在服务器操作中,您可以使用访问环境变量、读取 cookie 和执行其他仅限服务器端的操作。

// src/app/pages/newsletter.server.ts
import {
type PageServerAction,
redirect,
json,
fail,
} from '@analogjs/router/server/actions';
import { readFormData } from 'h3';

export async function action({ event }: PageServerAction) {
const body = await readFormData(event);
const email = body.get('email') as string;

if (!email) {
return fail(422, { email: 'Email is required' });
}

if (email.length < 10) {
return redirect('/');
}

return json({ type: 'success' });
}
  • json 函数返回 JSON 响应。
  • redirect 函数返回重定向响应给客户端。这个路径应该是绝对路径。
  • fail 函数用于返回表单验证错误。

处理多个表单

要在同一页面上处理多个表单,请添加一个隐藏输入以区分每个表单。

<form method="post">
<div>
<label for="email"> Email </label>
<input type="email" name="email" />
</div>

<input type="hidden" name="action" value="register" />

<button class="button" type="submit">Submit</button>
</form>

在服务器操作中,使用 action 值。

export async function action({ event }: PageServerAction) {
const body = await readFormData(event);
const action = body.get('action') as string;

if (action === 'register') {
// 处理注册表单
}
}

处理 GET 请求

具有 GET 操作的表单可用于导航到相同的 URL,并将表单输入作为查询参数传递。

下面的示例定义了一个搜索表单,其中 search 字段作为查询参数。

// src/app/pages/search.page.ts
import { Component, computed } from '@angular/core';
import { toSignal } from '@angular/core/rxjs-interop';
import { injectLoad, FormAction } from '@analogjs/router';

import type { load } from './search.server';

@Component({
selector: 'app-search-page',
standalone: true,
imports: [FormAction],
template: `
<h3>Search</h3>

<form method="get">
<div>
<label for="search"> Search </label>
<input type="text" name="search" [value]="searchTerm()" />
</div>

<button class="button" type="submit">Submit</button>
</form>

@if (searchTerm()) {
<p>Search Term: {{ searchTerm() }}</p>
}
`,
})
export default class NewsletterComponent {
loader = toSignal(injectLoad<typeof load>(), { requireSync: true });
searchTerm = computed(() => this.loader().searchTerm);
}

可以通过服务器表单操作访问查询参数。

// src/app/pages/search.server.ts
import type { PageServerLoad } from '@analogjs/router';
import { getQuery } from 'h3';

export async function load({ event }: PageServerLoad) {
const query = getQuery(event);
console.log('loaded search', query['search']);

return {
loaded: true,
searchTerm: `${query['search']}`,
};
}