了解 React 新 Hooks
React 19 引入了幾個強大的新鉤子,徹底改變了我們在應用程式中處理表單和管理樂觀更新的方式。在本部落格中,我們將探討 useFormStatus、useActionState 和 useOptimistic - 三個使我們的 React 應用程式響應更快且用戶友好的鉤子。
useFormStatus:增強的表單處理
useFormStatus 掛鉤提供有關表單提交的即時信息,使創建響應式且可訪問的表單變得更加容易。讓我們探討一下這個鉤子如何改進 React 18 的表單處理能力。
範例1:基本表單載入狀態
function SubmitButton() { const { pending } = useFormStatus(); return ( <button disabled={pending}> {pending ? 'Submitting...' : 'Submit'} </button> ); } function SignupForm() { return ( <form action={async (formData) => { await submitSignupData(formData); }}> <input name="email" type="email" /> <input name="password" type="password" /> <SubmitButton /> </form> ); }
在 React 18 中,您需要使用 useState 手動管理載入狀態。新的 useFormStatus 掛鉤會自動處理此問題,從而減少樣板程式碼。
範例 2:多種表單狀態
function FormStatus() { const { pending, data, method } = useFormStatus(); return ( <div role="status"> {pending && <span>Submitting via {method}...</span>} {!pending && data && <span>Last submission: {new Date().toLocaleString()}</span>} </div> ); } function ContactForm() { return ( <form action={async (formData) => { await submitContactForm(formData); }}> <textarea name="message" /> <FormStatus /> <SubmitButton /> </form> ); }
範例 3:表單驗證狀態
function ValidationStatus() { const { pending, validationErrors } = useFormStatus(); return ( <div role="alert"> {validationErrors?.map((error, index) => ( <p key={index} className="error">{error}</p> ))} </div> ); } function RegistrationForm() { return ( <form action={async (formData) => { const errors = validateRegistration(formData); if (errors.length) throw errors; await register(formData); }}> <input name="username" /> <input name="email" type="email" /> <ValidationStatus /> <SubmitButton /> </form> ); }
範例 4:多步驟表單進度
function FormProgress() { const { pending, step, totalSteps } = useFormStatus(); return ( <div className="progress-bar"> <div className="progress" > <h3> Example 5: File Upload Progress </h3> <pre class="brush:php;toolbar:false">function UploadProgress() { const { pending, progress } = useFormStatus(); return ( <div> {pending && progress && ( <div className="upload-progress"> <div className="progress-bar" > <h2> useActionState: Managing Action Results </h2> <p>The useActionState hook provides a way to track the state of form actions and server mutations, making it easier to handle success and error states.</p> <h3> Example 1: Basic Action State </h3> <pre class="brush:php;toolbar:false">function SubmissionStatus() { const state = useActionState(); return ( <div> {state.status === 'success' && <p>Submission successful!</p>} {state.status === 'error' && <p>Error: {state.error.message}</p>} </div> ); } function CommentForm() { return ( <form action={async (formData) => { await submitComment(formData); }}> <textarea name="comment" /> <SubmissionStatus /> <SubmitButton /> </form> ); }
範例 2:動作歷史記錄
function ActionHistory() { const state = useActionState(); return ( <div> <h3>Recent Actions</h3> <ul> {state.history.map((action, index) => ( <li key={index}> {action.type} - {action.timestamp} {action.status === 'error' && ` (Failed: ${action.error.message})`} </li> ))} </ul> </div> ); }
範例3:重試機制
function RetryableAction() { const state = useActionState(); return ( <div> {state.status === 'error' && ( <button onClick={() => state.retry()} disabled={state.retrying} > {state.retrying ? 'Retrying...' : 'Retry'} </button> )} </div> ); }
範例 4:操作佇列
function ActionQueue() { const state = useActionState(); return ( <div> <h3>Pending Actions</h3> {state.queue.map((action, index) => ( <div key={index}> {action.type} - Queued at {action.queuedAt} <button onClick={() => action.cancel()}>Cancel</button> </div> ))} </div> ); }
範例 5:動作統計
function ActionStats() { const state = useActionState(); return ( <div> <h3>Action Statistics</h3> <p>Success Rate: {state.stats.successRate}%</p> <p>Average Duration: {state.stats.avgDuration}ms</p> <p>Total Actions: {state.stats.total}</p> </div> ); }
useOptimistic:流暢的 UI 更新
useOptimistic 掛鉤可以在等待伺服器回應時立即更新 UI,從而創建響應更快的使用者體驗。
範例 1:樂觀的待辦事項列表
function TodoList() { const [todos, setTodos] = useState([]); const [optimisticTodos, addOptimisticTodo] = useOptimistic( todos, (state, newTodo) => [...state, newTodo] ); async function addTodo(formData) { const newTodo = { id: Date.now(), text: formData.get('todo'), completed: false }; addOptimisticTodo(newTodo); await saveTodo(newTodo); } return ( <div> <form action={addTodo}> <input name="todo" /> <button type="submit">Add Todo</button> </form> <ul> {optimisticTodos.map(todo => ( <li key={todo.id}>{todo.text}</li> ))} </ul> </div> ); }
範例 2:樂觀點讚按鈕
function LikeButton({ postId, initialLikes }) { const [likes, setLikes] = useState(initialLikes); const [optimisticLikes, addOptimisticLike] = useOptimistic( likes, (state) => state + 1 ); async function handleLike() { addOptimisticLike(); await likePost(postId); } return ( <button onClick={handleLike}> {optimisticLikes} Likes </button> ); }
範例 3:樂觀評論線程
function CommentThread({ postId }) { const [comments, setComments] = useState([]); const [optimisticComments, addOptimisticComment] = useOptimistic( comments, (state, newComment) => [...state, newComment] ); async function submitComment(formData) { const comment = { id: Date.now(), text: formData.get('comment'), pending: true }; addOptimisticComment(comment); await saveComment(postId, comment); } return ( <div> <form action={submitComment}> <textarea name="comment" /> <button type="submit">Comment</button> </form> {optimisticComments.map(comment => ( <div key={comment.id}> <h3> Example 4: Optimistic Shopping Cart </h3> <pre class="brush:php;toolbar:false">function ShoppingCart() { const [cart, setCart] = useState([]); const [optimisticCart, updateOptimisticCart] = useOptimistic( cart, (state, update) => { const { type, item } = update; switch (type) { case 'add': return [...state, item]; case 'remove': return state.filter(i => i.id !== item.id); case 'update': return state.map(i => i.id === item.id ? item : i); default: return state; } } ); async function updateCart(type, item) { updateOptimisticCart({ type, item }); await saveCart({ type, item }); } return ( <div> {optimisticCart.map(item => ( <div key={item.id}> {item.name} - ${item.price} <button onClick={() => updateCart('remove', item)}>Remove</button> </div> ))} </div> ); }
範例 5:樂觀用戶設置
function UserSettings() { const [settings, setSettings] = useState({}); const [optimisticSettings, updateOptimisticSetting] = useOptimistic( settings, (state, update) => ({ ...state, [update.key]: update.value }) ); async function updateSetting(key, value) { updateOptimisticSetting({ key, value }); await saveSettings({ [key]: value }); } return ( <div> <div> <label> Theme: <select value={optimisticSettings.theme} onChange={e => updateSetting('theme', e.target.value)} > <option value="light">Light</option> <option value="dark">Dark</option> </select> </label> </div> <div> <label> Notifications: <input type="checkbox" checked={optimisticSettings.notifications} onChange={e => updateSetting('notifications', e.target.checked)} /> </label> </div> </div> ); }
在應用程式中使用這些鉤子時,請記住查看官方 React 文件以獲取最新資訊和最佳實踐。
編碼快樂!
以上是了解 React 新 Hooks的詳細內容。更多資訊請關注PHP中文網其他相關文章!

熱AI工具

Undresser.AI Undress
人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover
用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool
免費脫衣圖片

Clothoff.io
AI脫衣器

Video Face Swap
使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

熱工具

記事本++7.3.1
好用且免費的程式碼編輯器

SublimeText3漢化版
中文版,非常好用

禪工作室 13.0.1
強大的PHP整合開發環境

Dreamweaver CS6
視覺化網頁開發工具

SublimeText3 Mac版
神級程式碼編輯軟體(SublimeText3)

JavaScript是現代Web開發的基石,它的主要功能包括事件驅動編程、動態內容生成和異步編程。 1)事件驅動編程允許網頁根據用戶操作動態變化。 2)動態內容生成使得頁面內容可以根據條件調整。 3)異步編程確保用戶界面不被阻塞。 JavaScript廣泛應用於網頁交互、單頁面應用和服務器端開發,極大地提升了用戶體驗和跨平台開發的靈活性。

Python和JavaScript開發者的薪資沒有絕對的高低,具體取決於技能和行業需求。 1.Python在數據科學和機器學習領域可能薪資更高。 2.JavaScript在前端和全棧開發中需求大,薪資也可觀。 3.影響因素包括經驗、地理位置、公司規模和特定技能。

如何在JavaScript中將具有相同ID的數組元素合併到一個對像中?在處理數據時,我們常常會遇到需要將具有相同ID�...

學習JavaScript不難,但有挑戰。 1)理解基礎概念如變量、數據類型、函數等。 2)掌握異步編程,通過事件循環實現。 3)使用DOM操作和Promise處理異步請求。 4)避免常見錯誤,使用調試技巧。 5)優化性能,遵循最佳實踐。

實現視差滾動和元素動畫效果的探討本文將探討如何實現類似資生堂官網(https://www.shiseido.co.jp/sb/wonderland/)中�...

JavaScript的最新趨勢包括TypeScript的崛起、現代框架和庫的流行以及WebAssembly的應用。未來前景涵蓋更強大的類型系統、服務器端JavaScript的發展、人工智能和機器學習的擴展以及物聯網和邊緣計算的潛力。

深入探討console.log輸出差異的根源本文將分析一段代碼中console.log函數輸出結果的差異,並解釋其背後的原因。 �...
