Mengapa tindakan server Next.js mengubah permainan!
Saya tidak tahu tentang Anda, tetapi setiap kali rilis Next.js baru diumumkan, saya sangat bersemangat untuk melihat fitur baru apa yang telah dirilis. Saat ini, Vercel sedang dalam langkah kunci dengan React dalam memindahkan sebanyak mungkin ke server, yang sangat menarik untuk dilihat! Minggu lalu, Vercel mengumumkan perilisan Next.js 13.4 , yang menyertakan paradigma baru untuk memperbarui data di server yang disebut Server Actions.
Tindakan Server dirancang untuk mengaktifkan pembaruan data sisi server, mengurangi JavaScript sisi klien, dan menyediakan formulir yang ditingkatkan secara progresif. Dalam artikel ini, kita akan mendalami dasar-dasar Server Actions, menjelajahi cara membuat dan memanggilnya, dan mendiskusikan berbagai peningkatan dan manfaat yang dibawanya ke aplikasi Next.js Anda.
ℹ️ Perhatikan bahwa contoh-contoh ini diambil langsung dari situs web dokumentasi Next.js. Lihat di bawah untuk contoh panduan formulir multi-langkah.
Mengaktifkan Tindakan Server
Untuk mengaktifkan Tindakan Server di proyek Next.js Anda, Anda perlu menyetel serverActions
bendera eksperimental di next.config.js
file Anda:
const nextConfig = {
experimental: {
serverActions: true,
},
};
Membuat Server Action semudah membuat fungsi asinkron dengan arahan "use server" di bagian atas badan fungsi. Fungsi harus memiliki argumen serializable dan nilai kembalian serializable, berdasarkan protokol React Server Components. Berikut contohnya:
async function myAction() {
"use server"
// Your server action logic here
}
Anda juga dapat menggunakan arahan "gunakan server" tingkat atas di atas file, yang berguna jika Anda memiliki satu file yang mengekspor beberapa tindakan server, dan diperlukan jika Anda mengimpor tindakan server di komponen klien.
"use server"
export async function myAction() {
// Your server action logic here
}
Menurut situs web dokumentasi Next.js, ada beberapa cara untuk menjalankan Server Actions:
- Menggunakan
action
prop: Prop Reactaction
memungkinkan Anda untuk menjalankan aksi server pada suatu<form>
elemen.
export default function AddToCart({ productId }) {
async function addItem(data) {
'use server'
const cartId = cookies().get('cartId')?.value;
await saveToDb({ cartId, data });
}
return (
<form action={addItem}>
<button type="submit">Add to Cart</button>
</form>
)
}
export default function Form() {
async function handleSubmit() {
"use server"
// Your server action logic here
}
async function submitImage() {
"use server"
// Your server action logic here
}
return (
<form action={handleSubmit}>
<input type="text" name="name" />
<input type="image" formAction={submitImage} />
<button type="submit">Submit</button>
</form>
)
}
'use client'
import { experimental_useOptimistic as useOptimistic } from 'react'
import { send } from './_actions.js'
export function Thread({ messages }) {
const [optimisticMessages, addOptimisticMessage] = useOptimistic(
messages,
(state, newMessage) => [...state, { message: newMessage, sending: true }],
)
const formRef = useRef();
return (
<div>
{optimisticMessages.map((m) => (
<div>
{m.message}
{m.sending ? 'Sending...' : ''}
</div>
))}
<form
action={async (formData) => {
const message = formData.get('message')
formRef.current.reset()
addOptimisticMessage(message)
await send(message)
}}
ref={formRef}
>
<input type="text" name="message" />
</form>
</div>
)
}
- Jika Tindakan Klien diteruskan ke
<form>
, formulir tetap interaktif, tetapi tindakan ditempatkan dalam antrean hingga formulir terhidrasi. Diutamakan<form>
dengan Hidrasi Selektif, sehingga terjadi dengan cepat.
// src/components/MyForm.js
'use client'
import { useState } from 'react'
import { handleSubmit } from './actions.js'
export default function MyForm({ myAction }) {
const [input, setInput] = useState();
return (
<form action={handleSubmit} onChange={(e) => setInput(e.target.value)}>
...
</form>
)
}
Mari satukan semua ini untuk membuat contoh alur pendaftaran wizard menggunakan tindakan server untuk menyimpan data di setiap langkah alur.
Pertama, kami akan menambahkan halaman:
// app/register/page.tsx
import RegistrationForm from 'src/components/RegistrationForm'
export default function Register() {
return (
<div>
<h1>Registration</h1>
<RegistrationForm />
</div>
)
}
// components/RegistrationForm.tsx
'use client'
import React, { useState } from 'react';
import {
savePersonalInformation,
saveContactInformation,
saveAccountDetails
} from '../../actions'
import PersonalInformationStep from './PersonalInformationStep'
import ContactInformationStep from './ContactInformationStep'
import AccountDetailsStep from './AccountDetailsStep'
const WIZARD_STEPS = [
{
Component: PersonalInformationStep,
action: savePersonalInformation,
},
{
Component: ContactInformationStep,
action: saveContactInformation,
},
{
Component: AccountDetailsStep,
action: saveAccountDetails,
},
]
export default function RegistrationForm() {
const [step, setStep] = useState(0);
const { Component, action } = WIZARD_STEPS[step]
const handleSubmit = async (data: FormData) => {
await action(data);
if (step < WIZARD_STEPS.length - 1) {
setStep((currentStep) => currentStep + 1)
} else {
// Submit multi-step wizard form and navigate
}
}
return (
<Component onSubmit={handleSubmit} />
)
}
// actions/index.ts
'use server'
import { cookies } from 'next/headers'
import prisma from 'prisma'
import { hashPassword } from 'src/utils/auth'
export async function savePersonalInformation(data: FormData) {
const user = await prisma.user.create({
data: {
firstName: data.get('firstName').toString(),
lastName: data.get('lastName').toString(),
},
})
cookies().set('userId', user.id)
}
export async function saveContactInformation(data: FormData) {
// Save contact information to the database
const id = cookies().get('userId')?.value
await prisma.user.update({
where: { id },
data: {
email: data.get('email').toString(),
phone: data.get('phone').toString(),
},
})
}
export async function saveAccountDetails(data: FormData) {
// Save account details
const id = cookies().get('userId')?.value
const password = await hashPassword(
data.get('password').toString()
);
await prisma.user.update({
where: { id },
data: {
username: data.get('username').toString(),
password,
},
})
}
// components/WizardStep.tsx
import React from 'react'
import { experimental_useFormStatus as useFormStatus } from 'react-dom'
type Props = {
title: string;
onSubmit: (data: FormData) => Promise<void>
children: React.ReactNode
}
export default function FormStep({ title, onSubmit, children }: Props) {
const { pending } = useFormStatus();
return (
<form
action={async (formData) => {
await onSubmit(formData);
}}
>
<h2>{title}</h2>
{children}
<button
type="submit"
className={pending ? 'button-submitting' : 'button'}
>
Next
</button>
</form>
)
}
// ./PersonalInformationStep.tsx
import React from 'react'
import WizardStep from '../WizardStep'
type Props = {
onSubmit: (data: FormData) => Promise<void>
}
export default function PersonalInformationStep({ onSubmit }: Props) {
return (
<WizardStep title="Personal Information" onSubmit={onSubmit}>
<label htmlFor="firstName">First Name:</label>
<input type="text" id="firstName" name="firstName" required />
<label htmlFor="lastName">Last Name:</label>
<input type="text" id="lastName" name="lastName" required />
</WizardStep>
)
}
Mengapa tindakan server mengubah permainan
Sebelum kita menyimpulkan, mari luangkan waktu sejenak untuk membahas mengapa Tindakan Server Next.js merupakan tambahan yang sangat kuat untuk ekosistem Next.js dan bagaimana mereka siap untuk mendefinisikan ulang praktik terbaik pengembangan web.
- JavaScript Sisi Klien yang Dikurangi: Next.js dan React bergerak ke arah meminimalkan bundel sisi klien dengan membongkar logika ke server. Next.js memungkinkan kami untuk memindahkan banyak kode ke server, tetapi sebagian besar kami masih membangun formulir di klien. Itu berubah dengan tindakan server!
- Pengalaman pengguna yang ditingkatkan: Dengan fitur seperti pembaruan yang optimis dan peningkatan progresif, pengembang dapat membuat aplikasi yang terasa lebih responsif dan mulus. Pengguna tidak perlu lagi menunggu respons server untuk melihat efek dari tindakan mereka, dan formulir tetap interaktif bahkan tanpa JavaScript, yang menghasilkan pengalaman pengguna yang lebih baik secara keseluruhan bahkan sebelum klien terhidrasi.
- Pengalaman pengembang yang luar biasa: Tindakan Server memudahkan pemindahan pembaruan data ke server, yang sebelumnya membutuhkan API. API yang disediakan oleh Next.js dan React membuatnya sangat mudah digunakan.
Singkatnya, Next.js Server Actions adalah fitur pengubah permainan yang berpotensi merevolusi cara pengembang web mendekati mutasi data sisi server dan penanganan formulir. Dengan menawarkan pengalaman pengembangan yang lebih responsif, terukur, dan efisien, Tindakan Server siap untuk menjadi alat yang sangat diperlukan untuk pengembangan web modern.