From 6e238878685e1051c70aa08abff927ac9c1b2da8 Mon Sep 17 00:00:00 2001 From: Edoardo Ranghieri Date: Wed, 14 Aug 2024 18:38:38 +0200 Subject: [PATCH] fix(hooks): add `isTransitioning` and `isPending` shorthand statuses to return object (#231) This PR introduces a workaround for the issues described in #227, by adding `isTransitioning` and `isPending` shorthand statuses to the hooks return object. `isTransitioning` is the value returned from the `useTransition` hook that is used under the hood, while `isPending` has the value of `isExecuting || isTransitioning`, for convenience. --- apps/playground/package.json | 4 +- .../src/app/(examples)/hook/page.tsx | 12 +- packages/next-safe-action/src/hooks-utils.ts | 22 +++- packages/next-safe-action/src/hooks.ts | 8 +- .../next-safe-action/src/stateful-hooks.ts | 4 +- pnpm-lock.yaml | 112 +++++++++--------- .../execution/hooks/check-action-status.md | 23 ++++ .../docs/execution/hooks/hook-callbacks.md | 2 +- website/docs/execution/hooks/useaction.md | 2 + .../execution/hooks/useoptimisticaction.md | 2 + .../docs/execution/hooks/usestateaction.md | 2 + website/docs/safe-action-client/middleware.md | 4 +- 12 files changed, 124 insertions(+), 73 deletions(-) create mode 100644 website/docs/execution/hooks/check-action-status.md diff --git a/apps/playground/package.json b/apps/playground/package.json index c890065c..cca628a6 100644 --- a/apps/playground/package.json +++ b/apps/playground/package.json @@ -12,7 +12,7 @@ "dependencies": { "@hookform/resolvers": "^3.3.4", "lucide-react": "^0.378.0", - "next": "15.0.0-canary.75", + "next": "15.0.0-canary.115", "next-safe-action": "workspace:*", "react": "19.0.0-rc-512b09b2-20240718", "react-dom": "19.0.0-rc-512b09b2-20240718", @@ -26,7 +26,7 @@ "@types/react-dom": "18.3.0", "autoprefixer": "10.4.19", "eslint": "^8.57.0", - "eslint-config-next": "15.0.0-canary.75", + "eslint-config-next": "15.0.0-canary.115", "postcss": "8.4.38", "tailwindcss": "3.4.3", "typescript": "^5.5.4" diff --git a/apps/playground/src/app/(examples)/hook/page.tsx b/apps/playground/src/app/(examples)/hook/page.tsx index 9d24977a..4ae7a45e 100644 --- a/apps/playground/src/app/(examples)/hook/page.tsx +++ b/apps/playground/src/app/(examples)/hook/page.tsx @@ -17,6 +17,8 @@ export default function Hook() { reset, isIdle, isExecuting, + isTransitioning, + isPending, hasSucceeded, hasErrored, } = useAction(deleteUser, { @@ -34,7 +36,15 @@ export default function Hook() { }, }); - console.dir({ status, isIdle, isExecuting, hasSucceeded, hasErrored }); + console.dir({ + status, + isIdle, + isExecuting, + isTransitioning, + isPending, + hasSucceeded, + hasErrored, + }); return (
diff --git a/packages/next-safe-action/src/hooks-utils.ts b/packages/next-safe-action/src/hooks-utils.ts index 7a9af7b4..b770af8e 100644 --- a/packages/next-safe-action/src/hooks-utils.ts +++ b/packages/next-safe-action/src/hooks-utils.ts @@ -36,10 +36,18 @@ export const getActionStatus = < } }; -export const getActionShorthandStatusObject = (status: HookActionStatus) => { +export const getActionShorthandStatusObject = ({ + status, + isTransitioning, +}: { + status: HookActionStatus; + isTransitioning: boolean; +}) => { return { isIdle: status === "idle", isExecuting: status === "executing", + isTransitioning, + isPending: status === "executing" || isTransitioning, hasSucceeded: status === "hasSucceeded", hasErrored: status === "hasErrored", }; @@ -102,12 +110,16 @@ export const useActionCallbacks = < await Promise.resolve(onExecute?.({ input })); break; case "hasSucceeded": - await Promise.resolve(onSuccess?.({ data: result?.data, input })); - await Promise.resolve(onSettled?.({ result, input })); + await Promise.all([ + Promise.resolve(onSuccess?.({ data: result?.data, input })), + Promise.resolve(onSettled?.({ result, input })), + ]); break; case "hasErrored": - await Promise.resolve(onError?.({ error: result, input })); - await Promise.resolve(onSettled?.({ result, input })); + await Promise.all([ + Promise.resolve(onError?.({ error: result, input })), + Promise.resolve(onSettled?.({ result, input })), + ]); break; } }; diff --git a/packages/next-safe-action/src/hooks.ts b/packages/next-safe-action/src/hooks.ts index 490de54a..3c7d5c9c 100644 --- a/packages/next-safe-action/src/hooks.ts +++ b/packages/next-safe-action/src/hooks.ts @@ -30,7 +30,7 @@ export const useAction = < safeActionFn: HookSafeActionFn, utils?: HookBaseUtils & HookCallbacks ) => { - const [, startTransition] = React.useTransition(); + const [isTransitioning, startTransition] = React.useTransition(); const [result, setResult] = React.useState>({}); const [clientInput, setClientInput] = React.useState : void>(); const [isExecuting, setIsExecuting] = React.useState(false); @@ -128,7 +128,7 @@ export const useAction = < result, reset, status, - ...getActionShorthandStatusObject(status), + ...getActionShorthandStatusObject({ status, isTransitioning }), }; }; @@ -155,7 +155,7 @@ export const useOptimisticAction = < } & HookBaseUtils & HookCallbacks ) => { - const [, startTransition] = React.useTransition(); + const [isTransitioning, startTransition] = React.useTransition(); const [result, setResult] = React.useState>({}); const [clientInput, setClientInput] = React.useState : void>(); const [isExecuting, setIsExecuting] = React.useState(false); @@ -260,7 +260,7 @@ export const useOptimisticAction = < optimisticState, reset, status, - ...getActionShorthandStatusObject(status), + ...getActionShorthandStatusObject({ status, isTransitioning }), }; }; diff --git a/packages/next-safe-action/src/stateful-hooks.ts b/packages/next-safe-action/src/stateful-hooks.ts index 524838d9..049f6c57 100644 --- a/packages/next-safe-action/src/stateful-hooks.ts +++ b/packages/next-safe-action/src/stateful-hooks.ts @@ -34,7 +34,7 @@ export const useStateAction = < utils?.permalink ); const [isIdle, setIsIdle] = React.useState(true); - const [, startTransition] = React.useTransition(); + const [isTransitioning, startTransition] = React.useTransition(); const [clientInput, setClientInput] = React.useState : void>(); const status = getActionStatus({ isExecuting, @@ -78,6 +78,6 @@ export const useStateAction = < input: clientInput, result, status, - ...getActionShorthandStatusObject(status), + ...getActionShorthandStatusObject({ status, isTransitioning }), }; }; diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 494096fa..e69f8d30 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -42,8 +42,8 @@ importers: specifier: ^0.378.0 version: 0.378.0(react@19.0.0-rc-512b09b2-20240718) next: - specifier: 15.0.0-canary.75 - version: 15.0.0-canary.75(react-dom@19.0.0-rc-512b09b2-20240718(react@19.0.0-rc-512b09b2-20240718))(react@19.0.0-rc-512b09b2-20240718) + specifier: 15.0.0-canary.115 + version: 15.0.0-canary.115(react-dom@19.0.0-rc-512b09b2-20240718(react@19.0.0-rc-512b09b2-20240718))(react@19.0.0-rc-512b09b2-20240718) next-safe-action: specifier: workspace:* version: link:../../packages/next-safe-action @@ -79,8 +79,8 @@ importers: specifier: ^8.57.0 version: 8.57.0 eslint-config-next: - specifier: 15.0.0-canary.75 - version: 15.0.0-canary.75(eslint@8.57.0)(typescript@5.5.4) + specifier: 15.0.0-canary.115 + version: 15.0.0-canary.115(eslint@8.57.0)(typescript@5.5.4) postcss: specifier: 8.4.38 version: 8.4.38 @@ -729,11 +729,11 @@ packages: '@next/env@14.3.0-canary.42': resolution: {integrity: sha512-1WqxEMzSd9nULVLMy0ctLsX9T934jQDblMyO5SujzXURG4jCfZBJud4LNamU748Iu/s1ScR7PsDyPmR1wZlKeQ==} - '@next/env@15.0.0-canary.75': - resolution: {integrity: sha512-sVRrLrZsNacn/S3Bp3uO3WSyMs+xJkQ6DRKwxPRto3tsoLit2jyoLnBjqs35gP4qlbXdoiEpC+MgGa0I2l44HQ==} + '@next/env@15.0.0-canary.115': + resolution: {integrity: sha512-pbiR6CUF2cfl48JvUkuL57z1DHPelMBUPLy+RtaW6rBQbhfjMiiEMBHlW2OZk+MI9To0JH0Qvh08H5VkHu0tNw==} - '@next/eslint-plugin-next@15.0.0-canary.75': - resolution: {integrity: sha512-DkIbyO2vyfxmCBiTPedkclyELSQGX5KNTcTYac7nrnYZPxVqC9WocJT7QJoeLpPIKrZBDDNl2XGYbNara04mZg==} + '@next/eslint-plugin-next@15.0.0-canary.115': + resolution: {integrity: sha512-LLHWknVxxPaBkLFAnFR4zQ691UoFlM6+h0LyHEBjIU7ZO1v28fqvZe26r9R5TN7I5VZnypBb/pbJ47oYi+ddjw==} '@next/swc-darwin-arm64@14.3.0-canary.42': resolution: {integrity: sha512-0FToZPLxC6QprBwHZMfEpVsvfWyD0owlh4/R3bYH8qJ/L+jFFwnrfE9TgQf+xQPQEps/9D5b4Z6yA70BKNaFiA==} @@ -741,8 +741,8 @@ packages: cpu: [arm64] os: [darwin] - '@next/swc-darwin-arm64@15.0.0-canary.75': - resolution: {integrity: sha512-Q5QJzdgitI7hyiJx9S+1mn5JxmbAdEfysritXnTBIXgSLmuwb+BIOn26yxwzg/RpFju7OiX5zuZB6tY9Uc8nsw==} + '@next/swc-darwin-arm64@15.0.0-canary.115': + resolution: {integrity: sha512-VFFuMiVH+xE45r0hgLzhO1LnumLvppTnWraAM4f1VMQE3SJH6z+NaXpGoxGYI+F271pGxoRR2MCs6l8FP3n5MQ==} engines: {node: '>= 10'} cpu: [arm64] os: [darwin] @@ -753,8 +753,8 @@ packages: cpu: [x64] os: [darwin] - '@next/swc-darwin-x64@15.0.0-canary.75': - resolution: {integrity: sha512-59dBcPrwOl2sd41LnWSAm48LkixbzyCIn4saaZ9eqW9zCst80PuqerI2oyPELPv38XpTomNTquEwIb60ZwUszQ==} + '@next/swc-darwin-x64@15.0.0-canary.115': + resolution: {integrity: sha512-UXIc4ChPOh2Epm2J+a819M6G3Hdj+7NT0yWJfHzNDc+YLIrHGWfualP7o92U5z0uXDjf5kJodfZTND2Y5WmXUw==} engines: {node: '>= 10'} cpu: [x64] os: [darwin] @@ -765,8 +765,8 @@ packages: cpu: [arm64] os: [linux] - '@next/swc-linux-arm64-gnu@15.0.0-canary.75': - resolution: {integrity: sha512-IQZcCOWCJhQzxz7cGONKWp11VStinWwz33AfVNwHMIx2+UVr2+BhaShVIsqa7PnFBY7DvXoQHJawK2suc0LMsw==} + '@next/swc-linux-arm64-gnu@15.0.0-canary.115': + resolution: {integrity: sha512-/xmn1z2gfUGvt0QhBpe4UyTUKOnDuH/OnCPTjmw/RhLrDTlTwFxNNVQ5ydwtvOpdPAhKn3PyGPfdQxbViysThg==} engines: {node: '>= 10'} cpu: [arm64] os: [linux] @@ -777,8 +777,8 @@ packages: cpu: [arm64] os: [linux] - '@next/swc-linux-arm64-musl@15.0.0-canary.75': - resolution: {integrity: sha512-Ls8TRqkOiIgPZoDcsD6P8z4qgQu7kAeeDfXNDwYuT+wf91qH0rCwPt0mBpHxk1jjs7y0auQ82uJAjIXgp5kupw==} + '@next/swc-linux-arm64-musl@15.0.0-canary.115': + resolution: {integrity: sha512-lG0pWv++A/nf9ePfH1nX+D1Gpc2TO4aPjl+DV9oZEreDG76gUP8+woGD+o1Y+Yrb1ffCrMYdsVf+zWiWBrcWDQ==} engines: {node: '>= 10'} cpu: [arm64] os: [linux] @@ -789,8 +789,8 @@ packages: cpu: [x64] os: [linux] - '@next/swc-linux-x64-gnu@15.0.0-canary.75': - resolution: {integrity: sha512-GONyJruBfUct03xyL+WzsgZQ9pd/z8jYRtYgdfavWJTvnj4a5s0L6ywZ31dzXIpPu8HyUV+IJeO9zWOn6GEufg==} + '@next/swc-linux-x64-gnu@15.0.0-canary.115': + resolution: {integrity: sha512-1t5zlrDBA9ZgJIv7eEqnItgyNEdPA0rlrckzhFqPU9aD+j4GJSPt9edTcyKnNv88+uGSMvghTepxn4M5mJBJow==} engines: {node: '>= 10'} cpu: [x64] os: [linux] @@ -801,8 +801,8 @@ packages: cpu: [x64] os: [linux] - '@next/swc-linux-x64-musl@15.0.0-canary.75': - resolution: {integrity: sha512-qCGNY2ox1+KGv0t+6ODm5qvWTKjP36mkFiEhWeGk1P7jw1pV8rLenHo6dDD+v10A1C5oaq5UWNWFSdC7AIvh2Q==} + '@next/swc-linux-x64-musl@15.0.0-canary.115': + resolution: {integrity: sha512-sR8MyiHiruUXACSLD3r8BOCueeZ25PDgFArw4cGxoBZ+Ts9t7n24nuCHJ1BNKWZP89LgKZ5F48+xcVfJ/7SuhQ==} engines: {node: '>= 10'} cpu: [x64] os: [linux] @@ -813,8 +813,8 @@ packages: cpu: [arm64] os: [win32] - '@next/swc-win32-arm64-msvc@15.0.0-canary.75': - resolution: {integrity: sha512-NOgRVnEU5DG3CB/gssEOmuRva5kyzKW36iJmb7LQHMVShzi/lg0eZI9vfGcNS/NWhN1qlKoVhw0e9BEXEZkFNw==} + '@next/swc-win32-arm64-msvc@15.0.0-canary.115': + resolution: {integrity: sha512-ItnToj9A0kpI1lBTHhVL87cWBAiqjRbts6ocGCTcn9w3rxuZ/h+DKlnAWDe6mX5PFxt3mndQQhX53GWrZ2sK4w==} engines: {node: '>= 10'} cpu: [arm64] os: [win32] @@ -825,8 +825,8 @@ packages: cpu: [ia32] os: [win32] - '@next/swc-win32-ia32-msvc@15.0.0-canary.75': - resolution: {integrity: sha512-XLosgh64YZV83fgYPX5Rn38ocqRjHXbDWXtZ9Yhc4BfPZnE5OahNabJylaHyJRvbdj4JnCqfgjY/DJm5iWIf0g==} + '@next/swc-win32-ia32-msvc@15.0.0-canary.115': + resolution: {integrity: sha512-zN2wCf1/bZpQmGeSUlx70Su0JfyxryUDfjUkIs1qRxQol2DeGaL4u4Sym7VIuq8gF99CicyfNfvinKUGBoT02g==} engines: {node: '>= 10'} cpu: [ia32] os: [win32] @@ -837,8 +837,8 @@ packages: cpu: [x64] os: [win32] - '@next/swc-win32-x64-msvc@15.0.0-canary.75': - resolution: {integrity: sha512-4vgzoMmqSM1PqmKKNpySnzzyRZsxRNHog43+qKN5l2oxWmITHeM2Vc98ThLm5lk/EJDsA4PQeN0XQ4sBHGRZFg==} + '@next/swc-win32-x64-msvc@15.0.0-canary.115': + resolution: {integrity: sha512-bbWwWzycLUSv8ypCEhoRbgQlu5xlH6QGhmj8msuhWIOg1PFFDplQtwqnvwISgxFnWTm+dxegRGeFR6CDQNRhQQ==} engines: {node: '>= 10'} cpu: [x64] os: [win32] @@ -1754,8 +1754,8 @@ packages: resolution: {integrity: sha512-/veY75JbMK4j1yjvuUxuVsiS/hr/4iHs9FTT6cgTexxdE0Ly/glccBAkloH/DofkjRbZU3bnoj38mOmhkZ0lHw==} engines: {node: '>=12'} - eslint-config-next@15.0.0-canary.75: - resolution: {integrity: sha512-ccfLBYKsrUffAduX+933mH0Qm3DVWV/U7ELl2SQuiZth/A0jh7ZkgAlsXwwY56pZssXa3xdLyN+jFCGDvdvmmw==} + eslint-config-next@15.0.0-canary.115: + resolution: {integrity: sha512-PvP0Z29oDZvs5oss7diO6CQ4d1X8qHldgmZ8T8XxAFXowlxck3/h4nEltsAf7uM9zdwxB7AJolIRXWhuVZ3ciQ==} peerDependencies: eslint: ^7.23.0 || ^8.0.0 typescript: '>=3.3.1' @@ -2749,16 +2749,16 @@ packages: sass: optional: true - next@15.0.0-canary.75: - resolution: {integrity: sha512-RJdWBrAt0CqyIEqHoz8MtJe6Mlttl8nArveh3+XTTGSCccedG+fiHWpzufgro1prVrItUAEu8yTtqENTDFLlvA==} + next@15.0.0-canary.115: + resolution: {integrity: sha512-kvIyTRt1LBnSriDw9OMMW1VaX8r4yVZ0DBCLw0iv5TXPf+Q4wUwu7VluKH+bnbYOK7eNe2denxvFLHt1lRLGUg==} engines: {node: '>=18.18.0'} hasBin: true peerDependencies: '@opentelemetry/api': ^1.1.0 '@playwright/test': ^1.41.2 babel-plugin-react-compiler: '*' - react: 19.0.0-rc.0 - react-dom: 19.0.0-rc.0 + react: 19.0.0-rc-187dd6a7-20240806 + react-dom: 19.0.0-rc-187dd6a7-20240806 sass: ^1.3.0 peerDependenciesMeta: '@opentelemetry/api': @@ -4433,64 +4433,64 @@ snapshots: '@next/env@14.3.0-canary.42': {} - '@next/env@15.0.0-canary.75': {} + '@next/env@15.0.0-canary.115': {} - '@next/eslint-plugin-next@15.0.0-canary.75': + '@next/eslint-plugin-next@15.0.0-canary.115': dependencies: fast-glob: 3.3.1 '@next/swc-darwin-arm64@14.3.0-canary.42': optional: true - '@next/swc-darwin-arm64@15.0.0-canary.75': + '@next/swc-darwin-arm64@15.0.0-canary.115': optional: true '@next/swc-darwin-x64@14.3.0-canary.42': optional: true - '@next/swc-darwin-x64@15.0.0-canary.75': + '@next/swc-darwin-x64@15.0.0-canary.115': optional: true '@next/swc-linux-arm64-gnu@14.3.0-canary.42': optional: true - '@next/swc-linux-arm64-gnu@15.0.0-canary.75': + '@next/swc-linux-arm64-gnu@15.0.0-canary.115': optional: true '@next/swc-linux-arm64-musl@14.3.0-canary.42': optional: true - '@next/swc-linux-arm64-musl@15.0.0-canary.75': + '@next/swc-linux-arm64-musl@15.0.0-canary.115': optional: true '@next/swc-linux-x64-gnu@14.3.0-canary.42': optional: true - '@next/swc-linux-x64-gnu@15.0.0-canary.75': + '@next/swc-linux-x64-gnu@15.0.0-canary.115': optional: true '@next/swc-linux-x64-musl@14.3.0-canary.42': optional: true - '@next/swc-linux-x64-musl@15.0.0-canary.75': + '@next/swc-linux-x64-musl@15.0.0-canary.115': optional: true '@next/swc-win32-arm64-msvc@14.3.0-canary.42': optional: true - '@next/swc-win32-arm64-msvc@15.0.0-canary.75': + '@next/swc-win32-arm64-msvc@15.0.0-canary.115': optional: true '@next/swc-win32-ia32-msvc@14.3.0-canary.42': optional: true - '@next/swc-win32-ia32-msvc@15.0.0-canary.75': + '@next/swc-win32-ia32-msvc@15.0.0-canary.115': optional: true '@next/swc-win32-x64-msvc@14.3.0-canary.42': optional: true - '@next/swc-win32-x64-msvc@15.0.0-canary.75': + '@next/swc-win32-x64-msvc@15.0.0-canary.115': optional: true '@nodelib/fs.scandir@2.1.5': @@ -5580,9 +5580,9 @@ snapshots: escape-string-regexp@5.0.0: {} - eslint-config-next@15.0.0-canary.75(eslint@8.57.0)(typescript@5.5.4): + eslint-config-next@15.0.0-canary.115(eslint@8.57.0)(typescript@5.5.4): dependencies: - '@next/eslint-plugin-next': 15.0.0-canary.75 + '@next/eslint-plugin-next': 15.0.0-canary.115 '@rushstack/eslint-patch': 1.10.3 '@typescript-eslint/eslint-plugin': 7.10.0(@typescript-eslint/parser@7.10.0(eslint@8.57.0)(typescript@5.5.4))(eslint@8.57.0)(typescript@5.5.4) '@typescript-eslint/parser': 7.10.0(eslint@8.57.0)(typescript@5.5.4) @@ -6678,9 +6678,9 @@ snapshots: - '@babel/core' - babel-plugin-macros - next@15.0.0-canary.75(react-dom@19.0.0-rc-512b09b2-20240718(react@19.0.0-rc-512b09b2-20240718))(react@19.0.0-rc-512b09b2-20240718): + next@15.0.0-canary.115(react-dom@19.0.0-rc-512b09b2-20240718(react@19.0.0-rc-512b09b2-20240718))(react@19.0.0-rc-512b09b2-20240718): dependencies: - '@next/env': 15.0.0-canary.75 + '@next/env': 15.0.0-canary.115 '@swc/counter': 0.1.3 '@swc/helpers': 0.5.12 busboy: 1.6.0 @@ -6691,15 +6691,15 @@ snapshots: react-dom: 19.0.0-rc-512b09b2-20240718(react@19.0.0-rc-512b09b2-20240718) styled-jsx: 5.1.6(react@19.0.0-rc-512b09b2-20240718) optionalDependencies: - '@next/swc-darwin-arm64': 15.0.0-canary.75 - '@next/swc-darwin-x64': 15.0.0-canary.75 - '@next/swc-linux-arm64-gnu': 15.0.0-canary.75 - '@next/swc-linux-arm64-musl': 15.0.0-canary.75 - '@next/swc-linux-x64-gnu': 15.0.0-canary.75 - '@next/swc-linux-x64-musl': 15.0.0-canary.75 - '@next/swc-win32-arm64-msvc': 15.0.0-canary.75 - '@next/swc-win32-ia32-msvc': 15.0.0-canary.75 - '@next/swc-win32-x64-msvc': 15.0.0-canary.75 + '@next/swc-darwin-arm64': 15.0.0-canary.115 + '@next/swc-darwin-x64': 15.0.0-canary.115 + '@next/swc-linux-arm64-gnu': 15.0.0-canary.115 + '@next/swc-linux-arm64-musl': 15.0.0-canary.115 + '@next/swc-linux-x64-gnu': 15.0.0-canary.115 + '@next/swc-linux-x64-musl': 15.0.0-canary.115 + '@next/swc-win32-arm64-msvc': 15.0.0-canary.115 + '@next/swc-win32-ia32-msvc': 15.0.0-canary.115 + '@next/swc-win32-x64-msvc': 15.0.0-canary.115 sharp: 0.33.4 transitivePeerDependencies: - '@babel/core' diff --git a/website/docs/execution/hooks/check-action-status.md b/website/docs/execution/hooks/check-action-status.md new file mode 100644 index 00000000..29a27d84 --- /dev/null +++ b/website/docs/execution/hooks/check-action-status.md @@ -0,0 +1,23 @@ +--- +sidebar_position: 6 +description: Check the current action execution status. +--- + +# Check action status + +You can check the current action execution status using either the `status` property or the shorthand properties returned by all hooks. + +## Via `status` property + +`status` property is a discriminated string, so it can only be one of the following values at any given time: `idle`, `executing`, `hasSucceeded`, `hasErrored`. + + +## Via shorthand properties + +Shorthand properties are convenience booleans that are `true` if the corresponding action status string is of the same value. Other than `isIdle`, `isExecuting`, `hasSucceeded` and `hasErrored`, also `isTransitioning` and `isPending` are returned by all three hooks. + +### Difference between `isExecuting`, `isTransitioning`, and `isPending` + +The difference between these three properties is that `isExecuting` is `true` when the Server Action is actually being executed, `isTransitioning` is true when the under the hood value from `useTransition` hook is `true` (so, when the transition is in progress), and `isPending` is `true` when `isExecuting` or `isTransitioning` are `true`. + +The safest and recommended way to check if the action is in progress is to use `isPending` property, because using just `isExecuting` could cause some weird glitches when navigation functions like `redirect` are used inside the Server Action. \ No newline at end of file diff --git a/website/docs/execution/hooks/hook-callbacks.md b/website/docs/execution/hooks/hook-callbacks.md index 261e1793..e208fc4a 100644 --- a/website/docs/execution/hooks/hook-callbacks.md +++ b/website/docs/execution/hooks/hook-callbacks.md @@ -23,4 +23,4 @@ Here is the full list of callbacks, with their behavior explained. All of them a | `onExecute?` | `"executing"` | | `onSuccess?` | `"hasSucceeded"` | | `onError?` | `"hasErrored"` | -| `onSettled?` | `"hasSucceeded"` or `"hasErrored"` (after `onSuccess` or `onError` call) | \ No newline at end of file +| `onSettled?` | `"hasSucceeded"` or `"hasErrored"` | \ No newline at end of file diff --git a/website/docs/execution/hooks/useaction.md b/website/docs/execution/hooks/useaction.md index b6bb272a..493eb330 100644 --- a/website/docs/execution/hooks/useaction.md +++ b/website/docs/execution/hooks/useaction.md @@ -81,7 +81,9 @@ As you can see, here we display a greet message after the action is performed, i | `reset` | `() => void` | Programmatically reset `input` and `result` object with this function. | | `status` | [`HookActionStatus`](/docs/types#hookresult) | The action current status. | | `isIdle` | `boolean` | True if the action status is `idle`. | +| `isTransitioning` | `boolean` | True if the transition status from the `useTransition` hook used under the hood is `true`. | | `isExecuting` | `boolean` | True if the action status is `executing`. | +| `isPending` | `boolean` | True if either `isTransitioning` or `isExecuting` are `true`. | | `hasSucceeded` | `boolean` | True if the action status is `hasSucceeded`. | | `hasErrored` | `boolean` | True if the action status is `hasErrored`. | diff --git a/website/docs/execution/hooks/useoptimisticaction.md b/website/docs/execution/hooks/useoptimisticaction.md index d2774973..1483df27 100644 --- a/website/docs/execution/hooks/useoptimisticaction.md +++ b/website/docs/execution/hooks/useoptimisticaction.md @@ -143,7 +143,9 @@ export default function TodosBox({ todos }: Props) { | `reset` | `() => void` | Programmatically reset `input` and `result` object with this function. | | `status` | [`HookActionStatus`](/docs/types#hookresult) | The action current status. | | `isIdle` | `boolean` | True if the action status is `idle`. | +| `isTransitioning` | `boolean` | True if the transition status from the `useTransition` hook used under the hood is `true`. | | `isExecuting` | `boolean` | True if the action status is `executing`. | +| `isPending` | `boolean` | True if either `isTransitioning` or `isExecuting` are `true`. | | `hasSucceeded` | `boolean` | True if the action status is `hasSucceeded`. | | `hasErrored` | `boolean` | True if the action status is `hasErrored`. | diff --git a/website/docs/execution/hooks/usestateaction.md b/website/docs/execution/hooks/usestateaction.md index a7b6d3c5..84affc5e 100644 --- a/website/docs/execution/hooks/usestateaction.md +++ b/website/docs/execution/hooks/usestateaction.md @@ -107,7 +107,9 @@ You can pass an optional initial result to `useStateAction`, with the `initResul | `result` | [`HookResult`](/docs/types#hookresult) | When the action gets called via `execute`, this is the result object. | | `status` | [`HookActionStatus`](/docs/types#hookresult) | The action current status. | | `isIdle` | `boolean` | True if the action status is `idle`. | +| `isTransitioning` | `boolean` | True if the transition status from the `useTransition` hook used under the hood is `true`. | | `isExecuting` | `boolean` | True if the action status is `executing`. | +| `isPending` | `boolean` | True if either `isTransitioning` or `isExecuting` are `true`. | | `hasSucceeded` | `boolean` | True if the action status is `hasSucceeded`. | | `hasErrored` | `boolean` | True if the action status is `hasErrored`. | diff --git a/website/docs/safe-action-client/middleware.md b/website/docs/safe-action-client/middleware.md index fa6eb1ff..0b0536b8 100644 --- a/website/docs/safe-action-client/middleware.md +++ b/website/docs/safe-action-client/middleware.md @@ -246,7 +246,7 @@ export const testAction = actionClient Experimental feature ::: -Starting from version 7.6.0, you can create standalone middleware functions using the built-in `experimental_createMiddleware()` function. It's labelled as experimental because the API could change in the future, but it's perfectly fine to use it, as it's a pretty simple function that just wraps the creation of middleware. +Starting from version 7.6.0, you can create standalone middleware functions using the built-in `experimental_createMiddleware()` function. It's labeled as experimental because the API could change in the future, but it's perfectly fine to use it, as it's a pretty simple function that just wraps the creation of middleware. Thanks to this feature, and the previously mentioned [context extension](#extend-context), you can now define standalone middleware functions and even publish them as packages, if you want to. @@ -290,4 +290,4 @@ export const actionClientWithMyMiddleware = actionClient.use(myMiddleware1).use( An action defined using the `actionClientWithMyMiddleware` will contain `foo`, `baz` and `john` in its context. -\* Note that you can pass, **but not required to**, an object with two generic properties to the `experimental_createMiddleware()` function: `ctx` \[1\], `metadata` \[2\] and `serverError` \[3\]. Those keys are optional, and you should only provide them if you want your middleware to require **at minimum** the shape you passed in as generic. By doing that, following the above example, you can then access `ctx.foo` and `metadata.actionName` in the middleware you're defining, and by await the `next` function you'll see that `serverError` is an object with the `message` property. If you pass a middleware that requires those properties to a client that doesn't have them, you'll get an error in `use()` method. \ No newline at end of file +\* Note that you can pass, **but not required to**, an object with two generic properties to the `experimental_createMiddleware()` function: `ctx` \[1\], `metadata` \[2\] and `serverError` \[3\]. Those keys are optional, and you should only provide them if you want your middleware to require **at minimum** the shape you passed in as generic. By doing that, following the above example, you can then access `ctx.foo` and `metadata.actionName` in the middleware you're defining, and by awaiting the `next` function you'll see that `serverError` is an object with the `message` property. If you pass a middleware that requires those properties to a client that doesn't have them, you'll get an error in `use()` method. \ No newline at end of file