diff --git a/examples/with-react-hook-form/.gitignore b/examples/with-react-hook-form/.gitignore new file mode 100644 index 0000000000000..e3b3fe7726885 --- /dev/null +++ b/examples/with-react-hook-form/.gitignore @@ -0,0 +1,34 @@ +# See https://help.github.com/articles/ignoring-files/ for more about ignoring files. + +# dependencies +/node_modules +/.pnp +.pnp.js + +# testing +/coverage + +# next.js +/.next/ +/out/ + +# production +/build + +# misc +.DS_Store +*.pem + +# debug +npm-debug.log* +yarn-debug.log* +yarn-error.log* + +# local env files +.env.local +.env.development.local +.env.test.local +.env.production.local + +# vercel +.vercel \ No newline at end of file diff --git a/examples/with-react-hook-form/README.md b/examples/with-react-hook-form/README.md new file mode 100644 index 0000000000000..20ef12175a311 --- /dev/null +++ b/examples/with-react-hook-form/README.md @@ -0,0 +1,23 @@ +# with react-hook-form + +This example shows how to integrate react-hook-form in Next.js + +Form handling doesn't have to be painful. React Hook Form will help you write less code while achieving better performance. For more information, see [react-hook-form](https://react-hook-form.com) + +## Deploy your own + +Deploy the example using [Vercel](https://vercel.com/now): + +[![Deploy with Vercel](https://vercel.com/button)](https://vercel.com/new/git/external?repository-url=https://github.com/vercel/next.js/tree/canary/examples/with-react-hook-form&project-name=with-react-hook-form&repository-name=with-react-hook-form) + +## How to use + +Execute [`create-next-app`](https://github.com/vercel/next.js/tree/canary/packages/create-next-app) with [npm](https://docs.npmjs.com/cli/init) or [Yarn](https://yarnpkg.com/lang/en/docs/cli/create/) to bootstrap the example: + +```bash +npx create-next-app --example with-react-hook-form with-react-hook-form-app +# or +yarn create next-app --example with-react-hook-form with-react-hook-form-app +``` + +Deploy it to the cloud with [Vercel](https://vercel.com/new?utm_source=github&utm_medium=readme&utm_campaign=next-example) ([Documentation](https://nextjs.org/docs/deployment)). diff --git a/examples/with-react-hook-form/package.json b/examples/with-react-hook-form/package.json new file mode 100644 index 0000000000000..60fa4a2191e6d --- /dev/null +++ b/examples/with-react-hook-form/package.json @@ -0,0 +1,16 @@ +{ + "name": "with-react-hook-form", + "version": "1.0.0", + "scripts": { + "dev": "next dev", + "build": "next build", + "start": "next start" + }, + "dependencies": { + "next": "latest", + "react": "17.0.1", + "react-dom": "17.0.1", + "react-hook-form": "7.4.0" + }, + "license": "MIT" +} diff --git a/examples/with-react-hook-form/pages/_app.js b/examples/with-react-hook-form/pages/_app.js new file mode 100644 index 0000000000000..0bd950249faeb --- /dev/null +++ b/examples/with-react-hook-form/pages/_app.js @@ -0,0 +1,5 @@ +import '../styles/global.css' + +export default function MyApp({ Component, pageProps }) { + return +} diff --git a/examples/with-react-hook-form/pages/index.js b/examples/with-react-hook-form/pages/index.js new file mode 100644 index 0000000000000..b063590ee6044 --- /dev/null +++ b/examples/with-react-hook-form/pages/index.js @@ -0,0 +1,75 @@ +import { useState } from 'react' +import { useForm } from 'react-hook-form' + +const IndexPage = () => { + const [user, setUser] = useState() + const { + register, + formState: { errors }, + handleSubmit, + } = useForm() + const onSubmit = ({ username, password, remember }) => { + // You should handle login logic with username, password and remember form data + setUser({ name: username }) + } + + return ( +
+ {user ? ( + Hello, {user.name}! + ) : ( +
+
+

LOGIN

+
+
+ + {errors.username && ( + {errors.username.message} + )} +
+
+ + {errors.password && ( + {errors.password.message} + )} +
+
+ + +
+
+ +
+
+ )} +
+ ) +} + +export default IndexPage diff --git a/examples/with-react-hook-form/styles/global.css b/examples/with-react-hook-form/styles/global.css new file mode 100644 index 0000000000000..9cfcfe629d991 --- /dev/null +++ b/examples/with-react-hook-form/styles/global.css @@ -0,0 +1,97 @@ +@import url('https://fonts.googleapis.com/css2?family=Poppins&display=swap'); + +:root { + --white: #fff; + --light-border: #ccc; + --danger: #ff0000; + --primary-color: #a4a4f7; + --primary-color-light: #7373f9; + --secondary-color: #f0f8ff; + --shadow-color: #ddd; + font-family: 'Poppins', sans-serif; + font-size: 12px; +} + +* { + margin: 0; + padding: 0; + box-sizing: border-box; +} + +.container { + display: flex; + height: 100vh; + justify-content: center; + align-items: center; +} + +.row { + margin: 10px 20px; +} + +.row-remember { + display: flex; + align-items: center; + border-bottom: 1px solid var(--light-border); + padding: 0 0 15px 0; +} + +.container form { + background-color: var(--secondary-color); + padding: 2rem; + border-radius: 5px; + box-shadow: 12px 12px 10px var(--shadow-color); + flex: 1; +} + +@media screen and (min-width: 768px) { + .container form { + flex: 0.5; + max-width: 600px; + } +} + +.form-header { + text-align: center; + font-size: 1.5rem; +} + +.form-field { + width: 100%; + padding: 10px; + outline: none; + border: 1px solid var(--light-border); + border-radius: 5px; +} + +.form-field.has-error { + border: 1px solid var(--danger); +} + +.error-label { + color: var(--danger); +} + +.remember-label { + margin: 0 10px; +} + +.btn { + width: 100%; + height: 3rem; + border: 1px solid var(--light-border); + border-radius: 5px; +} + +.btn:hover { + background-color: var(--primary-color-light); +} + +.login-btn { + background-color: var(--primary-color); + color: var(--white); +} + +.hello-user { + font-size: 2rem; +}