From 08ca5ec8a4d46782b5fdb98745ba53640235b8eb Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 11:50:25 +0200
Subject: [PATCH 1/8] Upgraded deps
---
.eslintrc | 3 +-
package.json | 86 ++++++++++++++++++++++++++--------------------------
2 files changed, 45 insertions(+), 44 deletions(-)
diff --git a/.eslintrc b/.eslintrc
index b76c7847..7939e29a 100755
--- a/.eslintrc
+++ b/.eslintrc
@@ -4,6 +4,7 @@
"rules": {
"jsx-a11y/href-no-hash": 0,
"react-hooks/rules-of-hooks": "error",
- "react-hooks/exhaustive-deps": "warn"
+ "react-hooks/exhaustive-deps": "warn",
+ "import/no-anonymous-default-export": 0
}
}
diff --git a/package.json b/package.json
index c4e1abf1..873f60e6 100644
--- a/package.json
+++ b/package.json
@@ -29,67 +29,67 @@
},
"homepage": "https://github.com/final-form/react-final-form#readme",
"devDependencies": {
- "@babel/core": "^7.12.3",
- "@babel/plugin-proposal-class-properties": "^7.12.1",
- "@babel/plugin-proposal-decorators": "^7.12.1",
- "@babel/plugin-proposal-export-namespace-from": "^7.12.1",
- "@babel/plugin-proposal-function-sent": "^7.12.1",
- "@babel/plugin-proposal-json-strings": "^7.12.1",
- "@babel/plugin-proposal-numeric-separator": "^7.12.1",
- "@babel/plugin-proposal-throw-expressions": "^7.12.1",
+ "@babel/core": "^7.15.5",
+ "@babel/plugin-proposal-class-properties": "^7.14.5",
+ "@babel/plugin-proposal-decorators": "^7.15.4",
+ "@babel/plugin-proposal-export-namespace-from": "^7.14.5",
+ "@babel/plugin-proposal-function-sent": "^7.14.5",
+ "@babel/plugin-proposal-json-strings": "^7.14.5",
+ "@babel/plugin-proposal-numeric-separator": "^7.14.5",
+ "@babel/plugin-proposal-throw-expressions": "^7.14.5",
"@babel/plugin-syntax-dynamic-import": "^7.8.3",
"@babel/plugin-syntax-import-meta": "^7.10.4",
- "@babel/plugin-transform-flow-strip-types": "^7.12.1",
- "@babel/plugin-transform-react-jsx-source": "^7.12.13",
- "@babel/plugin-transform-runtime": "^7.12.1",
- "@babel/preset-env": "^7.12.1",
- "@babel/preset-flow": "^7.12.1",
- "@babel/preset-react": "^7.12.1",
- "@testing-library/jest-dom": "^5.11.4",
- "@testing-library/react": "^11.1.0",
- "@types/react": "^16.9.53",
- "@typescript-eslint/eslint-plugin": "^4.5.0",
- "@typescript-eslint/parser": "^4.5.0",
+ "@babel/plugin-transform-flow-strip-types": "^7.14.5",
+ "@babel/plugin-transform-react-jsx-source": "^7.14.5",
+ "@babel/plugin-transform-runtime": "^7.15.0",
+ "@babel/preset-env": "^7.15.6",
+ "@babel/preset-flow": "^7.14.5",
+ "@babel/preset-react": "^7.14.5",
+ "@testing-library/jest-dom": "^5.14.1",
+ "@testing-library/react": "^12.1.0",
+ "@types/react": "^17.0.24",
+ "@typescript-eslint/eslint-plugin": "^4.31.2",
+ "@typescript-eslint/parser": "^4.31.2",
"babel-core": "^7.0.0-bridge.0",
"babel-eslint": "^10.1.0",
- "babel-jest": "^26.6.0",
- "bundlesize": "^0.18.0",
- "doctoc": "^1.4.0",
- "dtslint": "^4.0.4",
- "eslint": "^7.11.0",
- "eslint-config-react-app": "^5.2.1",
+ "babel-jest": "^27.2.1",
+ "bundlesize": "^0.18.1",
+ "doctoc": "^2.0.1",
+ "dtslint": "^4.1.6",
+ "eslint": "^7.32.0",
+ "eslint-config-react-app": "^6.0.0",
"eslint-plugin-babel": "^5.3.1",
- "eslint-plugin-flowtype": "^5.2.0",
- "eslint-plugin-import": "^2.22.1",
- "eslint-plugin-jsx-a11y": "^6.3.1",
- "eslint-plugin-react": "^7.21.5",
+ "eslint-plugin-flowtype": "^6.1.0",
+ "eslint-plugin-import": "^2.24.2",
+ "eslint-plugin-jsx-a11y": "^6.4.1",
+ "eslint-plugin-react": "^7.26.0",
"eslint-plugin-react-hooks": "^4.2.0",
"fast-deep-equal": "^3.1.3",
- "final-form": "^4.20.1",
- "flow-bin": "^0.136.0",
+ "final-form": "^4.20.2",
+ "flow-bin": "^0.160.2",
"glow": "^1.2.2",
"husky": "^4.3.0",
- "jest": "^26.6.0",
- "jest-mock-console": "^1.0.1",
- "lint-staged": "^10.4.2",
+ "jest": "^27.2.1",
+ "jest-mock-console": "^1.1.0",
+ "lint-staged": "^11.1.2",
"nps": "^5.10.0",
"nps-utils": "^1.7.0",
"opencollective": "^1.0.3",
- "prettier": "^2.1.2",
- "prettier-eslint-cli": "^5.0.0",
- "react": "^17.0.0",
- "react-dom": "^17.0.0",
- "rollup": "^2.32.1",
+ "prettier": "^2.4.1",
+ "prettier-eslint-cli": "^5.0.1",
+ "react": "^17.0.2",
+ "react-dom": "^17.0.2",
+ "rollup": "^2.57.0",
"rollup-plugin-babel": "^4.4.0",
"rollup-plugin-commonjs": "^10.1.0",
"rollup-plugin-json": "^4.0.0",
"rollup-plugin-node-resolve": "^5.2.0",
"rollup-plugin-replace": "^2.2.0",
"rollup-plugin-uglify": "^6.0.4",
- "tar": "^6.0.5",
- "ts-essentials": "^7.0.1",
+ "tar": "^6.1.11",
+ "ts-essentials": "^8.1.0",
"tslint": "^6.1.3",
- "typescript": "^4.0.3"
+ "typescript": "^4.4.3"
},
"peerDependencies": {
"final-form": "^4.20.0",
@@ -130,6 +130,6 @@
"url": "https://opencollective.com/final-form"
},
"dependencies": {
- "@babel/runtime": "^7.12.1"
+ "@babel/runtime": "^7.15.4"
}
}
From c2f5a182cf1fb5dc71e5fa92b8537a3efa434ea1 Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 11:52:54 +0200
Subject: [PATCH 2/8] Upgraded react in examples
---
examples/async-field-level-validation/package.json | 6 +++---
examples/async-redux-submission/package.json | 6 +++---
examples/async-typeahead-redux/package.json | 6 +++---
examples/auto-save-field-blur/package.json | 6 +++---
examples/auto-save-selective-debounce/package.json | 6 +++---
examples/auto-save-with-debounce/package.json | 6 +++---
examples/calculated-fields/package.json | 6 +++---
examples/chakra/package.json | 4 ++--
examples/conditional-fields/package.json | 6 +++---
examples/credit-card/package.json | 6 +++---
examples/custom-validation-engine/package.json | 6 +++---
examples/debounced-record-level-validation/package.json | 6 +++---
examples/declarative-form-rules/package.json | 6 +++---
examples/downshift-typeahead/package.json | 6 +++---
examples/external-submit/package.json | 6 +++---
examples/field-arrays/package.json | 6 +++---
examples/field-warnings/package.json | 6 +++---
examples/focus-first-error/package.json | 6 +++---
examples/format-on-blur/package.json | 6 +++---
examples/format-string-by-pattern/package.json | 6 +++---
.../hybrid-sync-async-record-level-validation/package.json | 6 +++---
.../independent-error-component-render-props/package.json | 6 +++---
.../independent-error-component-with-hooks/package.json | 6 +++---
examples/listening-for-external-changes/package.json | 6 +++---
examples/loading-initializing-values/package.json | 6 +++---
examples/loading-saving-reinitializing/package.json | 6 +++---
examples/parse-format/package.json | 6 +++---
examples/redux/package.json | 6 +++---
examples/reusable-field-groups/package.json | 6 +++---
examples/styling-with-smooth-ui/package.json | 6 +++---
examples/third-party-components/package.json | 6 +++---
31 files changed, 92 insertions(+), 92 deletions(-)
diff --git a/examples/async-field-level-validation/package.json b/examples/async-field-level-validation/package.json
index d42c59c5..33349807 100644
--- a/examples/async-field-level-validation/package.json
+++ b/examples/async-field-level-validation/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"final-form": "4.16.1",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "6.3.0",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/async-redux-submission/package.json b/examples/async-redux-submission/package.json
index 3791a97a..f816abdd 100644
--- a/examples/async-redux-submission/package.json
+++ b/examples/async-redux-submission/package.json
@@ -13,8 +13,8 @@
"main": "src/index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-redux": "7.0.3",
"react-redux-promise-listener": "1.0.0",
@@ -30,4 +30,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/async-typeahead-redux/package.json b/examples/async-typeahead-redux/package.json
index af0755c6..b2fc1d48 100644
--- a/examples/async-typeahead-redux/package.json
+++ b/examples/async-typeahead-redux/package.json
@@ -9,13 +9,13 @@
"final-form": "4.18.5",
"final-form-set-field-data": "1.0.2",
"ramda": "0.25.0",
- "react": "16.9.0",
+ "react": "17.0.2",
"react-bootstrap-typeahead": "3.1.4",
- "react-dom": "16.9.0",
+ "react-dom": "17.0.2",
"react-final-form": "6.3.0",
"react-redux": "7.1.0",
"redux": "4.0.4",
"redux-thunk": "2.3.0",
"styled-components": "2.4.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/auto-save-field-blur/package.json b/examples/auto-save-field-blur/package.json
index 391f4668..4ae3471b 100644
--- a/examples/auto-save-field-blur/package.json
+++ b/examples/auto-save-field-blur/package.json
@@ -8,9 +8,9 @@
"final-form": "4.12.0",
"final-form-set-field-data": "1.0.2",
"object-diff": "0.0.4",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/auto-save-selective-debounce/package.json b/examples/auto-save-selective-debounce/package.json
index cb32429b..416810e5 100644
--- a/examples/auto-save-selective-debounce/package.json
+++ b/examples/auto-save-selective-debounce/package.json
@@ -7,9 +7,9 @@
"dependencies": {
"final-form": "4.12.0",
"object-diff": "0.0.4",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/auto-save-with-debounce/package.json b/examples/auto-save-with-debounce/package.json
index ff6b38a2..3358038e 100644
--- a/examples/auto-save-with-debounce/package.json
+++ b/examples/auto-save-with-debounce/package.json
@@ -13,9 +13,9 @@
"dependencies": {
"final-form": "4.12.0",
"object-diff": "0.0.4",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/calculated-fields/package.json b/examples/calculated-fields/package.json
index 81196f07..6bfd7caa 100644
--- a/examples/calculated-fields/package.json
+++ b/examples/calculated-fields/package.json
@@ -8,9 +8,9 @@
"dependencies": {
"final-form": "4.12.0",
"final-form-calculate": "1.3.1",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/chakra/package.json b/examples/chakra/package.json
index 57156329..09b0d19a 100644
--- a/examples/chakra/package.json
+++ b/examples/chakra/package.json
@@ -16,8 +16,8 @@
"@emotion/styled": "^10.0.17",
"emotion-theming": "^10.0.17",
"final-form": "4.18.6",
- "react": "16.12.0",
- "react-dom": "16.12.0",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "latest",
"react-scripts": "3.2.0"
},
diff --git a/examples/conditional-fields/package.json b/examples/conditional-fields/package.json
index 406f44ab..d13b371a 100644
--- a/examples/conditional-fields/package.json
+++ b/examples/conditional-fields/package.json
@@ -7,8 +7,8 @@
"main": "src/index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
@@ -20,4 +20,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/credit-card/package.json b/examples/credit-card/package.json
index 441024b4..5e611713 100644
--- a/examples/credit-card/package.json
+++ b/examples/credit-card/package.json
@@ -13,9 +13,9 @@
"main": "src/index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
+ "react": "17.0.2",
"react-credit-cards": "0.7.0",
- "react-dom": "16.8.6",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
@@ -27,4 +27,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/custom-validation-engine/package.json b/examples/custom-validation-engine/package.json
index ae3b6898..f4c04b4a 100644
--- a/examples/custom-validation-engine/package.json
+++ b/examples/custom-validation-engine/package.json
@@ -7,10 +7,10 @@
"dependencies": {
"final-form": "4.12.0",
"final-form-set-field-data": "1.0.2",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-fontawesome": "1.6.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/debounced-record-level-validation/package.json b/examples/debounced-record-level-validation/package.json
index ace6b53d..4603f089 100644
--- a/examples/debounced-record-level-validation/package.json
+++ b/examples/debounced-record-level-validation/package.json
@@ -6,8 +6,8 @@
"main": "src/index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "2.0.3",
"styled-components": "4.2.0"
@@ -25,4 +25,4 @@
"not ie <= 11",
"not op_mini all"
]
-}
\ No newline at end of file
+}
diff --git a/examples/declarative-form-rules/package.json b/examples/declarative-form-rules/package.json
index 120ec310..60a9b17b 100644
--- a/examples/declarative-form-rules/package.json
+++ b/examples/declarative-form-rules/package.json
@@ -13,8 +13,8 @@
"main": "src/index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-final-form-listeners": "1.0.2",
"react-scripts": "3.0.1",
@@ -27,4 +27,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/downshift-typeahead/package.json b/examples/downshift-typeahead/package.json
index b0596c5a..7fc70bfc 100644
--- a/examples/downshift-typeahead/package.json
+++ b/examples/downshift-typeahead/package.json
@@ -15,8 +15,8 @@
"downshift": "1.28.0",
"final-form": "4.12.0",
"match-sorter": "3.0.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
@@ -28,4 +28,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/external-submit/package.json b/examples/external-submit/package.json
index 2efa334b..463c7f66 100644
--- a/examples/external-submit/package.json
+++ b/examples/external-submit/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/field-arrays/package.json b/examples/field-arrays/package.json
index f714e770..f3114c6a 100644
--- a/examples/field-arrays/package.json
+++ b/examples/field-arrays/package.json
@@ -7,10 +7,10 @@
"dependencies": {
"final-form": "4.12.0",
"final-form-arrays": "1.1.2",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "4.1.0",
"react-final-form-arrays": "2.0.3",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/field-warnings/package.json b/examples/field-warnings/package.json
index 6967fd01..b15673fd 100644
--- a/examples/field-warnings/package.json
+++ b/examples/field-warnings/package.json
@@ -8,9 +8,9 @@
"dependencies": {
"styled-components": "2.2.4",
"react-final-form": "1.2.0",
- "react-dom": "16.0.0",
- "react": "16.0.0",
+ "react-dom": "17.0.2",
+ "react": "17.0.2",
"final-form-set-field-data": "1.0.0",
"final-form": "1.3.1"
}
-}
\ No newline at end of file
+}
diff --git a/examples/focus-first-error/package.json b/examples/focus-first-error/package.json
index 3187a4fa..06748335 100644
--- a/examples/focus-first-error/package.json
+++ b/examples/focus-first-error/package.json
@@ -8,8 +8,8 @@
"dependencies": {
"final-form": "4.12.0",
"final-form-focus": "1.1.2",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
@@ -21,4 +21,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/format-on-blur/package.json b/examples/format-on-blur/package.json
index 19875403..6c4ec408 100644
--- a/examples/format-on-blur/package.json
+++ b/examples/format-on-blur/package.json
@@ -13,8 +13,8 @@
"dependencies": {
"final-form": "4.12.0",
"numeral": "2.0.6",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
@@ -26,4 +26,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/format-string-by-pattern/package.json b/examples/format-string-by-pattern/package.json
index f3bf437e..edf88836 100644
--- a/examples/format-string-by-pattern/package.json
+++ b/examples/format-string-by-pattern/package.json
@@ -7,9 +7,9 @@
"dependencies": {
"final-form": "latest",
"format-string-by-pattern": "1.1.1",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "latest",
"styled-components": "latest"
}
-}
\ No newline at end of file
+}
diff --git a/examples/hybrid-sync-async-record-level-validation/package.json b/examples/hybrid-sync-async-record-level-validation/package.json
index ecbc8105..d4e4d61d 100644
--- a/examples/hybrid-sync-async-record-level-validation/package.json
+++ b/examples/hybrid-sync-async-record-level-validation/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"final-form": "latest",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "latest",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/independent-error-component-render-props/package.json b/examples/independent-error-component-render-props/package.json
index 581e7556..1767a6b5 100644
--- a/examples/independent-error-component-render-props/package.json
+++ b/examples/independent-error-component-render-props/package.json
@@ -6,9 +6,9 @@
"main": "index.js",
"dependencies": {
"final-form": "4.13.1",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "6.0.1",
"styled-components": "4.2.1"
}
-}
\ No newline at end of file
+}
diff --git a/examples/independent-error-component-with-hooks/package.json b/examples/independent-error-component-with-hooks/package.json
index b6ac2eb1..d9e8b4da 100644
--- a/examples/independent-error-component-with-hooks/package.json
+++ b/examples/independent-error-component-with-hooks/package.json
@@ -6,9 +6,9 @@
"main": "index.js",
"dependencies": {
"final-form": "4.13.1",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "6.0.1",
"styled-components": "4.2.1"
}
-}
\ No newline at end of file
+}
diff --git a/examples/listening-for-external-changes/package.json b/examples/listening-for-external-changes/package.json
index ccd8e3a7..1dc75a88 100644
--- a/examples/listening-for-external-changes/package.json
+++ b/examples/listening-for-external-changes/package.json
@@ -14,8 +14,8 @@
"dependencies": {
"final-form": "4.12.0",
"final-form-calculate": "1.3.1",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
@@ -27,4 +27,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/loading-initializing-values/package.json b/examples/loading-initializing-values/package.json
index 901a11d7..98c563ed 100644
--- a/examples/loading-initializing-values/package.json
+++ b/examples/loading-initializing-values/package.json
@@ -6,9 +6,9 @@
"main": "index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/loading-saving-reinitializing/package.json b/examples/loading-saving-reinitializing/package.json
index 4ff65ddf..65ec1320 100644
--- a/examples/loading-saving-reinitializing/package.json
+++ b/examples/loading-saving-reinitializing/package.json
@@ -12,10 +12,10 @@
"main": "index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-fontawesome": "1.6.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/parse-format/package.json b/examples/parse-format/package.json
index 685f0410..e340241d 100644
--- a/examples/parse-format/package.json
+++ b/examples/parse-format/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"final-form": "latest",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "latest",
"styled-components": "latest"
}
-}
\ No newline at end of file
+}
diff --git a/examples/redux/package.json b/examples/redux/package.json
index e711b406..a914eafa 100644
--- a/examples/redux/package.json
+++ b/examples/redux/package.json
@@ -13,8 +13,8 @@
"main": "src/index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-redux": "7.0.3",
"react-scripts": "3.0.1",
@@ -28,4 +28,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/reusable-field-groups/package.json b/examples/reusable-field-groups/package.json
index d747c85b..1236cdf1 100644
--- a/examples/reusable-field-groups/package.json
+++ b/examples/reusable-field-groups/package.json
@@ -6,9 +6,9 @@
"main": "index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
diff --git a/examples/styling-with-smooth-ui/package.json b/examples/styling-with-smooth-ui/package.json
index d0702baa..1d2bb3a6 100644
--- a/examples/styling-with-smooth-ui/package.json
+++ b/examples/styling-with-smooth-ui/package.json
@@ -12,8 +12,8 @@
"main": "src/index.js",
"dependencies": {
"final-form": "4.12.0",
- "react": "16.8.6",
- "react-dom": "16.8.6",
+ "react": "17.0.2",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-scripts": "3.0.1",
"smooth-ui": "4.3.2",
@@ -26,4 +26,4 @@
"test": "react-scripts test --env=jsdom",
"eject": "react-scripts eject"
}
-}
\ No newline at end of file
+}
diff --git a/examples/third-party-components/package.json b/examples/third-party-components/package.json
index 6d1fc879..19339d91 100644
--- a/examples/third-party-components/package.json
+++ b/examples/third-party-components/package.json
@@ -13,11 +13,11 @@
"dependencies": {
"final-form": "4.12.0",
"material-ui": "0.20.2",
- "react": "16.8.6",
+ "react": "17.0.2",
"react-addons-shallow-compare": "15.6.2",
- "react-dom": "16.8.6",
+ "react-dom": "17.0.2",
"react-final-form": "5.0.1",
"react-select": "2.4.3",
"styled-components": "4.2.0"
}
-}
\ No newline at end of file
+}
From 36d053c131a097c942defcfd2acd4c25306b2a4e Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 11:55:35 +0200
Subject: [PATCH 3/8] Upgraded FF and RFF in examples
---
examples/async-field-level-validation/package.json | 4 ++--
examples/async-redux-submission/package.json | 4 ++--
examples/async-typeahead-redux/package.json | 4 ++--
examples/auto-save-field-blur/package.json | 4 ++--
examples/auto-save-selective-debounce/package.json | 4 ++--
examples/auto-save-with-debounce/package.json | 4 ++--
examples/calculated-fields/package.json | 4 ++--
examples/chakra/package.json | 4 ++--
examples/conditional-fields/package.json | 4 ++--
examples/credit-card/package.json | 4 ++--
examples/custom-validation-engine/package.json | 4 ++--
.../debounced-record-level-validation/package.json | 4 ++--
examples/declarative-form-rules/package.json | 4 ++--
examples/downshift-typeahead/package.json | 4 ++--
examples/external-submit/package.json | 4 ++--
examples/field-arrays/package.json | 4 ++--
examples/field-level-validation/package.json | 4 ++--
examples/field-warnings/package.json | 4 ++--
examples/fields-component/package.json | 4 ++--
examples/focus-first-error/package.json | 4 ++--
examples/format-on-blur/package.json | 4 ++--
examples/format-string-by-pattern/package.json | 4 ++--
.../package.json | 4 ++--
.../package.json | 4 ++--
.../package.json | 4 ++--
examples/listening-for-external-changes/package.json | 4 ++--
examples/loading-initializing-values/package.json | 4 ++--
examples/loading-saving-reinitializing/package.json | 4 ++--
examples/material-ui/package.json | 8 +++++---
examples/parse-format/package.json | 4 ++--
examples/prefixed-fields/package.json | 6 +++---
examples/record-level-validation/package.json | 4 ++--
examples/redux/package.json | 4 ++--
examples/reusable-field-groups/package.json | 4 ++--
examples/simple/package.json | 4 ++--
examples/styling-with-smooth-ui/package.json | 4 ++--
examples/submission-errors/package.json | 4 ++--
examples/subscriptions/package.json | 4 ++--
examples/third-party-components/package.json | 4 ++--
examples/wizard/package.json | 11 ++++++++---
package.json | 4 ++--
41 files changed, 92 insertions(+), 85 deletions(-)
diff --git a/examples/async-field-level-validation/package.json b/examples/async-field-level-validation/package.json
index 33349807..80517548 100644
--- a/examples/async-field-level-validation/package.json
+++ b/examples/async-field-level-validation/package.json
@@ -11,10 +11,10 @@
],
"main": "index.js",
"dependencies": {
- "final-form": "4.16.1",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "6.3.0",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/async-redux-submission/package.json b/examples/async-redux-submission/package.json
index f816abdd..7d9aac2d 100644
--- a/examples/async-redux-submission/package.json
+++ b/examples/async-redux-submission/package.json
@@ -12,10 +12,10 @@
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-redux": "7.0.3",
"react-redux-promise-listener": "1.0.0",
"react-scripts": "3.0.1",
diff --git a/examples/async-typeahead-redux/package.json b/examples/async-typeahead-redux/package.json
index b2fc1d48..63ba49ae 100644
--- a/examples/async-typeahead-redux/package.json
+++ b/examples/async-typeahead-redux/package.json
@@ -6,13 +6,13 @@
"main": "index.js",
"dependencies": {
"arrify": "2.0.1",
- "final-form": "4.18.5",
+ "final-form": "4.20.2",
"final-form-set-field-data": "1.0.2",
"ramda": "0.25.0",
"react": "17.0.2",
"react-bootstrap-typeahead": "3.1.4",
"react-dom": "17.0.2",
- "react-final-form": "6.3.0",
+ "react-final-form": "6.5.3",
"react-redux": "7.1.0",
"redux": "4.0.4",
"redux-thunk": "2.3.0",
diff --git a/examples/auto-save-field-blur/package.json b/examples/auto-save-field-blur/package.json
index 4ae3471b..3b15de72 100644
--- a/examples/auto-save-field-blur/package.json
+++ b/examples/auto-save-field-blur/package.json
@@ -5,12 +5,12 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"final-form-set-field-data": "1.0.2",
"object-diff": "0.0.4",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/auto-save-selective-debounce/package.json b/examples/auto-save-selective-debounce/package.json
index 416810e5..5fdf0364 100644
--- a/examples/auto-save-selective-debounce/package.json
+++ b/examples/auto-save-selective-debounce/package.json
@@ -5,11 +5,11 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"object-diff": "0.0.4",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/auto-save-with-debounce/package.json b/examples/auto-save-with-debounce/package.json
index 3358038e..6b2b96dc 100644
--- a/examples/auto-save-with-debounce/package.json
+++ b/examples/auto-save-with-debounce/package.json
@@ -11,11 +11,11 @@
],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"object-diff": "0.0.4",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/calculated-fields/package.json b/examples/calculated-fields/package.json
index 6bfd7caa..3137f693 100644
--- a/examples/calculated-fields/package.json
+++ b/examples/calculated-fields/package.json
@@ -6,11 +6,11 @@
"homepage": "https://codesandbox.io/s/oq52p6v96y",
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"final-form-calculate": "1.3.1",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/chakra/package.json b/examples/chakra/package.json
index 09b0d19a..e2b9708c 100644
--- a/examples/chakra/package.json
+++ b/examples/chakra/package.json
@@ -15,10 +15,10 @@
"@emotion/core": "^10.0.17",
"@emotion/styled": "^10.0.17",
"emotion-theming": "^10.0.17",
- "final-form": "4.18.6",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react-scripts": "3.2.0"
},
"scripts": {
diff --git a/examples/conditional-fields/package.json b/examples/conditional-fields/package.json
index d13b371a..0e5c6895 100644
--- a/examples/conditional-fields/package.json
+++ b/examples/conditional-fields/package.json
@@ -6,10 +6,10 @@
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
},
diff --git a/examples/credit-card/package.json b/examples/credit-card/package.json
index 5e611713..d12ddd33 100644
--- a/examples/credit-card/package.json
+++ b/examples/credit-card/package.json
@@ -12,11 +12,11 @@
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-credit-cards": "0.7.0",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
},
diff --git a/examples/custom-validation-engine/package.json b/examples/custom-validation-engine/package.json
index f4c04b4a..49480589 100644
--- a/examples/custom-validation-engine/package.json
+++ b/examples/custom-validation-engine/package.json
@@ -5,11 +5,11 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"final-form-set-field-data": "1.0.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-fontawesome": "1.6.1",
"styled-components": "4.2.0"
}
diff --git a/examples/debounced-record-level-validation/package.json b/examples/debounced-record-level-validation/package.json
index 4603f089..5f8186ba 100644
--- a/examples/debounced-record-level-validation/package.json
+++ b/examples/debounced-record-level-validation/package.json
@@ -5,10 +5,10 @@
"keywords": [],
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "2.0.3",
"styled-components": "4.2.0"
},
diff --git a/examples/declarative-form-rules/package.json b/examples/declarative-form-rules/package.json
index 60a9b17b..0affb2b3 100644
--- a/examples/declarative-form-rules/package.json
+++ b/examples/declarative-form-rules/package.json
@@ -12,10 +12,10 @@
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-final-form-listeners": "1.0.2",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
diff --git a/examples/downshift-typeahead/package.json b/examples/downshift-typeahead/package.json
index 7fc70bfc..ce89fc5f 100644
--- a/examples/downshift-typeahead/package.json
+++ b/examples/downshift-typeahead/package.json
@@ -13,11 +13,11 @@
"main": "src/index.js",
"dependencies": {
"downshift": "1.28.0",
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"match-sorter": "3.0.0",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
},
diff --git a/examples/external-submit/package.json b/examples/external-submit/package.json
index 463c7f66..f31c35ad 100644
--- a/examples/external-submit/package.json
+++ b/examples/external-submit/package.json
@@ -11,10 +11,10 @@
],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/field-arrays/package.json b/examples/field-arrays/package.json
index f3114c6a..8c0de9c9 100644
--- a/examples/field-arrays/package.json
+++ b/examples/field-arrays/package.json
@@ -5,11 +5,11 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"final-form-arrays": "1.1.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "4.1.0",
+ "react-final-form": "6.5.3",
"react-final-form-arrays": "2.0.3",
"styled-components": "4.2.0"
}
diff --git a/examples/field-level-validation/package.json b/examples/field-level-validation/package.json
index 5c7ed26b..13c27b58 100644
--- a/examples/field-level-validation/package.json
+++ b/examples/field-level-validation/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"styled-components": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react": "latest",
"react-dom": "latest",
- "final-form": "latest"
+ "final-form": "4.20.2"
}
}
diff --git a/examples/field-warnings/package.json b/examples/field-warnings/package.json
index b15673fd..286204df 100644
--- a/examples/field-warnings/package.json
+++ b/examples/field-warnings/package.json
@@ -7,10 +7,10 @@
"main": "index.js",
"dependencies": {
"styled-components": "2.2.4",
- "react-final-form": "1.2.0",
+ "react-final-form": "6.5.3",
"react-dom": "17.0.2",
"react": "17.0.2",
"final-form-set-field-data": "1.0.0",
- "final-form": "1.3.1"
+ "final-form": "4.20.2"
}
}
diff --git a/examples/fields-component/package.json b/examples/fields-component/package.json
index 9cb2fd3e..9b3e8589 100644
--- a/examples/fields-component/package.json
+++ b/examples/fields-component/package.json
@@ -8,9 +8,9 @@
"dependencies": {
"styled-components": "latest",
"react-fontawesome": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react-dom": "latest",
"react": "latest",
- "final-form": "latest"
+ "final-form": "4.20.2"
}
}
diff --git a/examples/focus-first-error/package.json b/examples/focus-first-error/package.json
index 06748335..53bb9072 100644
--- a/examples/focus-first-error/package.json
+++ b/examples/focus-first-error/package.json
@@ -6,11 +6,11 @@
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"final-form-focus": "1.1.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
},
diff --git a/examples/format-on-blur/package.json b/examples/format-on-blur/package.json
index 6c4ec408..dc3e17ad 100644
--- a/examples/format-on-blur/package.json
+++ b/examples/format-on-blur/package.json
@@ -11,11 +11,11 @@
],
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"numeral": "2.0.6",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
},
diff --git a/examples/format-string-by-pattern/package.json b/examples/format-string-by-pattern/package.json
index edf88836..dbf36dd4 100644
--- a/examples/format-string-by-pattern/package.json
+++ b/examples/format-string-by-pattern/package.json
@@ -5,11 +5,11 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "latest",
+ "final-form": "4.20.2",
"format-string-by-pattern": "1.1.1",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"styled-components": "latest"
}
}
diff --git a/examples/hybrid-sync-async-record-level-validation/package.json b/examples/hybrid-sync-async-record-level-validation/package.json
index d4e4d61d..439c31d5 100644
--- a/examples/hybrid-sync-async-record-level-validation/package.json
+++ b/examples/hybrid-sync-async-record-level-validation/package.json
@@ -11,10 +11,10 @@
],
"main": "index.js",
"dependencies": {
- "final-form": "latest",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/independent-error-component-render-props/package.json b/examples/independent-error-component-render-props/package.json
index 1767a6b5..4dc2c0db 100644
--- a/examples/independent-error-component-render-props/package.json
+++ b/examples/independent-error-component-render-props/package.json
@@ -5,10 +5,10 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.13.1",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "6.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.1"
}
}
diff --git a/examples/independent-error-component-with-hooks/package.json b/examples/independent-error-component-with-hooks/package.json
index d9e8b4da..3989d545 100644
--- a/examples/independent-error-component-with-hooks/package.json
+++ b/examples/independent-error-component-with-hooks/package.json
@@ -5,10 +5,10 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.13.1",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "6.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.1"
}
}
diff --git a/examples/listening-for-external-changes/package.json b/examples/listening-for-external-changes/package.json
index 1dc75a88..efad68ad 100644
--- a/examples/listening-for-external-changes/package.json
+++ b/examples/listening-for-external-changes/package.json
@@ -12,11 +12,11 @@
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"final-form-calculate": "1.3.1",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "3.0.1",
"styled-components": "4.2.0"
},
diff --git a/examples/loading-initializing-values/package.json b/examples/loading-initializing-values/package.json
index 98c563ed..7c35737f 100644
--- a/examples/loading-initializing-values/package.json
+++ b/examples/loading-initializing-values/package.json
@@ -5,10 +5,10 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/loading-saving-reinitializing/package.json b/examples/loading-saving-reinitializing/package.json
index 65ec1320..3deeb6fb 100644
--- a/examples/loading-saving-reinitializing/package.json
+++ b/examples/loading-saving-reinitializing/package.json
@@ -11,10 +11,10 @@
],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-fontawesome": "1.6.1",
"styled-components": "4.2.0"
}
diff --git a/examples/material-ui/package.json b/examples/material-ui/package.json
index b47a8d12..9851ef2f 100644
--- a/examples/material-ui/package.json
+++ b/examples/material-ui/package.json
@@ -2,7 +2,9 @@
"name": "react-final-form-material-ui-example",
"version": "1.0.0",
"description": "Demonstrates how to use Material-UI form controls.",
- "keywords": ["material-ui"],
+ "keywords": [
+ "material-ui"
+ ],
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
@@ -11,11 +13,11 @@
"@material-ui/lab": "latest",
"@material-ui/pickers": "latest",
"date-fns": "next",
- "final-form": "latest",
+ "final-form": "4.20.2",
"mui-rff": "latest",
"react": "latest",
"react-dom": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react-scripts": "latest",
"tslib": "latest"
},
diff --git a/examples/parse-format/package.json b/examples/parse-format/package.json
index e340241d..21d68520 100644
--- a/examples/parse-format/package.json
+++ b/examples/parse-format/package.json
@@ -11,10 +11,10 @@
],
"main": "index.js",
"dependencies": {
- "final-form": "latest",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"styled-components": "latest"
}
}
diff --git a/examples/prefixed-fields/package.json b/examples/prefixed-fields/package.json
index 6e135eab..663b9715 100644
--- a/examples/prefixed-fields/package.json
+++ b/examples/prefixed-fields/package.json
@@ -8,9 +8,9 @@
"dependencies": {
"styled-components": "latest",
"react-fontawesome": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react-dom": "latest",
"react": "latest",
- "final-form": "latest"
+ "final-form": "4.20.2"
}
-}
\ No newline at end of file
+}
diff --git a/examples/record-level-validation/package.json b/examples/record-level-validation/package.json
index a12b89de..54fab590 100644
--- a/examples/record-level-validation/package.json
+++ b/examples/record-level-validation/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"styled-components": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react": "latest",
"react-dom": "latest",
- "final-form": "latest"
+ "final-form": "4.20.2"
}
}
diff --git a/examples/redux/package.json b/examples/redux/package.json
index a914eafa..047a16e9 100644
--- a/examples/redux/package.json
+++ b/examples/redux/package.json
@@ -12,10 +12,10 @@
"homepage": "https://codesandbox.io/s/new",
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-redux": "7.0.3",
"react-scripts": "3.0.1",
"redux": "4.0.1",
diff --git a/examples/reusable-field-groups/package.json b/examples/reusable-field-groups/package.json
index 1236cdf1..b88b7e18 100644
--- a/examples/reusable-field-groups/package.json
+++ b/examples/reusable-field-groups/package.json
@@ -5,10 +5,10 @@
"keywords": [],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"styled-components": "4.2.0"
}
}
diff --git a/examples/simple/package.json b/examples/simple/package.json
index e3c0b3e9..4d9699b3 100644
--- a/examples/simple/package.json
+++ b/examples/simple/package.json
@@ -12,10 +12,10 @@
"main": "index.js",
"dependencies": {
"styled-components": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react": "latest",
"react-dom": "latest",
- "final-form": "latest",
+ "final-form": "4.20.2",
"prop-types": "latest"
}
}
diff --git a/examples/styling-with-smooth-ui/package.json b/examples/styling-with-smooth-ui/package.json
index 1d2bb3a6..6bb3042d 100644
--- a/examples/styling-with-smooth-ui/package.json
+++ b/examples/styling-with-smooth-ui/package.json
@@ -11,10 +11,10 @@
],
"main": "src/index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"react": "17.0.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-scripts": "3.0.1",
"smooth-ui": "4.3.2",
"styled-components": "3.4.10"
diff --git a/examples/submission-errors/package.json b/examples/submission-errors/package.json
index 78423610..bea07a74 100644
--- a/examples/submission-errors/package.json
+++ b/examples/submission-errors/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"styled-components": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react": "latest",
"react-dom": "latest",
- "final-form": "latest"
+ "final-form": "4.20.2"
}
}
diff --git a/examples/subscriptions/package.json b/examples/subscriptions/package.json
index 8173001e..bec54dd8 100644
--- a/examples/subscriptions/package.json
+++ b/examples/subscriptions/package.json
@@ -12,9 +12,9 @@
"main": "index.js",
"dependencies": {
"styled-components": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react": "latest",
"react-dom": "latest",
- "final-form": "latest"
+ "final-form": "4.20.2"
}
}
diff --git a/examples/third-party-components/package.json b/examples/third-party-components/package.json
index 19339d91..85829306 100644
--- a/examples/third-party-components/package.json
+++ b/examples/third-party-components/package.json
@@ -11,12 +11,12 @@
],
"main": "index.js",
"dependencies": {
- "final-form": "4.12.0",
+ "final-form": "4.20.2",
"material-ui": "0.20.2",
"react": "17.0.2",
"react-addons-shallow-compare": "15.6.2",
"react-dom": "17.0.2",
- "react-final-form": "5.0.1",
+ "react-final-form": "6.5.3",
"react-select": "2.4.3",
"styled-components": "4.2.0"
}
diff --git a/examples/wizard/package.json b/examples/wizard/package.json
index 9e68a5dc..ae7f52f6 100644
--- a/examples/wizard/package.json
+++ b/examples/wizard/package.json
@@ -13,16 +13,21 @@
"dependencies": {
"prop-types": "latest",
"styled-components": "latest",
- "react-final-form": "latest",
+ "react-final-form": "6.5.3",
"react": "latest",
"react-dom": "latest",
"react-scripts": "3.0.1",
- "final-form": "latest",
+ "final-form": "4.20.2",
"prop-types": "latest"
},
"scripts": {
"start": "react-scripts start",
"build": "react-scripts build"
},
- "browserslist": [">0.2%", "not dead", "not ie <= 11", "not op_mini all"]
+ "browserslist": [
+ ">0.2%",
+ "not dead",
+ "not ie <= 11",
+ "not op_mini all"
+ ]
}
diff --git a/package.json b/package.json
index 873f60e6..fd2db392 100644
--- a/package.json
+++ b/package.json
@@ -65,7 +65,7 @@
"eslint-plugin-react": "^7.26.0",
"eslint-plugin-react-hooks": "^4.2.0",
"fast-deep-equal": "^3.1.3",
- "final-form": "^4.20.2",
+ "final-form": "4.20.2",
"flow-bin": "^0.160.2",
"glow": "^1.2.2",
"husky": "^4.3.0",
@@ -92,7 +92,7 @@
"typescript": "^4.4.3"
},
"peerDependencies": {
- "final-form": "^4.20.0",
+ "final-form": "4.20.2",
"react": "^16.8.0 || ^17.0.0"
},
"lint-staged": {
From 809e8c9a9aeb5959b79242d738e8428f7532d6c4 Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 12:30:19 +0200
Subject: [PATCH 4/8] Prettier
---
.babelrc.js | 46 +-
.github/ci.yml | 49 +
.prettierignore | 3 +
.prettierrc | 17 +-
.../async-field-level-validation/Styles.js | 4 +-
.../async-field-level-validation/index.js | 73 +-
examples/async-redux-submission/Styles.js | 20 +-
.../asyncSubmissionMiddleware.js | 49 +-
examples/async-redux-submission/index.js | 48 +-
.../registrationDuck.js | 20 +-
examples/async-redux-submission/store.js | 41 +-
.../GithubUserTypeahead.jsx | 10 +-
examples/async-typeahead-redux/Styles.js | 22 +-
examples/async-typeahead-redux/actions.js | 10 +-
examples/async-typeahead-redux/index.js | 36 +-
examples/async-typeahead-redux/store.js | 8 +-
examples/async-typeahead-redux/useKeyword.js | 6 +-
examples/auto-save-field-blur/AutoSave.js | 42 +-
examples/auto-save-field-blur/Styles.js | 22 +-
examples/auto-save-field-blur/index.js | 62 +-
.../auto-save-selective-debounce/AutoSave.js | 20 +-
.../auto-save-selective-debounce/Styles.js | 22 +-
.../auto-save-selective-debounce/index.js | 58 +-
examples/auto-save-with-debounce/AutoSave.js | 38 +-
examples/auto-save-with-debounce/Styles.js | 22 +-
examples/auto-save-with-debounce/index.js | 54 +-
examples/calculated-fields/Styles.js | 22 +-
examples/calculated-fields/index.js | 48 +-
examples/chakra/index.js | 92 +-
examples/chakra/validate.js | 28 +-
examples/conditional-fields/Styles.js | 20 +-
examples/conditional-fields/index.js | 62 +-
examples/credit-card/Card.js | 40 +-
examples/credit-card/Styles.js | 20 +-
examples/credit-card/cardUtils.js | 54 +-
examples/credit-card/index.js | 46 +-
examples/credit-card/sandbox.config.json | 2 +-
.../OnBlurValidation.js | 56 +-
examples/custom-validation-engine/Styles.js | 22 +-
examples/custom-validation-engine/index.js | 56 +-
.../ErrorWithDelay.js | 39 +-
.../Styles.js | 18 +-
.../index.js | 56 +-
examples/declarative-form-rules/Styles.js | 20 +-
examples/declarative-form-rules/index.js | 44 +-
.../downshift-typeahead/DownshiftInput.js | 91 +-
examples/downshift-typeahead/Styles.js | 20 +-
examples/downshift-typeahead/fruit.js | 24 +-
examples/downshift-typeahead/index.js | 50 +-
examples/external-submit/Styles.js | 22 +-
examples/external-submit/index.js | 54 +-
examples/field-arrays/Styles.js | 20 +-
examples/field-arrays/index.js | 42 +-
examples/field-level-validation/Styles.js | 18 +-
examples/field-level-validation/index.js | 39 +-
examples/field-warnings/Styles.js | 22 +-
examples/field-warnings/index.js | 38 +-
examples/field-warnings/warning-engine.js | 4 +-
examples/fields-component/index.js | 40 +-
examples/focus-first-error/Styles.js | 20 +-
examples/focus-first-error/index.js | 46 +-
examples/focus-first-error/validate.js | 18 +-
examples/format-on-blur/Styles.js | 20 +-
examples/format-on-blur/index.js | 36 +-
examples/format-string-by-pattern/Styles.js | 22 +-
examples/format-string-by-pattern/index.js | 42 +-
.../Styles.js | 4 +-
.../index.js | 54 +-
.../Styles.js | 22 +-
.../index.js | 42 +-
.../Styles.js | 22 +-
.../index.js | 48 +-
.../BooleanDecay.js | 36 +-
.../ExternalModificationDetector.js | 33 +-
.../listening-for-external-changes/Styles.js | 20 +-
.../listening-for-external-changes/index.js | 44 +-
.../loading-initializing-values/Styles.js | 22 +-
examples/loading-initializing-values/index.js | 44 +-
.../LoadSaveReinitializeForm.js | 60 +-
.../loading-saving-reinitializing/Styles.js | 20 +-
.../loading-saving-reinitializing/index.js | 94 +-
examples/material-ui/index.js | 6 +-
examples/parse-format/Styles.js | 22 +-
examples/parse-format/index.js | 44 +-
examples/prefixed-fields/index.js | 36 +-
examples/record-level-validation/Styles.js | 18 +-
examples/record-level-validation/index.js | 36 +-
examples/redux/FormStateFromRedux.js | 12 +-
examples/redux/FormStateToRedux.js | 14 +-
examples/redux/Styles.js | 20 +-
examples/redux/finalFormDuck.js | 15 +-
examples/redux/index.js | 78 +-
examples/redux/store.js | 18 +-
examples/reusable-field-groups/Styles.js | 22 +-
examples/reusable-field-groups/index.js | 26 +-
examples/simple/Styles.js | 18 +-
examples/simple/index.js | 38 +-
.../components/CheckboxInput.tsx | 2 +-
.../components/MultiCheckboxInput.tsx | 2 +-
.../index.tsx | 48 +-
examples/styling-with-smooth-ui/index.js | 95 +-
examples/submission-errors/Styles.js | 18 +-
examples/submission-errors/index.js | 46 +-
examples/subscriptions/RenderCount.js | 12 +-
examples/subscriptions/Styles.js | 18 +-
examples/subscriptions/index.js | 35 +-
examples/third-party-components/Styles.js | 16 +-
examples/third-party-components/index.js | 50 +-
examples/third-party-components/states.js | 126 +-
examples/wizard/Styles.js | 22 +-
examples/wizard/Wizard.js | 64 +-
examples/wizard/index.js | 66 +-
package-scripts.js | 110 +-
rollup.config.js | 90 +-
src/Field.js | 34 +-
src/Field.test.js | 1108 ++++++++---------
src/FormSpy.js | 41 +-
src/FormSpy.test.js | 480 +++----
src/ReactFinalForm.js | 178 +--
src/ReactFinalForm.test.js | 1008 +++++++--------
src/context.js | 6 +-
src/getValue.js | 48 +-
src/getValue.test.js | 442 +++----
src/getters.js | 102 +-
src/index.js | 18 +-
src/index.js.flow | 36 +-
src/isReactNative.js | 6 +-
src/isSyntheticEvent.js | 4 +-
src/renderComponent.js | 26 +-
src/renderComponent.test.js | 104 +-
src/shallowEqual.js | 24 +-
src/shallowEqual.test.js | 88 +-
src/testUtils.js | 20 +-
src/types.js.flow | 77 +-
src/useConstant.js | 8 +-
src/useConstantCallback.js | 10 +-
src/useConstantCallback.test.js | 172 +--
src/useField.js | 180 +--
src/useField.test.js | 588 ++++-----
src/useForm.js | 18 +-
src/useForm.test.js | 100 +-
src/useFormState.js | 58 +-
src/useFormState.test.js | 58 +-
src/useLatest.js | 10 +-
src/useWhenValueChanges.js | 12 +-
typescript/Field.test.tsx | 4 +-
typescript/FormSpy.test.tsx | 4 +-
typescript/ReactFinalForm.test.tsx | 22 +-
typescript/index.d.ts | 94 +-
typescript/useFormState.test.tsx | 4 +-
150 files changed, 4667 insertions(+), 4568 deletions(-)
create mode 100644 .github/ci.yml
create mode 100644 .prettierignore
diff --git a/.babelrc.js b/.babelrc.js
index 9a36788b..688a14dd 100644
--- a/.babelrc.js
+++ b/.babelrc.js
@@ -1,35 +1,35 @@
-const { NODE_ENV } = process.env
-const test = NODE_ENV === 'test'
-const loose = true
+const { NODE_ENV } = process.env;
+const test = NODE_ENV === "test";
+const loose = true;
module.exports = {
presets: [
[
- '@babel/preset-env',
+ "@babel/preset-env",
{
loose,
- ...(test ? { targets: { node: '8' } } : {})
- }
+ ...(test ? { targets: { node: "8" } } : {}),
+ },
],
- '@babel/preset-react',
- '@babel/preset-flow'
+ "@babel/preset-react",
+ "@babel/preset-flow",
],
plugins: [
- '@babel/plugin-transform-flow-strip-types',
- '@babel/plugin-syntax-dynamic-import',
- '@babel/plugin-syntax-import-meta',
- ['@babel/plugin-proposal-class-properties', { loose }],
- '@babel/plugin-proposal-json-strings',
+ "@babel/plugin-transform-flow-strip-types",
+ "@babel/plugin-syntax-dynamic-import",
+ "@babel/plugin-syntax-import-meta",
+ ["@babel/plugin-proposal-class-properties", { loose }],
+ "@babel/plugin-proposal-json-strings",
[
- '@babel/plugin-proposal-decorators',
+ "@babel/plugin-proposal-decorators",
{
- legacy: true
- }
+ legacy: true,
+ },
],
- '@babel/plugin-proposal-function-sent',
- '@babel/plugin-proposal-export-namespace-from',
- '@babel/plugin-proposal-numeric-separator',
- '@babel/plugin-proposal-throw-expressions',
- test && '@babel/plugin-transform-react-jsx-source'
- ].filter(Boolean)
-}
+ "@babel/plugin-proposal-function-sent",
+ "@babel/plugin-proposal-export-namespace-from",
+ "@babel/plugin-proposal-numeric-separator",
+ "@babel/plugin-proposal-throw-expressions",
+ test && "@babel/plugin-transform-react-jsx-source",
+ ].filter(Boolean),
+};
diff --git a/.github/ci.yml b/.github/ci.yml
new file mode 100644
index 00000000..d93f882f
--- /dev/null
+++ b/.github/ci.yml
@@ -0,0 +1,49 @@
+name: CI
+
+on: [push]
+
+jobs:
+ lint:
+ name: Prettier Check
+ runs-on: ubuntu-latest
+
+ steps:
+ - uses: actions/checkout@v2
+ - name: Use Node.js ${{ matrix.node_version }}
+ uses: actions/setup-node@v2
+ with:
+ node-version: "14"
+ - name: Prepare env
+ run: yarn install --ignore-scripts --frozen-lockfile
+ - name: Run linter
+ run: yarn start lint
+
+ prettier:
+ name: Prettier Check
+ runs-on: ubuntu-latest
+
+ steps:
+ - uses: actions/checkout@v2
+ - name: Use Node.js ${{ matrix.node_version }}
+ uses: actions/setup-node@v2
+ with:
+ node-version: "14"
+ - name: Prepare env
+ run: yarn install --ignore-scripts --frozen-lockfile
+ - name: Run prettier
+ run: yarn start prettier
+
+ test:
+ name: Unit Tests
+ runs-on: ubuntu-latest
+
+ steps:
+ - uses: actions/checkout@v2
+ - name: Use Node.js ${{ matrix.node_version }}
+ uses: actions/setup-node@v2
+ with:
+ node-version: "14"
+ - name: Prepare env
+ run: yarn install --ignore-scripts --frozen-lockfile
+ - name: Run unit tests
+ run: yarn start test
diff --git a/.prettierignore b/.prettierignore
new file mode 100644
index 00000000..5de02b14
--- /dev/null
+++ b/.prettierignore
@@ -0,0 +1,3 @@
+coverage
+dist
+node_modules
\ No newline at end of file
diff --git a/.prettierrc b/.prettierrc
index 4bc38160..bf357fbb 100644
--- a/.prettierrc
+++ b/.prettierrc
@@ -1,18 +1,3 @@
{
- "semi": false,
- "singleQuote": true,
- "trailingComma": "none",
- "arrowParens": "avoid",
- "overrides": [
- {
- "files": [".prettierrc"],
- "options": { "parser": "json" }
- },
- {
- "files": ["*.ts*"],
- "options": {
- "semi": true
- }
- }
- ]
+ "trailingComma": "all"
}
diff --git a/examples/async-field-level-validation/Styles.js b/examples/async-field-level-validation/Styles.js
index a4edc9af..b8accf84 100644
--- a/examples/async-field-level-validation/Styles.js
+++ b/examples/async-field-level-validation/Styles.js
@@ -27,7 +27,9 @@ const btn = (light, dark) => css`
}
`;
-const btnDefault = css`${btn("#ffffff", "#d5d5d5")} color: #555;`;
+const btnDefault = css`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
const btnPrimary = btn("#4f93ce", "#285f8f");
diff --git a/examples/async-field-level-validation/index.js b/examples/async-field-level-validation/index.js
index d9940823..fda6a04a 100644
--- a/examples/async-field-level-validation/index.js
+++ b/examples/async-field-level-validation/index.js
@@ -1,53 +1,58 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import Spinner from './Spinner'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import Spinner from "./Spinner";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, undefined, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, undefined, 2));
+};
-const required = value => (value ? undefined : 'Required')
-const mustBeNumber = value => (isNaN(value) ? 'Must be a number' : undefined)
-const minValue = min => value =>
- isNaN(value) || value >= min ? undefined : `Should be greater than ${min}`
-const composeValidators = (...validators) => value =>
- validators.reduce((error, validator) => error || validator(value), undefined)
+const required = (value) => (value ? undefined : "Required");
+const mustBeNumber = (value) => (isNaN(value) ? "Must be a number" : undefined);
+const minValue = (min) => (value) =>
+ isNaN(value) || value >= min ? undefined : `Should be greater than ${min}`;
+const composeValidators =
+ (...validators) =>
+ (value) =>
+ validators.reduce(
+ (error, validator) => error || validator(value),
+ undefined,
+ );
-const simpleMemoize = fn => {
- let lastArg
- let lastResult
- return arg => {
+const simpleMemoize = (fn) => {
+ let lastArg;
+ let lastResult;
+ return (arg) => {
if (arg !== lastArg) {
- lastArg = arg
- lastResult = fn(arg)
+ lastArg = arg;
+ lastResult = fn(arg);
}
- return lastResult
- }
-}
+ return lastResult;
+ };
+};
-const usernameAvailable = simpleMemoize(async value => {
+const usernameAvailable = simpleMemoize(async (value) => {
if (!value) {
- return 'Required'
+ return "Required";
}
- await sleep(400)
+ await sleep(400);
if (
- ~['john', 'paul', 'george', 'ringo'].indexOf(value && value.toLowerCase())
+ ~["john", "paul", "george", "ringo"].indexOf(value && value.toLowerCase())
) {
- return 'Username taken!'
+ return "Username taken!";
}
-})
+});
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Asynchronous Field-Level Validation
@@ -107,6 +112,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/async-redux-submission/Styles.js b/examples/async-redux-submission/Styles.js
index 12d16329..75deb222 100644
--- a/examples/async-redux-submission/Styles.js
+++ b/examples/async-redux-submission/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -116,7 +116,7 @@ export default styled.div`
background: #eee;
}
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -196,10 +196,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -210,4 +210,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/async-redux-submission/asyncSubmissionMiddleware.js b/examples/async-redux-submission/asyncSubmissionMiddleware.js
index b1d54d15..f1158550 100644
--- a/examples/async-redux-submission/asyncSubmissionMiddleware.js
+++ b/examples/async-redux-submission/asyncSubmissionMiddleware.js
@@ -1,30 +1,31 @@
-import { REGISTER, REGISTER_SUCCESS } from './registrationDuck'
+import { REGISTER, REGISTER_SUCCESS } from "./registrationDuck";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const submit = async values => {
- await sleep(200)
- if (values.firstName === 'John') {
- throw Error({ firstName: "No John's Allowed!" })
+const submit = async (values) => {
+ await sleep(200);
+ if (values.firstName === "John") {
+ throw Error({ firstName: "No John's Allowed!" });
}
- window.alert(JSON.stringify(values, 0, 2))
-}
+ window.alert(JSON.stringify(values, 0, 2));
+};
/** This is to mimic the behavior of one of the various Redux async middlewares */
-const asyncSubmissionMiddleware = store => (next: Next) => (
- action: Action
-): State => {
- if (action && action.type === REGISTER) {
- submit(action.payload).then(
- () => store.dispatch({ type: REGISTER_SUCCESS }),
- errors => {
- // NOTE!! We are passing REGISTER_SUCCESS here because 🏁 Final Form expects
- // submit errors to come back in a *resolved* promise.
- store.dispatch({ type: REGISTER_SUCCESS, payload: errors })
- }
- )
- }
- return next(action)
-}
+const asyncSubmissionMiddleware =
+ (store) =>
+ (next: Next) =>
+ (action: Action): State => {
+ if (action && action.type === REGISTER) {
+ submit(action.payload).then(
+ () => store.dispatch({ type: REGISTER_SUCCESS }),
+ (errors) => {
+ // NOTE!! We are passing REGISTER_SUCCESS here because 🏁 Final Form expects
+ // submit errors to come back in a *resolved* promise.
+ store.dispatch({ type: REGISTER_SUCCESS, payload: errors });
+ },
+ );
+ }
+ return next(action);
+ };
-export default asyncSubmissionMiddleware
+export default asyncSubmissionMiddleware;
diff --git a/examples/async-redux-submission/index.js b/examples/async-redux-submission/index.js
index e08ee288..afd2f57f 100644
--- a/examples/async-redux-submission/index.js
+++ b/examples/async-redux-submission/index.js
@@ -1,15 +1,15 @@
-import React from 'react'
-import { render } from 'react-dom'
-import { Provider } from 'react-redux'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import store, { promiseListener } from './store'
+import React from "react";
+import { render } from "react-dom";
+import { Provider } from "react-redux";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import store, { promiseListener } from "./store";
import {
REGISTER,
REGISTER_SUCCESS,
- REGISTER_FAILURE
-} from './registrationDuck'
-import MakeAsyncFunction from 'react-redux-promise-listener'
+ REGISTER_FAILURE,
+} from "./registrationDuck";
+import MakeAsyncFunction from "react-redux-promise-listener";
const SubmitError = ({ name }) => (
(
submitError && !dirtySinceLastSubmit ? {submitError} : null
}
-)
+);
const App = () => (
@@ -28,7 +28,7 @@ const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Async Redux Submission
@@ -36,27 +36,27 @@ const App = () => (
Read Docs
- Demonstrates how to use an async Redux side-effects library, like{' '}
- redux-saga to manage form submissions using{' '}
+ Demonstrates how to use an async Redux side-effects library, like{" "}
+ redux-saga to manage form submissions using{" "}
🏁
- {' '}
- React Final Form. Uses{' '}
+ {" "}
+ React Final Form. Uses{" "}
redux-promise-listener
- {' '}
- and{' '}
+ {" "}
+ and{" "}
react-redux-promise-listener
- {' '}
+ {" "}
libraries.
Dispatched Redux actions can be found in the console.
@@ -69,7 +69,7 @@ const App = () => (
resolve={REGISTER_SUCCESS}
reject={REGISTER_FAILURE}
>
- {onSubmit => (
+ {(onSubmit) => (
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/async-redux-submission/registrationDuck.js b/examples/async-redux-submission/registrationDuck.js
index 3c96a7b7..261169bb 100644
--- a/examples/async-redux-submission/registrationDuck.js
+++ b/examples/async-redux-submission/registrationDuck.js
@@ -1,11 +1,11 @@
// QUACK! This is a duck. https://github.com/erikras/ducks-modular-redux
// Actions
-export const REGISTER = 'final-form-examples/registration/REGISTER'
+export const REGISTER = "final-form-examples/registration/REGISTER";
export const REGISTER_SUCCESS =
- 'final-form-examples/registration/REGISTER_SUCCESS'
+ "final-form-examples/registration/REGISTER_SUCCESS";
export const REGISTER_FAILURE =
- 'final-form-examples/registration/REGISTER_FAILURE'
+ "final-form-examples/registration/REGISTER_FAILURE";
// Reducer
export default function reducer(state = {}, action = {}) {
@@ -13,20 +13,20 @@ export default function reducer(state = {}, action = {}) {
case REGISTER:
return {
...state,
- registering: true
- }
+ registering: true,
+ };
case REGISTER_SUCCESS:
return {
...state,
- registering: false
- }
+ registering: false,
+ };
case REGISTER_FAILURE:
return {
...state,
- registering: false
- }
+ registering: false,
+ };
default:
- return state
+ return state;
}
}
diff --git a/examples/async-redux-submission/store.js b/examples/async-redux-submission/store.js
index 13422689..738cfde6 100644
--- a/examples/async-redux-submission/store.js
+++ b/examples/async-redux-submission/store.js
@@ -1,23 +1,26 @@
-import { createStore, combineReducers, applyMiddleware, compose } from 'redux'
-import createReduxPromiseListener from 'redux-promise-listener'
-import registration from './registrationDuck'
-import asyncSubmissionMiddleware from './asyncSubmissionMiddleware'
+import { createStore, combineReducers, applyMiddleware, compose } from "redux";
+import createReduxPromiseListener from "redux-promise-listener";
+import registration from "./registrationDuck";
+import asyncSubmissionMiddleware from "./asyncSubmissionMiddleware";
-const reduxPromiseListener = createReduxPromiseListener()
+const reduxPromiseListener = createReduxPromiseListener();
-const logger = store => (next: Next) => (action: Action): State => {
- console.log(action)
- return next(action)
-}
+const logger =
+ (store) =>
+ (next: Next) =>
+ (action: Action): State => {
+ console.log(action);
+ return next(action);
+ };
const reducer = combineReducers({
- registration
-})
+ registration,
+});
const composeEnhancers =
- (typeof window !== 'undefined' &&
+ (typeof window !== "undefined" &&
window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__) ||
- compose
+ compose;
const store = createStore(
reducer,
@@ -26,11 +29,11 @@ const store = createStore(
applyMiddleware(
reduxPromiseListener.middleware,
asyncSubmissionMiddleware,
- logger
- )
- )
-)
+ logger,
+ ),
+ ),
+);
-export const promiseListener = reduxPromiseListener // <---------- IMPORTANT
+export const promiseListener = reduxPromiseListener; // <---------- IMPORTANT
-export default store
+export default store;
diff --git a/examples/async-typeahead-redux/GithubUserTypeahead.jsx b/examples/async-typeahead-redux/GithubUserTypeahead.jsx
index 33105149..e9e4be59 100644
--- a/examples/async-typeahead-redux/GithubUserTypeahead.jsx
+++ b/examples/async-typeahead-redux/GithubUserTypeahead.jsx
@@ -18,13 +18,9 @@ const GithubUserTypeahead = ({ name, ...props }) => {
const dispatch = useDispatch();
const getOptions = useCallback(pathOr([], [keyword, "value"]), [keyword]);
const isLoading = useCallback(pathOr(false, [keyword, "loading"]), [keyword]);
- const handleOnSearch = useCallback(
- compose(
- dispatch,
- searchGithubUsers
- ),
- [dispatch]
- );
+ const handleOnSearch = useCallback(compose(dispatch, searchGithubUsers), [
+ dispatch,
+ ]);
const options = useSelector(getOptions);
const loading = useSelector(isLoading);
diff --git a/examples/async-typeahead-redux/Styles.js b/examples/async-typeahead-redux/Styles.js
index 127e3598..833d8dd0 100644
--- a/examples/async-typeahead-redux/Styles.js
+++ b/examples/async-typeahead-redux/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -81,7 +81,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -116,7 +116,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -182,11 +182,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/async-typeahead-redux/actions.js b/examples/async-typeahead-redux/actions.js
index 78f389e4..bd14b1cf 100644
--- a/examples/async-typeahead-redux/actions.js
+++ b/examples/async-typeahead-redux/actions.js
@@ -1,18 +1,18 @@
-export const requestGithubUsers = query => ({
+export const requestGithubUsers = (query) => ({
type: "GITHUB_USERS_REQUEST",
- query
+ query,
});
export const storeGithubUsers = (query, users) => ({
type: "GITHUB_USERS_RESPONSE",
query,
- users
+ users,
});
-export const searchGithubUsers = query => dispatch => {
+export const searchGithubUsers = (query) => (dispatch) => {
dispatch(requestGithubUsers(query));
fetch(`https://api.github.com/search/users?q=${query}`)
- .then(res => res.json())
+ .then((res) => res.json())
.then(({ items: users }) => dispatch(storeGithubUsers(query, users)));
};
diff --git a/examples/async-typeahead-redux/index.js b/examples/async-typeahead-redux/index.js
index 91f67580..787a53b0 100644
--- a/examples/async-typeahead-redux/index.js
+++ b/examples/async-typeahead-redux/index.js
@@ -1,28 +1,28 @@
-import React from 'react'
-import { render } from 'react-dom'
-import { Provider } from 'react-redux'
-import Styles from './Styles'
-import { Form } from 'react-final-form'
-import setFieldData from 'final-form-set-field-data'
+import React from "react";
+import { render } from "react-dom";
+import { Provider } from "react-redux";
+import Styles from "./Styles";
+import { Form } from "react-final-form";
+import setFieldData from "final-form-set-field-data";
-import configureStore from './store'
-import GithubUserTypeahead from './GithubUserTypeahead'
+import configureStore from "./store";
+import GithubUserTypeahead from "./GithubUserTypeahead";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
-const store = configureStore()
+const store = configureStore();
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
@@ -52,11 +52,11 @@ const App = () => (
)}
/>
-)
+);
render(
,
- document.getElementById('root')
-)
+ document.getElementById("root"),
+);
diff --git a/examples/async-typeahead-redux/store.js b/examples/async-typeahead-redux/store.js
index 76f0a68c..58fcc8a8 100644
--- a/examples/async-typeahead-redux/store.js
+++ b/examples/async-typeahead-redux/store.js
@@ -9,16 +9,16 @@ const reducer = (state = initialState, action) => {
...state,
[action.query]: {
...state[action.query],
- loading: true
- }
+ loading: true,
+ },
};
case "GITHUB_USERS_RESPONSE":
return {
...state,
[action.query]: {
value: action.users,
- loading: false
- }
+ loading: false,
+ },
};
default:
return state;
diff --git a/examples/async-typeahead-redux/useKeyword.js b/examples/async-typeahead-redux/useKeyword.js
index 5d5c3c27..8cb01a5a 100644
--- a/examples/async-typeahead-redux/useKeyword.js
+++ b/examples/async-typeahead-redux/useKeyword.js
@@ -1,15 +1,15 @@
import { useForm, useField } from "react-final-form";
import { propOr } from "ramda";
-const useKeyword = name => {
+const useKeyword = (name) => {
const {
- mutators: { setFieldData }
+ mutators: { setFieldData },
} = useForm();
const { meta } = useField(name, { subscription: { data: true } });
return {
keyword: propOr(null, "keyword", meta.data),
- updateKeyword: keyword => setFieldData(name, { keyword })
+ updateKeyword: (keyword) => setFieldData(name, { keyword }),
};
};
diff --git a/examples/auto-save-field-blur/AutoSave.js b/examples/auto-save-field-blur/AutoSave.js
index b036e076..2d517ac3 100644
--- a/examples/auto-save-field-blur/AutoSave.js
+++ b/examples/auto-save-field-blur/AutoSave.js
@@ -1,44 +1,44 @@
-import React from 'react'
-import { FormSpy } from 'react-final-form'
-import diff from 'object-diff'
+import React from "react";
+import { FormSpy } from "react-final-form";
+import diff from "object-diff";
class AutoSave extends React.Component {
constructor(props) {
- super(props)
- this.state = { values: props.values, submitting: false }
+ super(props);
+ this.state = { values: props.values, submitting: false };
}
componentWillReceiveProps(nextProps) {
if (this.props.active && this.props.active !== nextProps.active) {
// blur occurred
- this.save(this.props.active)
+ this.save(this.props.active);
}
}
- save = async blurredField => {
+ save = async (blurredField) => {
if (this.promise) {
- await this.promise
+ await this.promise;
}
- const { values, setFieldData, save } = this.props
+ const { values, setFieldData, save } = this.props;
// This diff step is totally optional
- const difference = diff(this.state.values, values)
+ const difference = diff(this.state.values, values);
if (Object.keys(difference).length) {
// values have changed
- this.setState({ submitting: true, values })
- setFieldData(blurredField, { saving: true })
- this.promise = save(difference)
- await this.promise
- delete this.promise
- this.setState({ submitting: false })
- setFieldData(blurredField, { saving: false })
+ this.setState({ submitting: true, values });
+ setFieldData(blurredField, { saving: true });
+ this.promise = save(difference);
+ await this.promise;
+ delete this.promise;
+ this.setState({ submitting: false });
+ setFieldData(blurredField, { saving: false });
}
- }
+ };
render() {
// This component doesn't have to render anything, but it can render
// submitting state.
- return null
+ return null;
}
}
@@ -48,10 +48,10 @@ class AutoSave extends React.Component {
// - Maintain state of when we are submitting
// - Render a message when submitting
// - Pass in save prop nicely
-export default props => (
+export default (props) => (
-)
+);
diff --git a/examples/auto-save-field-blur/Styles.js b/examples/auto-save-field-blur/Styles.js
index 1aa7b27e..a686b959 100644
--- a/examples/auto-save-field-blur/Styles.js
+++ b/examples/auto-save-field-blur/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -81,7 +81,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -116,7 +116,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -182,11 +182,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/auto-save-field-blur/index.js b/examples/auto-save-field-blur/index.js
index 4dc1a874..0044ba8d 100644
--- a/examples/auto-save-field-blur/index.js
+++ b/examples/auto-save-field-blur/index.js
@@ -1,16 +1,16 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import setFieldData from 'final-form-set-field-data'
-import AutoSave from './AutoSave'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import setFieldData from "final-form-set-field-data";
+import AutoSave from "./AutoSave";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const save = async values => {
- console.log('Saving', values)
- await sleep(1000)
-}
+const save = async (values) => {
+ console.log("Saving", values);
+ await sleep(1000);
+};
const SavingIndicator = ({ name }) => (
(
subscription={{ data: true }}
render={({
meta: {
- data: { saving }
- }
+ data: { saving },
+ },
}) => (saving ? Saving
: null)}
/>
-)
+);
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Auto-Save on Field Blur
@@ -37,21 +37,21 @@ const App = () => (
Read Docs
- The AutoSave component uses{' '}
+ The AutoSave component uses{" "}
FormSpy
- {' '}
+ {" "}
to listen to changes to values and which field is currently active and
auto-saves changes when a field is blurred. Look in the console for the
save events.
@@ -129,31 +129,31 @@ const App = () => (
🐷
- {' '}
+ {" "}
Ham
🍄
- {' '}
+ {" "}
Mushrooms
🧀
- {' '}
+ {" "}
Cheese
🐓
- {' '}
+ {" "}
Chicken
🍍
- {' '}
+ {" "}
Pinapple
@@ -168,7 +168,7 @@ const App = () => (
component="input"
type="radio"
value="larry"
- />{' '}
+ />{" "}
Larry
@@ -177,7 +177,7 @@ const App = () => (
component="input"
type="radio"
value="moe"
- />{' '}
+ />{" "}
Moe
@@ -186,7 +186,7 @@ const App = () => (
component="input"
type="radio"
value="curly"
- />{' '}
+ />{" "}
Curly
@@ -201,6 +201,6 @@ const App = () => (
)}
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/auto-save-selective-debounce/AutoSave.js b/examples/auto-save-selective-debounce/AutoSave.js
index 594ba79b..fde3ed32 100644
--- a/examples/auto-save-selective-debounce/AutoSave.js
+++ b/examples/auto-save-selective-debounce/AutoSave.js
@@ -7,14 +7,14 @@ const areObjectsIdentical = (a, b) =>
class AutoSave extends React.Component {
static defaultProps = {
- debounced: []
+ debounced: [],
};
constructor(props) {
super(props);
this.state = {
submitting: false,
- ...this.splitValues(props.values)
+ ...this.splitValues(props.values),
};
}
@@ -36,13 +36,13 @@ class AutoSave extends React.Component {
}
}
- splitValues = values => {
+ splitValues = (values) => {
const { debounced } = this.props;
const debouncedValues = {};
const immediateValues = {};
- Object.keys(values).forEach(key => {
+ Object.keys(values).forEach((key) => {
if (debounced.includes(key)) {
debouncedValues[key] = values[key];
} else {
@@ -52,7 +52,7 @@ class AutoSave extends React.Component {
return {
debouncedValues,
- immediateValues
+ immediateValues,
};
};
@@ -65,22 +65,22 @@ class AutoSave extends React.Component {
const { debouncedValues, immediateValues } = this.splitValues(values);
this.setState(
- state => ({
+ (state) => ({
submitting: true,
immediateValues: { ...immediateValues },
- debouncedValues: { ...debouncedValues }
+ debouncedValues: { ...debouncedValues },
}),
async () => {
this.promise = save({
...this.state.immediateValues,
- ...this.state.debouncedValues
+ ...this.state.debouncedValues,
});
await this.promise;
delete this.promise;
this.setState({ submitting: false });
- }
+ },
);
};
@@ -99,6 +99,6 @@ class AutoSave extends React.Component {
// - Maintain state of when we are submitting
// - Render a message when submitting
// - Pass in debounce and save props nicely
-export default props => (
+export default (props) => (
);
diff --git a/examples/auto-save-selective-debounce/Styles.js b/examples/auto-save-selective-debounce/Styles.js
index d46ad21e..b7fb0008 100644
--- a/examples/auto-save-selective-debounce/Styles.js
+++ b/examples/auto-save-selective-debounce/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -81,7 +81,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -116,7 +116,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -176,11 +176,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/auto-save-selective-debounce/index.js b/examples/auto-save-selective-debounce/index.js
index 22b1c481..91d22e47 100644
--- a/examples/auto-save-selective-debounce/index.js
+++ b/examples/auto-save-selective-debounce/index.js
@@ -1,22 +1,22 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import AutoSave from './AutoSave'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import AutoSave from "./AutoSave";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const save = async values => {
- console.log('Saving', values)
- await sleep(2000)
-}
+const save = async (values) => {
+ console.log("Saving", values);
+ await sleep(2000);
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
@@ -26,22 +26,22 @@ const App = () => (
Read Docs
- The AutoSave component uses{' '}
+ The AutoSave component uses{" "}
FormSpy
- {' '}
+ {" "}
to listen to changes to values and auto-save changes. But it only
- debounces changes to a specific list of fields given to the{' '}
+ debounces changes to a specific list of fields given to the{" "}
AutoSave component, in this case, all the text inputs. Look
in the console for the save events.
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/auto-save-with-debounce/AutoSave.js b/examples/auto-save-with-debounce/AutoSave.js
index ed1bc5b2..7763ef63 100644
--- a/examples/auto-save-with-debounce/AutoSave.js
+++ b/examples/auto-save-with-debounce/AutoSave.js
@@ -1,44 +1,44 @@
-import React from 'react'
-import { FormSpy } from 'react-final-form'
-import diff from 'object-diff'
+import React from "react";
+import { FormSpy } from "react-final-form";
+import diff from "object-diff";
class AutoSave extends React.Component {
constructor(props) {
- super(props)
- this.state = { values: props.values, submitting: false }
+ super(props);
+ this.state = { values: props.values, submitting: false };
}
componentWillReceiveProps(nextProps) {
if (this.timeout) {
- clearTimeout(this.timeout)
+ clearTimeout(this.timeout);
}
- this.timeout = setTimeout(this.save, this.props.debounce)
+ this.timeout = setTimeout(this.save, this.props.debounce);
}
save = async () => {
if (this.promise) {
- await this.promise
+ await this.promise;
}
- const { values, save } = this.props
+ const { values, save } = this.props;
// This diff step is totally optional
- const difference = diff(this.state.values, values)
+ const difference = diff(this.state.values, values);
if (Object.keys(difference).length) {
// values have changed
- this.setState({ submitting: true, values })
- this.promise = save(difference)
- await this.promise
- delete this.promise
- this.setState({ submitting: false })
+ this.setState({ submitting: true, values });
+ this.promise = save(difference);
+ await this.promise;
+ delete this.promise;
+ this.setState({ submitting: false });
}
- }
+ };
render() {
// This component doesn't have to render anything, but it can render
// submitting state.
return (
this.state.submitting && Submitting...
- )
+ );
}
}
@@ -48,6 +48,6 @@ class AutoSave extends React.Component {
// - Maintain state of when we are submitting
// - Render a message when submitting
// - Pass in debounce and save props nicely
-export default props => (
+export default (props) => (
-)
+);
diff --git a/examples/auto-save-with-debounce/Styles.js b/examples/auto-save-with-debounce/Styles.js
index d46ad21e..b7fb0008 100644
--- a/examples/auto-save-with-debounce/Styles.js
+++ b/examples/auto-save-with-debounce/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -81,7 +81,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -116,7 +116,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -176,11 +176,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/auto-save-with-debounce/index.js b/examples/auto-save-with-debounce/index.js
index fc6e9332..7c80d1aa 100644
--- a/examples/auto-save-with-debounce/index.js
+++ b/examples/auto-save-with-debounce/index.js
@@ -1,22 +1,22 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import AutoSave from './AutoSave'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import AutoSave from "./AutoSave";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const save = async values => {
- console.log('Saving', values)
- await sleep(2000)
-}
+const save = async (values) => {
+ console.log("Saving", values);
+ await sleep(2000);
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Auto-Save with Debounce
@@ -24,20 +24,20 @@ const App = () => (
Read Docs
- The AutoSave component uses{' '}
+ The AutoSave component uses{" "}
FormSpy
- {' '}
+ {" "}
to listen to changes to values and auto-save changes. Look in the console
for the save events.
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/calculated-fields/Styles.js b/examples/calculated-fields/Styles.js
index 2a2bfadd..3f54fa69 100644
--- a/examples/calculated-fields/Styles.js
+++ b/examples/calculated-fields/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -144,10 +144,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -166,4 +166,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/calculated-fields/index.js b/examples/calculated-fields/index.js
index 4e920e66..a7cb1af2 100644
--- a/examples/calculated-fields/index.js
+++ b/examples/calculated-fields/index.js
@@ -1,32 +1,32 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import createDecorator from 'final-form-calculate'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import createDecorator from "final-form-calculate";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const calculator = createDecorator(
{
- field: 'minimum', // when minimum changes...
+ field: "minimum", // when minimum changes...
updates: {
// ...update maximum to the result of this function
maximum: (minimumValue, allValues) =>
- Math.max(minimumValue || 0, allValues.maximum || 0)
- }
+ Math.max(minimumValue || 0, allValues.maximum || 0),
+ },
},
{
- field: 'maximum', // when maximum changes...
+ field: "maximum", // when maximum changes...
updates: {
// update minimum to the result of this function
minimum: (maximumValue, allValues) =>
- Math.min(maximumValue || 0, allValues.minimum || 0)
- }
+ Math.min(maximumValue || 0, allValues.minimum || 0),
+ },
},
{
field: /day\[\d\]/, // when a field matching this pattern changes...
@@ -35,18 +35,18 @@ const calculator = createDecorator(
total: (ignoredValue, allValues) =>
(allValues.day || []).reduce(
(sum, value) => sum + Number(value || 0),
- 0
- )
- }
- }
-)
+ 0,
+ ),
+ },
+ },
+);
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Calculated Fields
@@ -159,6 +159,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/chakra/index.js b/examples/chakra/index.js
index 12601ae9..0f9da298 100644
--- a/examples/chakra/index.js
+++ b/examples/chakra/index.js
@@ -1,6 +1,6 @@
/* eslint-disable jsx-a11y/accessible-emoji */
-import React from 'react'
-import { render } from 'react-dom'
+import React from "react";
+import { render } from "react-dom";
import {
Box,
Button,
@@ -20,17 +20,17 @@ import {
Radio,
RadioGroup,
Stack,
- Textarea
-} from '@chakra-ui/core'
-import { Form, Field, useField, useForm } from 'react-final-form'
-import validate from './validate'
+ Textarea,
+} from "@chakra-ui/core";
+import { Form, Field, useField, useForm } from "react-final-form";
+import validate from "./validate";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => (
@@ -43,7 +43,7 @@ const App = () => (
Chakra Example
- Example using React Final Form and{' '}
+ Example using React Final Form and{" "}
Chakra
@@ -63,7 +63,7 @@ const App = () => (
errors,
submitting,
pristine,
- values
+ values,
}) => (
(
/>
-)
+);
const AdaptedTextarea = ({ input, meta, ...rest }) => (
-)
+);
const CheckboxControl = ({ name, value, children }) => {
const {
input: { checked, ...input },
- meta: { error, touched, invalid }
+ meta: { error, touched, invalid },
} = useField(name, {
- type: 'checkbox' // important for RFF to manage the checked prop
- })
+ type: "checkbox", // important for RFF to manage the checked prop
+ });
return (
@@ -168,23 +168,23 @@ const CheckboxControl = ({ name, value, children }) => {
{error}
- )
-}
+ );
+};
const CheckboxArrayControl = ({ name, value, children }) => {
const {
input: { checked, ...input },
- meta: { error, touched }
+ meta: { error, touched },
} = useField(name, {
- type: 'checkbox', // important for RFF to manage the checked prop
- value // important for RFF to manage list of strings
- })
+ type: "checkbox", // important for RFF to manage the checked prop
+ value, // important for RFF to manage list of strings
+ });
return (
{children}
- )
-}
+ );
+};
const AdaptedRadioGroup = ({ input, meta, label, children }) => (
@@ -192,24 +192,24 @@ const AdaptedRadioGroup = ({ input, meta, label, children }) => (
{children}
{meta.error}
-)
+);
const Control = ({ name, ...rest }) => {
const {
- meta: { error, touched }
- } = useField(name, { subscription: { touched: true, error: true } })
- return
-}
+ meta: { error, touched },
+ } = useField(name, { subscription: { touched: true, error: true } });
+ return ;
+};
const Error = ({ name }) => {
const {
- meta: { error }
- } = useField(name, { subscription: { error: true } })
- return {error}
-}
+ meta: { error },
+ } = useField(name, { subscription: { error: true } });
+ return {error} ;
+};
const InputControl = ({ name, label }) => {
- const { input, meta } = useField(name)
+ const { input, meta } = useField(name);
return (
{label}
@@ -221,8 +221,8 @@ const InputControl = ({ name, label }) => {
/>
- )
-}
+ );
+};
const TextareaControl = ({ name, label }) => (
@@ -235,18 +235,18 @@ const TextareaControl = ({ name, label }) => (
/>
-)
+);
-const PercentComplete = props => {
- const form = useForm()
- const numFields = form.getRegisteredFields().length
- const numErrors = Object.keys(form.getState().errors).length
+const PercentComplete = (props) => {
+ const form = useForm();
+ const numFields = form.getRegisteredFields().length;
+ const numErrors = Object.keys(form.getState().errors).length;
return (
- )
-}
+ );
+};
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/chakra/validate.js b/examples/chakra/validate.js
index d49b6ed3..19d9e21c 100644
--- a/examples/chakra/validate.js
+++ b/examples/chakra/validate.js
@@ -1,27 +1,27 @@
-const validate = values => {
- const errors = {}
+const validate = (values) => {
+ const errors = {};
if (!values.firstName) {
- errors.firstName = 'Required'
+ errors.firstName = "Required";
}
if (!values.lastName) {
- errors.lastName = 'Required'
+ errors.lastName = "Required";
}
if (values.employed) {
- errors.employed = "We're only accepted unemployed applicants at the moment"
+ errors.employed = "We're only accepted unemployed applicants at the moment";
}
if (!values.favoriteColor) {
- errors.favoriteColor = 'Required'
- } else if (values.favoriteColor === '#00ff00') {
- errors.favoriteColor = 'Not green! Gross!'
+ errors.favoriteColor = "Required";
+ } else if (values.favoriteColor === "#00ff00") {
+ errors.favoriteColor = "Not green! Gross!";
}
if (!values.toppings || values.toppings.length < 2) {
- errors.toppings = 'You need at least two toppings'
+ errors.toppings = "You need at least two toppings";
} else if (values.toppings && values.toppings.length > 3) {
- errors.toppings = 'No more than three toppings'
+ errors.toppings = "No more than three toppings";
}
if (!values.notes) {
- errors.notes = 'Required'
+ errors.notes = "Required";
}
- return errors
-}
-export default validate
+ return errors;
+};
+export default validate;
diff --git a/examples/conditional-fields/Styles.js b/examples/conditional-fields/Styles.js
index dd091033..698c8005 100644
--- a/examples/conditional-fields/Styles.js
+++ b/examples/conditional-fields/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -108,7 +108,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -188,10 +188,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -202,4 +202,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/conditional-fields/index.js b/examples/conditional-fields/index.js
index 666873d7..7ed0afd6 100644
--- a/examples/conditional-fields/index.js
+++ b/examples/conditional-fields/index.js
@@ -1,15 +1,15 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import pickupTimes from './pickupTimes'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import pickupTimes from "./pickupTimes";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const Error = ({ name }) => (
@@ -17,20 +17,20 @@ const Error = ({ name }) => (
error && touched ? {error} : null
}
-)
+);
const Condition = ({ when, is, children }) => (
{({ input: { value } }) => (value === is ? children : null)}
-)
+);
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Conditional Fields
@@ -40,34 +40,34 @@ const App = () => (
Sometimes you might want to conditionally show or hide some parts of your
form depending on values the user has already provided for other form
- inputs.{' '}
+ inputs.{" "}
🏁
- {' '}
- React Final Form makes that very easy to do by creating a{' '}
+ {" "}
+ React Final Form makes that very easy to do by creating a{" "}
Condition component out of a Field component.
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/credit-card/Card.js b/examples/credit-card/Card.js
index 62964fcf..ab6ef7b0 100644
--- a/examples/credit-card/Card.js
+++ b/examples/credit-card/Card.js
@@ -1,6 +1,6 @@
-import React from 'react'
-import styled from 'styled-components'
-import Cards from 'react-credit-cards'
+import React from "react";
+import styled from "styled-components";
+import Cards from "react-credit-cards";
const Container = styled.div`
.rccs {
@@ -91,7 +91,7 @@ const Container = styled.div`
background: linear-gradient(25deg, #308c67, #a3f2cf);
}
.rccs__card--amex .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--amex .rccs__cvc__front {
opacity: 0.5;
@@ -101,7 +101,7 @@ const Container = styled.div`
background: linear-gradient(25deg, #ccc, #999);
}
.rccs__card--dankort .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--dinersclub > div {
color: #555;
@@ -110,7 +110,7 @@ const Container = styled.div`
background: linear-gradient(25deg, #fff, #eee);
}
.rccs__card--dinersclub .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--discover > div {
color: #555;
@@ -119,38 +119,38 @@ const Container = styled.div`
background: linear-gradient(25deg, #fff, #eee);
}
.rccs__card--discover .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--elo .rccs__card__background {
background: linear-gradient(25deg, #211c18, #aaa7a2);
}
.rccs__card--elo .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--hipercard .rccs__card__background {
background: linear-gradient(25deg, #8b181b, #de1f27);
}
.rccs__card--hipercard .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--jcb .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--laser .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--maestro .rccs__card__background,
.rccs__card--mastercard .rccs__card__background {
background: linear-gradient(25deg, #f37b26, #fdb731);
}
.rccs__card--maestro .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--mastercard .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--unionpay .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--visa .rccs__card__background,
.rccs__card--visaelectron .rccs__card__background {
@@ -161,10 +161,10 @@ const Container = styled.div`
background-size: 75%;
}
.rccs__card--visa .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__card--visaelectron .rccs__issuer {
- background-image: url('');
+ background-image: url("");
}
.rccs__number {
clear: both;
@@ -217,7 +217,7 @@ const Container = styled.div`
transition: opacity 0.3s;
}
.rccs__chip {
- background-image: url('');
+ background-image: url("");
background-repeat: no-repeat;
background-size: contain;
height: 26.36364px;
@@ -284,10 +284,10 @@ const Container = styled.div`
font-weight: 700;
opacity: 1 !important;
}
-`
+`;
-export default props => (
+export default (props) => (
-)
+);
diff --git a/examples/credit-card/Styles.js b/examples/credit-card/Styles.js
index 12d16329..75deb222 100644
--- a/examples/credit-card/Styles.js
+++ b/examples/credit-card/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -116,7 +116,7 @@ export default styled.div`
background: #eee;
}
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -196,10 +196,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -210,4 +210,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/credit-card/cardUtils.js b/examples/credit-card/cardUtils.js
index a005004f..bcc860c6 100644
--- a/examples/credit-card/cardUtils.js
+++ b/examples/credit-card/cardUtils.js
@@ -1,60 +1,60 @@
-import Payment from 'payment'
+import Payment from "payment";
-function clearNumber(value = '') {
- return value.replace(/\D+/g, '')
+function clearNumber(value = "") {
+ return value.replace(/\D+/g, "");
}
export function formatCreditCardNumber(value) {
if (!value) {
- return value
+ return value;
}
- const issuer = Payment.fns.cardType(value)
- const clearValue = clearNumber(value)
- let nextValue
+ const issuer = Payment.fns.cardType(value);
+ const clearValue = clearNumber(value);
+ let nextValue;
switch (issuer) {
- case 'amex':
+ case "amex":
nextValue = `${clearValue.slice(0, 4)} ${clearValue.slice(
4,
- 10
- )} ${clearValue.slice(10, 15)}`
- break
- case 'dinersclub':
+ 10,
+ )} ${clearValue.slice(10, 15)}`;
+ break;
+ case "dinersclub":
nextValue = `${clearValue.slice(0, 4)} ${clearValue.slice(
4,
- 10
- )} ${clearValue.slice(10, 14)}`
- break
+ 10,
+ )} ${clearValue.slice(10, 14)}`;
+ break;
default:
nextValue = `${clearValue.slice(0, 4)} ${clearValue.slice(
4,
- 8
- )} ${clearValue.slice(8, 12)} ${clearValue.slice(12, 19)}`
- break
+ 8,
+ )} ${clearValue.slice(8, 12)} ${clearValue.slice(12, 19)}`;
+ break;
}
- return nextValue.trim()
+ return nextValue.trim();
}
export function formatCVC(value, prevValue, allValues = {}) {
- const clearValue = clearNumber(value)
- let maxLength = 4
+ const clearValue = clearNumber(value);
+ let maxLength = 4;
if (allValues.number) {
- const issuer = Payment.fns.cardType(allValues.number)
- maxLength = issuer === 'amex' ? 4 : 3
+ const issuer = Payment.fns.cardType(allValues.number);
+ maxLength = issuer === "amex" ? 4 : 3;
}
- return clearValue.slice(0, maxLength)
+ return clearValue.slice(0, maxLength);
}
export function formatExpirationDate(value) {
- const clearValue = clearNumber(value)
+ const clearValue = clearNumber(value);
if (clearValue.length >= 3) {
- return `${clearValue.slice(0, 2)}/${clearValue.slice(2, 4)}`
+ return `${clearValue.slice(0, 2)}/${clearValue.slice(2, 4)}`;
}
- return clearValue
+ return clearValue;
}
diff --git a/examples/credit-card/index.js b/examples/credit-card/index.js
index eb8032aa..a1b9e31b 100644
--- a/examples/credit-card/index.js
+++ b/examples/credit-card/index.js
@@ -1,27 +1,27 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import Card from './Card'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import Card from "./Card";
import {
formatCreditCardNumber,
formatCVC,
- formatExpirationDate
-} from './cardUtils'
+ formatExpirationDate,
+} from "./cardUtils";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Credit Card Example
@@ -29,14 +29,14 @@ const App = () => (
Read Docs
- This example demonstrates how to use the amazing{' '}
+ This example demonstrates how to use the amazing{" "}
React Credit Cards
- {' '}
+ {" "}
library with your form.
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/declarative-form-rules/Styles.js b/examples/declarative-form-rules/Styles.js
index 9d887124..7dfd0c60 100644
--- a/examples/declarative-form-rules/Styles.js
+++ b/examples/declarative-form-rules/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -116,7 +116,7 @@ export default styled.div`
background: #eee;
}
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -196,10 +196,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -210,4 +210,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/declarative-form-rules/index.js b/examples/declarative-form-rules/index.js
index 6ddb9e64..e1061231 100644
--- a/examples/declarative-form-rules/index.js
+++ b/examples/declarative-form-rules/index.js
@@ -1,39 +1,39 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import { OnChange } from 'react-final-form-listeners'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import { OnChange } from "react-final-form-listeners";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const WhenFieldChanges = ({ field, becomes, set, to }) => (
{(
// No subscription. We only use Field to get to the change function
- { input: { onChange } }
+ { input: { onChange } },
) => (
- {value => {
+ {(value) => {
if (value === becomes) {
- onChange(to)
+ onChange(to);
}
}}
)}
-)
+);
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Declarative Form Rules
@@ -41,7 +41,7 @@ const App = () => (
Read Docs
- This example demonstrates how to use{' '}
+ This example demonstrates how to use{" "}
(
>
🏁
- {' '}
+ {" "}
React Final Form Listeners
- {' '}
+ {" "}
to listen to the change of one field to then update the value of other
fields.
@@ -68,7 +68,7 @@ const App = () => (
submitting,
pristine,
values,
- errors
+ errors,
}) => {
return (
@@ -121,9 +121,9 @@ const App = () => (
Data
{JSON.stringify(values, 0, 2)}
- )
+ );
}}
/>
-)
-render( , document.getElementById('root'))
+);
+render( , document.getElementById("root"));
diff --git a/examples/downshift-typeahead/DownshiftInput.js b/examples/downshift-typeahead/DownshiftInput.js
index c35659ae..7a7c1cd1 100644
--- a/examples/downshift-typeahead/DownshiftInput.js
+++ b/examples/downshift-typeahead/DownshiftInput.js
@@ -1,14 +1,14 @@
-import React from 'react'
-import Downshift from 'downshift'
-import matchSorter from 'match-sorter'
+import React from "react";
+import Downshift from "downshift";
+import matchSorter from "match-sorter";
-const itemToString = item => (item ? item : '')
+const itemToString = (item) => (item ? item : "");
const DownshiftInput = ({ input, meta, placeholder, items, ...rest }) => (
{
- input.onChange(inputValue)
+ onInputValueChange={(inputValue) => {
+ input.onChange(inputValue);
}}
itemToString={itemToString}
selectedItem={input.value}
@@ -20,55 +20,54 @@ const DownshiftInput = ({ input, meta, placeholder, items, ...rest }) => (
isOpen,
inputValue,
highlightedIndex,
- selectedItem
+ selectedItem,
}) => {
const filteredItems = matchSorter(items, inputValue, {
- keys: ['label'],
- maxRanking: matchSorter.rankings.STARTS_WITH
- })
+ keys: ["label"],
+ maxRanking: matchSorter.rankings.STARTS_WITH,
+ });
return (
-
+
- {isOpen &&
- !!filteredItems.length && (
-
- {filteredItems.map(({ value, label }, index) => (
-
- {label}
-
- ))}
-
- )}
+ {isOpen && !!filteredItems.length && (
+
+ {filteredItems.map(({ value, label }, index) => (
+
+ {label}
+
+ ))}
+
+ )}
- )
+ );
}}
-)
+);
-export default DownshiftInput
+export default DownshiftInput;
diff --git a/examples/downshift-typeahead/Styles.js b/examples/downshift-typeahead/Styles.js
index dd091033..698c8005 100644
--- a/examples/downshift-typeahead/Styles.js
+++ b/examples/downshift-typeahead/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -108,7 +108,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -188,10 +188,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -202,4 +202,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/downshift-typeahead/fruit.js b/examples/downshift-typeahead/fruit.js
index c324128e..926cbeca 100644
--- a/examples/downshift-typeahead/fruit.js
+++ b/examples/downshift-typeahead/fruit.js
@@ -1,13 +1,13 @@
export default [
- { value: 'Apple', label: '🍎 Apple' },
- { value: 'Banana', label: '🍌 Banana' },
- { value: 'Cherry', label: '🍒 Cherry' },
- { value: 'Grape', label: '🍇 Grape' },
- { value: 'Kiwi', label: '🥝 Kiwi' },
- { value: 'Orange', label: '🍊 Orange' },
- { value: 'Peach', label: '🍑 Peach' },
- { value: 'Pear', label: '🍐 Pear' },
- { value: 'Pineapple', label: '🍍 Pineapple' },
- { value: 'Strawberry', label: '🍓 Strawberry' },
- { value: 'Watermelon', label: '🍉 Watermelon' }
-]
+ { value: "Apple", label: "🍎 Apple" },
+ { value: "Banana", label: "🍌 Banana" },
+ { value: "Cherry", label: "🍒 Cherry" },
+ { value: "Grape", label: "🍇 Grape" },
+ { value: "Kiwi", label: "🥝 Kiwi" },
+ { value: "Orange", label: "🍊 Orange" },
+ { value: "Peach", label: "🍑 Peach" },
+ { value: "Pear", label: "🍐 Pear" },
+ { value: "Pineapple", label: "🍍 Pineapple" },
+ { value: "Strawberry", label: "🍓 Strawberry" },
+ { value: "Watermelon", label: "🍉 Watermelon" },
+];
diff --git a/examples/downshift-typeahead/index.js b/examples/downshift-typeahead/index.js
index cf2019e9..6865ca85 100644
--- a/examples/downshift-typeahead/index.js
+++ b/examples/downshift-typeahead/index.js
@@ -1,26 +1,26 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import DownshiftInput from './DownshiftInput'
-import fruit from './fruit'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import DownshiftInput from "./DownshiftInput";
+import fruit from "./fruit";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
-const validate = values => {
- const errors = {}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
+const validate = (values) => {
+ const errors = {};
if (!values.firstName) {
- errors.firstName = 'Required'
+ errors.firstName = "Required";
}
if (!values.fruit) {
- errors.fruit = 'Required'
+ errors.fruit = "Required";
}
- return errors
-}
+ return errors;
+};
const Error = ({ name }) => (
(
touched && error ? {error} : null
}
/>
-)
+);
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
🏎️
- {' '}
+ {" "}
Downshift Example
Read Docs
- This example demonstrates using a{' '}
+ This example demonstrates using a{" "}
🏎️
- {' '}
+ {" "}
Downshift
- {' '}
+ {" "}
type-ahead component.
(
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/external-submit/Styles.js b/examples/external-submit/Styles.js
index 06a77f51..0b4ad747 100644
--- a/examples/external-submit/Styles.js
+++ b/examples/external-submit/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -160,11 +160,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/external-submit/index.js b/examples/external-submit/index.js
index a858e3a8..f9957060 100644
--- a/examples/external-submit/index.js
+++ b/examples/external-submit/index.js
@@ -1,23 +1,23 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => {
- let submit
+ let submit;
return (
🏁
- {' '}
+ {" "}
React Final Form Example
External Submit
@@ -31,33 +31,31 @@ const App = () => {
// { cancelable: true } required for Firefox
// https://github.com/facebook/react/issues/12639#issuecomment-382519193
document
- .getElementById('exampleForm')
- .dispatchEvent(new Event('submit', { cancelable: true, bubbles:true }))
+ .getElementById("exampleForm")
+ .dispatchEvent(
+ new Event("submit", { cancelable: true, bubbles: true }),
+ )
}
>
External Submit via document.getElementById()
{
- submit(event)
+ onClick={(event) => {
+ submit(event);
}}
style={{ marginTop: 10 }}
>
External Submit via closure
-
+
External Submit via form attribute
{
- submit = handleSubmit
+ submit = handleSubmit;
return (
@@ -85,19 +83,19 @@ const App = () => {
❤️
- {' '}
+ {" "}
Red
💚
- {' '}
+ {" "}
Green
💙
- {' '}
+ {" "}
Blue
@@ -116,11 +114,11 @@ const App = () => {
{JSON.stringify(values, 0, 2)}
- )
+ );
}}
/>
- )
-}
+ );
+};
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/field-arrays/Styles.js b/examples/field-arrays/Styles.js
index bba6ab2d..b90cade7 100644
--- a/examples/field-arrays/Styles.js
+++ b/examples/field-arrays/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,12 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
-const btnDefault = css`${btn('#ffffff', '#d5d5d5')} color: #555;`
+const btnDefault = css`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -96,7 +98,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -126,10 +128,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -148,4 +150,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/field-arrays/index.js b/examples/field-arrays/index.js
index 1a39cb06..95f3735e 100644
--- a/examples/field-arrays/index.js
+++ b/examples/field-arrays/index.js
@@ -1,23 +1,23 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import arrayMutators from 'final-form-arrays'
-import { FieldArray } from 'react-final-form-arrays'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import arrayMutators from "final-form-arrays";
+import { FieldArray } from "react-final-form-arrays";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form - Array Fields
@@ -26,17 +26,17 @@ const App = () => (
{
return (
@@ -47,11 +47,11 @@ const App = () => (
push('customers', undefined)}
+ onClick={() => push("customers", undefined)}
>
Add Customer
- pop('customers')}>
+ pop("customers")}>
Remove Customer
@@ -72,7 +72,7 @@ const App = () => (
/>
fields.remove(index)}
- style={{ cursor: 'pointer' }}
+ style={{ cursor: "pointer" }}
role="img"
aria-label="remove customer icon"
>
@@ -97,10 +97,10 @@ const App = () => (
{JSON.stringify(values, 0, 2)}
- )
+ );
}}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/field-level-validation/Styles.js b/examples/field-level-validation/Styles.js
index 0a371734..a1c46db5 100644
--- a/examples/field-level-validation/Styles.js
+++ b/examples/field-level-validation/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,13 +25,13 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
+const btnPrimary = btn("#4f93ce", "#285f8f");
export default styled.div`
font-family: sans-serif;
@@ -85,7 +85,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -112,10 +112,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -126,4 +126,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/field-level-validation/index.js b/examples/field-level-validation/index.js
index cea9c24b..e4658091 100644
--- a/examples/field-level-validation/index.js
+++ b/examples/field-level-validation/index.js
@@ -1,21 +1,26 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
-const required = value => (value ? undefined : 'Required')
-const mustBeNumber = value => (isNaN(value) ? 'Must be a number' : undefined)
-const minValue = min => value =>
- isNaN(value) || value >= min ? undefined : `Should be greater than ${min}`
-const composeValidators = (...validators) => value =>
- validators.reduce((error, validator) => error || validator(value), undefined)
+const required = (value) => (value ? undefined : "Required");
+const mustBeNumber = (value) => (isNaN(value) ? "Must be a number" : undefined);
+const minValue = (min) => (value) =>
+ isNaN(value) || value >= min ? undefined : `Should be greater than ${min}`;
+const composeValidators =
+ (...validators) =>
+ (value) =>
+ validators.reduce(
+ (error, validator) => error || validator(value),
+ undefined,
+ );
const App = () => (
@@ -79,6 +84,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/field-warnings/Styles.js b/examples/field-warnings/Styles.js
index 2a2bfadd..3f54fa69 100644
--- a/examples/field-warnings/Styles.js
+++ b/examples/field-warnings/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -144,10 +144,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -166,4 +166,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/field-warnings/index.js b/examples/field-warnings/index.js
index 23fbea02..7adcfdd0 100644
--- a/examples/field-warnings/index.js
+++ b/examples/field-warnings/index.js
@@ -1,30 +1,30 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { WarningEngine } from './warning-engine'
-import { Form, Field } from 'react-final-form'
-import setFieldData from 'final-form-set-field-data'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { WarningEngine } from "./warning-engine";
+import { Form, Field } from "react-final-form";
+import setFieldData from "final-form-set-field-data";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
⚠️
- {' '}
- Warnings{' '}
+ {" "}
+ Warnings{" "}
⚠️
@@ -34,7 +34,7 @@ const App = () => (
Warnings, in this example, are defined as: suggestions to the user, like
- validation errors, but that do not prevent submission. Note that the{' '}
+ validation errors, but that do not prevent submission. Note that the{" "}
<WarningEngine/> component must be at the bottom of the
form to guarantee that all the fields have registered.
@@ -46,7 +46,7 @@ const App = () => (
form: { mutators, reset },
submitting,
pristine,
- values
+ values,
}) => {
return (
@@ -87,10 +87,10 @@ const App = () => (
{JSON.stringify(values, 0, 2)}
- )
+ );
}}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/field-warnings/warning-engine.js b/examples/field-warnings/warning-engine.js
index 0aada156..91102f54 100644
--- a/examples/field-warnings/warning-engine.js
+++ b/examples/field-warnings/warning-engine.js
@@ -6,10 +6,10 @@ export const WarningEngine = ({ mutators: { setFieldData } }) => (
subscription={{ values: true }}
onChange={({ values }) => {
setFieldData("firstName", {
- warning: values.firstName ? undefined : "Recommended"
+ warning: values.firstName ? undefined : "Recommended",
});
setFieldData("lastName", {
- warning: values.lastName ? undefined : "Recommended"
+ warning: values.lastName ? undefined : "Recommended",
});
}}
/>
diff --git a/examples/fields-component/index.js b/examples/fields-component/index.js
index 7105ae8b..5f6a0229 100644
--- a/examples/fields-component/index.js
+++ b/examples/fields-component/index.js
@@ -1,29 +1,29 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const Fields = ({
names,
subscription,
fieldsState = {},
children,
- originalRender
+ originalRender,
}) => {
if (!names.length) {
- return (originalRender || children)(fieldsState)
+ return (originalRender || children)(fieldsState);
}
- const [name, ...rest] = names
+ const [name, ...rest] = names;
return (
- {fieldState => (
+ {(fieldState) => (
)}
- )
-}
+ );
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Fields Component
@@ -90,8 +90,8 @@ const App = () => (
Reset
-
- {fieldsState => (
+
+ {(fieldsState) => (
{JSON.stringify(fieldsState, undefined, 2)}
)}
@@ -99,6 +99,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/focus-first-error/Styles.js b/examples/focus-first-error/Styles.js
index d40d2e49..fbb99a53 100644
--- a/examples/focus-first-error/Styles.js
+++ b/examples/focus-first-error/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -113,7 +113,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -193,10 +193,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -207,4 +207,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/focus-first-error/index.js b/examples/focus-first-error/index.js
index 53806f0e..b9167c8e 100644
--- a/examples/focus-first-error/index.js
+++ b/examples/focus-first-error/index.js
@@ -1,31 +1,31 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import createDecorator from 'final-form-focus'
-import validate from './validate'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import createDecorator from "final-form-focus";
+import validate from "./validate";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const InputRow = ({ label, type, input, meta: { active, error, touched } }) => (
-
+
{label}
{error && touched && {error} }
-)
+);
-const focusOnError = createDecorator()
+const focusOnError = createDecorator();
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Focus On First Error
@@ -33,7 +33,7 @@ const App = () => (
Read Docs
- Demonstrates how to use the{' '}
+ Demonstrates how to use the{" "}
(
>
🏁
- {' '}
- Final Form Focus{' '}
+ {" "}
+ Final Form Focus{" "}
🧐
- {' '}
- library as a pluggable{' '}
+ {" "}
+ library as a pluggable{" "}
🏁
- {' '}
+ {" "}
Final Form decorator to provide "focus on first error" functionality.
Notice what when you click the Submit button, the focus is placed on the
first field with an error.
@@ -97,6 +97,6 @@ const App = () => (
)}
/>
-)
+);
-render(
, document.getElementById('root'))
+render(
, document.getElementById("root"));
diff --git a/examples/focus-first-error/validate.js b/examples/focus-first-error/validate.js
index de6a71f3..39f80f9c 100644
--- a/examples/focus-first-error/validate.js
+++ b/examples/focus-first-error/validate.js
@@ -1,18 +1,18 @@
-export default values => {
- const errors = {}
+export default (values) => {
+ const errors = {};
if (!values.firstName) {
- errors.firstName = 'Required'
+ errors.firstName = "Required";
}
if (!values.lastName) {
- errors.lastName = 'Required'
+ errors.lastName = "Required";
}
if (!values.street) {
- errors.street = 'Required'
+ errors.street = "Required";
}
if (!values.age) {
- errors.age = 'Required'
+ errors.age = "Required";
} else if (isNaN(values.age)) {
- errors.age = 'Must be a number'
+ errors.age = "Must be a number";
}
- return errors
-}
+ return errors;
+};
diff --git a/examples/format-on-blur/Styles.js b/examples/format-on-blur/Styles.js
index ea36d7b2..48f75bad 100644
--- a/examples/format-on-blur/Styles.js
+++ b/examples/format-on-blur/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -128,7 +128,7 @@ export default styled.div`
font-family: monospace, monospace;
}
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -208,10 +208,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -222,4 +222,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/format-on-blur/index.js b/examples/format-on-blur/index.js
index fd2dc938..c42ded40 100644
--- a/examples/format-on-blur/index.js
+++ b/examples/format-on-blur/index.js
@@ -1,27 +1,27 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import numeral from 'numeral'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import numeral from "numeral";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
-const formatPrice = value =>
+const formatPrice = (value) =>
value === undefined
- ? '' // make controlled
- : numeral(value).format('$0,0.00')
+ ? "" // make controlled
+ : numeral(value).format("$0,0.00");
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Format On Blur Example
@@ -31,8 +31,8 @@ const App = () => (
By default the format function given to Field is
called every time the component is rendered. But that can lead to a
- difficult UX for some types of values. That's why there is a{' '}
- formatOnBlur flag that will prevent the format{' '}
+ difficult UX for some types of values. That's why there is a{" "}
+ formatOnBlur flag that will prevent the format{" "}
function from being called until the field is blurred.
(
)}
/>
-)
+);
-render(
, document.getElementById('root'))
+render(
, document.getElementById("root"));
diff --git a/examples/format-string-by-pattern/Styles.js b/examples/format-string-by-pattern/Styles.js
index 06a77f51..0b4ad747 100644
--- a/examples/format-string-by-pattern/Styles.js
+++ b/examples/format-string-by-pattern/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -160,11 +160,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/format-string-by-pattern/index.js b/examples/format-string-by-pattern/index.js
index 686de466..ce8f125c 100644
--- a/examples/format-string-by-pattern/index.js
+++ b/examples/format-string-by-pattern/index.js
@@ -1,38 +1,38 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import formatString from 'format-string-by-pattern'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import formatString from "format-string-by-pattern";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const masks = [
- { name: 'phone-1', parse: '999-999-9999' },
- { name: 'phone-2', parse: '(999) 999-9999' },
- { name: 'phone-3', parse: '+49 (AAAA) BBBBBB' },
- { name: 'cep 🇧🇷', parse: '12345-678' },
- { name: 'cpf 🇧🇷', parse: 'XXX.XXX.XXX-XX' },
- { name: 'cnpj 🇧🇷', parse: 'XX.XXX.XXX/XXXX-XX' }
-]
+ { name: "phone-1", parse: "999-999-9999" },
+ { name: "phone-2", parse: "(999) 999-9999" },
+ { name: "phone-3", parse: "+49 (AAAA) BBBBBB" },
+ { name: "cep 🇧🇷", parse: "12345-678" },
+ { name: "cpf 🇧🇷", parse: "XXX.XXX.XXX-XX" },
+ { name: "cnpj 🇧🇷", parse: "XX.XXX.XXX/XXXX-XX" },
+];
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
(
- {masks.map(mask => (
+ {masks.map((mask) => (
{mask.name}
(
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/hybrid-sync-async-record-level-validation/Styles.js b/examples/hybrid-sync-async-record-level-validation/Styles.js
index a4edc9af..b8accf84 100644
--- a/examples/hybrid-sync-async-record-level-validation/Styles.js
+++ b/examples/hybrid-sync-async-record-level-validation/Styles.js
@@ -27,7 +27,9 @@ const btn = (light, dark) => css`
}
`;
-const btnDefault = css`${btn("#ffffff", "#d5d5d5")} color: #555;`;
+const btnDefault = css`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
const btnPrimary = btn("#4f93ce", "#285f8f");
diff --git a/examples/hybrid-sync-async-record-level-validation/index.js b/examples/hybrid-sync-async-record-level-validation/index.js
index f78a61cd..4ac45603 100644
--- a/examples/hybrid-sync-async-record-level-validation/index.js
+++ b/examples/hybrid-sync-async-record-level-validation/index.js
@@ -1,33 +1,33 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import Spinner from './Spinner'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import Spinner from "./Spinner";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
-const verifyUsername = async values => {
- await sleep(400)
+const verifyUsername = async (values) => {
+ await sleep(400);
if (
- ~['john', 'paul', 'george', 'ringo'].indexOf(
- values.username && values.username.toLowerCase()
+ ~["john", "paul", "george", "ringo"].indexOf(
+ values.username && values.username.toLowerCase(),
)
) {
- return { username: 'Username taken!' }
+ return { username: "Username taken!" };
}
-}
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Hybrid Synchronous/Asynchronous Record-Level Validation
@@ -37,20 +37,20 @@ const App = () => (
Usernames John, Paul, George or Ringo will fail async validation.
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.username) {
- errors.username = 'Required'
+ errors.username = "Required";
}
if (!values.password) {
- errors.password = 'Required'
+ errors.password = "Required";
}
if (!values.confirm) {
- errors.confirm = 'Required'
+ errors.confirm = "Required";
} else if (values.confirm !== values.password) {
- errors.confirm = 'Does not match'
+ errors.confirm = "Does not match";
}
- return Object.keys(errors).length ? errors : verifyUsername(values)
+ return Object.keys(errors).length ? errors : verifyUsername(values);
}}
render={({
handleSubmit,
@@ -58,7 +58,7 @@ const App = () => (
submitting,
pristine,
validating,
- values
+ values,
}) => (
{validating && }
@@ -110,6 +110,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/independent-error-component-render-props/Styles.js b/examples/independent-error-component-render-props/Styles.js
index 2a2bfadd..3f54fa69 100644
--- a/examples/independent-error-component-render-props/Styles.js
+++ b/examples/independent-error-component-render-props/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -144,10 +144,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -166,4 +166,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/independent-error-component-render-props/index.js b/examples/independent-error-component-render-props/index.js
index c4ef8892..d8abc769 100644
--- a/examples/independent-error-component-render-props/index.js
+++ b/examples/independent-error-component-render-props/index.js
@@ -1,14 +1,14 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const Error = ({ name }) => (
(
touched && error ? {error} : null
}
/>
-)
+);
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Reusable Independent Error Component
@@ -34,22 +34,22 @@ const App = () => (
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.firstName) {
- errors.firstName = 'Required'
+ errors.firstName = "Required";
}
if (!values.lastName) {
- errors.lastName = 'Required'
+ errors.lastName = "Required";
}
if (!values.age) {
- errors.age = 'Required'
+ errors.age = "Required";
} else if (isNaN(values.age)) {
- errors.age = 'Must be a number'
+ errors.age = "Must be a number";
} else if (values.age < 18) {
- errors.age = 'No kids allowed'
+ errors.age = "No kids allowed";
}
- return errors
+ return errors;
}}
render={({ handleSubmit, reset, submitting, pristine, values }) => (
@@ -94,6 +94,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/independent-error-component-with-hooks/Styles.js b/examples/independent-error-component-with-hooks/Styles.js
index 2a2bfadd..3f54fa69 100644
--- a/examples/independent-error-component-with-hooks/Styles.js
+++ b/examples/independent-error-component-with-hooks/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -144,10 +144,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -166,4 +166,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/independent-error-component-with-hooks/index.js b/examples/independent-error-component-with-hooks/index.js
index bd0cd175..9572509b 100644
--- a/examples/independent-error-component-with-hooks/index.js
+++ b/examples/independent-error-component-with-hooks/index.js
@@ -1,28 +1,28 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field, useField } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field, useField } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const Error = ({ name }) => {
const {
- meta: { touched, error }
- } = useField(name, { subscription: { touched: true, error: true } })
- return touched && error ? {error} : null
-}
+ meta: { touched, error },
+ } = useField(name, { subscription: { touched: true, error: true } });
+ return touched && error ? {error} : null;
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Reusable Independent Error Component
@@ -31,22 +31,22 @@ const App = () => (
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.firstName) {
- errors.firstName = 'Required'
+ errors.firstName = "Required";
}
if (!values.lastName) {
- errors.lastName = 'Required'
+ errors.lastName = "Required";
}
if (!values.age) {
- errors.age = 'Required'
+ errors.age = "Required";
} else if (isNaN(values.age)) {
- errors.age = 'Must be a number'
+ errors.age = "Must be a number";
} else if (values.age < 18) {
- errors.age = 'No kids allowed'
+ errors.age = "No kids allowed";
}
- return errors
+ return errors;
}}
render={({ handleSubmit, reset, submitting, pristine, values }) => (
@@ -91,6 +91,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/listening-for-external-changes/BooleanDecay.js b/examples/listening-for-external-changes/BooleanDecay.js
index da01d011..b2cbb420 100644
--- a/examples/listening-for-external-changes/BooleanDecay.js
+++ b/examples/listening-for-external-changes/BooleanDecay.js
@@ -1,5 +1,5 @@
-import React from 'react'
-import PropTypes from 'prop-types'
+import React from "react";
+import PropTypes from "prop-types";
/**
* 👋 Hey! Thanks for being curious about what this component does!
@@ -15,52 +15,52 @@ export default class BooleanDecay extends React.Component {
static propTypes = {
children: PropTypes.func.isRequired,
delay: PropTypes.number.isRequired,
- value: PropTypes.bool.isRequired
- }
+ value: PropTypes.bool.isRequired,
+ };
constructor(props) {
- super(props)
+ super(props);
this.state = {
- value: props.value
- }
+ value: props.value,
+ };
}
startTimer() {
- this.stopTimer()
+ this.stopTimer();
this.timeout = setTimeout(() => {
- this.setState({ value: false })
- }, this.props.delay)
+ this.setState({ value: false });
+ }, this.props.delay);
}
stopTimer() {
if (this.timeout) {
- clearTimeout(this.timeout)
+ clearTimeout(this.timeout);
}
}
componentDidMount() {
if (this.state.value) {
- this.startTimer()
+ this.startTimer();
}
}
componentWillUnmount() {
- this.stopTimer()
+ this.stopTimer();
}
componentWillReceiveProps(nextProps) {
- const { value } = nextProps
+ const { value } = nextProps;
if (value !== this.state.value) {
- this.setState({ value })
+ this.setState({ value });
if (value) {
- this.startTimer()
+ this.startTimer();
} else {
- this.stopTimer()
+ this.stopTimer();
}
}
}
render() {
- return this.props.children(this.state.value)
+ return this.props.children(this.state.value);
}
}
diff --git a/examples/listening-for-external-changes/ExternalModificationDetector.js b/examples/listening-for-external-changes/ExternalModificationDetector.js
index b07f9865..40169e8b 100644
--- a/examples/listening-for-external-changes/ExternalModificationDetector.js
+++ b/examples/listening-for-external-changes/ExternalModificationDetector.js
@@ -1,5 +1,5 @@
-import React from 'react'
-import { Field } from 'react-final-form'
+import React from "react";
+import { Field } from "react-final-form";
/**
* Listens for changes to a field's value, and, if the value changes
@@ -8,31 +8,34 @@ import { Field } from 'react-final-form'
*/
class ExternalModificationDetector extends React.Component {
constructor(props) {
- super(props)
+ super(props);
this.state = {
externallyModified: false,
- value: props.input.value
- }
+ value: props.input.value,
+ };
}
componentWillReceiveProps(nextProps) {
- const { input: { value }, meta: { active } } = nextProps
+ const {
+ input: { value },
+ meta: { active },
+ } = nextProps;
if (value !== this.state.value) {
this.setState({
value,
- externallyModified: !active
- })
+ externallyModified: !active,
+ });
} else if (this.state.externallyModified) {
this.setState({
- externallyModified: false
- })
+ externallyModified: false,
+ });
}
}
render() {
- const { children } = this.props
- const { externallyModified } = this.state
- return children(externallyModified)
+ const { children } = this.props;
+ const { externallyModified } = this.state;
+ return children(externallyModified);
}
}
@@ -41,10 +44,10 @@ export default ({ name, children }) => (
(
+ render={(props) => (
{children}
)}
/>
-)
+);
diff --git a/examples/listening-for-external-changes/Styles.js b/examples/listening-for-external-changes/Styles.js
index dd091033..698c8005 100644
--- a/examples/listening-for-external-changes/Styles.js
+++ b/examples/listening-for-external-changes/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -108,7 +108,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -188,10 +188,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -202,4 +202,4 @@ export default styled.div`
padding: 3px 5px;
}
}
-`
+`;
diff --git a/examples/listening-for-external-changes/index.js b/examples/listening-for-external-changes/index.js
index 72220fcf..ebee728a 100644
--- a/examples/listening-for-external-changes/index.js
+++ b/examples/listening-for-external-changes/index.js
@@ -1,33 +1,33 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import createDecorator from 'final-form-calculate'
-import ExternalModificationDetector from './ExternalModificationDetector'
-import BooleanDecay from './BooleanDecay'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import createDecorator from "final-form-calculate";
+import ExternalModificationDetector from "./ExternalModificationDetector";
+import BooleanDecay from "./BooleanDecay";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const calculator = createDecorator({
field: /day\[\d\]/, // when a field matching this pattern changes...
updates: {
// ...update the total to the result of this function
total: (ignoredValue, allValues) =>
- (allValues.day || []).reduce((sum, value) => sum + Number(value || 0), 0)
- }
-})
+ (allValues.day || []).reduce((sum, value) => sum + Number(value || 0), 0),
+ },
+});
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Listening for External Changes
@@ -98,17 +98,17 @@ const App = () => (
Total
- {externallyModified => (
+ {(externallyModified) => (
- {highlight => (
+ {(highlight) => (
)}
@@ -134,6 +134,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/loading-initializing-values/Styles.js b/examples/loading-initializing-values/Styles.js
index 42a76a3e..3f54fa69 100644
--- a/examples/loading-initializing-values/Styles.js
+++ b/examples/loading-initializing-values/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,12 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
-const btnDefault = css`${btn('#ffffff', '#d5d5d5')} color: #555;`
+const btnDefault = css`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -77,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -112,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -142,10 +144,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -164,4 +166,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/loading-initializing-values/index.js b/examples/loading-initializing-values/index.js
index 46101198..ac2369ae 100644
--- a/examples/loading-initializing-values/index.js
+++ b/examples/loading-initializing-values/index.js
@@ -1,29 +1,29 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
const load = async () => {
- await sleep(2000)
+ await sleep(2000);
return {
- username: 'erikras',
- firstName: 'Erik'
- }
-}
+ username: "erikras",
+ firstName: "Erik",
+ };
+};
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
class App extends React.Component {
- state = { data: {} }
+ state = { data: {} };
async componentDidMount() {
- this.setState({ loading: true })
- const data = await load()
- this.setState({ loading: false, data })
+ this.setState({ loading: true });
+ const data = await load();
+ this.setState({ loading: false, data });
}
render() {
@@ -32,7 +32,7 @@ class App extends React.Component {
🏁
- {' '}
+ {" "}
React Final Form - Loading and Initializing Values
@@ -75,12 +75,12 @@ class App extends React.Component {
{JSON.stringify(values, 0, 2)}
- )
+ );
}}
/>
- )
+ );
}
}
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/loading-saving-reinitializing/LoadSaveReinitializeForm.js b/examples/loading-saving-reinitializing/LoadSaveReinitializeForm.js
index f1678681..ad753d9d 100644
--- a/examples/loading-saving-reinitializing/LoadSaveReinitializeForm.js
+++ b/examples/loading-saving-reinitializing/LoadSaveReinitializeForm.js
@@ -1,6 +1,6 @@
-import React from 'react'
-import PropTypes from 'prop-types'
-import { Form } from 'react-final-form'
+import React from "react";
+import PropTypes from "prop-types";
+import { Form } from "react-final-form";
export default class LoadSaveReinitializeForm extends React.Component {
static propTypes = {
@@ -8,62 +8,56 @@ export default class LoadSaveReinitializeForm extends React.Component {
loading: PropTypes.node.isRequired,
postLoadFormat: PropTypes.func,
preSaveFormat: PropTypes.func,
- save: PropTypes.func.isRequired
- }
+ save: PropTypes.func.isRequired,
+ };
state = {
isLoading: false,
originalValues: undefined,
- initialValues: undefined
- }
+ initialValues: undefined,
+ };
load = async () => {
- const { load, postLoadFormat } = this.props
- this.setState({ isLoading: true })
- const originalValues = await load()
+ const { load, postLoadFormat } = this.props;
+ this.setState({ isLoading: true });
+ const originalValues = await load();
const initialValues = postLoadFormat
? postLoadFormat(originalValues)
- : originalValues
+ : originalValues;
this.setState({
isLoading: false,
originalValues,
- initialValues
- })
- }
+ initialValues,
+ });
+ };
- save = async values => {
- const { postLoadFormat, preSaveFormat, save } = this.props
+ save = async (values) => {
+ const { postLoadFormat, preSaveFormat, save } = this.props;
let valuesToSave = preSaveFormat
? preSaveFormat(values, this.state.originalValues)
- : values
- const result = await save(valuesToSave)
+ : values;
+ const result = await save(valuesToSave);
this.setState({
originalValues: valuesToSave,
initialValues: postLoadFormat
? postLoadFormat(valuesToSave)
- : valuesToSave
- })
- return result
- }
+ : valuesToSave,
+ });
+ return result;
+ };
componentDidMount() {
- this.load()
+ this.load();
}
render() {
- const {
- load,
- loading,
- postLoadFormat,
- preSaveFormat,
- save,
- ...rest
- } = this.props
- const { isLoading, initialValues } = this.state
+ const { load, loading, postLoadFormat, preSaveFormat, save, ...rest } =
+ this.props;
+ const { isLoading, initialValues } = this.state;
return isLoading || !initialValues ? (
loading
) : (
- )
+ );
}
}
diff --git a/examples/loading-saving-reinitializing/Styles.js b/examples/loading-saving-reinitializing/Styles.js
index aeb652d7..2c842de6 100644
--- a/examples/loading-saving-reinitializing/Styles.js
+++ b/examples/loading-saving-reinitializing/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -107,7 +107,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -173,11 +173,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/loading-saving-reinitializing/index.js b/examples/loading-saving-reinitializing/index.js
index 466bfd26..4f102d85 100644
--- a/examples/loading-saving-reinitializing/index.js
+++ b/examples/loading-saving-reinitializing/index.js
@@ -1,55 +1,55 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Field } from 'react-final-form'
-import LoadSaveReinitializeForm from './LoadSaveReinitializeForm'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Field } from "react-final-form";
+import LoadSaveReinitializeForm from "./LoadSaveReinitializeForm";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
let record = {
primaryKey: 42,
- phone: '2045551234',
- name: 'John Doe',
- email: 'john.doe@final-form.org',
+ phone: "2045551234",
+ name: "John Doe",
+ email: "john.doe@final-form.org",
otherExtraneousInfo: {
- creditScore: 800
- }
-}
+ creditScore: 800,
+ },
+};
const load = async () => {
- console.info('Loading...')
- await sleep(1500)
- console.info('Loaded...')
- return record
-}
+ console.info("Loading...");
+ await sleep(1500);
+ console.info("Loaded...");
+ return record;
+};
-const save = async values => {
- console.info('Saving', values)
- await sleep(1500)
- record = values
-}
+const save = async (values) => {
+ console.info("Saving", values);
+ await sleep(1500);
+ record = values;
+};
-const postLoadFormat = values => {
- const { name, email, phone } = values
- const [firstName, lastName] = name.split(/ /, 2)
+const postLoadFormat = (values) => {
+ const { name, email, phone } = values;
+ const [firstName, lastName] = name.split(/ /, 2);
return {
firstName,
lastName,
email,
phone: phone
? `${phone.slice(0, 3)}-${phone.slice(3, 6)}-${phone.slice(6, 10)}`
- : ''
- }
-}
+ : "",
+ };
+};
const preSaveFormat = (values, originalValues) => {
return {
...originalValues,
- name: `${values.firstName || ''} ${values.lastName || ''}`,
+ name: `${values.firstName || ""} ${values.lastName || ""}`,
email: values.email,
- phone: values.phone.replace(/-/g, '')
- }
-}
+ phone: values.phone.replace(/-/g, ""),
+ };
+};
const Error = ({ name }) => (
(
touched && error ? {error} : null
}
/>
-)
+);
-const loading = Loading...
+const loading = Loading...
;
-const validate = values => {
- const errors = {}
+const validate = (values) => {
+ const errors = {};
if (!values.firstName) {
- errors.firstName = 'Required'
+ errors.firstName = "Required";
}
if (!values.lastName) {
- errors.lastName = 'Required'
+ errors.lastName = "Required";
}
if (!values.email) {
- errors.email = 'Required'
+ errors.email = "Required";
}
if (!values.phone) {
- errors.phone = 'Required'
+ errors.phone = "Required";
}
- return errors
-}
+ return errors;
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Load, Save, and Reinitialize
@@ -92,10 +92,10 @@ const App = () => (
Read Docs
- LoadSaveReinitializeForm is a wrapper for a{' '}
+ LoadSaveReinitializeForm is a wrapper for a{" "}
🏁
- {' '}
+ {" "}
React Final Form component. On mount, it loads a record from the
database/API, and formats it to the shape of the form inputs. On submit,
it converts the data back to the format that the database/API wants
@@ -177,6 +177,6 @@ const App = () => (
)}
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/material-ui/index.js b/examples/material-ui/index.js
index de0b5c82..e7849135 100644
--- a/examples/material-ui/index.js
+++ b/examples/material-ui/index.js
@@ -21,13 +21,13 @@ import {
// Picker
import DateFnsUtils from '@date-io/date-fns';
-const onSubmit = async values => {
- const sleep = ms => new Promise(resolve => setTimeout(resolve, ms));
+const onSubmit = async (values) => {
+ const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
await sleep(300);
window.alert(JSON.stringify(values, 0, 2));
};
-const validate = values => {
+const validate = (values) => {
const errors = {};
if (!values.firstName) {
errors.firstName = 'Required';
diff --git a/examples/parse-format/Styles.js b/examples/parse-format/Styles.js
index 06a77f51..0b4ad747 100644
--- a/examples/parse-format/Styles.js
+++ b/examples/parse-format/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -160,11 +160,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/parse-format/index.js b/examples/parse-format/index.js
index f38c668c..9af32c72 100644
--- a/examples/parse-format/index.js
+++ b/examples/parse-format/index.js
@@ -1,33 +1,33 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
-const normalizePhone = value => {
- if (!value) return value
- const onlyNums = value.replace(/[^\d]/g, '')
- if (onlyNums.length <= 3) return onlyNums
+const normalizePhone = (value) => {
+ if (!value) return value;
+ const onlyNums = value.replace(/[^\d]/g, "");
+ if (onlyNums.length <= 3) return onlyNums;
if (onlyNums.length <= 7)
- return `(${onlyNums.slice(0, 3)}) ${onlyNums.slice(3, 7)}`
+ return `(${onlyNums.slice(0, 3)}) ${onlyNums.slice(3, 7)}`;
return `(${onlyNums.slice(0, 3)}) ${onlyNums.slice(3, 6)}-${onlyNums.slice(
6,
- 10
- )}`
-}
+ 10,
+ )}`;
+};
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Parse and Format
@@ -46,8 +46,8 @@ const App = () => (
component="input"
type="text"
placeholder="Username"
- parse={value => value && value.toUpperCase()}
- format={value => (value ? value.toLowerCase() : '')}
+ parse={(value) => value && value.toUpperCase()}
+ format={(value) => (value ? value.toLowerCase() : "")}
/>
@@ -77,6 +77,6 @@ const App = () => (
)}
/>
-)
+);
-render(
, document.getElementById('root'))
+render(
, document.getElementById("root"));
diff --git a/examples/prefixed-fields/index.js b/examples/prefixed-fields/index.js
index 5bff040a..20c584dc 100644
--- a/examples/prefixed-fields/index.js
+++ b/examples/prefixed-fields/index.js
@@ -1,27 +1,27 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
/************ IMPORTANT CODE STARTS HERE **************/
-const FieldPrefixContext = React.createContext()
+const FieldPrefixContext = React.createContext();
const FieldPrefix = ({ prefix, children }) => (
{children}
-)
+);
const PrefixedField = ({ name, ...props }) => (
- {prefix => }
+ {(prefix) => }
-)
+);
/************* IMPORTANT CODE ENDS HERE ***************/
const App = () => (
@@ -29,7 +29,7 @@ const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Prefixed Fields Example
@@ -39,14 +39,14 @@ const App = () => (
This example shows how to use React Context to create sections of your
form that have their fields' names prefixed, to structure the resulting
- form data. This provides similar functionality to Redux Form's{' '}
+ form data. This provides similar functionality to Redux Form's{" "}
FormSection
- {' '}
+ {" "}
component.
(
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/record-level-validation/Styles.js b/examples/record-level-validation/Styles.js
index 0a371734..a1c46db5 100644
--- a/examples/record-level-validation/Styles.js
+++ b/examples/record-level-validation/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,13 +25,13 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
+const btnPrimary = btn("#4f93ce", "#285f8f");
export default styled.div`
font-family: sans-serif;
@@ -85,7 +85,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -112,10 +112,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -126,4 +126,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/record-level-validation/index.js b/examples/record-level-validation/index.js
index 1b8c6d3f..96109338 100644
--- a/examples/record-level-validation/index.js
+++ b/examples/record-level-validation/index.js
@@ -1,14 +1,14 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => (
@@ -23,20 +23,20 @@ const App = () => (
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.username) {
- errors.username = 'Required'
+ errors.username = "Required";
}
if (!values.password) {
- errors.password = 'Required'
+ errors.password = "Required";
}
if (!values.confirm) {
- errors.confirm = 'Required'
+ errors.confirm = "Required";
} else if (values.confirm !== values.password) {
- errors.confirm = 'Must match'
+ errors.confirm = "Must match";
}
- return errors
+ return errors;
}}
render={({ handleSubmit, form, submitting, pristine, values }) => (
@@ -84,6 +84,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/redux/FormStateFromRedux.js b/examples/redux/FormStateFromRedux.js
index c0b1e767..8d672cb7 100644
--- a/examples/redux/FormStateFromRedux.js
+++ b/examples/redux/FormStateFromRedux.js
@@ -1,11 +1,11 @@
-import React from 'react'
-import { connect } from 'react-redux'
-import { getFormState } from './finalFormDuck'
+import React from "react";
+import { connect } from "react-redux";
+import { getFormState } from "./finalFormDuck";
const FormStateFromRedux = ({ state }) => (
{JSON.stringify(state, 0, 2)}
-)
+);
export default connect((state, ownProps) => ({
- state: getFormState(state, ownProps.form)
-}))(FormStateFromRedux)
+ state: getFormState(state, ownProps.form),
+}))(FormStateFromRedux);
diff --git a/examples/redux/FormStateToRedux.js b/examples/redux/FormStateToRedux.js
index c74a3bc3..28d234d5 100644
--- a/examples/redux/FormStateToRedux.js
+++ b/examples/redux/FormStateToRedux.js
@@ -1,10 +1,10 @@
-import React from 'react'
-import { connect } from 'react-redux'
-import { FormSpy } from 'react-final-form'
-import { updateFormState } from './finalFormDuck'
+import React from "react";
+import { connect } from "react-redux";
+import { FormSpy } from "react-final-form";
+import { updateFormState } from "./finalFormDuck";
const FormStateToRedux = ({ form, updateFormState }) => (
- updateFormState(form, state)} />
-)
+ updateFormState(form, state)} />
+);
-export default connect(undefined, { updateFormState })(FormStateToRedux)
+export default connect(undefined, { updateFormState })(FormStateToRedux);
diff --git a/examples/redux/Styles.js b/examples/redux/Styles.js
index aeb652d7..2c842de6 100644
--- a/examples/redux/Styles.js
+++ b/examples/redux/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -107,7 +107,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -173,11 +173,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/redux/finalFormDuck.js b/examples/redux/finalFormDuck.js
index eacedceb..723be3e2 100644
--- a/examples/redux/finalFormDuck.js
+++ b/examples/redux/finalFormDuck.js
@@ -1,7 +1,8 @@
// QUACK! This is a duck. https://github.com/erikras/ducks-modular-redux
// Actions
-const UPDATE_FORM_STATE = 'final-form-redux-example/finalForm/UPDATE_FORM_STATE'
+const UPDATE_FORM_STATE =
+ "final-form-redux-example/finalForm/UPDATE_FORM_STATE";
// Reducer
export default function reducer(state = {}, action = {}) {
@@ -9,10 +10,10 @@ export default function reducer(state = {}, action = {}) {
case UPDATE_FORM_STATE:
return {
...state,
- [action.form]: action.payload
- }
+ [action.form]: action.payload,
+ };
default:
- return state
+ return state;
}
}
@@ -20,9 +21,9 @@ export default function reducer(state = {}, action = {}) {
export const updateFormState = (form, state) => ({
type: UPDATE_FORM_STATE,
form,
- payload: state
-})
+ payload: state,
+});
// Selectors
export const getFormState = (state, form) =>
- (state && state.finalForm && state.finalForm[form]) || {}
+ (state && state.finalForm && state.finalForm[form]) || {};
diff --git a/examples/redux/index.js b/examples/redux/index.js
index 809599c1..c341d0e3 100644
--- a/examples/redux/index.js
+++ b/examples/redux/index.js
@@ -1,18 +1,18 @@
-import React from 'react'
-import { render } from 'react-dom'
-import { Provider } from 'react-redux'
-import store from './store'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import FormStateToRedux from './FormStateToRedux'
-import FormStateFromRedux from './FormStateFromRedux'
+import React from "react";
+import { render } from "react-dom";
+import { Provider } from "react-redux";
+import store from "./store";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import FormStateToRedux from "./FormStateToRedux";
+import FormStateFromRedux from "./FormStateFromRedux";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => (
@@ -20,7 +20,7 @@ const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
Redux Example
@@ -28,25 +28,25 @@ const App = () => (
Read Docs
- The only reason to keep your{' '}
+ The only reason to keep your{" "}
🏁
- {' '}
+ {" "}
Final Form form data in Redux is if you need to be able to read it from
- outside your form. This example demonstrates how to use a{' '}
+ outside your form. This example demonstrates how to use a{" "}
FormSpy to keep a copy of your form data in the Redux
store. Note that the canonical authoritative version of the data still
- lives in{' '}
+ lives in{" "}
🏁
- {' '}
+ {" "}
Final Form. If you need to mutate your data via dispatching
- Redux actions, you should probably use{' '}
+ Redux actions, you should probably use{" "}
Redux Form .
{({ handleSubmit, form, submitting, pristine }) => (
@@ -86,19 +86,19 @@ const App = () => (
❤️
- {' '}
+ {" "}
Red
💚
- {' '}
+ {" "}
Green
💙
- {' '}
+ {" "}
Blue
@@ -113,31 +113,31 @@ const App = () => (
🐷
- {' '}
+ {" "}
Ham
🍄
- {' '}
+ {" "}
Mushrooms
🧀
- {' '}
+ {" "}
Cheese
🐓
- {' '}
+ {" "}
Chicken
🍍
- {' '}
+ {" "}
Pinapple
@@ -151,7 +151,7 @@ const App = () => (
component="input"
type="radio"
value="larry"
- />{' '}
+ />{" "}
Larry
@@ -160,7 +160,7 @@ const App = () => (
component="input"
type="radio"
value="moe"
- />{' '}
+ />{" "}
Moe
@@ -169,7 +169,7 @@ const App = () => (
component="input"
type="radio"
value="curly"
- />{' '}
+ />{" "}
Curly
@@ -183,7 +183,7 @@ const App = () => (
component="input"
type="checkbox"
value="ketchup"
- />{' '}
+ />{" "}
Ketchup
@@ -192,7 +192,7 @@ const App = () => (
component="input"
type="checkbox"
value="mustard"
- />{' '}
+ />{" "}
Mustard
@@ -201,7 +201,7 @@ const App = () => (
component="input"
type="checkbox"
value="salsa"
- />{' '}
+ />{" "}
Salsa
@@ -210,8 +210,8 @@ const App = () => (
component="input"
type="checkbox"
value="guacamole"
- />{' '}
- Guacamole{' '}
+ />{" "}
+ Guacamole{" "}
🥑
@@ -241,6 +241,6 @@ const App = () => (
-)
+);
-render(
, document.getElementById('root'))
+render(
, document.getElementById("root"));
diff --git a/examples/redux/store.js b/examples/redux/store.js
index 833857cf..3823e64f 100644
--- a/examples/redux/store.js
+++ b/examples/redux/store.js
@@ -1,11 +1,13 @@
-import { createStore, combineReducers } from 'redux'
-import finalFormReducer from './finalFormDuck'
+import { createStore, combineReducers } from "redux";
+import finalFormReducer from "./finalFormDuck";
const reducer = combineReducers({
- finalForm: finalFormReducer
-})
-const store = (typeof window !== 'undefined' && window.devToolsExtension
- ? window.devToolsExtension()(createStore)
- : createStore)(reducer)
+ finalForm: finalFormReducer,
+});
+const store = (
+ typeof window !== "undefined" && window.devToolsExtension
+ ? window.devToolsExtension()(createStore)
+ : createStore
+)(reducer);
-export default store
+export default store;
diff --git a/examples/reusable-field-groups/Styles.js b/examples/reusable-field-groups/Styles.js
index 6821a1f1..c27cface 100644
--- a/examples/reusable-field-groups/Styles.js
+++ b/examples/reusable-field-groups/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -144,10 +144,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -166,4 +166,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/reusable-field-groups/index.js b/examples/reusable-field-groups/index.js
index fade75b8..48fc34d8 100644
--- a/examples/reusable-field-groups/index.js
+++ b/examples/reusable-field-groups/index.js
@@ -1,14 +1,14 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const Address = ({ name, label }) => (
@@ -37,14 +37,14 @@ const Address = ({ name, label }) => (
/>
-)
+);
const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Reusable Field Groups
@@ -78,6 +78,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/simple/Styles.js b/examples/simple/Styles.js
index e5d1235a..80ba86b6 100644
--- a/examples/simple/Styles.js
+++ b/examples/simple/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,13 +25,13 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
+const btnPrimary = btn("#4f93ce", "#285f8f");
export default styled.div`
font-family: sans-serif;
@@ -80,7 +80,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -101,10 +101,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -115,4 +115,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/simple/index.js b/examples/simple/index.js
index f6737b1d..b5803871 100644
--- a/examples/simple/index.js
+++ b/examples/simple/index.js
@@ -1,15 +1,15 @@
/* eslint-disable jsx-a11y/accessible-emoji */
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const App = () => (
@@ -23,7 +23,7 @@ const App = () => (
(
@@ -77,7 +77,7 @@ const App = () => (
component="input"
type="checkbox"
value="ketchup"
- />{' '}
+ />{" "}
Ketchup
@@ -86,7 +86,7 @@ const App = () => (
component="input"
type="checkbox"
value="mustard"
- />{' '}
+ />{" "}
Mustard
@@ -95,7 +95,7 @@ const App = () => (
component="input"
type="checkbox"
value="mayonnaise"
- />{' '}
+ />{" "}
Mayonnaise
@@ -104,7 +104,7 @@ const App = () => (
component="input"
type="checkbox"
value="guacamole"
- />{' '}
+ />{" "}
Guacamole 🥑
@@ -118,7 +118,7 @@ const App = () => (
component="input"
type="radio"
value="larry"
- />{' '}
+ />{" "}
Larry
@@ -127,7 +127,7 @@ const App = () => (
component="input"
type="radio"
value="moe"
- />{' '}
+ />{" "}
Moe
@@ -136,7 +136,7 @@ const App = () => (
component="input"
type="radio"
value="curly"
- />{' '}
+ />{" "}
Curly
@@ -162,6 +162,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/strongly-typed-values-typescript/components/CheckboxInput.tsx b/examples/strongly-typed-values-typescript/components/CheckboxInput.tsx
index ed28a97f..54d8e714 100644
--- a/examples/strongly-typed-values-typescript/components/CheckboxInput.tsx
+++ b/examples/strongly-typed-values-typescript/components/CheckboxInput.tsx
@@ -4,7 +4,7 @@ import { FieldRenderProps } from "react-final-form";
type Props = FieldRenderProps;
const CheckboxInput: React.FC = ({
- input: { value, ...input }
+ input: { value, ...input },
}: Props) => ;
export default CheckboxInput;
diff --git a/examples/strongly-typed-values-typescript/components/MultiCheckboxInput.tsx b/examples/strongly-typed-values-typescript/components/MultiCheckboxInput.tsx
index d1ee51ce..9683151d 100644
--- a/examples/strongly-typed-values-typescript/components/MultiCheckboxInput.tsx
+++ b/examples/strongly-typed-values-typescript/components/MultiCheckboxInput.tsx
@@ -4,7 +4,7 @@ import { FieldRenderProps } from "react-final-form";
type Props = FieldRenderProps;
const MultiCheckboxInput: React.FC = ({
- input: { value, ...input }
+ input: { value, ...input },
}: Props) => ;
export default MultiCheckboxInput;
diff --git a/examples/strongly-typed-values-typescript/index.tsx b/examples/strongly-typed-values-typescript/index.tsx
index 3b5f6e4f..2f8e4ea4 100644
--- a/examples/strongly-typed-values-typescript/index.tsx
+++ b/examples/strongly-typed-values-typescript/index.tsx
@@ -1,20 +1,20 @@
/* eslint-disable jsx-a11y/accessible-emoji */
-import React from 'react';
-import { render } from 'react-dom';
-import Styles from './Styles';
-import { Form, Field } from 'react-final-form';
-import CheckboxInput from './components/CheckboxInput';
-import RadioInput from './components/RadioInput';
-import TextInput from './components/TextInput';
-import NumberInput from './components/NumberInput';
-import TextAreaInput from './components/TextAreaInput';
-import SelectInput from './components/SelectInput';
-import MultiSelectInput from './components/MultiSelectInput';
-import MultiCheckboxInput from './components/MultiCheckboxInput';
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import CheckboxInput from "./components/CheckboxInput";
+import RadioInput from "./components/RadioInput";
+import TextInput from "./components/TextInput";
+import NumberInput from "./components/NumberInput";
+import TextAreaInput from "./components/TextAreaInput";
+import SelectInput from "./components/SelectInput";
+import MultiSelectInput from "./components/MultiSelectInput";
+import MultiCheckboxInput from "./components/MultiCheckboxInput";
-const sleep = (ms: number) => new Promise(resolve => setTimeout(resolve, ms));
+const sleep = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms));
-type Stooge = 'larry' | 'moe' | 'curly';
+type Stooge = "larry" | "moe" | "curly";
interface Values {
firstName?: string;
lastName?: string;
@@ -36,7 +36,7 @@ const App: React.FC = () => (
🏁
- {' '}
+ {" "}
React Final Form
Strongly Typed Values with TypeScript
@@ -50,7 +50,7 @@ const App: React.FC = () => (
(
@@ -114,7 +114,7 @@ const App: React.FC = () => (
component={MultiCheckboxInput}
type="checkbox"
value="ketchup"
- />{' '}
+ />{" "}
Ketchup
@@ -123,7 +123,7 @@ const App: React.FC = () => (
component="input"
type="checkbox"
value="mustard"
- />{' '}
+ />{" "}
Mustard
@@ -132,7 +132,7 @@ const App: React.FC = () => (
component="input"
type="checkbox"
value="mayonnaise"
- />{' '}
+ />{" "}
Mayonnaise
@@ -141,7 +141,7 @@ const App: React.FC = () => (
component="input"
type="checkbox"
value="guacamole"
- />{' '}
+ />{" "}
Guacamole 🥑
@@ -155,7 +155,7 @@ const App: React.FC = () => (
component={RadioInput}
type="radio"
value="larry"
- />{' '}
+ />{" "}
Larry
@@ -164,7 +164,7 @@ const App: React.FC = () => (
component={RadioInput}
type="radio"
value="moe"
- />{' '}
+ />{" "}
Moe
@@ -173,7 +173,7 @@ const App: React.FC = () => (
component={RadioInput}
type="radio"
value="curly"
- />{' '}
+ />{" "}
Curly
@@ -201,5 +201,5 @@ const App: React.FC = () => (
);
-const rootElement = document.getElementById('root');
+const rootElement = document.getElementById("root");
render( , rootElement);
diff --git a/examples/styling-with-smooth-ui/index.js b/examples/styling-with-smooth-ui/index.js
index 148d84ae..992f1c57 100644
--- a/examples/styling-with-smooth-ui/index.js
+++ b/examples/styling-with-smooth-ui/index.js
@@ -1,6 +1,6 @@
-import React from 'react'
-import { render } from 'react-dom'
-import { Form, Field } from 'react-final-form'
+import React from "react";
+import { render } from "react-dom";
+import { Form, Field } from "react-final-form";
import {
Box,
Button,
@@ -17,29 +17,28 @@ import {
Row,
Select,
Textarea,
- Typography
-} from 'smooth-ui'
+ Typography,
+} from "smooth-ui";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
// ****************************************
//⬇️ THIS IS WHERE ALL THE MAGIC HAPPENS ⬇️
// ****************************************
-const adapt /* ⬅️ this is a HOC */ = Component => ({
- input,
- meta: { valid },
- ...rest
-}) =>
-const AdaptedInput = adapt(Input)
-const AdaptedCheckbox = adapt(Checkbox)
-const AdaptedRadio = adapt(Radio)
-const AdaptedSelect = adapt(Select)
-const AdaptedTextarea = adapt(Textarea)
+const adapt /* ⬅️ this is a HOC */ =
+ (Component) =>
+ ({ input, meta: { valid }, ...rest }) =>
+ ;
+const AdaptedInput = adapt(Input);
+const AdaptedCheckbox = adapt(Checkbox);
+const AdaptedRadio = adapt(Radio);
+const AdaptedSelect = adapt(Select);
+const AdaptedTextarea = adapt(Textarea);
const Error = ({ name }) => (
@@ -49,12 +48,12 @@ const Error = ({ name }) => (
) : null
}
-)
+);
// ****************************************
//⬆️ THIS IS WHERE ALL THE MAGIC HAPPENS ⬆️
// ****************************************
-const required = value => (value ? undefined : 'Required')
+const required = (value) => (value ? undefined : "Required");
const App = () => (
@@ -62,17 +61,17 @@ const App = () => (
🏁
- {' '}
+ {" "}
React Final Form
🍭
- {' '}
+ {" "}
Smooth-UI Example
- This example demonstrates how to use{' '}
+ This example demonstrates how to use{" "}
(
>
🍭
- {' '}
+ {" "}
Smooth-UI
- {' '}
+ {" "}
to make your forms look fabulous. All you really need is the
- higher-order component that adapts the{' '}
+ higher-order component that adapts the{" "}
🍭
- {' '}
- Smooth-UI form components to be compatible with{' '}
+ {" "}
+ Smooth-UI form components to be compatible with{" "}
🏁
- {' '}
+ {" "}
React Final Form.
(
@@ -138,9 +137,9 @@ const App = () => (
validate={required}
options={[
{},
- { value: '#ff0000', label: '❤️ Red' },
- { value: '#00ff00', label: '💚 Green' },
- { value: '#0000ff', label: '💙 Blue' }
+ { value: "#ff0000", label: "❤️ Red" },
+ { value: "#00ff00", label: "💚 Green" },
+ { value: "#0000ff", label: "💙 Blue" },
]}
control
/>
@@ -155,12 +154,12 @@ const App = () => (
multiple
arrow={false}
options={[
- { value: 'chicken', label: '🐓 Chicken' },
- { value: 'ham', label: '🐷 Ham' },
- { value: 'mushrooms', label: '🍄 Mushrooms' },
- { value: 'cheese', label: '🧀 Cheese' },
- { value: 'tuna', label: '🐟 Tuna' },
- { value: 'pineapple', label: '🍍 Pineapple' }
+ { value: "chicken", label: "🐓 Chicken" },
+ { value: "ham", label: "🐷 Ham" },
+ { value: "mushrooms", label: "🍄 Mushrooms" },
+ { value: "cheese", label: "🧀 Cheese" },
+ { value: "tuna", label: "🐟 Tuna" },
+ { value: "pineapple", label: "🍍 Pineapple" },
]}
control
/>
@@ -197,7 +196,7 @@ const App = () => (
value="guacamole"
/>
- Guacamole{' '}
+ Guacamole{" "}
🥑
@@ -270,10 +269,10 @@ const App = () => (
{JSON.stringify(values, 0, 2)}
@@ -284,6 +283,6 @@ const App = () => (
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/submission-errors/Styles.js b/examples/submission-errors/Styles.js
index 68d1047e..060cf8ab 100644
--- a/examples/submission-errors/Styles.js
+++ b/examples/submission-errors/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,13 +25,13 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
+const btnPrimary = btn("#4f93ce", "#285f8f");
export default styled.div`
font-family: sans-serif;
@@ -87,7 +87,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -114,10 +114,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -135,4 +135,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/submission-errors/index.js b/examples/submission-errors/index.js
index c91aef72..c2f2037a 100644
--- a/examples/submission-errors/index.js
+++ b/examples/submission-errors/index.js
@@ -1,21 +1,21 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import { FORM_ERROR } from 'final-form'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import { FORM_ERROR } from "final-form";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- if (values.username !== 'erikras') {
- return { username: 'Unknown username' }
+const onSubmit = async (values) => {
+ await sleep(300);
+ if (values.username !== "erikras") {
+ return { username: "Unknown username" };
}
- if (values.password !== 'finalformrocks') {
- return { [FORM_ERROR]: 'Login Failed' }
+ if (values.password !== "finalformrocks") {
+ return { [FORM_ERROR]: "Login Failed" };
}
- window.alert('LOGIN SUCCESS!')
-}
+ window.alert("LOGIN SUCCESS!");
+};
const App = () => (
@@ -29,20 +29,20 @@ const App = () => (
Read Docs
- Only successful credentials are erikras and{' '}
+ Only successful credentials are erikras and{" "}
finalformrocks.
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.username) {
- errors.username = 'Required'
+ errors.username = "Required";
}
if (!values.password) {
- errors.password = 'Required'
+ errors.password = "Required";
}
- return errors
+ return errors;
}}
render={({
submitError,
@@ -50,7 +50,7 @@ const App = () => (
form,
submitting,
pristine,
- values
+ values,
}) => (
@@ -91,6 +91,6 @@ const App = () => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/subscriptions/RenderCount.js b/examples/subscriptions/RenderCount.js
index 051e06c3..a783309d 100644
--- a/examples/subscriptions/RenderCount.js
+++ b/examples/subscriptions/RenderCount.js
@@ -1,13 +1,13 @@
-import React from 'react'
-import styled from 'styled-components'
+import React from "react";
+import styled from "styled-components";
export default function RenderCount() {
- const renders = React.useRef(0)
+ const renders = React.useRef(0);
- return {++renders.current}
+ return {++renders.current} ;
}
-const size = 30
+const size = 30;
const Circle = styled.i`
position: absolute;
top: 0;
@@ -20,4 +20,4 @@ const Circle = styled.i`
border-radius: ${size / 2}px;
border: 1px solid #ddd;
background: #eee;
-`
+`;
diff --git a/examples/subscriptions/Styles.js b/examples/subscriptions/Styles.js
index a14e5eaa..16dcd028 100644
--- a/examples/subscriptions/Styles.js
+++ b/examples/subscriptions/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,13 +25,13 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
+const btnPrimary = btn("#4f93ce", "#285f8f");
export default styled.div`
font-family: sans-serif;
@@ -82,7 +82,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > span {
@@ -109,10 +109,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -124,4 +124,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/subscriptions/index.js b/examples/subscriptions/index.js
index 9657afaf..8468bc6c 100644
--- a/examples/subscriptions/index.js
+++ b/examples/subscriptions/index.js
@@ -1,16 +1,16 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field, FormSpy } from 'react-final-form'
-import RenderCount from './RenderCount'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field, FormSpy } from "react-final-form";
+import RenderCount from "./RenderCount";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
-const required = value => (value ? undefined : 'Required')
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
+const required = (value) => (value ? undefined : "Required");
const App = () => (
@@ -36,7 +36,7 @@ const App = () => (
-)
+);
const MyForm = ({ subscription }) => (
(
{values ? (
-
- {JSON.stringify(values, 0, 2)}
+
+ {JSON.stringify(values, 0, 2)}
+
) : (
{({ values }) => (
-
+
{JSON.stringify(values, 0, 2)}
)}
@@ -94,6 +95,6 @@ const MyForm = ({ subscription }) => (
)}
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/third-party-components/Styles.js b/examples/third-party-components/Styles.js
index e41efd20..886ce8ad 100644
--- a/examples/third-party-components/Styles.js
+++ b/examples/third-party-components/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,11 +25,13 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
-const btnDefault = css`${btn('#ffffff', '#d5d5d5')} color: #555;`
+const btnDefault = css`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
+const btnPrimary = btn("#4f93ce", "#285f8f");
export default styled.div`
font-family: sans-serif;
@@ -72,10 +74,10 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
@@ -96,4 +98,4 @@ export default styled.div`
padding: 20px;
}
}
-`
+`;
diff --git a/examples/third-party-components/index.js b/examples/third-party-components/index.js
index 94539ac3..4b9c5591 100644
--- a/examples/third-party-components/index.js
+++ b/examples/third-party-components/index.js
@@ -1,22 +1,22 @@
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Form, Field } from 'react-final-form'
-import MuiThemeProvider from 'material-ui/styles/MuiThemeProvider'
-import getMuiTheme from 'material-ui/styles/getMuiTheme'
-import TextField from 'material-ui/TextField'
-import Toggle from 'material-ui/Toggle'
-import Select from 'react-select'
-import states from './states'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Form, Field } from "react-final-form";
+import MuiThemeProvider from "material-ui/styles/MuiThemeProvider";
+import getMuiTheme from "material-ui/styles/getMuiTheme";
+import TextField from "material-ui/TextField";
+import Toggle from "material-ui/Toggle";
+import Select from "react-select";
+import states from "./states";
const TextFieldAdapter = ({ input, meta, ...rest }) => (
input.onChange(value)}
- errorText={meta.touched ? meta.error : ''}
+ errorText={meta.touched ? meta.error : ""}
/>
-)
+);
const ToggleAdapter = ({ input: { onChange, value }, label, ...rest }) => (
(
onToggle={(event, isInputChecked) => onChange(isInputChecked)}
{...rest}
/>
-)
+);
const ReactSelectAdapter = ({ input, ...rest }) => (
-)
+);
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
-const required = value => (value ? undefined : 'Required')
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
+const required = (value) => (value ? undefined : "Required");
const App = () => (
@@ -45,7 +45,7 @@ const App = () => (
🏁
- {' '}
+ {" "}
React Final Form Example
Third Party Components
@@ -53,8 +53,8 @@ const App = () => (
Read Docs
(
/>
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/examples/third-party-components/states.js b/examples/third-party-components/states.js
index d8e84794..e2c7cbbc 100644
--- a/examples/third-party-components/states.js
+++ b/examples/third-party-components/states.js
@@ -1,65 +1,65 @@
const states = {
- AL: 'Alabama',
- AK: 'Alaska',
- AS: 'American Samoa',
- AZ: 'Arizona',
- AR: 'Arkansas',
- CA: 'California',
- CO: 'Colorado',
- CT: 'Connecticut',
- DE: 'Delaware',
- DC: 'District Of Columbia',
- FM: 'Federated States Of Micronesia',
- FL: 'Florida',
- GA: 'Georgia',
- GU: 'Guam',
- HI: 'Hawaii',
- ID: 'Idaho',
- IL: 'Illinois',
- IN: 'Indiana',
- IA: 'Iowa',
- KS: 'Kansas',
- KY: 'Kentucky',
- LA: 'Louisiana',
- ME: 'Maine',
- MH: 'Marshall Islands',
- MD: 'Maryland',
- MA: 'Massachusetts',
- MI: 'Michigan',
- MN: 'Minnesota',
- MS: 'Mississippi',
- MO: 'Missouri',
- MT: 'Montana',
- NE: 'Nebraska',
- NV: 'Nevada',
- NH: 'New Hampshire',
- NJ: 'New Jersey',
- NM: 'New Mexico',
- NY: 'New York',
- NC: 'North Carolina',
- ND: 'North Dakota',
- MP: 'Northern Mariana Islands',
- OH: 'Ohio',
- OK: 'Oklahoma',
- OR: 'Oregon',
- PW: 'Palau',
- PA: 'Pennsylvania',
- PR: 'Puerto Rico',
- RI: 'Rhode Island',
- SC: 'South Carolina',
- SD: 'South Dakota',
- TN: 'Tennessee',
- TX: 'Texas',
- UT: 'Utah',
- VT: 'Vermont',
- VI: 'Virgin Islands',
- VA: 'Virginia',
- WA: 'Washington',
- WV: 'West Virginia',
- WI: 'Wisconsin',
- WY: 'Wyoming'
-}
-export default Object.keys(states).map(value => ({
+ AL: "Alabama",
+ AK: "Alaska",
+ AS: "American Samoa",
+ AZ: "Arizona",
+ AR: "Arkansas",
+ CA: "California",
+ CO: "Colorado",
+ CT: "Connecticut",
+ DE: "Delaware",
+ DC: "District Of Columbia",
+ FM: "Federated States Of Micronesia",
+ FL: "Florida",
+ GA: "Georgia",
+ GU: "Guam",
+ HI: "Hawaii",
+ ID: "Idaho",
+ IL: "Illinois",
+ IN: "Indiana",
+ IA: "Iowa",
+ KS: "Kansas",
+ KY: "Kentucky",
+ LA: "Louisiana",
+ ME: "Maine",
+ MH: "Marshall Islands",
+ MD: "Maryland",
+ MA: "Massachusetts",
+ MI: "Michigan",
+ MN: "Minnesota",
+ MS: "Mississippi",
+ MO: "Missouri",
+ MT: "Montana",
+ NE: "Nebraska",
+ NV: "Nevada",
+ NH: "New Hampshire",
+ NJ: "New Jersey",
+ NM: "New Mexico",
+ NY: "New York",
+ NC: "North Carolina",
+ ND: "North Dakota",
+ MP: "Northern Mariana Islands",
+ OH: "Ohio",
+ OK: "Oklahoma",
+ OR: "Oregon",
+ PW: "Palau",
+ PA: "Pennsylvania",
+ PR: "Puerto Rico",
+ RI: "Rhode Island",
+ SC: "South Carolina",
+ SD: "South Dakota",
+ TN: "Tennessee",
+ TX: "Texas",
+ UT: "Utah",
+ VT: "Vermont",
+ VI: "Virgin Islands",
+ VA: "Virginia",
+ WA: "Washington",
+ WV: "West Virginia",
+ WI: "Wisconsin",
+ WY: "Wyoming",
+};
+export default Object.keys(states).map((value) => ({
value,
- label: states[value]
-}))
+ label: states[value],
+}));
diff --git a/examples/wizard/Styles.js b/examples/wizard/Styles.js
index 06a77f51..0b4ad747 100644
--- a/examples/wizard/Styles.js
+++ b/examples/wizard/Styles.js
@@ -1,4 +1,4 @@
-import styled, { css } from 'styled-components'
+import styled, { css } from "styled-components";
const btn = (light, dark) => css`
white-space: nowrap;
@@ -25,14 +25,14 @@ const btn = (light, dark) => css`
opacity: 0.6;
cursor: not-allowed;
}
-`
+`;
const btnDefault = css`
- ${btn('#ffffff', '#d5d5d5')} color: #555;
-`
+ ${btn("#ffffff", "#d5d5d5")} color: #555;
+`;
-const btnPrimary = btn('#4f93ce', '#285f8f')
-const btnDanger = btn('#e27c79', '#c9302c')
+const btnPrimary = btn("#4f93ce", "#285f8f");
+const btnDanger = btn("#e27c79", "#c9302c");
export default styled.div`
font-family: sans-serif;
@@ -79,7 +79,7 @@ export default styled.div`
text-align: center;
display: block;
position: absolute;
- background: url('https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif')
+ background: url("https://media.giphy.com/media/130AxGoOaR6t0I/giphy.gif")
center center;
background-size: fill;
font-size: 2em;
@@ -114,7 +114,7 @@ export default styled.div`
border: 1px solid #ccc;
border-radius: 3px;
}
- & > input[type='checkbox'] {
+ & > input[type="checkbox"] {
margin-top: 7px;
}
& > div {
@@ -160,11 +160,11 @@ export default styled.div`
}
button {
margin: 0 10px;
- &[type='submit'] {
+ &[type="submit"] {
${btnPrimary};
}
- &[type='button'] {
+ &[type="button"] {
${btnDefault};
}
}
-`
+`;
diff --git a/examples/wizard/Wizard.js b/examples/wizard/Wizard.js
index 10e085b3..72e3990f 100644
--- a/examples/wizard/Wizard.js
+++ b/examples/wizard/Wizard.js
@@ -1,30 +1,30 @@
-import React from 'react'
-import PropTypes from 'prop-types'
-import { Form } from 'react-final-form'
+import React from "react";
+import PropTypes from "prop-types";
+import { Form } from "react-final-form";
export default class Wizard extends React.Component {
static propTypes = {
- onSubmit: PropTypes.func.isRequired
- }
- static Page = ({ children }) => children
+ onSubmit: PropTypes.func.isRequired,
+ };
+ static Page = ({ children }) => children;
constructor(props) {
- super(props)
+ super(props);
this.state = {
page: 0,
- values: props.initialValues || {}
- }
+ values: props.initialValues || {},
+ };
}
- next = values =>
- this.setState(state => ({
+ next = (values) =>
+ this.setState((state) => ({
page: Math.min(state.page + 1, this.props.children.length - 1),
- values
- }))
+ values,
+ }));
previous = () =>
- this.setState(state => ({
- page: Math.max(state.page - 1, 0)
- }))
+ this.setState((state) => ({
+ page: Math.max(state.page - 1, 0),
+ }));
/**
* NOTE: Both validate and handleSubmit switching are implemented
@@ -32,29 +32,29 @@ export default class Wizard extends React.Component {
* functions once the form has been defined.
*/
- validate = values => {
+ validate = (values) => {
const activePage = React.Children.toArray(this.props.children)[
this.state.page
- ]
- return activePage.props.validate ? activePage.props.validate(values) : {}
- }
+ ];
+ return activePage.props.validate ? activePage.props.validate(values) : {};
+ };
- handleSubmit = values => {
- const { children, onSubmit } = this.props
- const { page } = this.state
- const isLastPage = page === React.Children.count(children) - 1
+ handleSubmit = (values) => {
+ const { children, onSubmit } = this.props;
+ const { page } = this.state;
+ const isLastPage = page === React.Children.count(children) - 1;
if (isLastPage) {
- return onSubmit(values)
+ return onSubmit(values);
} else {
- this.next(values)
+ this.next(values);
}
- }
+ };
render() {
- const { children } = this.props
- const { page, values } = this.state
- const activePage = React.Children.toArray(children)[page]
- const isLastPage = page === React.Children.count(children) - 1
+ const { children } = this.props;
+ const { page, values } = this.state;
+ const activePage = React.Children.toArray(children)[page];
+ const isLastPage = page === React.Children.count(children) - 1;
return (
)}
- )
+ );
}
}
diff --git a/examples/wizard/index.js b/examples/wizard/index.js
index efff5fdd..6312fe75 100644
--- a/examples/wizard/index.js
+++ b/examples/wizard/index.js
@@ -1,16 +1,16 @@
/* eslint-disable jsx-a11y/accessible-emoji */
-import React from 'react'
-import { render } from 'react-dom'
-import Styles from './Styles'
-import { Field } from 'react-final-form'
-import Wizard from './Wizard'
+import React from "react";
+import { render } from "react-dom";
+import Styles from "./Styles";
+import { Field } from "react-final-form";
+import Wizard from "./Wizard";
-const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
+const sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
-const onSubmit = async values => {
- await sleep(300)
- window.alert(JSON.stringify(values, 0, 2))
-}
+const onSubmit = async (values) => {
+ await sleep(300);
+ window.alert(JSON.stringify(values, 0, 2));
+};
const Error = ({ name }) => (
(
touched && error ? {error} : null
}
/>
-)
+);
-const required = value => (value ? undefined : 'Required')
+const required = (value) => (value ? undefined : "Required");
const App = () => (
@@ -36,11 +36,11 @@ const App = () => (
Read Docs
- Notice the mixture of field-level and record-level (or page-level {' '}
+ Notice the mixture of field-level and record-level (or page-level {" "}
in this case) validation.
@@ -68,15 +68,15 @@ const App = () => (
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.email) {
- errors.email = 'Required'
+ errors.email = "Required";
}
if (!values.favoriteColor) {
- errors.favoriteColor = 'Required'
+ errors.favoriteColor = "Required";
}
- return errors
+ return errors;
}}
>
@@ -101,14 +101,14 @@ const App = () => (
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.toppings) {
- errors.toppings = 'Required'
+ errors.toppings = "Required";
} else if (values.toppings.length < 2) {
- errors.toppings = 'Choose more'
+ errors.toppings = "Choose more";
}
- return errors
+ return errors;
}}
>
@@ -128,12 +128,12 @@ const App = () => (
{
- const errors = {}
+ validate={(values) => {
+ const errors = {};
if (!values.notes) {
- errors.notes = 'Required'
+ errors.notes = "Required";
}
- return errors
+ return errors;
}}
>
@@ -145,11 +145,11 @@ const App = () => (
component="input"
type="radio"
value="larry"
- />{' '}
+ />{" "}
Larry
- {' '}
+ {" "}
Moe
@@ -158,7 +158,7 @@ const App = () => (
component="input"
type="radio"
value="curly"
- />{' '}
+ />{" "}
Curly
@@ -171,6 +171,6 @@ const App = () => (
-)
+);
-render( , document.getElementById('root'))
+render( , document.getElementById("root"));
diff --git a/package-scripts.js b/package-scripts.js
index a8fe31af..a61fc887 100644
--- a/package-scripts.js
+++ b/package-scripts.js
@@ -1,99 +1,103 @@
-const npsUtils = require('nps-utils')
+const npsUtils = require("nps-utils");
-const series = npsUtils.series
-const concurrent = npsUtils.concurrent
-const rimraf = npsUtils.rimraf
-const crossEnv = npsUtils.crossEnv
+const series = npsUtils.series;
+const concurrent = npsUtils.concurrent;
+const rimraf = npsUtils.rimraf;
+const crossEnv = npsUtils.crossEnv;
module.exports = {
scripts: {
test: {
- default: crossEnv('NODE_ENV=test jest --coverage'),
- update: crossEnv('NODE_ENV=test jest --coverage --updateSnapshot'),
- watch: crossEnv('NODE_ENV=test jest --watch'),
+ default: crossEnv("NODE_ENV=test jest --coverage"),
+ update: crossEnv("NODE_ENV=test jest --coverage --updateSnapshot"),
+ watch: crossEnv("NODE_ENV=test jest --watch"),
codeCov: crossEnv(
- 'cat ./coverage/lcov.info | ./node_modules/codecov.io/bin/codecov.io.js'
+ "cat ./coverage/lcov.info | ./node_modules/codecov.io/bin/codecov.io.js",
),
size: {
- description: 'check the size of the bundle',
- script: 'bundlesize'
- }
+ description: "check the size of the bundle",
+ script: "bundlesize",
+ },
},
build: {
- description: 'delete the dist directory and run all builds',
+ description: "delete the dist directory and run all builds",
default: series(
- rimraf('dist'),
+ rimraf("dist"),
concurrent.nps(
- 'build.es',
- 'build.cjs',
- 'build.umd.main',
- 'build.umd.min',
- 'copyTypes'
- )
+ "build.es",
+ "build.cjs",
+ "build.umd.main",
+ "build.umd.min",
+ "copyTypes",
+ ),
),
es: {
- description: 'run the build with rollup (uses rollup.config.js)',
- script: 'rollup --config --environment FORMAT:es'
+ description: "run the build with rollup (uses rollup.config.js)",
+ script: "rollup --config --environment FORMAT:es",
},
cjs: {
- description: 'run rollup build with CommonJS format',
- script: 'rollup --config --environment FORMAT:cjs'
+ description: "run rollup build with CommonJS format",
+ script: "rollup --config --environment FORMAT:cjs",
},
umd: {
min: {
- description: 'run the rollup build with sourcemaps',
- script: 'rollup --config --sourcemap --environment MINIFY,FORMAT:umd'
+ description: "run the rollup build with sourcemaps",
+ script: "rollup --config --sourcemap --environment MINIFY,FORMAT:umd",
},
main: {
- description: 'builds the cjs and umd files',
- script: 'rollup --config --sourcemap --environment FORMAT:umd'
- }
+ description: "builds the cjs and umd files",
+ script: "rollup --config --sourcemap --environment FORMAT:umd",
+ },
},
- andTest: series.nps('build', 'test.size')
+ andTest: series.nps("build", "test.size"),
},
copyTypes: series(
- npsUtils.copy('src/*.js.flow dist'),
+ npsUtils.copy("src/*.js.flow dist"),
npsUtils.copy(
- 'dist/index.js.flow dist --rename="react-final-form.cjs.js.flow"'
+ 'dist/index.js.flow dist --rename="react-final-form.cjs.js.flow"',
),
npsUtils.copy(
- 'dist/index.js.flow dist --rename="react-final-form.es.js.flow"'
- )
+ 'dist/index.js.flow dist --rename="react-final-form.es.js.flow"',
+ ),
),
docs: {
- description: 'Generates table of contents in README',
- script: 'doctoc README.md'
+ description: "Generates table of contents in README",
+ script: "doctoc README.md",
+ },
+ prettier: {
+ description: "Runs prettier on everything",
+ script: 'prettier --write "**/*.([jt]s*)"',
},
lint: {
- description: 'lint the entire project',
- script: 'eslint .'
+ description: "lint the entire project",
+ script: "eslint .",
},
flow: {
- description: 'flow check the entire project',
- script: 'flow check'
+ description: "flow check the entire project",
+ script: "flow check",
},
typescript: {
default: {
- description: 'typescript',
+ description: "typescript",
script:
- 'dtslint --localTs ./node_modules/typescript/lib --expectOnly ./typescript'
- }
+ "dtslint --localTs ./node_modules/typescript/lib --expectOnly ./typescript",
+ },
},
validate: {
description:
- 'This runs several scripts to make sure things look good before committing or on clean install',
+ "This runs several scripts to make sure things look good before committing or on clean install",
default: concurrent.nps(
- 'lint',
+ "lint",
// Flow has been causing headaches in this lib for years.
// Disabling now to ship some features. -@erikras, 2020-04-19
// 'flow',
- 'typescript',
- 'build.andTest',
- 'test'
- )
- }
+ "typescript",
+ "build.andTest",
+ "test",
+ ),
+ },
},
options: {
- silent: false
- }
-}
+ silent: false,
+ },
+};
diff --git a/rollup.config.js b/rollup.config.js
index 10cce73a..6183f630 100644
--- a/rollup.config.js
+++ b/rollup.config.js
@@ -1,83 +1,83 @@
-import resolve from 'rollup-plugin-node-resolve'
-import babel from 'rollup-plugin-babel'
-import commonjs from 'rollup-plugin-commonjs'
-import json from 'rollup-plugin-json'
-import { uglify } from 'rollup-plugin-uglify'
-import replace from 'rollup-plugin-replace'
-import pkg from './package.json'
+import resolve from "rollup-plugin-node-resolve";
+import babel from "rollup-plugin-babel";
+import commonjs from "rollup-plugin-commonjs";
+import json from "rollup-plugin-json";
+import { uglify } from "rollup-plugin-uglify";
+import replace from "rollup-plugin-replace";
+import pkg from "./package.json";
-const makeExternalPredicate = externalArr => {
+const makeExternalPredicate = (externalArr) => {
if (externalArr.length === 0) {
- return () => false
+ return () => false;
}
- const pattern = new RegExp(`^(${externalArr.join('|')})($|/)`)
- return id => pattern.test(id)
-}
+ const pattern = new RegExp(`^(${externalArr.join("|")})($|/)`);
+ return (id) => pattern.test(id);
+};
-const minify = process.env.MINIFY
-const format = process.env.FORMAT
-const es = format === 'es'
-const umd = format === 'umd'
-const cjs = format === 'cjs'
+const minify = process.env.MINIFY;
+const format = process.env.FORMAT;
+const es = format === "es";
+const umd = format === "umd";
+const cjs = format === "cjs";
-let output
+let output;
if (es) {
- output = { file: `dist/react-final-form.es.js`, format: 'es' }
+ output = { file: `dist/react-final-form.es.js`, format: "es" };
} else if (umd) {
if (minify) {
output = {
file: `dist/react-final-form.umd.min.js`,
- format: 'umd'
- }
+ format: "umd",
+ };
} else {
- output = { file: `dist/react-final-form.umd.js`, format: 'umd' }
+ output = { file: `dist/react-final-form.umd.js`, format: "umd" };
}
} else if (cjs) {
- output = { file: `dist/react-final-form.cjs.js`, format: 'cjs' }
+ output = { file: `dist/react-final-form.cjs.js`, format: "cjs" };
} else if (format) {
- throw new Error(`invalid format specified: "${format}".`)
+ throw new Error(`invalid format specified: "${format}".`);
} else {
- throw new Error('no format specified. --environment FORMAT:xxx')
+ throw new Error("no format specified. --environment FORMAT:xxx");
}
export default {
- input: 'src/index.js',
+ input: "src/index.js",
output: Object.assign(
{
- name: 'react-final-form',
- exports: 'named',
+ name: "react-final-form",
+ exports: "named",
globals: {
- react: 'React',
- 'final-form': 'FinalForm'
- }
+ react: "React",
+ "final-form": "FinalForm",
+ },
},
- output
+ output,
),
external: makeExternalPredicate(
umd
? Object.keys(pkg.peerDependencies || {})
: [
...Object.keys(pkg.dependencies || {}),
- ...Object.keys(pkg.peerDependencies || {})
- ]
+ ...Object.keys(pkg.peerDependencies || {}),
+ ],
),
plugins: [
- resolve({ mainFields: ['jsnext:main'] }),
+ resolve({ mainFields: ["jsnext:main"] }),
json(),
- commonjs({ include: 'node_modules/**' }),
+ commonjs({ include: "node_modules/**" }),
babel({
- exclude: 'node_modules/**',
- plugins: [['@babel/plugin-transform-runtime', { useESModules: !cjs }]],
- runtimeHelpers: true
+ exclude: "node_modules/**",
+ plugins: [["@babel/plugin-transform-runtime", { useESModules: !cjs }]],
+ runtimeHelpers: true,
}),
umd
? replace({
- 'process.env.NODE_ENV': JSON.stringify(
- minify ? 'production' : 'development'
- )
+ "process.env.NODE_ENV": JSON.stringify(
+ minify ? "production" : "development",
+ ),
})
: null,
- minify ? uglify() : null
- ].filter(Boolean)
-}
+ minify ? uglify() : null,
+ ].filter(Boolean),
+};
diff --git a/src/Field.js b/src/Field.js
index 2f46efe3..00937012 100644
--- a/src/Field.js
+++ b/src/Field.js
@@ -1,8 +1,8 @@
// @flow
-import * as React from 'react'
-import type { FieldProps as Props, FieldRenderProps } from './types'
-import renderComponent from './renderComponent'
-import useField from './useField'
+import * as React from "react";
+import type { FieldProps as Props, FieldRenderProps } from "./types";
+import renderComponent from "./renderComponent";
+import useField from "./useField";
const Field = React.forwardRef(function Field(
{
@@ -27,7 +27,7 @@ const Field = React.forwardRef(function Field(
value,
...rest
}: Props,
- ref
+ ref,
) {
const field: FieldRenderProps = useField(name, {
afterSubmit,
@@ -47,32 +47,32 @@ const Field = React.forwardRef(function Field(
type,
validate,
validateFields,
- value
- })
+ value,
+ });
- if (typeof children === 'function') {
- return (children: Function)({ ...field, ...rest })
+ if (typeof children === "function") {
+ return (children: Function)({ ...field, ...rest });
}
- if (typeof component === 'string') {
+ if (typeof component === "string") {
// ignore meta, combine input with any other props
return React.createElement(component, {
...field.input,
children,
ref,
- ...rest
- })
+ ...rest,
+ });
}
if (!name) {
- throw new Error('prop name cannot be undefined in component')
+ throw new Error("prop name cannot be undefined in component");
}
return renderComponent(
{ children, component, ref, ...rest },
field,
- `Field(${name})`
- )
-})
+ `Field(${name})`,
+ );
+});
-export default Field
+export default Field;
diff --git a/src/Field.test.js b/src/Field.test.js
index 1e62f154..afd08104 100644
--- a/src/Field.test.js
+++ b/src/Field.test.js
@@ -1,50 +1,50 @@
-import React from 'react'
-import { render, fireEvent, cleanup, act } from '@testing-library/react'
-import '@testing-library/jest-dom/extend-expect'
-import { ErrorBoundary, Toggle, wrapWith } from './testUtils'
-import Form from './ReactFinalForm'
-import Field from './Field'
+import React from "react";
+import { render, fireEvent, cleanup, act } from "@testing-library/react";
+import "@testing-library/jest-dom/extend-expect";
+import { ErrorBoundary, Toggle, wrapWith } from "./testUtils";
+import Form from "./ReactFinalForm";
+import Field from "./Field";
-const onSubmitMock = values => {}
+const onSubmitMock = (values) => {};
-const timeout = ms => new Promise(resolve => setTimeout(resolve, ms))
+const timeout = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
async function sleep(ms) {
await act(async () => {
- await timeout(ms)
- })
+ await timeout(ms);
+ });
}
-describe('Field', () => {
- afterEach(cleanup)
+describe("Field", () => {
+ afterEach(cleanup);
- it('should warn if not used inside a form', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ it("should warn if not used inside a form", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
render(
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'useField must be used inside of a component'
- )
- console.error.mockRestore()
- })
+ "useField must be used inside of a component",
+ );
+ console.error.mockRestore();
+ });
- it('should resubscribe if name changes', () => {
+ it("should resubscribe if name changes", () => {
const { getByTestId, getByText } = render(
- {isCat => (
+ {(isCat) => (
{({ handleSubmit }) => (
@@ -52,16 +52,16 @@ describe('Field', () => {
)}
)}
-
- )
- expect(getByTestId('name').value).toBe('Odie')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('Garfield')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('Odie')
- })
-
- it('should render via children render function', () => {
+ ,
+ );
+ expect(getByTestId("name").value).toBe("Odie");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("Garfield");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("Odie");
+ });
+
+ it("should render via children render function", () => {
const { getByTestId } = render(
{() => (
@@ -71,12 +71,12 @@ describe('Field', () => {
)}
-
- )
- expect(getByTestId('name')).toBeDefined()
- })
+ ,
+ );
+ expect(getByTestId("name")).toBeDefined();
+ });
- it('should render via render prop function', () => {
+ it("should render via render prop function", () => {
const { getByTestId } = render(
{() => (
@@ -87,12 +87,12 @@ describe('Field', () => {
/>
)}
-
- )
- expect(getByTestId('name')).toBeDefined()
- })
+ ,
+ );
+ expect(getByTestId("name")).toBeDefined();
+ });
- it('should include children when rendering via render prop function', () => {
+ it("should include children when rendering via render prop function", () => {
const { getByTestId } = render(
{() => (
@@ -117,22 +117,22 @@ describe('Field', () => {
)}
-
- )
- expect(getByTestId('color')).toBeDefined()
- expect(getByTestId('red')).toBeDefined()
- expect(getByTestId('green')).toBeDefined()
- expect(getByTestId('blue')).toBeDefined()
- })
-
- it('should unsubscribe on unmount', () => {
+ ,
+ );
+ expect(getByTestId("color")).toBeDefined();
+ expect(getByTestId("red")).toBeDefined();
+ expect(getByTestId("green")).toBeDefined();
+ expect(getByTestId("blue")).toBeDefined();
+ });
+
+ it("should unsubscribe on unmount", () => {
// This is mainly here for code coverage. 🧐
const { getByText } = render(
- {hidden => (
+ {(hidden) => (
{({ handleSubmit }) => (
@@ -143,13 +143,13 @@ describe('Field', () => {
)}
)}
-
- )
- fireEvent.click(getByText('Toggle'))
- })
+ ,
+ );
+ fireEvent.click(getByText("Toggle"));
+ });
- it('should focus, change, and blur', () => {
- const spy = jest.fn()
+ it("should focus, change, and blur", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{() => (
@@ -161,27 +161,27 @@ describe('Field', () => {
)}
-
- )
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].meta.active).toBe(false)
- expect(spy.mock.calls[0][0].input.value).toBe('')
- fireEvent.focus(getByTestId('name'))
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].meta.active).toBe(true)
- expect(spy.mock.calls[1][0].input.value).toBe('')
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0].meta.active).toBe(true)
- expect(spy.mock.calls[2][0].input.value).toBe('erikras')
- fireEvent.blur(getByTestId('name'))
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[3][0].meta.active).toBe(false)
- expect(spy.mock.calls[3][0].input.value).toBe('erikras')
- })
+ ,
+ );
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].meta.active).toBe(false);
+ expect(spy.mock.calls[0][0].input.value).toBe("");
+ fireEvent.focus(getByTestId("name"));
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].meta.active).toBe(true);
+ expect(spy.mock.calls[1][0].input.value).toBe("");
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0].meta.active).toBe(true);
+ expect(spy.mock.calls[2][0].input.value).toBe("erikras");
+ fireEvent.blur(getByTestId("name"));
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[3][0].meta.active).toBe(false);
+ expect(spy.mock.calls[3][0].input.value).toBe("erikras");
+ });
it("should convert '' to undefined on change", () => {
- const spy = jest.fn()
+ const spy = jest.fn();
const { getByTestId } = render(
{wrapWith(spy, () => (
@@ -189,51 +189,51 @@ describe('Field', () => {
))}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].values).toEqual({})
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].values).toEqual({ name: 'erikras' })
- fireEvent.change(getByTestId('name'), { target: { value: '' } })
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0].values).toEqual({})
- })
-
- it('should accept an identity parse prop to preserve empty strings', () => {
- const spy = jest.fn()
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].values).toEqual({});
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].values).toEqual({ name: "erikras" });
+ fireEvent.change(getByTestId("name"), { target: { value: "" } });
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0].values).toEqual({});
+ });
+
+ it("should accept an identity parse prop to preserve empty strings", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{wrapWith(spy, () => (
- v}>
+ v}>
{({ input: { value, ...props } }) => (
)}
))}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].values).toEqual({})
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].values).toEqual({ name: 'erikras' })
- fireEvent.change(getByTestId('name'), { target: { value: '' } })
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0].values).toEqual({ name: '' })
- })
-
- it('should accept a format function prop', () => {
- const spy = jest.fn()
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].values).toEqual({});
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].values).toEqual({ name: "erikras" });
+ fireEvent.change(getByTestId("name"), { target: { value: "" } });
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0].values).toEqual({ name: "" });
+ });
+
+ it("should accept a format function prop", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{wrapWith(spy, () => (
@@ -241,24 +241,24 @@ describe('Field', () => {
(value ? value.toUpperCase() : '')}
+ format={(value) => (value ? value.toUpperCase() : "")}
data-testid="name"
/>
))}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].values).toEqual({})
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].values).toEqual({ name: 'erikras' })
- expect(getByTestId('name').value).toBe('ERIKRAS')
- })
-
- it('should only format on blur if formatOnBlur is true', () => {
- const format = jest.fn(value => (value ? value.toUpperCase() : ''))
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].values).toEqual({});
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].values).toEqual({ name: "erikras" });
+ expect(getByTestId("name").value).toBe("ERIKRAS");
+ });
+
+ it("should only format on blur if formatOnBlur is true", () => {
+ const format = jest.fn((value) => (value ? value.toUpperCase() : ""));
const { getByTestId } = render(
{() => (
@@ -272,21 +272,21 @@ describe('Field', () => {
/>
)}
-
- )
- fireEvent.focus(getByTestId('name'))
- expect(getByTestId('name').value).toBe('')
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(getByTestId('name').value).toBe('erikras')
- expect(format).not.toHaveBeenCalled()
- fireEvent.blur(getByTestId('name'))
- expect(format).toHaveBeenCalled()
- expect(format).toHaveBeenCalledTimes(1)
- expect(getByTestId('name').value).toBe('ERIKRAS')
- })
-
- it('should `formatOnBlur` most updated value', () => {
- const format = jest.fn(value => (value ? value.trim() : ''))
+ ,
+ );
+ fireEvent.focus(getByTestId("name"));
+ expect(getByTestId("name").value).toBe("");
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(getByTestId("name").value).toBe("erikras");
+ expect(format).not.toHaveBeenCalled();
+ fireEvent.blur(getByTestId("name"));
+ expect(format).toHaveBeenCalled();
+ expect(format).toHaveBeenCalledTimes(1);
+ expect(getByTestId("name").value).toBe("ERIKRAS");
+ });
+
+ it("should `formatOnBlur` most updated value", () => {
+ const format = jest.fn((value) => (value ? value.trim() : ""));
const { getByTestId } = render(
{() => (
@@ -296,79 +296,79 @@ describe('Field', () => {
{
+ onBlur={(e) => {
input.onChange(
- e.target.value && e.target.value.toUpperCase()
- )
- input.onBlur(e)
+ e.target.value && e.target.value.toUpperCase(),
+ );
+ input.onBlur(e);
}}
/>
)}
)}
-
- )
- const inputText = ' erikras'
- fireEvent.focus(getByTestId('name'))
- expect(getByTestId('name').value).toBe('')
- fireEvent.change(getByTestId('name'), { target: { value: inputText } })
- expect(getByTestId('name').value).toBe(inputText)
- fireEvent.blur(getByTestId('name'))
- expect(format.mock.calls[0][0]).toBe(inputText.toUpperCase())
- expect(getByTestId('name').value).toBe(inputText.trim().toUpperCase())
- })
-
- it('should not format value at all when formatOnBlur and render prop', () => {
- const format = jest.fn(value => (value ? value.toUpperCase() : ''))
+ ,
+ );
+ const inputText = " erikras";
+ fireEvent.focus(getByTestId("name"));
+ expect(getByTestId("name").value).toBe("");
+ fireEvent.change(getByTestId("name"), { target: { value: inputText } });
+ expect(getByTestId("name").value).toBe(inputText);
+ fireEvent.blur(getByTestId("name"));
+ expect(format.mock.calls[0][0]).toBe(inputText.toUpperCase());
+ expect(getByTestId("name").value).toBe(inputText.trim().toUpperCase());
+ });
+
+ it("should not format value at all when formatOnBlur and render prop", () => {
+ const format = jest.fn((value) => (value ? value.toUpperCase() : ""));
render(
{() => (
{({ input }) => {
- expect(input.value).toBeUndefined()
- expect(format).not.toHaveBeenCalled()
- return
+ expect(input.value).toBeUndefined();
+ expect(format).not.toHaveBeenCalled();
+ return ;
}}
)}
-
- )
- })
+ ,
+ );
+ });
- it('should accept an identity format prop to preserve undefined values', () => {
- const spy = jest.fn()
+ it("should accept an identity format prop to preserve undefined values", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{() => (
- v}>
+ v}>
{wrapWith(spy, ({ input: { value, ...props } }) => (
))}
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].input.value).toBeUndefined()
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].input.value).toBe('erikras')
- fireEvent.change(getByTestId('name'), { target: { value: '' } })
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0].input.value).toBeUndefined()
- })
-
- it('should provide a value of [] when empty on a select multiple', () => {
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].input.value).toBeUndefined();
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].input.value).toBe("erikras");
+ fireEvent.change(getByTestId("name"), { target: { value: "" } });
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0].input.value).toBeUndefined();
+ });
+
+ it("should provide a value of [] when empty on a select multiple", () => {
const { getByTestId } = render(
{() => (
@@ -376,16 +376,16 @@ describe('Field', () => {
)}
-
- )
+ ,
+ );
// This test is mostly for code coverage. Is there a way to assure that the value prop
// passed to the is []?
- expect(getByTestId('name').value).toBe('')
- })
+ expect(getByTestId("name").value).toBe("");
+ });
- it('should pass multiple through to custom components', () => {
- const CustomSelect = jest.fn(({ input }) => )
+ it("should pass multiple through to custom components", () => {
+ const CustomSelect = jest.fn(({ input }) => );
render(
{
)}
-
- )
+ ,
+ );
- expect(CustomSelect).toHaveBeenCalled()
- expect(CustomSelect).toHaveBeenCalledTimes(1)
- expect(CustomSelect.mock.calls[0][0].input.multiple).toBe(true)
- })
+ expect(CustomSelect).toHaveBeenCalled();
+ expect(CustomSelect).toHaveBeenCalledTimes(1);
+ expect(CustomSelect.mock.calls[0][0].input.multiple).toBe(true);
+ });
- it('should pass ref through to the input', () => {
- const ref = React.createRef()
+ it("should pass ref through to the input", () => {
+ const ref = React.createRef();
render(
{() => (
@@ -414,15 +414,15 @@ describe('Field', () => {
)}
-
- )
+ ,
+ );
- expect(ref.current).not.toBe(null)
- expect(ref.current instanceof HTMLInputElement).toBe(true)
- })
+ expect(ref.current).not.toBe(null);
+ expect(ref.current instanceof HTMLInputElement).toBe(true);
+ });
- it('should not pass an undefined type through to the input', () => {
- const MyInput = jest.fn(({ input }) => )
+ it("should not pass an undefined type through to the input", () => {
+ const MyInput = jest.fn(({ input }) => );
render(
{() => (
@@ -430,16 +430,16 @@ describe('Field', () => {
)}
-
- )
+ ,
+ );
- expect(MyInput).toHaveBeenCalled()
- expect(MyInput).toHaveBeenCalledTimes(1)
- expect(MyInput.mock.calls[0][0].input).not.toHaveProperty('type')
- })
+ expect(MyInput).toHaveBeenCalled();
+ expect(MyInput).toHaveBeenCalledTimes(1);
+ expect(MyInput.mock.calls[0][0].input).not.toHaveProperty("type");
+ });
- it('should optionally allow null values', () => {
- const spy = jest.fn()
+ it("should optionally allow null values", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{
{wrapWith(spy, ({ input: { value, ...props } }) => (
))}
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].input.value).toBe(null)
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].input.value).toBe('erikras')
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].input.value).toBe(null);
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].input.value).toBe("erikras");
act(() => {
- spy.mock.calls[1][0].input.onChange(null)
- })
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0].input.value).toBe(null)
- })
-
- it('should not allow null values when allowNull not true', () => {
- const spy = jest.fn()
+ spy.mock.calls[1][0].input.onChange(null);
+ });
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0].input.value).toBe(null);
+ });
+
+ it("should not allow null values when allowNull not true", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{
{wrapWith(spy, ({ input: { value, ...props } }) => (
))}
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].input.value).toBe('')
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].input.value).toBe('erikras')
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].input.value).toBe("");
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].input.value).toBe("erikras");
act(() => {
- spy.mock.calls[1][0].input.onChange(null)
- })
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0].input.value).toBe('')
- })
-
- it('should not let validate prop bleed through', () => {
- const spy = jest.fn()
+ spy.mock.calls[1][0].input.onChange(null);
+ });
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0].input.value).toBe("");
+ });
+
+ it("should not let validate prop bleed through", () => {
+ const spy = jest.fn();
render(
{() => (
@@ -523,15 +523,15 @@ describe('Field', () => {
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].validate).toBeUndefined()
- })
-
- it('should not let subscription prop bleed through', () => {
- const spy = jest.fn()
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].validate).toBeUndefined();
+ });
+
+ it("should not let subscription prop bleed through", () => {
+ const spy = jest.fn();
render(
{() => (
@@ -543,27 +543,27 @@ describe('Field', () => {
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].subscription).toBeUndefined()
- })
-
- it('should allow changing field-level validation function', () => {
- const simpleValidate = value => (value ? undefined : 'Required')
- const complexValidate = value => {
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].subscription).toBeUndefined();
+ });
+
+ it("should allow changing field-level validation function", () => {
+ const simpleValidate = (value) => (value ? undefined : "Required");
+ const complexValidate = (value) => {
if (value) {
if (value !== value.toUpperCase()) {
- return 'SHOULD BE UPPERCASE!'
+ return "SHOULD BE UPPERCASE!";
}
} else {
- return 'Required'
+ return "Required";
}
- }
+ };
const { getByTestId, getByText } = render(
- {useComplexValidation => (
+ {(useComplexValidation) => (
{({ handleSubmit }) => (
@@ -585,16 +585,16 @@ describe('Field', () => {
)}
)}
-
- )
- expect(getByTestId('error')).toHaveTextContent('Required')
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(getByTestId('error')).toHaveTextContent('')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('error')).toHaveTextContent('SHOULD BE UPPERCASE!')
- fireEvent.change(getByTestId('name'), { target: { value: 'ERIKRAS' } })
- expect(getByTestId('error')).toHaveTextContent('')
- })
+ ,
+ );
+ expect(getByTestId("error")).toHaveTextContent("Required");
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(getByTestId("error")).toHaveTextContent("");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("error")).toHaveTextContent("SHOULD BE UPPERCASE!");
+ fireEvent.change(getByTestId("name"), { target: { value: "ERIKRAS" } });
+ expect(getByTestId("error")).toHaveTextContent("");
+ });
/**
* Allow me to explain this. If we allow field level validation functions
@@ -609,18 +609,18 @@ describe('Field', () => {
* from the newly unmounted Field, it would need to run validation on the entire
* form.
*/
- it('should ignore changes field-level validation function', () => {
- const createValidator = isRequired =>
- isRequired ? value => (value ? undefined : 'Required') : undefined
+ it("should ignore changes field-level validation function", () => {
+ const createValidator = (isRequired) =>
+ isRequired ? (value) => (value ? undefined : "Required") : undefined;
const Error = ({ name }) => (
{({ meta: { error } }) => {error}
}
- )
+ );
const { getByTestId, getByText } = render(
- {isRequired => (
+ {(isRequired) => (
{({ handleSubmit }) => (
@@ -641,23 +641,23 @@ describe('Field', () => {
)}
)}
-
- )
- expect(getByTestId('error')).toBeEmptyDOMElement()
- expect(getByTestId('error2')).toBeEmptyDOMElement()
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('error')).toHaveTextContent('Required')
- expect(getByTestId('error2')).toHaveTextContent('Required')
- fireEvent.click(getByText('Toggle'))
+ ,
+ );
+ expect(getByTestId("error")).toBeEmptyDOMElement();
+ expect(getByTestId("error2")).toBeEmptyDOMElement();
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("error")).toHaveTextContent("Required");
+ expect(getByTestId("error2")).toHaveTextContent("Required");
+ fireEvent.click(getByText("Toggle"));
// ERROR IS NOT CLEARED (see comment above)
- expect(getByTestId('error')).toHaveTextContent('Required')
- expect(getByTestId('error2')).toHaveTextContent('Required')
- })
+ expect(getByTestId("error")).toHaveTextContent("Required");
+ expect(getByTestId("error2")).toHaveTextContent("Required");
+ });
- it('should not rerender if validateFields is !== every time', () => {
+ it("should not rerender if validateFields is !== every time", () => {
// https://github.com/final-form/react-final-form/issues/502
- const required = value => (value ? undefined : 'Required')
- const spy = jest.fn()
+ const required = (value) => (value ? undefined : "Required");
+ const spy = jest.fn();
const { getByTestId } = render(
{({ handleSubmit }) => (
@@ -672,14 +672,14 @@ describe('Field', () => {
)}
-
- )
+ ,
+ );
// first render registered validation, second contains error
- expect(spy).toHaveBeenCalledTimes(2)
- expect(getByTestId('error')).toHaveTextContent('Required')
- })
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(getByTestId("error")).toHaveTextContent("Required");
+ });
- it('should pass along type prop', () => {
+ it("should pass along type prop", () => {
const { getByTestId } = render(
{({ handleSubmit }) => (
@@ -704,14 +704,14 @@ describe('Field', () => {
/>
)}
-
- )
- expect(getByTestId('checkbox').type).toBe('checkbox')
- expect(getByTestId('password').type).toBe('password')
- expect(getByTestId('radio').type).toBe('radio')
- })
-
- it('should render checkboxes with checked prop', () => {
+ ,
+ );
+ expect(getByTestId("checkbox").type).toBe("checkbox");
+ expect(getByTestId("password").type).toBe("password");
+ expect(getByTestId("radio").type).toBe("radio");
+ });
+
+ it("should render checkboxes with checked prop", () => {
const { getByTestId } = render(
{({ handleSubmit }) => (
@@ -724,17 +724,17 @@ describe('Field', () => {
/>
)}
-
- )
- expect(getByTestId('employed').type).toBe('checkbox')
- expect(getByTestId('employed').checked).toBe(false)
- fireEvent.change(getByTestId('employed'), { target: { checked: true } })
- expect(getByTestId('employed').checked).toBe(true)
- })
+ ,
+ );
+ expect(getByTestId("employed").type).toBe("checkbox");
+ expect(getByTestId("employed").checked).toBe(false);
+ fireEvent.change(getByTestId("employed"), { target: { checked: true } });
+ expect(getByTestId("employed").checked).toBe(true);
+ });
it('should render "array" checkboxes with checked prop when value is included in array', () => {
const { getByTestId } = render(
-
+
{({ handleSubmit }) => (
{
/>
)}
-
- )
- expect(getByTestId('red').checked).toBe(true)
- expect(getByTestId('green').checked).toBe(false)
- expect(getByTestId('blue').checked).toBe(true)
- })
+ ,
+ );
+ expect(getByTestId("red").checked).toBe(true);
+ expect(getByTestId("green").checked).toBe(false);
+ expect(getByTestId("blue").checked).toBe(true);
+ });
it('should render "array" custom checkboxes with checked prop when value is included in array', () => {
- const red = jest.fn()
- const green = jest.fn()
- const blue = jest.fn()
+ const red = jest.fn();
+ const green = jest.fn();
+ const blue = jest.fn();
render(
-
+
{({ handleSubmit }) => (
@@ -792,28 +792,28 @@ describe('Field', () => {
)}
-
- )
+ ,
+ );
// All forms without restricted subscriptions render twice at first because they
// need to update their validation and touched/modified/visited maps every time
// new fields are registered.
- expect(red).toHaveBeenCalled()
- expect(red).toHaveBeenCalledTimes(2)
- expect(red.mock.calls[0][0].input.checked).toBe(true)
- expect(red.mock.calls[1][0].input.checked).toBe(true)
- expect(green).toHaveBeenCalled()
- expect(green).toHaveBeenCalledTimes(2)
- expect(green.mock.calls[0][0].input.checked).toBe(false)
- expect(green.mock.calls[1][0].input.checked).toBe(false)
- expect(blue).toHaveBeenCalled()
- expect(blue).toHaveBeenCalledTimes(2)
- expect(blue.mock.calls[0][0].input.checked).toBe(true)
- expect(blue.mock.calls[1][0].input.checked).toBe(true)
- })
-
- it('should render radio buttons with checked prop', () => {
+ expect(red).toHaveBeenCalled();
+ expect(red).toHaveBeenCalledTimes(2);
+ expect(red.mock.calls[0][0].input.checked).toBe(true);
+ expect(red.mock.calls[1][0].input.checked).toBe(true);
+ expect(green).toHaveBeenCalled();
+ expect(green).toHaveBeenCalledTimes(2);
+ expect(green.mock.calls[0][0].input.checked).toBe(false);
+ expect(green.mock.calls[1][0].input.checked).toBe(false);
+ expect(blue).toHaveBeenCalled();
+ expect(blue).toHaveBeenCalledTimes(2);
+ expect(blue.mock.calls[0][0].input.checked).toBe(true);
+ expect(blue.mock.calls[1][0].input.checked).toBe(true);
+ });
+
+ it("should render radio buttons with checked prop", () => {
const { getByTestId } = render(
-
+
{({ handleSubmit }) => (
{
/>
)}
-
- )
- expect(getByTestId('red').type).toBe('radio')
- expect(getByTestId('red').checked).toBe(false)
- expect(getByTestId('green').type).toBe('radio')
- expect(getByTestId('green').checked).toBe(true)
- expect(getByTestId('blue').type).toBe('radio')
- expect(getByTestId('blue').checked).toBe(false)
- })
-
- it('should render custom radio component with checked prop', () => {
- const red = jest.fn()
- const green = jest.fn()
- const blue = jest.fn()
+ ,
+ );
+ expect(getByTestId("red").type).toBe("radio");
+ expect(getByTestId("red").checked).toBe(false);
+ expect(getByTestId("green").type).toBe("radio");
+ expect(getByTestId("green").checked).toBe(true);
+ expect(getByTestId("blue").type).toBe("radio");
+ expect(getByTestId("blue").checked).toBe(false);
+ });
+
+ it("should render custom radio component with checked prop", () => {
+ const red = jest.fn();
+ const green = jest.fn();
+ const blue = jest.fn();
render(
-
+
{({ handleSubmit }) => (
@@ -874,36 +874,36 @@ describe('Field', () => {
)}
-
- )
+ ,
+ );
// All forms without restricted subscriptions render twice at first because they
// need to update their validation and touched/modified/visited maps every time
// new fields are registered.
- expect(red).toHaveBeenCalled()
- expect(red).toHaveBeenCalledTimes(2)
- expect(red.mock.calls[0][0].input.checked).toBe(false)
- expect(red.mock.calls[1][0].input.checked).toBe(false)
- expect(green).toHaveBeenCalled()
- expect(green).toHaveBeenCalledTimes(2)
- expect(green.mock.calls[0][0].input.checked).toBe(true)
- expect(green.mock.calls[1][0].input.checked).toBe(true)
- expect(blue).toHaveBeenCalled()
- expect(blue).toHaveBeenCalledTimes(2)
- expect(blue.mock.calls[0][0].input.checked).toBe(false)
- expect(blue.mock.calls[1][0].input.checked).toBe(false)
- })
-
- it('should use isEqual to calculate dirty/pristine', () => {
- const isEqual = (a, b) => (a && a.toUpperCase()) === (b && b.toUpperCase())
+ expect(red).toHaveBeenCalled();
+ expect(red).toHaveBeenCalledTimes(2);
+ expect(red.mock.calls[0][0].input.checked).toBe(false);
+ expect(red.mock.calls[1][0].input.checked).toBe(false);
+ expect(green).toHaveBeenCalled();
+ expect(green).toHaveBeenCalledTimes(2);
+ expect(green.mock.calls[0][0].input.checked).toBe(true);
+ expect(green.mock.calls[1][0].input.checked).toBe(true);
+ expect(blue).toHaveBeenCalled();
+ expect(blue).toHaveBeenCalledTimes(2);
+ expect(blue.mock.calls[0][0].input.checked).toBe(false);
+ expect(blue.mock.calls[1][0].input.checked).toBe(false);
+ });
+
+ it("should use isEqual to calculate dirty/pristine", () => {
+ const isEqual = (a, b) => (a && a.toUpperCase()) === (b && b.toUpperCase());
const { getByTestId } = render(
-
+
{() => (
{({ input, meta }) => (
@@ -911,19 +911,19 @@ describe('Field', () => {
)}
-
- )
- expect(getByTestId('input').value).toBe('bob')
- expect(getByTestId('dirty')).toHaveTextContent('Pristine')
- fireEvent.change(getByTestId('input'), { target: { value: 'bobby' } })
- expect(getByTestId('dirty')).toHaveTextContent('Dirty')
- fireEvent.change(getByTestId('input'), { target: { value: 'BOB' } })
- expect(getByTestId('dirty')).toHaveTextContent('Pristine')
- })
-
- it('should be able to use inline isEqual to calculate dirty/pristine without falling into infinite rerender loop', () => {
+ ,
+ );
+ expect(getByTestId("input").value).toBe("bob");
+ expect(getByTestId("dirty")).toHaveTextContent("Pristine");
+ fireEvent.change(getByTestId("input"), { target: { value: "bobby" } });
+ expect(getByTestId("dirty")).toHaveTextContent("Dirty");
+ fireEvent.change(getByTestId("input"), { target: { value: "BOB" } });
+ expect(getByTestId("dirty")).toHaveTextContent("Pristine");
+ });
+
+ it("should be able to use inline isEqual to calculate dirty/pristine without falling into infinite rerender loop", () => {
const { getByTestId } = render(
-
+
{() => (
{
{({ input, meta }) => (
@@ -943,20 +943,20 @@ describe('Field', () => {
)}
-
- )
- expect(getByTestId('input').value).toBe('bob')
- expect(getByTestId('dirty')).toHaveTextContent('Pristine')
- fireEvent.change(getByTestId('input'), { target: { value: 'bobby' } })
- expect(getByTestId('dirty')).toHaveTextContent('Dirty')
- fireEvent.change(getByTestId('input'), { target: { value: 'BOB' } })
- expect(getByTestId('dirty')).toHaveTextContent('Pristine')
- })
-
- it('should only call each field-level validation once upon initial mount', () => {
- const fooValidate = jest.fn()
- const barValidate = jest.fn()
- const bazValidate = jest.fn()
+ ,
+ );
+ expect(getByTestId("input").value).toBe("bob");
+ expect(getByTestId("dirty")).toHaveTextContent("Pristine");
+ fireEvent.change(getByTestId("input"), { target: { value: "bobby" } });
+ expect(getByTestId("dirty")).toHaveTextContent("Dirty");
+ fireEvent.change(getByTestId("input"), { target: { value: "BOB" } });
+ expect(getByTestId("dirty")).toHaveTextContent("Pristine");
+ });
+
+ it("should only call each field-level validation once upon initial mount", () => {
+ const fooValidate = jest.fn();
+ const barValidate = jest.fn();
+ const bazValidate = jest.fn();
render(
{() => (
@@ -966,15 +966,15 @@ describe('Field', () => {
)}
-
- )
- expect(fooValidate).toHaveBeenCalledTimes(1)
- expect(barValidate).toHaveBeenCalledTimes(1)
- expect(bazValidate).toHaveBeenCalledTimes(1)
- })
-
- it('should warn when used without type prop and rendering radio, checkbox or multiple select indirectly', () => {
- const errorSpy = jest.spyOn(console, 'error').mockImplementation(() => {})
+ ,
+ );
+ expect(fooValidate).toHaveBeenCalledTimes(1);
+ expect(barValidate).toHaveBeenCalledTimes(1);
+ expect(bazValidate).toHaveBeenCalledTimes(1);
+ });
+
+ it("should warn when used without type prop and rendering radio, checkbox or multiple select indirectly", () => {
+ const errorSpy = jest.spyOn(console, "error").mockImplementation(() => {});
const { getByTestId } = render(
{() => (
@@ -992,7 +992,7 @@ describe('Field', () => {
{({ input }) => (
- {'Option'}
+ {"Option"}
)}
@@ -1002,48 +1002,48 @@ describe('Field', () => {
data-testid="selectMultipleWithoutRenderProp"
multiple
>
- {'Option'}
+ {"Option"}
)}
-
- )
-
- expect(errorSpy).not.toHaveBeenCalled()
- fireEvent.click(getByTestId('checkbox'), {
- target: { type: 'checkbox', checked: true }
- })
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+
+ expect(errorSpy).not.toHaveBeenCalled();
+ fireEvent.click(getByTestId("checkbox"), {
+ target: { type: "checkbox", checked: true },
+ });
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0]).toBe(
'You must pass `type="checkbox"` prop to your Field(checkboxInput) component.\n' +
- 'Without it we don\'t know how to unpack your `value` prop - "checkboxValue".'
- )
- fireEvent.click(getByTestId('radio'), {
- target: { type: 'radio', value: 'radio value' }
- })
- expect(errorSpy).toHaveBeenCalledTimes(2)
+ 'Without it we don\'t know how to unpack your `value` prop - "checkboxValue".',
+ );
+ fireEvent.click(getByTestId("radio"), {
+ target: { type: "radio", value: "radio value" },
+ });
+ expect(errorSpy).toHaveBeenCalledTimes(2);
expect(errorSpy.mock.calls[1][0]).toBe(
'You must pass `type="radio"` prop to your Field(radioInput) component.\n' +
- 'Without it we don\'t know how to unpack your `value` prop - "radioValue".'
- )
- fireEvent.change(getByTestId('select'), {
- target: { value: ['some value'] }
- })
- expect(errorSpy).toHaveBeenCalledTimes(3)
+ 'Without it we don\'t know how to unpack your `value` prop - "radioValue".',
+ );
+ fireEvent.change(getByTestId("select"), {
+ target: { value: ["some value"] },
+ });
+ expect(errorSpy).toHaveBeenCalledTimes(3);
expect(errorSpy.mock.calls[2][0]).toBe(
'You must pass `type="select"` prop to your Field(selectMultipleInput) component.\n' +
- "Without it we don't know how to unpack your `value` prop - []."
- )
- fireEvent.change(getByTestId('selectMultipleWithoutRenderProp'), {
- target: { value: ['some value'] }
- })
+ "Without it we don't know how to unpack your `value` prop - [].",
+ );
+ fireEvent.change(getByTestId("selectMultipleWithoutRenderProp"), {
+ target: { value: ["some value"] },
+ });
// error not given, since we can deduce that it's a "select"
- expect(errorSpy).toHaveBeenCalledTimes(3)
- errorSpy.mockRestore()
- })
+ expect(errorSpy).toHaveBeenCalledTimes(3);
+ errorSpy.mockRestore();
+ });
- it('should formatOnBlur on submit', () => {
- const onSubmit = jest.fn()
+ it("should formatOnBlur on submit", () => {
+ const onSubmit = jest.fn();
const { getByTestId, getByText } = render(
{({ handleSubmit }) => (
@@ -1051,41 +1051,41 @@ describe('Field', () => {
value && value.toUpperCase()}
+ format={(value) => value && value.toUpperCase()}
formatOnBlur
data-testid="name"
/>
Submit
)}
-
- )
- expect(getByTestId('name').value).toBe('')
- fireEvent.focus(getByTestId('name'))
- fireEvent.change(getByTestId('name'), { target: { value: 'erik' } })
- expect(getByTestId('name').value).toBe('erik')
- fireEvent.blur(getByTestId('name'))
- expect(getByTestId('name').value).toBe('ERIK')
-
- fireEvent.focus(getByTestId('name'))
- fireEvent.change(getByTestId('name'), { target: { value: 'ERIKras' } })
- expect(getByTestId('name').value).toBe('ERIKras')
-
- expect(onSubmit).not.toHaveBeenCalled()
- fireEvent.click(getByText('Submit'))
- expect(onSubmit).toHaveBeenCalled()
- expect(onSubmit).toHaveBeenCalledTimes(1)
- expect(onSubmit.mock.calls[0][0]).toEqual({ name: 'ERIKRAS' })
+ ,
+ );
+ expect(getByTestId("name").value).toBe("");
+ fireEvent.focus(getByTestId("name"));
+ fireEvent.change(getByTestId("name"), { target: { value: "erik" } });
+ expect(getByTestId("name").value).toBe("erik");
+ fireEvent.blur(getByTestId("name"));
+ expect(getByTestId("name").value).toBe("ERIK");
+
+ fireEvent.focus(getByTestId("name"));
+ fireEvent.change(getByTestId("name"), { target: { value: "ERIKras" } });
+ expect(getByTestId("name").value).toBe("ERIKras");
+
+ expect(onSubmit).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Submit"));
+ expect(onSubmit).toHaveBeenCalled();
+ expect(onSubmit).toHaveBeenCalledTimes(1);
+ expect(onSubmit.mock.calls[0][0]).toEqual({ name: "ERIKRAS" });
// submit again with no need for format
- fireEvent.click(getByText('Submit'))
- expect(onSubmit).toHaveBeenCalledTimes(2)
- expect(onSubmit.mock.calls[1][0]).toEqual({ name: 'ERIKRAS' })
- })
-
- it('should allow submission to be cancelled in beforeSubmit', () => {
- const onSubmit = jest.fn()
- const beforeSubmit = jest.fn(() => false)
+ fireEvent.click(getByText("Submit"));
+ expect(onSubmit).toHaveBeenCalledTimes(2);
+ expect(onSubmit.mock.calls[1][0]).toEqual({ name: "ERIKRAS" });
+ });
+
+ it("should allow submission to be cancelled in beforeSubmit", () => {
+ const onSubmit = jest.fn();
+ const beforeSubmit = jest.fn(() => false);
const { getByTestId, getByText } = render(
{({ handleSubmit }) => (
@@ -1099,20 +1099,20 @@ describe('Field', () => {
Submit
)}
-
- )
- expect(getByTestId('name').value).toBe('')
- fireEvent.focus(getByTestId('name'))
- fireEvent.change(getByTestId('name'), { target: { value: 'erik' } })
-
- expect(onSubmit).not.toHaveBeenCalled()
- expect(beforeSubmit).not.toHaveBeenCalled()
- fireEvent.click(getByText('Submit'))
- expect(onSubmit).not.toHaveBeenCalled()
- expect(beforeSubmit).toHaveBeenCalled()
- })
-
- it('update validating flag on async field-level validation', async () => {
+ ,
+ );
+ expect(getByTestId("name").value).toBe("");
+ fireEvent.focus(getByTestId("name"));
+ fireEvent.change(getByTestId("name"), { target: { value: "erik" } });
+
+ expect(onSubmit).not.toHaveBeenCalled();
+ expect(beforeSubmit).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Submit"));
+ expect(onSubmit).not.toHaveBeenCalled();
+ expect(beforeSubmit).toHaveBeenCalled();
+ });
+
+ it("update validating flag on async field-level validation", async () => {
const { getByTestId } = render(
{({ handleSubmit }) => (
@@ -1120,52 +1120,52 @@ describe('Field', () => {
{
- await timeout(5)
- return value === 'erikras' ? 'Username taken' : undefined
+ validate={async (value) => {
+ await timeout(5);
+ return value === "erikras" ? "Username taken" : undefined;
}}
data-testid="name"
/>
{({ meta: { validating } }) => (
- {validating === true ? 'Spinner' : 'Not Validating'}
+ {validating === true ? "Spinner" : "Not Validating"}
)}
Submit
)}
-
- )
- expect(getByTestId('validating')).toHaveTextContent('Spinner')
+ ,
+ );
+ expect(getByTestId("validating")).toHaveTextContent("Spinner");
- await sleep(6)
+ await sleep(6);
- expect(getByTestId('name').value).toBe('')
- fireEvent.focus(getByTestId('name'))
- expect(getByTestId('validating')).toHaveTextContent('Not Validating')
+ expect(getByTestId("name").value).toBe("");
+ fireEvent.focus(getByTestId("name"));
+ expect(getByTestId("validating")).toHaveTextContent("Not Validating");
- fireEvent.change(getByTestId('name'), { target: { value: 'erik' } })
- expect(getByTestId('validating')).toHaveTextContent('Spinner')
+ fireEvent.change(getByTestId("name"), { target: { value: "erik" } });
+ expect(getByTestId("validating")).toHaveTextContent("Spinner");
- await sleep(6)
+ await sleep(6);
- expect(getByTestId('validating')).toHaveTextContent('Not Validating')
+ expect(getByTestId("validating")).toHaveTextContent("Not Validating");
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(getByTestId('validating')).toHaveTextContent('Spinner')
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(getByTestId("validating")).toHaveTextContent("Spinner");
- await sleep(6)
+ await sleep(6);
- expect(getByTestId('validating')).toHaveTextContent('Not Validating')
- })
+ expect(getByTestId("validating")).toHaveTextContent("Not Validating");
+ });
- it('not call record-level validation on Field mount', () => {
- const validate = jest.fn()
+ it("not call record-level validation on Field mount", () => {
+ const validate = jest.fn();
const { getByText } = render(
- {showOtherFields => (
+ {(showOtherFields) => (
{({ handleSubmit }) => (
@@ -1187,26 +1187,26 @@ describe('Field', () => {
)}
)}
-
- )
- expect(validate).toHaveBeenCalledTimes(1)
- fireEvent.click(getByText('Toggle'))
- expect(validate).toHaveBeenCalledTimes(1)
- })
+ ,
+ );
+ expect(validate).toHaveBeenCalledTimes(1);
+ fireEvent.click(getByText("Toggle"));
+ expect(validate).toHaveBeenCalledTimes(1);
+ });
- it('submit should not throw when field with enabled `formatOnBlur` changes name `prop`', () => {
- const onSubmit = jest.fn()
+ it("submit should not throw when field with enabled `formatOnBlur` changes name `prop`", () => {
+ const onSubmit = jest.fn();
- const trim = value => value && value.trim()
+ const trim = (value) => value && value.trim();
const { getByTestId, getByText } = render(
{({ handleSubmit }) => (
- {newFieldName => (
+ {(newFieldName) => (
{
)}
)}
-
- )
-
- fireEvent.click(getByText('Toggle'))
- fireEvent.change(getByTestId('field'), {
- target: { value: 'trailing space ' }
- })
- fireEvent.click(getByText('Submit'))
- expect(onSubmit).toHaveBeenCalled()
- expect(onSubmit.mock.calls[0][0]).toEqual({ newName: 'trailing space' })
- })
-
- it('should throw an error if name prop is undefined', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
-
- const errorSpy = jest.fn()
+ ,
+ );
+
+ fireEvent.click(getByText("Toggle"));
+ fireEvent.change(getByTestId("field"), {
+ target: { value: "trailing space " },
+ });
+ fireEvent.click(getByText("Submit"));
+ expect(onSubmit).toHaveBeenCalled();
+ expect(onSubmit.mock.calls[0][0]).toEqual({ newName: "trailing space" });
+ });
+
+ it("should throw an error if name prop is undefined", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+
+ const errorSpy = jest.fn();
render(
{() => } />}
-
- )
+ ,
+ );
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'prop name cannot be undefined in component'
- )
- console.error.mockRestore()
- })
-})
+ "prop name cannot be undefined in component",
+ );
+ console.error.mockRestore();
+ });
+});
diff --git a/src/FormSpy.js b/src/FormSpy.js
index ca86f340..281b7d4e 100644
--- a/src/FormSpy.js
+++ b/src/FormSpy.js
@@ -1,43 +1,46 @@
// @flow
-import renderComponent from './renderComponent'
-import type { FormSpyPropsWithForm as Props, FormSpyRenderProps } from './types'
-import type { FormValuesShape } from 'final-form'
-import isSyntheticEvent from './isSyntheticEvent'
-import useForm from './useForm'
-import useFormState from './useFormState'
+import renderComponent from "./renderComponent";
+import type {
+ FormSpyPropsWithForm as Props,
+ FormSpyRenderProps,
+} from "./types";
+import type { FormValuesShape } from "final-form";
+import isSyntheticEvent from "./isSyntheticEvent";
+import useForm from "./useForm";
+import useFormState from "./useFormState";
function FormSpy({
onChange,
subscription,
...rest
}: Props) {
- const reactFinalForm = useForm('FormSpy')
- const state = useFormState({ onChange, subscription })
+ const reactFinalForm = useForm("FormSpy");
+ const state = useFormState({ onChange, subscription });
if (onChange) {
- return null
+ return null;
}
const renderProps: FormSpyRenderProps = {
form: {
...reactFinalForm,
- reset: eventOrValues => {
+ reset: (eventOrValues) => {
if (isSyntheticEvent(eventOrValues)) {
// it's a React SyntheticEvent, call reset with no arguments
- reactFinalForm.reset()
+ reactFinalForm.reset();
} else {
- reactFinalForm.reset(eventOrValues)
+ reactFinalForm.reset(eventOrValues);
}
- }
- }
- }
+ },
+ },
+ };
return renderComponent(
{
...rest,
- ...renderProps
+ ...renderProps,
},
state,
- 'FormSpy'
- )
+ "FormSpy",
+ );
}
-export default FormSpy
+export default FormSpy;
diff --git a/src/FormSpy.test.js b/src/FormSpy.test.js
index c48ea676..7a4f79ec 100644
--- a/src/FormSpy.test.js
+++ b/src/FormSpy.test.js
@@ -1,43 +1,43 @@
-import React from 'react'
-import { render, fireEvent, cleanup } from '@testing-library/react'
-import '@testing-library/jest-dom/extend-expect'
-import { ErrorBoundary, Toggle, wrapWith } from './testUtils'
-import Form from './ReactFinalForm'
-import Field from './Field'
-import FormSpy from './FormSpy'
+import React from "react";
+import { render, fireEvent, cleanup } from "@testing-library/react";
+import "@testing-library/jest-dom/extend-expect";
+import { ErrorBoundary, Toggle, wrapWith } from "./testUtils";
+import Form from "./ReactFinalForm";
+import Field from "./Field";
+import FormSpy from "./FormSpy";
-const onSubmitMock = values => {}
-const hasFormApi = props => {
- expect(props.form).toBeDefined()
- expect(typeof props.form.batch).toBe('function')
- expect(typeof props.form.blur).toBe('function')
- expect(typeof props.form.change).toBe('function')
- expect(typeof props.form.focus).toBe('function')
- expect(typeof props.form.initialize).toBe('function')
- expect(typeof props.form.reset).toBe('function')
-}
+const onSubmitMock = (values) => {};
+const hasFormApi = (props) => {
+ expect(props.form).toBeDefined();
+ expect(typeof props.form.batch).toBe("function");
+ expect(typeof props.form.blur).toBe("function");
+ expect(typeof props.form.change).toBe("function");
+ expect(typeof props.form.focus).toBe("function");
+ expect(typeof props.form.initialize).toBe("function");
+ expect(typeof props.form.reset).toBe("function");
+};
-describe('FormSpy', () => {
- afterEach(cleanup)
+describe("FormSpy", () => {
+ afterEach(cleanup);
- it('should warn if not used inside a form', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ it("should warn if not used inside a form", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
render(
} />
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'FormSpy must be used inside of a component'
- )
- console.error.mockRestore()
- })
+ "FormSpy must be used inside of a component",
+ );
+ console.error.mockRestore();
+ });
- it('should allow subscribing to everything', () => {
- const spy = jest.fn()
+ it("should allow subscribing to everything", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{() => (
@@ -50,63 +50,63 @@ describe('FormSpy', () => {
/>
)}
-
- )
- expect(spy).toHaveBeenCalled()
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
// All forms without restricted subscriptions render twice at first because they
// need to update their validation and touched/modified/visited maps every time
// new fields are registered.
- expect(spy).toHaveBeenCalledTimes(2)
- hasFormApi(spy.mock.calls[0][0])
- expect(spy.mock.calls[0][0].dirty).toBe(false)
- expect(spy.mock.calls[0][0].errors).toEqual({})
- expect(spy.mock.calls[0][0].invalid).toBe(false)
- expect(spy.mock.calls[0][0].pristine).toBe(true)
- expect(spy.mock.calls[0][0].submitFailed).toBe(false)
- expect(spy.mock.calls[0][0].submitSucceeded).toBe(false)
- expect(spy.mock.calls[0][0].submitting).toBe(false)
- expect(spy.mock.calls[0][0].valid).toBe(true)
- expect(spy.mock.calls[0][0].validating).toBe(false)
- expect(spy.mock.calls[0][0].values).toEqual({})
- hasFormApi(spy.mock.calls[1][0])
- expect(spy.mock.calls[1][0].dirty).toBe(false)
- expect(spy.mock.calls[1][0].errors).toEqual({})
- expect(spy.mock.calls[1][0].invalid).toBe(false)
- expect(spy.mock.calls[1][0].pristine).toBe(true)
- expect(spy.mock.calls[1][0].submitFailed).toBe(false)
- expect(spy.mock.calls[1][0].submitSucceeded).toBe(false)
- expect(spy.mock.calls[1][0].submitting).toBe(false)
- expect(spy.mock.calls[1][0].valid).toBe(true)
- expect(spy.mock.calls[1][0].validating).toBe(false)
- expect(spy.mock.calls[1][0].values).toEqual({})
+ expect(spy).toHaveBeenCalledTimes(2);
+ hasFormApi(spy.mock.calls[0][0]);
+ expect(spy.mock.calls[0][0].dirty).toBe(false);
+ expect(spy.mock.calls[0][0].errors).toEqual({});
+ expect(spy.mock.calls[0][0].invalid).toBe(false);
+ expect(spy.mock.calls[0][0].pristine).toBe(true);
+ expect(spy.mock.calls[0][0].submitFailed).toBe(false);
+ expect(spy.mock.calls[0][0].submitSucceeded).toBe(false);
+ expect(spy.mock.calls[0][0].submitting).toBe(false);
+ expect(spy.mock.calls[0][0].valid).toBe(true);
+ expect(spy.mock.calls[0][0].validating).toBe(false);
+ expect(spy.mock.calls[0][0].values).toEqual({});
+ hasFormApi(spy.mock.calls[1][0]);
+ expect(spy.mock.calls[1][0].dirty).toBe(false);
+ expect(spy.mock.calls[1][0].errors).toEqual({});
+ expect(spy.mock.calls[1][0].invalid).toBe(false);
+ expect(spy.mock.calls[1][0].pristine).toBe(true);
+ expect(spy.mock.calls[1][0].submitFailed).toBe(false);
+ expect(spy.mock.calls[1][0].submitSucceeded).toBe(false);
+ expect(spy.mock.calls[1][0].submitting).toBe(false);
+ expect(spy.mock.calls[1][0].valid).toBe(true);
+ expect(spy.mock.calls[1][0].validating).toBe(false);
+ expect(spy.mock.calls[1][0].values).toEqual({});
// change value
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
- expect(spy).toHaveBeenCalledTimes(3)
- hasFormApi(spy.mock.calls[2][0])
- expect(spy.mock.calls[2][0].dirty).toBe(true)
- expect(spy.mock.calls[2][0].errors).toEqual({})
- expect(spy.mock.calls[2][0].invalid).toBe(false)
- expect(spy.mock.calls[2][0].pristine).toBe(false)
- expect(spy.mock.calls[2][0].submitFailed).toBe(false)
- expect(spy.mock.calls[2][0].submitSucceeded).toBe(false)
- expect(spy.mock.calls[2][0].submitting).toBe(false)
- expect(spy.mock.calls[2][0].valid).toBe(true)
- expect(spy.mock.calls[2][0].validating).toBe(false)
- expect(spy.mock.calls[2][0].values).toEqual({ name: 'erikras' })
- })
+ expect(spy).toHaveBeenCalledTimes(3);
+ hasFormApi(spy.mock.calls[2][0]);
+ expect(spy.mock.calls[2][0].dirty).toBe(true);
+ expect(spy.mock.calls[2][0].errors).toEqual({});
+ expect(spy.mock.calls[2][0].invalid).toBe(false);
+ expect(spy.mock.calls[2][0].pristine).toBe(false);
+ expect(spy.mock.calls[2][0].submitFailed).toBe(false);
+ expect(spy.mock.calls[2][0].submitSucceeded).toBe(false);
+ expect(spy.mock.calls[2][0].submitting).toBe(false);
+ expect(spy.mock.calls[2][0].valid).toBe(true);
+ expect(spy.mock.calls[2][0].validating).toBe(false);
+ expect(spy.mock.calls[2][0].values).toEqual({ name: "erikras" });
+ });
- it('should NOT resubscribe if subscription changes', () => {
- const firstSubscription = { values: true, pristine: true }
- const secondSubscription = { dirty: true, submitting: true }
- const spy = jest.fn()
+ it("should NOT resubscribe if subscription changes", () => {
+ const firstSubscription = { values: true, pristine: true };
+ const secondSubscription = { dirty: true, submitting: true };
+ const spy = jest.fn();
const { getByText } = render(
- {useAlternateSubscription => (
+ {(useAlternateSubscription) => (
{() => (
@@ -119,7 +119,7 @@ describe('FormSpy', () => {
: firstSubscription
}
>
- {wrapWith(spy, props => (
+ {wrapWith(spy, (props) => (
))}
@@ -127,121 +127,121 @@ describe('FormSpy', () => {
)}
)}
-
- )
- expect(spy).toHaveBeenCalled()
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
// All forms without restricted subscriptions render twice at first because they
// need to update their validation and touched/modified/visited maps every time
// new fields are registered.
- expect(spy).toHaveBeenCalledTimes(2)
- hasFormApi(spy.mock.calls[0][0])
- expect(spy.mock.calls[0][0].dirty).toBeUndefined()
- expect(spy.mock.calls[0][0].errors).toBeUndefined()
- expect(spy.mock.calls[0][0].invalid).toBeUndefined()
- expect(spy.mock.calls[0][0].pristine).toBe(true)
- expect(spy.mock.calls[0][0].submitFailed).toBeUndefined()
- expect(spy.mock.calls[0][0].submitSucceeded).toBeUndefined()
- expect(spy.mock.calls[0][0].submitting).toBeUndefined()
- expect(spy.mock.calls[0][0].valid).toBeUndefined()
- expect(spy.mock.calls[0][0].validating).toBeUndefined()
+ expect(spy).toHaveBeenCalledTimes(2);
+ hasFormApi(spy.mock.calls[0][0]);
+ expect(spy.mock.calls[0][0].dirty).toBeUndefined();
+ expect(spy.mock.calls[0][0].errors).toBeUndefined();
+ expect(spy.mock.calls[0][0].invalid).toBeUndefined();
+ expect(spy.mock.calls[0][0].pristine).toBe(true);
+ expect(spy.mock.calls[0][0].submitFailed).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitSucceeded).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitting).toBeUndefined();
+ expect(spy.mock.calls[0][0].valid).toBeUndefined();
+ expect(spy.mock.calls[0][0].validating).toBeUndefined();
expect(spy.mock.calls[0][0].values).toEqual({
- dog: 'Odie',
- cat: 'Garfield'
- })
- hasFormApi(spy.mock.calls[1][0])
- expect(spy.mock.calls[1][0].dirty).toBeUndefined()
- expect(spy.mock.calls[1][0].errors).toBeUndefined()
- expect(spy.mock.calls[1][0].invalid).toBeUndefined()
- expect(spy.mock.calls[1][0].pristine).toBe(true)
- expect(spy.mock.calls[1][0].submitFailed).toBeUndefined()
- expect(spy.mock.calls[1][0].submitSucceeded).toBeUndefined()
- expect(spy.mock.calls[1][0].submitting).toBeUndefined()
- expect(spy.mock.calls[1][0].valid).toBeUndefined()
- expect(spy.mock.calls[1][0].validating).toBeUndefined()
+ dog: "Odie",
+ cat: "Garfield",
+ });
+ hasFormApi(spy.mock.calls[1][0]);
+ expect(spy.mock.calls[1][0].dirty).toBeUndefined();
+ expect(spy.mock.calls[1][0].errors).toBeUndefined();
+ expect(spy.mock.calls[1][0].invalid).toBeUndefined();
+ expect(spy.mock.calls[1][0].pristine).toBe(true);
+ expect(spy.mock.calls[1][0].submitFailed).toBeUndefined();
+ expect(spy.mock.calls[1][0].submitSucceeded).toBeUndefined();
+ expect(spy.mock.calls[1][0].submitting).toBeUndefined();
+ expect(spy.mock.calls[1][0].valid).toBeUndefined();
+ expect(spy.mock.calls[1][0].validating).toBeUndefined();
expect(spy.mock.calls[1][0].values).toEqual({
- dog: 'Odie',
- cat: 'Garfield'
- })
+ dog: "Odie",
+ cat: "Garfield",
+ });
- fireEvent.click(getByText('Toggle'))
+ fireEvent.click(getByText("Toggle"));
// one for new prop, and NOT again because no reregistering since v6
- expect(spy).toHaveBeenCalledTimes(3)
- })
+ expect(spy).toHaveBeenCalledTimes(3);
+ });
- it('should hear changes', () => {
- const spy = jest.fn()
+ it("should hear changes", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{() => (
- {wrapWith(spy, props => (
+ {wrapWith(spy, (props) => (
))}
)}
-
- )
- expect(spy).toHaveBeenCalled()
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
// All forms without restricted subscriptions render twice at first because they
// need to update their validation and touched/modified/visited maps every time
// new fields are registered.
- expect(spy).toHaveBeenCalledTimes(2)
- hasFormApi(spy.mock.calls[0][0])
- expect(spy.mock.calls[0][0].dirty).toBe(false)
- expect(spy.mock.calls[0][0].errors).toBeUndefined()
- expect(spy.mock.calls[0][0].invalid).toBeUndefined()
- expect(spy.mock.calls[0][0].pristine).toBeUndefined()
- expect(spy.mock.calls[0][0].submitFailed).toBeUndefined()
- expect(spy.mock.calls[0][0].submitSucceeded).toBeUndefined()
- expect(spy.mock.calls[0][0].submitting).toBeUndefined()
- expect(spy.mock.calls[0][0].valid).toBeUndefined()
- expect(spy.mock.calls[0][0].validating).toBeUndefined()
- expect(spy.mock.calls[0][0].values).toEqual({})
- hasFormApi(spy.mock.calls[1][0])
- expect(spy.mock.calls[1][0].dirty).toBe(false)
- expect(spy.mock.calls[1][0].errors).toBeUndefined()
- expect(spy.mock.calls[1][0].invalid).toBeUndefined()
- expect(spy.mock.calls[1][0].pristine).toBeUndefined()
- expect(spy.mock.calls[1][0].submitFailed).toBeUndefined()
- expect(spy.mock.calls[1][0].submitSucceeded).toBeUndefined()
- expect(spy.mock.calls[1][0].submitting).toBeUndefined()
- expect(spy.mock.calls[1][0].valid).toBeUndefined()
- expect(spy.mock.calls[1][0].validating).toBeUndefined()
- expect(spy.mock.calls[1][0].values).toEqual({})
+ expect(spy).toHaveBeenCalledTimes(2);
+ hasFormApi(spy.mock.calls[0][0]);
+ expect(spy.mock.calls[0][0].dirty).toBe(false);
+ expect(spy.mock.calls[0][0].errors).toBeUndefined();
+ expect(spy.mock.calls[0][0].invalid).toBeUndefined();
+ expect(spy.mock.calls[0][0].pristine).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitFailed).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitSucceeded).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitting).toBeUndefined();
+ expect(spy.mock.calls[0][0].valid).toBeUndefined();
+ expect(spy.mock.calls[0][0].validating).toBeUndefined();
+ expect(spy.mock.calls[0][0].values).toEqual({});
+ hasFormApi(spy.mock.calls[1][0]);
+ expect(spy.mock.calls[1][0].dirty).toBe(false);
+ expect(spy.mock.calls[1][0].errors).toBeUndefined();
+ expect(spy.mock.calls[1][0].invalid).toBeUndefined();
+ expect(spy.mock.calls[1][0].pristine).toBeUndefined();
+ expect(spy.mock.calls[1][0].submitFailed).toBeUndefined();
+ expect(spy.mock.calls[1][0].submitSucceeded).toBeUndefined();
+ expect(spy.mock.calls[1][0].submitting).toBeUndefined();
+ expect(spy.mock.calls[1][0].valid).toBeUndefined();
+ expect(spy.mock.calls[1][0].validating).toBeUndefined();
+ expect(spy.mock.calls[1][0].values).toEqual({});
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
- expect(spy).toHaveBeenCalledTimes(3)
- hasFormApi(spy.mock.calls[2][0])
- expect(spy.mock.calls[2][0].dirty).toBe(true)
- expect(spy.mock.calls[2][0].errors).toBeUndefined()
- expect(spy.mock.calls[2][0].invalid).toBeUndefined()
- expect(spy.mock.calls[2][0].pristine).toBeUndefined()
- expect(spy.mock.calls[2][0].submitFailed).toBeUndefined()
- expect(spy.mock.calls[2][0].submitSucceeded).toBeUndefined()
- expect(spy.mock.calls[2][0].submitting).toBeUndefined()
- expect(spy.mock.calls[2][0].valid).toBeUndefined()
- expect(spy.mock.calls[2][0].validating).toBeUndefined()
- expect(spy.mock.calls[2][0].values).toEqual({ name: 'erikras' })
- })
+ expect(spy).toHaveBeenCalledTimes(3);
+ hasFormApi(spy.mock.calls[2][0]);
+ expect(spy.mock.calls[2][0].dirty).toBe(true);
+ expect(spy.mock.calls[2][0].errors).toBeUndefined();
+ expect(spy.mock.calls[2][0].invalid).toBeUndefined();
+ expect(spy.mock.calls[2][0].pristine).toBeUndefined();
+ expect(spy.mock.calls[2][0].submitFailed).toBeUndefined();
+ expect(spy.mock.calls[2][0].submitSucceeded).toBeUndefined();
+ expect(spy.mock.calls[2][0].submitting).toBeUndefined();
+ expect(spy.mock.calls[2][0].valid).toBeUndefined();
+ expect(spy.mock.calls[2][0].validating).toBeUndefined();
+ expect(spy.mock.calls[2][0].values).toEqual({ name: "erikras" });
+ });
- it('should unsubscribe on unmount', () => {
+ it("should unsubscribe on unmount", () => {
// This is mainly here for code coverage. 🧐
- const spy = jest.fn()
+ const spy = jest.fn();
const { getByText } = render(
- {hidden => (
+ {(hidden) => (
{() => (
{!hidden && (
- {wrapWith(spy, props => (
+ {wrapWith(spy, (props) => (
))}
@@ -250,19 +250,19 @@ describe('FormSpy', () => {
)}
)}
-
- )
- expect(spy).toHaveBeenCalled()
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
// All forms without restricted subscriptions render twice at first because they
// need to update their validation and touched/modified/visited maps every time
// new fields are registered.
- expect(spy).toHaveBeenCalledTimes(2)
- fireEvent.click(getByText('Toggle'))
- expect(spy).toHaveBeenCalledTimes(2)
- })
+ expect(spy).toHaveBeenCalledTimes(2);
+ fireEvent.click(getByText("Toggle"));
+ expect(spy).toHaveBeenCalledTimes(2);
+ });
- it('should call onChange', () => {
- const spy = jest.fn()
+ it("should call onChange", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{() => (
@@ -274,30 +274,30 @@ describe('FormSpy', () => {
/>
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0].dirty).toBe(false)
- expect(spy.mock.calls[0][0].errors).toBeUndefined()
- expect(spy.mock.calls[0][0].invalid).toBeUndefined()
- expect(spy.mock.calls[0][0].pristine).toBeUndefined()
- expect(spy.mock.calls[0][0].submitFailed).toBeUndefined()
- expect(spy.mock.calls[0][0].submitSucceeded).toBeUndefined()
- expect(spy.mock.calls[0][0].submitting).toBeUndefined()
- expect(spy.mock.calls[0][0].valid).toBeUndefined()
- expect(spy.mock.calls[0][0].validating).toBeUndefined()
- expect(spy.mock.calls[0][0].values).toEqual({})
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0].dirty).toBe(false);
+ expect(spy.mock.calls[0][0].errors).toBeUndefined();
+ expect(spy.mock.calls[0][0].invalid).toBeUndefined();
+ expect(spy.mock.calls[0][0].pristine).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitFailed).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitSucceeded).toBeUndefined();
+ expect(spy.mock.calls[0][0].submitting).toBeUndefined();
+ expect(spy.mock.calls[0][0].valid).toBeUndefined();
+ expect(spy.mock.calls[0][0].validating).toBeUndefined();
+ expect(spy.mock.calls[0][0].values).toEqual({});
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0].dirty).toBe(true)
- expect(spy.mock.calls[1][0].values).toEqual({ name: 'erikras' })
- })
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0].dirty).toBe(true);
+ expect(spy.mock.calls[1][0].values).toEqual({ name: "erikras" });
+ });
- it('should not rerender when onChange changes, but SHOULD use latest onChange passed', () => {
- const spy = jest.fn()
+ it("should not rerender when onChange changes, but SHOULD use latest onChange passed", () => {
+ const spy = jest.fn();
const { getByTestId } = render(
{({ values }) => (
@@ -305,34 +305,34 @@ describe('FormSpy', () => {
{
- spy(values.name, formState.values.name)
+ onChange={(formState) => {
+ spy(values.name, formState.values.name);
}}
/>
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0]).toEqual([undefined, undefined])
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0]).toEqual([undefined, undefined]);
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1]).toEqual([undefined, 'erikras'])
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1]).toEqual([undefined, "erikras"]);
- fireEvent.change(getByTestId('name'), {
- target: { value: 'erikras rulez' }
- })
+ fireEvent.change(getByTestId("name"), {
+ target: { value: "erikras rulez" },
+ });
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2]).toEqual(['erikras', 'erikras rulez'])
- })
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2]).toEqual(["erikras", "erikras rulez"]);
+ });
- it('should not render with render prop when given onChange', () => {
- const spy = jest.fn()
- const renderSpy = jest.fn()
+ it("should not render with render prop when given onChange", () => {
+ const spy = jest.fn();
+ const renderSpy = jest.fn();
render(
{() => (
@@ -345,19 +345,19 @@ describe('FormSpy', () => {
/>
)}
-
- )
- expect(spy).toHaveBeenCalled()
- expect(spy).toHaveBeenCalledTimes(1)
- expect(renderSpy).not.toHaveBeenCalled()
- })
+ ,
+ );
+ expect(spy).toHaveBeenCalled();
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(renderSpy).not.toHaveBeenCalled();
+ });
- it('should ignore SyntheticEvents on form reset ', () => {
+ it("should ignore SyntheticEvents on form reset ", () => {
const { getByTestId, getByText } = render(
{() => (
@@ -371,23 +371,23 @@ describe('FormSpy', () => {
)}
-
- )
- expect(getByTestId('name').value).toBe('erikras')
- fireEvent.change(getByTestId('name'), {
- target: { value: 'erikrasmussen' }
- })
- expect(getByTestId('name').value).toBe('erikrasmussen')
- fireEvent.click(getByText('Reset'))
- expect(getByTestId('name').value).toBe('erikras')
- })
+ ,
+ );
+ expect(getByTestId("name").value).toBe("erikras");
+ fireEvent.change(getByTestId("name"), {
+ target: { value: "erikrasmussen" },
+ });
+ expect(getByTestId("name").value).toBe("erikrasmussen");
+ fireEvent.click(getByText("Reset"));
+ expect(getByTestId("name").value).toBe("erikras");
+ });
- it('should accept new initial values on form reset ', () => {
+ it("should accept new initial values on form reset ", () => {
const { getByTestId, getByText } = render(
{() => (
@@ -396,7 +396,7 @@ describe('FormSpy', () => {
{({ form }) => (
form.reset({ name: 'bob' })}
+ onClick={() => form.reset({ name: "bob" })}
>
Reset
@@ -404,14 +404,14 @@ describe('FormSpy', () => {
)}
-
- )
- expect(getByTestId('name').value).toBe('erikras')
- fireEvent.change(getByTestId('name'), {
- target: { value: 'erikrasmussen' }
- })
- expect(getByTestId('name').value).toBe('erikrasmussen')
- fireEvent.click(getByText('Reset'))
- expect(getByTestId('name').value).toBe('bob')
- })
-})
+ ,
+ );
+ expect(getByTestId("name").value).toBe("erikras");
+ fireEvent.change(getByTestId("name"), {
+ target: { value: "erikrasmussen" },
+ });
+ expect(getByTestId("name").value).toBe("erikrasmussen");
+ fireEvent.click(getByText("Reset"));
+ expect(getByTestId("name").value).toBe("bob");
+ });
+});
diff --git a/src/ReactFinalForm.js b/src/ReactFinalForm.js
index 2efdf31e..7c881be6 100644
--- a/src/ReactFinalForm.js
+++ b/src/ReactFinalForm.js
@@ -1,44 +1,44 @@
// @flow
-import * as React from 'react'
+import * as React from "react";
import {
createForm,
formSubscriptionItems,
- version as ffVersion
-} from 'final-form'
+ version as ffVersion,
+} from "final-form";
import type {
FormApi,
Config,
FormSubscription,
FormState,
FormValuesShape,
- Unsubscribe
-} from 'final-form'
-import type { FormProps as Props, SubmitEvent } from './types'
-import renderComponent from './renderComponent'
-import useWhenValueChanges from './useWhenValueChanges'
-import useConstant from './useConstant'
-import shallowEqual from './shallowEqual'
-import isSyntheticEvent from './isSyntheticEvent'
-import type { FormRenderProps } from './types.js.flow'
-import ReactFinalFormContext from './context'
-import useLatest from './useLatest'
-import { version } from '../package.json'
-import { addLazyFormState } from './getters'
+ Unsubscribe,
+} from "final-form";
+import type { FormProps as Props, SubmitEvent } from "./types";
+import renderComponent from "./renderComponent";
+import useWhenValueChanges from "./useWhenValueChanges";
+import useConstant from "./useConstant";
+import shallowEqual from "./shallowEqual";
+import isSyntheticEvent from "./isSyntheticEvent";
+import type { FormRenderProps } from "./types.js.flow";
+import ReactFinalFormContext from "./context";
+import useLatest from "./useLatest";
+import { version } from "../package.json";
+import { addLazyFormState } from "./getters";
-export { version }
+export { version };
const versions = {
- 'final-form': ffVersion,
- 'react-final-form': version
-}
+ "final-form": ffVersion,
+ "react-final-form": version,
+};
export const all: FormSubscription = formSubscriptionItems.reduce(
(result, key) => {
- result[key] = true
- return result
+ result[key] = true;
+ return result;
},
- {}
-)
+ {},
+);
function ReactFinalForm({
debug,
@@ -63,60 +63,60 @@ function ReactFinalForm({
mutators,
onSubmit,
validate,
- validateOnBlur
- }
+ validateOnBlur,
+ };
const form: FormApi = useConstant(() => {
- const f = alternateFormApi || createForm(config)
+ const f = alternateFormApi || createForm(config);
// pause validation until children register all fields on first render (unpaused in useEffect() below)
- f.pauseValidation()
- return f
- })
+ f.pauseValidation();
+ return f;
+ });
// synchronously register and unregister to query form state for our subscription on first render
const [state, setState] = React.useState>(
(): FormState => {
- let initialState: FormState = {}
- form.subscribe(state => {
- initialState = state
- }, subscription)()
- return initialState
- }
- )
+ let initialState: FormState = {};
+ form.subscribe((state) => {
+ initialState = state;
+ }, subscription)();
+ return initialState;
+ },
+ );
// save a copy of state that can break through the closure
// on the shallowEqual() line below.
- const stateRef = useLatest>(state)
+ const stateRef = useLatest>(state);
React.useEffect(() => {
// We have rendered, so all fields are now registered, so we can unpause validation
- form.isValidationPaused() && form.resumeValidation()
+ form.isValidationPaused() && form.resumeValidation();
const unsubscriptions: Unsubscribe[] = [
- form.subscribe(s => {
+ form.subscribe((s) => {
if (!shallowEqual(s, stateRef.current)) {
- setState(s)
+ setState(s);
}
}, subscription),
...(decorators
- ? decorators.map(decorator =>
+ ? decorators.map((decorator) =>
// this noop ternary is to appease the flow gods
// istanbul ignore next
- decorator(form)
+ decorator(form),
)
- : [])
- ]
+ : []),
+ ];
return () => {
- form.pauseValidation() // pause validation so we don't revalidate on every field deregistration
- unsubscriptions.reverse().forEach(unsubscribe => unsubscribe())
+ form.pauseValidation(); // pause validation so we don't revalidate on every field deregistration
+ unsubscriptions.reverse().forEach((unsubscribe) => unsubscribe());
// don't need to resume validation here; either unmounting, or will re-run this hook with new deps
- }
+ };
// eslint-disable-next-line react-hooks/exhaustive-deps
- }, decorators)
+ }, decorators);
// warn about decorator changes
// istanbul ignore next
- if (process.env.NODE_ENV !== 'production') {
+ if (process.env.NODE_ENV !== "production") {
// You're never supposed to use hooks inside a conditional, but in this
// case we can be certain that you're not going to be changing your
// NODE_ENV between renders, so this is safe.
@@ -126,84 +126,84 @@ function ReactFinalForm({
decorators,
() => {
console.error(
- 'Form decorators should not change from one render to the next as new values will be ignored'
- )
+ "Form decorators should not change from one render to the next as new values will be ignored",
+ );
},
- shallowEqual
- )
+ shallowEqual,
+ );
}
// allow updatable config
useWhenValueChanges(debug, () => {
- form.setConfig('debug', debug)
- })
+ form.setConfig("debug", debug);
+ });
useWhenValueChanges(destroyOnUnregister, () => {
- form.destroyOnUnregister = !!destroyOnUnregister
- })
+ form.destroyOnUnregister = !!destroyOnUnregister;
+ });
useWhenValueChanges(keepDirtyOnReinitialize, () => {
- form.setConfig('keepDirtyOnReinitialize', keepDirtyOnReinitialize)
- })
+ form.setConfig("keepDirtyOnReinitialize", keepDirtyOnReinitialize);
+ });
useWhenValueChanges(
initialValues,
() => {
- form.setConfig('initialValues', initialValues)
+ form.setConfig("initialValues", initialValues);
},
- initialValuesEqual || shallowEqual
- )
+ initialValuesEqual || shallowEqual,
+ );
useWhenValueChanges(mutators, () => {
- form.setConfig('mutators', mutators)
- })
+ form.setConfig("mutators", mutators);
+ });
useWhenValueChanges(onSubmit, () => {
- form.setConfig('onSubmit', onSubmit)
- })
+ form.setConfig("onSubmit", onSubmit);
+ });
useWhenValueChanges(validate, () => {
- form.setConfig('validate', validate)
- })
+ form.setConfig("validate", validate);
+ });
useWhenValueChanges(validateOnBlur, () => {
- form.setConfig('validateOnBlur', validateOnBlur)
- })
+ form.setConfig("validateOnBlur", validateOnBlur);
+ });
const handleSubmit = (event: ?SubmitEvent) => {
if (event) {
// sometimes not true, e.g. React Native
- if (typeof event.preventDefault === 'function') {
- event.preventDefault()
+ if (typeof event.preventDefault === "function") {
+ event.preventDefault();
}
- if (typeof event.stopPropagation === 'function') {
+ if (typeof event.stopPropagation === "function") {
// prevent any outer forms from receiving the event too
- event.stopPropagation()
+ event.stopPropagation();
}
}
- return form.submit()
- }
+ return form.submit();
+ };
const renderProps: FormRenderProps = {
form: {
...form,
- reset: eventOrValues => {
+ reset: (eventOrValues) => {
if (isSyntheticEvent(eventOrValues)) {
// it's a React SyntheticEvent, call reset with no arguments
- form.reset()
+ form.reset();
} else {
- form.reset(eventOrValues)
+ form.reset(eventOrValues);
}
- }
+ },
},
- handleSubmit
- }
- addLazyFormState(renderProps, state)
+ handleSubmit,
+ };
+ addLazyFormState(renderProps, state);
return React.createElement(
ReactFinalFormContext.Provider,
{ value: form },
renderComponent(
{
...rest,
- __versions: versions
+ __versions: versions,
},
renderProps,
- 'ReactFinalForm'
- )
- )
+ "ReactFinalForm",
+ ),
+ );
}
-export default ReactFinalForm
+export default ReactFinalForm;
diff --git a/src/ReactFinalForm.test.js b/src/ReactFinalForm.test.js
index 5e38ac37..ef429f65 100644
--- a/src/ReactFinalForm.test.js
+++ b/src/ReactFinalForm.test.js
@@ -1,93 +1,93 @@
-import React from 'react'
-import { render, fireEvent, cleanup, act } from '@testing-library/react'
-import '@testing-library/jest-dom/extend-expect'
-import deepEqual from 'fast-deep-equal'
-import { ErrorBoundary, Toggle, wrapWith } from './testUtils'
-import { createForm } from 'final-form'
-import { Form, Field, version, withTypes } from '.'
-
-const onSubmitMock = values => {}
-const timeout = ms => new Promise(resolve => setTimeout(resolve, ms))
+import React from "react";
+import { render, fireEvent, cleanup, act } from "@testing-library/react";
+import "@testing-library/jest-dom/extend-expect";
+import deepEqual from "fast-deep-equal";
+import { ErrorBoundary, Toggle, wrapWith } from "./testUtils";
+import { createForm } from "final-form";
+import { Form, Field, version, withTypes } from ".";
+
+const onSubmitMock = (values) => {};
+const timeout = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
async function sleep(ms) {
await act(async () => {
- await timeout(ms)
- })
+ await timeout(ms);
+ });
}
-describe('ReactFinalForm', () => {
- afterEach(cleanup)
+describe("ReactFinalForm", () => {
+ afterEach(cleanup);
- it('should export version', () => {
- expect(version).toBeDefined()
- })
+ it("should export version", () => {
+ expect(version).toBeDefined();
+ });
- it('should export withTypes', () => {
+ it("should export withTypes", () => {
// mostly for code coverage
- expect(withTypes).toBeDefined()
- expect(withTypes()).toBeDefined()
- })
+ expect(withTypes).toBeDefined();
+ expect(withTypes()).toBeDefined();
+ });
- it('should render with render function', () => {
+ it("should render with render function", () => {
const { getByTestId } = render(
}
- />
- )
- expect(getByTestId('myDiv')).toBeDefined()
- })
+ />,
+ );
+ expect(getByTestId("myDiv")).toBeDefined();
+ });
- it('should render with children render function', () => {
+ it("should render with children render function", () => {
const { getByTestId } = render(
- {() =>
}
- )
- expect(getByTestId('myDiv')).toBeDefined()
- })
-
- it('should print a warning with no render or children specified', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ {() =>
} ,
+ );
+ expect(getByTestId("myDiv")).toBeDefined();
+ });
+
+ it("should print a warning with no render or children specified", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
render(
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'Must specify either a render prop, a render function as children, or a component prop to ReactFinalForm'
- )
- console.error.mockRestore()
- })
-
- it('should print a warning with no onSubmit specified', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ "Must specify either a render prop, a render function as children, or a component prop to ReactFinalForm",
+ );
+ console.error.mockRestore();
+ });
+
+ it("should print a warning with no onSubmit specified", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
render(
} />
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'No onSubmit function specified'
- )
- console.error.mockRestore()
- })
+ "No onSubmit function specified",
+ );
+ console.error.mockRestore();
+ });
- it('should allow render to be a component', () => {
- const Component = () =>
+ it("should allow render to be a component", () => {
+ const Component = () =>
;
const { getByTestId } = render(
-
- )
- expect(getByTestId('myDiv')).toBeDefined()
- })
+ ,
+ );
+ expect(getByTestId("myDiv")).toBeDefined();
+ });
- it('should unsubscribe on unmount', () => {
+ it("should unsubscribe on unmount", () => {
// This is mainly here for code coverage. 🧐
const Container = () => {
- const [shown, setShown] = React.useState(true)
+ const [shown, setShown] = React.useState(true);
return (
{shown &&
} />}
@@ -95,17 +95,17 @@ describe('ReactFinalForm', () => {
Unmount
- )
- }
+ );
+ };
const { getByText } = render(
-
- )
- fireEvent.click(getByText('Unmount'))
- })
-
- it('should render with a field', () => {
- const formRender = jest.fn()
- const fieldRender = jest.fn()
+ ,
+ );
+ fireEvent.click(getByText("Unmount"));
+ });
+
+ it("should render with a field", () => {
+ const formRender = jest.fn();
+ const fieldRender = jest.fn();
const { getByTestId } = render(
{wrapWith(formRender, () => (
@@ -114,7 +114,7 @@ describe('ReactFinalForm', () => {
{wrapWith(fieldRender, ({ input, meta }) => (
<>
- {meta.active ? 'active' : 'inactive'}
+ {meta.active ? "active" : "inactive"}
>
@@ -122,45 +122,45 @@ describe('ReactFinalForm', () => {
))}
-
- )
- const firstName = getByTestId('firstName')
- const active = getByTestId('firstNameActive')
- expect(firstName).toBeDefined()
- expect(active).toBeDefined()
- expect(firstName.value).toBe('')
- expect(active).toHaveTextContent('inactive')
+ ,
+ );
+ const firstName = getByTestId("firstName");
+ const active = getByTestId("firstNameActive");
+ expect(firstName).toBeDefined();
+ expect(active).toBeDefined();
+ expect(firstName.value).toBe("");
+ expect(active).toHaveTextContent("inactive");
// All forms render twice at first because they need to update their validation
// and touched/modified/visited maps every time new fields are registered.
- expect(formRender).toHaveBeenCalledTimes(2)
- expect(fieldRender).toHaveBeenCalledTimes(2)
- fireEvent.focus(firstName)
- expect(formRender).toHaveBeenCalledTimes(3)
- expect(fieldRender).toHaveBeenCalledTimes(3)
- expect(active).toHaveTextContent('active')
- fireEvent.change(firstName, { target: { value: 'E' } })
- expect(formRender).toHaveBeenCalledTimes(4)
- expect(fieldRender).toHaveBeenCalledTimes(4)
- expect(firstName.value).toBe('E')
- fireEvent.change(firstName, { target: { value: 'Er' } })
- expect(formRender).toHaveBeenCalledTimes(5)
- expect(fieldRender).toHaveBeenCalledTimes(5)
- fireEvent.change(firstName, { target: { value: 'Eri' } })
- expect(formRender).toHaveBeenCalledTimes(6)
- expect(fieldRender).toHaveBeenCalledTimes(6)
- fireEvent.change(firstName, { target: { value: 'Erik' } })
- expect(formRender).toHaveBeenCalledTimes(7)
- expect(fieldRender).toHaveBeenCalledTimes(7)
- expect(active).toHaveTextContent('active')
- fireEvent.blur(firstName)
- expect(formRender).toHaveBeenCalledTimes(8)
- expect(fieldRender).toHaveBeenCalledTimes(8)
- expect(active).toHaveTextContent('inactive')
- })
-
- it('should render with a field with a limited subscription', () => {
- const formRender = jest.fn()
- const fieldRender = jest.fn()
+ expect(formRender).toHaveBeenCalledTimes(2);
+ expect(fieldRender).toHaveBeenCalledTimes(2);
+ fireEvent.focus(firstName);
+ expect(formRender).toHaveBeenCalledTimes(3);
+ expect(fieldRender).toHaveBeenCalledTimes(3);
+ expect(active).toHaveTextContent("active");
+ fireEvent.change(firstName, { target: { value: "E" } });
+ expect(formRender).toHaveBeenCalledTimes(4);
+ expect(fieldRender).toHaveBeenCalledTimes(4);
+ expect(firstName.value).toBe("E");
+ fireEvent.change(firstName, { target: { value: "Er" } });
+ expect(formRender).toHaveBeenCalledTimes(5);
+ expect(fieldRender).toHaveBeenCalledTimes(5);
+ fireEvent.change(firstName, { target: { value: "Eri" } });
+ expect(formRender).toHaveBeenCalledTimes(6);
+ expect(fieldRender).toHaveBeenCalledTimes(6);
+ fireEvent.change(firstName, { target: { value: "Erik" } });
+ expect(formRender).toHaveBeenCalledTimes(7);
+ expect(fieldRender).toHaveBeenCalledTimes(7);
+ expect(active).toHaveTextContent("active");
+ fireEvent.blur(firstName);
+ expect(formRender).toHaveBeenCalledTimes(8);
+ expect(fieldRender).toHaveBeenCalledTimes(8);
+ expect(active).toHaveTextContent("inactive");
+ });
+
+ it("should render with a field with a limited subscription", () => {
+ const formRender = jest.fn();
+ const fieldRender = jest.fn();
const { getByTestId } = render(
{wrapWith(formRender, () => (
@@ -169,7 +169,7 @@ describe('ReactFinalForm', () => {
{wrapWith(fieldRender, ({ input, meta }) => (
<>
- {meta.active ? 'active' : 'inactive'}
+ {meta.active ? "active" : "inactive"}
>
@@ -177,44 +177,44 @@ describe('ReactFinalForm', () => {
))}
-
- )
- const firstName = getByTestId('firstName')
- const active = getByTestId('firstNameActive')
- expect(firstName).toBeDefined()
- expect(active).toBeDefined()
- expect(firstName.value).toBe('')
- expect(active).toHaveTextContent('inactive')
- expect(formRender).toHaveBeenCalledTimes(1)
- expect(fieldRender).toHaveBeenCalledTimes(1)
- fireEvent.focus(firstName)
+ ,
+ );
+ const firstName = getByTestId("firstName");
+ const active = getByTestId("firstNameActive");
+ expect(firstName).toBeDefined();
+ expect(active).toBeDefined();
+ expect(firstName.value).toBe("");
+ expect(active).toHaveTextContent("inactive");
+ expect(formRender).toHaveBeenCalledTimes(1);
+ expect(fieldRender).toHaveBeenCalledTimes(1);
+ fireEvent.focus(firstName);
// not subscribing to active, so no rerender!
- expect(formRender).toHaveBeenCalledTimes(1)
- expect(fieldRender).toHaveBeenCalledTimes(1)
- expect(active).toHaveTextContent('inactive')
- fireEvent.change(firstName, { target: { value: 'E' } })
- expect(formRender).toHaveBeenCalledTimes(1)
- expect(fieldRender).toHaveBeenCalledTimes(2)
- expect(firstName.value).toBe('E')
- fireEvent.change(firstName, { target: { value: 'Er' } })
- expect(formRender).toHaveBeenCalledTimes(1)
- expect(fieldRender).toHaveBeenCalledTimes(3)
- fireEvent.change(firstName, { target: { value: 'Eri' } })
- expect(formRender).toHaveBeenCalledTimes(1)
- expect(fieldRender).toHaveBeenCalledTimes(4)
- fireEvent.change(firstName, { target: { value: 'Erik' } })
- expect(formRender).toHaveBeenCalledTimes(1)
- expect(fieldRender).toHaveBeenCalledTimes(5)
- expect(active).toHaveTextContent('inactive')
- fireEvent.blur(firstName)
+ expect(formRender).toHaveBeenCalledTimes(1);
+ expect(fieldRender).toHaveBeenCalledTimes(1);
+ expect(active).toHaveTextContent("inactive");
+ fireEvent.change(firstName, { target: { value: "E" } });
+ expect(formRender).toHaveBeenCalledTimes(1);
+ expect(fieldRender).toHaveBeenCalledTimes(2);
+ expect(firstName.value).toBe("E");
+ fireEvent.change(firstName, { target: { value: "Er" } });
+ expect(formRender).toHaveBeenCalledTimes(1);
+ expect(fieldRender).toHaveBeenCalledTimes(3);
+ fireEvent.change(firstName, { target: { value: "Eri" } });
+ expect(formRender).toHaveBeenCalledTimes(1);
+ expect(fieldRender).toHaveBeenCalledTimes(4);
+ fireEvent.change(firstName, { target: { value: "Erik" } });
+ expect(formRender).toHaveBeenCalledTimes(1);
+ expect(fieldRender).toHaveBeenCalledTimes(5);
+ expect(active).toHaveTextContent("inactive");
+ fireEvent.blur(firstName);
// no rerender
- expect(formRender).toHaveBeenCalledTimes(1)
- expect(fieldRender).toHaveBeenCalledTimes(5)
- expect(active).toHaveTextContent('inactive')
- })
+ expect(formRender).toHaveBeenCalledTimes(1);
+ expect(fieldRender).toHaveBeenCalledTimes(5);
+ expect(active).toHaveTextContent("inactive");
+ });
- it('should call onSubmit when form is submitted', () => {
- const onSubmit = jest.fn()
+ it("should call onSubmit when form is submitted", () => {
+ const onSubmit = jest.fn();
const { getByTestId, getByText } = render(
{({ handleSubmit }) => (
@@ -224,21 +224,21 @@ describe('ReactFinalForm', () => {
Submit
)}
-
- )
- fireEvent.change(getByTestId('firstName'), { target: { value: 'Erik' } })
- fireEvent.change(getByTestId('lastName'), {
- target: { value: 'Rasmussen' }
- })
- expect(onSubmit).not.toHaveBeenCalled()
- fireEvent.click(getByText('Submit'))
- expect(onSubmit).toHaveBeenCalled()
- expect(onSubmit).toHaveBeenCalledTimes(1)
+ ,
+ );
+ fireEvent.change(getByTestId("firstName"), { target: { value: "Erik" } });
+ fireEvent.change(getByTestId("lastName"), {
+ target: { value: "Rasmussen" },
+ });
+ expect(onSubmit).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Submit"));
+ expect(onSubmit).toHaveBeenCalled();
+ expect(onSubmit).toHaveBeenCalledTimes(1);
expect(onSubmit.mock.calls[0][0]).toEqual({
- firstName: 'Erik',
- lastName: 'Rasmussen'
- })
- })
+ firstName: "Erik",
+ lastName: "Rasmussen",
+ });
+ });
// it('should not throw when handleSubmit is called with no event', () => {
// const onSubmit = jest.fn()
@@ -272,16 +272,16 @@ describe('ReactFinalForm', () => {
// })
// })
- it('should reinitialize when initialValues prop changes', () => {
+ it("should reinitialize when initialValues prop changes", () => {
const initialValues = {
- name: 'Dr. Jekyll'
- }
+ name: "Dr. Jekyll",
+ };
const alternateInitialValues = {
- name: 'Mr. Hyde'
- }
+ name: "Mr. Hyde",
+ };
const { getByTestId, getByText } = render(
- {useAlternateInitialValues => (
+ {(useAlternateInitialValues) => (
{
)}
)}
-
- )
- expect(getByTestId('name').value).toBe('Dr. Jekyll')
- fireEvent.change(getByTestId('name'), { target: { value: 'Dr. Watson' } })
- expect(getByTestId('name').value).toBe('Dr. Watson')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('Mr. Hyde')
- })
+ ,
+ );
+ expect(getByTestId("name").value).toBe("Dr. Jekyll");
+ fireEvent.change(getByTestId("name"), { target: { value: "Dr. Watson" } });
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("Mr. Hyde");
+ });
it("should NOT reinitialize when initialValues prop doesn't change (shallowly) but rerendered", () => {
const { getByTestId, getByText } = render(
- {useAlternateInitialValues => (
+ {(useAlternateInitialValues) => (
{({ handleSubmit }) => (
@@ -321,29 +321,29 @@ describe('ReactFinalForm', () => {
)}
)}
-
- )
- expect(getByTestId('name').value).toBe('Dr. Jekyll')
- fireEvent.change(getByTestId('name'), { target: { value: 'Dr. Watson' } })
- expect(getByTestId('name').value).toBe('Dr. Watson')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('Dr. Watson')
- })
-
- it('should reinitialize when initialValues prop changes, deeply', () => {
+ ,
+ );
+ expect(getByTestId("name").value).toBe("Dr. Jekyll");
+ fireEvent.change(getByTestId("name"), { target: { value: "Dr. Watson" } });
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ });
+
+ it("should reinitialize when initialValues prop changes, deeply", () => {
const initialValues = {
professor: {
- name: 'Dr. Jekyll'
- }
- }
+ name: "Dr. Jekyll",
+ },
+ };
const alternateInitialValues = {
professor: {
- name: 'Mr. Hyde'
- }
- }
+ name: "Mr. Hyde",
+ },
+ };
const { getByTestId, getByText } = render(
- {useAlternateInitialValues => (
+ {(useAlternateInitialValues) => (
{
)}
)}
-
- )
- expect(getByTestId('name').value).toBe('Dr. Jekyll')
- fireEvent.change(getByTestId('name'), { target: { value: 'Dr. Watson' } })
- expect(getByTestId('name').value).toBe('Dr. Watson')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('Mr. Hyde')
- })
-
- it('should not reinitialize if initialValues prop is deep equal', () => {
+ ,
+ );
+ expect(getByTestId("name").value).toBe("Dr. Jekyll");
+ fireEvent.change(getByTestId("name"), { target: { value: "Dr. Watson" } });
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("Mr. Hyde");
+ });
+
+ it("should not reinitialize if initialValues prop is deep equal", () => {
const { getByTestId, getByText } = render(
- {useAlternateInitialValues => (
+ {(useAlternateInitialValues) => (
@@ -394,25 +394,25 @@ describe('ReactFinalForm', () => {
)}
)}
-
- )
- expect(getByTestId('name').value).toBe('Dr. Jekyll')
- fireEvent.change(getByTestId('name'), { target: { value: 'Dr. Watson' } })
- expect(getByTestId('name').value).toBe('Dr. Watson')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('Dr. Watson')
- })
-
- it('should respect keepDirtyOnReinitialize prop when initialValues prop changes', () => {
+ ,
+ );
+ expect(getByTestId("name").value).toBe("Dr. Jekyll");
+ fireEvent.change(getByTestId("name"), { target: { value: "Dr. Watson" } });
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ });
+
+ it("should respect keepDirtyOnReinitialize prop when initialValues prop changes", () => {
const initialValues = {
- name: 'Dr. Jekyll'
- }
+ name: "Dr. Jekyll",
+ };
const alternateInitialValues = {
- name: 'Mr. Hyde'
- }
+ name: "Mr. Hyde",
+ };
const { getByTestId, getByText } = render(
- {useAlternateInitialValues => (
+ {(useAlternateInitialValues) => (
{
)}
)}
-
- )
- expect(getByTestId('name').value).toBe('Dr. Jekyll')
- fireEvent.change(getByTestId('name'), { target: { value: 'Dr. Watson' } })
- expect(getByTestId('name').value).toBe('Dr. Watson')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('Dr. Watson')
- })
-
- it('should update when onSubmit changes', () => {
- const firstOnSubmit = jest.fn()
- const secondOnSubmit = jest.fn()
+ ,
+ );
+ expect(getByTestId("name").value).toBe("Dr. Jekyll");
+ fireEvent.change(getByTestId("name"), { target: { value: "Dr. Watson" } });
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("Dr. Watson");
+ });
+
+ it("should update when onSubmit changes", () => {
+ const firstOnSubmit = jest.fn();
+ const secondOnSubmit = jest.fn();
const { getByTestId, getByText } = render(
- {useAlternateOnSubmit => (
+ {(useAlternateOnSubmit) => (
@@ -453,40 +453,40 @@ describe('ReactFinalForm', () => {
)}
)}
-
- )
- fireEvent.change(getByTestId('name'), { target: { value: 'Erik' } })
- expect(firstOnSubmit).not.toHaveBeenCalled()
- fireEvent.click(getByText('Submit'))
- expect(firstOnSubmit).toHaveBeenCalled()
- expect(firstOnSubmit).toHaveBeenCalledTimes(1)
+ ,
+ );
+ fireEvent.change(getByTestId("name"), { target: { value: "Erik" } });
+ expect(firstOnSubmit).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Submit"));
+ expect(firstOnSubmit).toHaveBeenCalled();
+ expect(firstOnSubmit).toHaveBeenCalledTimes(1);
expect(firstOnSubmit.mock.calls[0][0]).toEqual({
- name: 'Erik'
- })
- expect(secondOnSubmit).not.toHaveBeenCalled()
- fireEvent.click(getByText('Toggle'))
- expect(firstOnSubmit).toHaveBeenCalledTimes(1)
- expect(secondOnSubmit).not.toHaveBeenCalled()
- fireEvent.click(getByText('Submit'))
- expect(firstOnSubmit).toHaveBeenCalledTimes(1)
- expect(secondOnSubmit).toHaveBeenCalled()
- expect(secondOnSubmit).toHaveBeenCalledTimes(1)
+ name: "Erik",
+ });
+ expect(secondOnSubmit).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Toggle"));
+ expect(firstOnSubmit).toHaveBeenCalledTimes(1);
+ expect(secondOnSubmit).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Submit"));
+ expect(firstOnSubmit).toHaveBeenCalledTimes(1);
+ expect(secondOnSubmit).toHaveBeenCalled();
+ expect(secondOnSubmit).toHaveBeenCalledTimes(1);
expect(secondOnSubmit.mock.calls[0][0]).toEqual({
- name: 'Erik'
- })
- })
-
- it('should warn if decorators change', () => {
- const decoratorA = form => () => {}
- const decoratorB = form => () => {}
- const decoratorC = form => () => {}
- const oldDecorators = [decoratorA, decoratorB]
- const newDecorators = [decoratorA, decoratorB, decoratorC]
- const errorSpy = jest.spyOn(console, 'error').mockImplementation(() => {})
+ name: "Erik",
+ });
+ });
+
+ it("should warn if decorators change", () => {
+ const decoratorA = (form) => () => {};
+ const decoratorB = (form) => () => {};
+ const decoratorC = (form) => () => {};
+ const oldDecorators = [decoratorA, decoratorB];
+ const newDecorators = [decoratorA, decoratorB, decoratorC];
+ const errorSpy = jest.spyOn(console, "error").mockImplementation(() => {});
const { getByText } = render(
- {useAlternateDecorators => (
+ {(useAlternateDecorators) => (
{
)}
)}
-
- )
- expect(errorSpy).not.toHaveBeenCalled()
- fireEvent.click(getByText('Toggle'))
- expect(errorSpy).toHaveBeenCalled()
+ ,
+ );
+ expect(errorSpy).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Toggle"));
+ expect(errorSpy).toHaveBeenCalled();
expect(errorSpy).toHaveBeenCalledWith(
- 'Form decorators should not change from one render to the next as new values will be ignored'
- )
- errorSpy.mockRestore()
- })
+ "Form decorators should not change from one render to the next as new values will be ignored",
+ );
+ errorSpy.mockRestore();
+ });
- it('should return a promise from handleSubmit when submission is async', () => {
+ it("should return a promise from handleSubmit when submission is async", () => {
const { getByTestId, getByText } = render(
{
- await timeout(2)
+ await timeout(2);
}}
>
{({ handleSubmit }) => (
{
- const promise = handleSubmit(event)
- expect(promise).toBeDefined()
- expect(typeof promise.then).toBe('function')
+ onSubmit={(event) => {
+ const promise = handleSubmit(event);
+ expect(promise).toBeDefined();
+ expect(typeof promise.then).toBe("function");
}}
>
Submit
)}
-
- )
- fireEvent.change(getByTestId('name'), { target: { value: 'Erik' } })
- fireEvent.click(getByText('Submit'))
- })
+ ,
+ );
+ fireEvent.change(getByTestId("name"), { target: { value: "Erik" } });
+ fireEvent.click(getByText("Submit"));
+ });
- it('should ignore SyntheticEvents on form reset ', () => {
+ it("should ignore SyntheticEvents on form reset ", () => {
const { getByTestId, getByText } = render(
-
+
{({ handleSubmit, form }) => (
@@ -547,47 +547,47 @@ describe('ReactFinalForm', () => {
)}
-
- )
- expect(getByTestId('name').value).toBe('John')
- fireEvent.change(getByTestId('name'), { target: { value: 'Paul' } })
- expect(getByTestId('name').value).toBe('Paul')
- fireEvent.click(getByText('Reset'))
- expect(getByTestId('name').value).toBe('John')
- })
-
- it('should accept new initial values on form reset ', () => {
+ ,
+ );
+ expect(getByTestId("name").value).toBe("John");
+ fireEvent.change(getByTestId("name"), { target: { value: "Paul" } });
+ expect(getByTestId("name").value).toBe("Paul");
+ fireEvent.click(getByText("Reset"));
+ expect(getByTestId("name").value).toBe("John");
+ });
+
+ it("should accept new initial values on form reset ", () => {
const { getByTestId, getByText } = render(
{({ form }) => (
- form.reset({ name: 'bob' })}>
+ form.reset({ name: "bob" })}>
Reset
)}
-
- )
- expect(getByTestId('name').value).toBe('erikras')
- fireEvent.change(getByTestId('name'), {
- target: { value: 'erikrasmussen' }
- })
- expect(getByTestId('name').value).toBe('erikrasmussen')
- fireEvent.click(getByText('Reset'))
- expect(getByTestId('name').value).toBe('bob')
- })
-
- it('should use decorators, and unsubscribe them on unmount', () => {
- const unsubscribe = jest.fn()
- const decorator = jest.fn(() => unsubscribe)
+ ,
+ );
+ expect(getByTestId("name").value).toBe("erikras");
+ fireEvent.change(getByTestId("name"), {
+ target: { value: "erikrasmussen" },
+ });
+ expect(getByTestId("name").value).toBe("erikrasmussen");
+ fireEvent.click(getByText("Reset"));
+ expect(getByTestId("name").value).toBe("bob");
+ });
+
+ it("should use decorators, and unsubscribe them on unmount", () => {
+ const unsubscribe = jest.fn();
+ const decorator = jest.fn(() => unsubscribe);
const { getByText } = render(
- {hideForm =>
+ {(hideForm) =>
!hideForm && (
{({ handleSubmit }) => (
@@ -598,35 +598,35 @@ describe('ReactFinalForm', () => {
)
}
-
- )
- expect(decorator).toHaveBeenCalled()
- expect(decorator).toHaveBeenCalledTimes(1)
- expect(unsubscribe).not.toHaveBeenCalled()
- fireEvent.click(getByText('Toggle'))
- expect(unsubscribe).toHaveBeenCalled()
- })
-
- it('should all record level validation function to change', () => {
- const simpleValidation = values => {
- const errors = {}
+ ,
+ );
+ expect(decorator).toHaveBeenCalled();
+ expect(decorator).toHaveBeenCalledTimes(1);
+ expect(unsubscribe).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Toggle"));
+ expect(unsubscribe).toHaveBeenCalled();
+ });
+
+ it("should all record level validation function to change", () => {
+ const simpleValidation = (values) => {
+ const errors = {};
if (!values.name) {
- errors.name = 'Required'
+ errors.name = "Required";
}
- return errors
- }
- const complexValidation = values => {
- const errors = {}
+ return errors;
+ };
+ const complexValidation = (values) => {
+ const errors = {};
if (!values.name) {
- errors.name = 'Required'
+ errors.name = "Required";
} else if (values.name.toUpperCase() !== values.name) {
- errors.name = 'SHOULD BE SHOUTING'
+ errors.name = "SHOULD BE SHOUTING";
}
- return errors
- }
+ return errors;
+ };
const { getByTestId, getByText } = render(
- {useComplexValidation => (
+ {(useComplexValidation) => (
{
)}
)}
-
- )
- expect(getByTestId('name').value).toBe('')
- expect(getByTestId('error')).toHaveTextContent('Required')
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(getByTestId('error')).toHaveTextContent('')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('error')).toHaveTextContent('SHOULD BE SHOUTING')
- fireEvent.change(getByTestId('name'), { target: { value: 'ERIKRAS' } })
- expect(getByTestId('error')).toHaveTextContent('')
- })
-
- it('should show form as invalid on second rerender if field-level validation errors are present', () => {
+ ,
+ );
+ expect(getByTestId("name").value).toBe("");
+ expect(getByTestId("error")).toHaveTextContent("Required");
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(getByTestId("error")).toHaveTextContent("");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("error")).toHaveTextContent("SHOULD BE SHOUTING");
+ fireEvent.change(getByTestId("name"), { target: { value: "ERIKRAS" } });
+ expect(getByTestId("error")).toHaveTextContent("");
+ });
+
+ it("should show form as invalid on second rerender if field-level validation errors are present", () => {
// Debugging https://github.com/final-form/react-final-form/issues/196
- const spy = jest.fn()
+ const spy = jest.fn();
render(
{({ handleSubmit, invalid }) => {
- spy(invalid)
+ spy(invalid);
return (
'Required'}
+ validate={() => "Required"}
/>
- )
+ );
}}
-
- )
+ ,
+ );
// On first render, we cannot know about any field level validation rules
// because none of the fields have yet had a chance to render and register.
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[0][0]).toBe(false)
- expect(spy.mock.calls[1][0]).toBe(true)
- })
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[0][0]).toBe(false);
+ expect(spy.mock.calls[1][0]).toBe(true);
+ });
- it('should work with server-side rendering', () => {
- const errorSpy = jest.spyOn(console, 'error').mockImplementation(() => {})
+ it("should work with server-side rendering", () => {
+ const errorSpy = jest.spyOn(console, "error").mockImplementation(() => {});
render(
{
)}
- />
- )
- expect(errorSpy).not.toHaveBeenCalled()
- errorSpy.mockRestore()
- })
+ />,
+ );
+ expect(errorSpy).not.toHaveBeenCalled();
+ errorSpy.mockRestore();
+ });
- it('should allow change to debug flag', () => {
+ it("should allow change to debug flag", () => {
// this is mostly for code coverage 😰
- const debugMock = jest.fn()
+ const debugMock = jest.fn();
const { getByText } = render(
- {debug => (
+ {(debug) => (
{({ handleSubmit }) => (
@@ -720,16 +720,16 @@ describe('ReactFinalForm', () => {
)}
)}
-
- )
- fireEvent.click(getByText('Toggle'))
- })
+ ,
+ );
+ fireEvent.click(getByText("Toggle"));
+ });
- it('should allow change to destroyOnUnregister flag', () => {
+ it("should allow change to destroyOnUnregister flag", () => {
// this is mostly for code coverage 😰
const { getByText } = render(
- {destroyOnUnregister => (
+ {(destroyOnUnregister) => (
{
)}
)}
-
- )
- fireEvent.click(getByText('Toggle'))
- })
+ ,
+ );
+ fireEvent.click(getByText("Toggle"));
+ });
- it('should allow change to keepDirtyOnReinitialize flag', () => {
+ it("should allow change to keepDirtyOnReinitialize flag", () => {
// this is mostly for code coverage 😰
const { getByText } = render(
- {keepDirtyOnReinitialize => (
+ {(keepDirtyOnReinitialize) => (
{
)}
)}
-
- )
- fireEvent.click(getByText('Toggle'))
- })
+ ,
+ );
+ fireEvent.click(getByText("Toggle"));
+ });
- it('should allow change to validateOnBlur flag', () => {
+ it("should allow change to validateOnBlur flag", () => {
// this is mostly for code coverage 😰
const { getByText } = render(
- {validateOnBlur => (
+ {(validateOnBlur) => (
{({ handleSubmit }) => (
@@ -780,23 +780,23 @@ describe('ReactFinalForm', () => {
)}
)}
-
- )
- fireEvent.click(getByText('Toggle'))
- })
+ ,
+ );
+ fireEvent.click(getByText("Toggle"));
+ });
- it('should allow change to mutators', () => {
+ it("should allow change to mutators", () => {
// this is mostly for code coverage 😰
- const oldMutators = undefined
+ const oldMutators = undefined;
const newMutators = {
clearField: ([name], state, { changeValue }) => {
- changeValue(state, name, () => undefined)
- }
- }
- const spy = jest.fn()
+ changeValue(state, name, () => undefined);
+ },
+ };
+ const spy = jest.fn();
const { getByTestId, getByText } = render(
- {swapMutators => (
+ {(swapMutators) => (
{
form.mutators.clearField('name')}
+ onClick={() => form.mutators.clearField("name")}
>
Clear Name
@@ -814,105 +814,105 @@ describe('ReactFinalForm', () => {
))}
)}
-
- )
- expect(getByTestId('name').value).toBe('')
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(getByTestId('name').value).toBe('erikras')
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('name').value).toBe('erikras')
- fireEvent.click(getByText('Clear Name'))
- expect(getByTestId('name').value).toBe('')
- })
+ ,
+ );
+ expect(getByTestId("name").value).toBe("");
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(getByTestId("name").value).toBe("erikras");
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("name").value).toBe("erikras");
+ fireEvent.click(getByText("Clear Name"));
+ expect(getByTestId("name").value).toBe("");
+ });
it("should allow handleSubmit to be called with an object that's not an event", () => {
- const onSubmitMock = jest.fn()
+ const onSubmitMock = jest.fn();
const { getByText } = render(
-
+
{({ handleSubmit }) => (
{
- handleSubmit({})
- event.preventDefault() // so react-testing-library doesn't freak out
+ onSubmit={(event) => {
+ handleSubmit({});
+ event.preventDefault(); // so react-testing-library doesn't freak out
}}
>
Submit
)}
-
- )
- expect(onSubmitMock).not.toHaveBeenCalled()
- fireEvent.click(getByText('Submit'))
- expect(onSubmitMock).toHaveBeenCalled()
- expect(onSubmitMock).toHaveBeenCalledTimes(1)
- expect(onSubmitMock.mock.calls[0][0]).toEqual({ name: 'erikras' })
- })
-
- it('should allow handleSubmit to be called with no parameters', () => {
- const onSubmitMock = jest.fn()
+ ,
+ );
+ expect(onSubmitMock).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Submit"));
+ expect(onSubmitMock).toHaveBeenCalled();
+ expect(onSubmitMock).toHaveBeenCalledTimes(1);
+ expect(onSubmitMock.mock.calls[0][0]).toEqual({ name: "erikras" });
+ });
+
+ it("should allow handleSubmit to be called with no parameters", () => {
+ const onSubmitMock = jest.fn();
const { getByText } = render(
-
+
{({ handleSubmit }) => (
{
- handleSubmit()
- event.preventDefault() // so react-testing-library doesn't freak out
+ onSubmit={(event) => {
+ handleSubmit();
+ event.preventDefault(); // so react-testing-library doesn't freak out
}}
>
Submit
)}
-
- )
- expect(onSubmitMock).not.toHaveBeenCalled()
- fireEvent.click(getByText('Submit'))
- expect(onSubmitMock).toHaveBeenCalled()
- expect(onSubmitMock).toHaveBeenCalledTimes(1)
- expect(onSubmitMock.mock.calls[0][0]).toEqual({ name: 'erikras' })
- })
-
- it('should set submitting back to false after submit', async () => {
+ ,
+ );
+ expect(onSubmitMock).not.toHaveBeenCalled();
+ fireEvent.click(getByText("Submit"));
+ expect(onSubmitMock).toHaveBeenCalled();
+ expect(onSubmitMock).toHaveBeenCalledTimes(1);
+ expect(onSubmitMock.mock.calls[0][0]).toEqual({ name: "erikras" });
+ });
+
+ it("should set submitting back to false after submit", async () => {
const onSubmit = jest.fn(async () => {
- await timeout(1)
- })
- const recordSubmitting = jest.fn()
+ await timeout(1);
+ });
+ const recordSubmitting = jest.fn();
const { getByText } = render(
{({ handleSubmit, submitting }) => {
- recordSubmitting(submitting)
+ recordSubmitting(submitting);
return (
Submit
- )
+ );
}}
-
- )
- expect(onSubmit).not.toHaveBeenCalled()
- expect(recordSubmitting).toHaveBeenCalled()
- expect(recordSubmitting).toHaveBeenCalledTimes(1)
- expect(recordSubmitting.mock.calls[0][0]).toBe(false)
-
- fireEvent.click(getByText('Submit'))
-
- expect(onSubmit).toHaveBeenCalled()
- expect(onSubmit).toHaveBeenCalledTimes(1)
- expect(recordSubmitting).toHaveBeenCalledTimes(2)
- expect(recordSubmitting.mock.calls[1][0]).toBe(true)
-
- await sleep(5)
-
- expect(recordSubmitting).toHaveBeenCalledTimes(3)
- expect(recordSubmitting.mock.calls[2][0]).toBe(false)
- })
-
- it('should allow an alternative form api to be passed in', () => {
- const onSubmit = jest.fn()
- const form = createForm({ onSubmit: onSubmit })
- const formMock = jest.spyOn(form, 'registerField')
+ ,
+ );
+ expect(onSubmit).not.toHaveBeenCalled();
+ expect(recordSubmitting).toHaveBeenCalled();
+ expect(recordSubmitting).toHaveBeenCalledTimes(1);
+ expect(recordSubmitting.mock.calls[0][0]).toBe(false);
+
+ fireEvent.click(getByText("Submit"));
+
+ expect(onSubmit).toHaveBeenCalled();
+ expect(onSubmit).toHaveBeenCalledTimes(1);
+ expect(recordSubmitting).toHaveBeenCalledTimes(2);
+ expect(recordSubmitting.mock.calls[1][0]).toBe(true);
+
+ await sleep(5);
+
+ expect(recordSubmitting).toHaveBeenCalledTimes(3);
+ expect(recordSubmitting.mock.calls[2][0]).toBe(false);
+ });
+
+ it("should allow an alternative form api to be passed in", () => {
+ const onSubmit = jest.fn();
+ const form = createForm({ onSubmit: onSubmit });
+ const formMock = jest.spyOn(form, "registerField");
render(
{({ handleSubmit }) => (
@@ -920,24 +920,24 @@ describe('ReactFinalForm', () => {
)}
-
- )
- expect(formMock).toHaveBeenCalled()
+ ,
+ );
+ expect(formMock).toHaveBeenCalled();
// called once on first render to get initial state, and then again to subscribe
- expect(formMock).toHaveBeenCalledTimes(2)
- expect(formMock.mock.calls[0][0]).toBe('name')
- expect(formMock.mock.calls[0][2].active).toBe(true) // default subscription
- expect(formMock.mock.calls[1][0]).toBe('name')
- expect(formMock.mock.calls[1][2].active).toBe(true) // default subscription
- })
-
- it('should not destroy on unregister on initial unregister', () => {
+ expect(formMock).toHaveBeenCalledTimes(2);
+ expect(formMock.mock.calls[0][0]).toBe("name");
+ expect(formMock.mock.calls[0][2].active).toBe(true); // default subscription
+ expect(formMock.mock.calls[1][0]).toBe("name");
+ expect(formMock.mock.calls[1][2].active).toBe(true); // default subscription
+ });
+
+ it("should not destroy on unregister on initial unregister", () => {
// https://github.com/final-form/react-final-form/issues/523
const { getByTestId } = render(
{({ handleSubmit }) => (
@@ -945,28 +945,28 @@ describe('ReactFinalForm', () => {
)}
-
- )
+ ,
+ );
- expect(getByTestId('name')).toBeDefined()
- expect(getByTestId('name').value).toBe('erikras')
- fireEvent.focus(getByTestId('name'))
- expect(getByTestId('name').value).toBe('erikras')
- })
+ expect(getByTestId("name")).toBeDefined();
+ expect(getByTestId("name").value).toBe("erikras");
+ fireEvent.focus(getByTestId("name"));
+ expect(getByTestId("name").value).toBe("erikras");
+ });
- it('should not destroy on unregister on initial register/unregister of new field', () => {
+ it("should not destroy on unregister on initial register/unregister of new field", () => {
// https://github.com/final-form/react-final-form/issues/523
const { getByTestId, queryByTestId, getByText } = render(
{({ handleSubmit }) => (
- {showPassword =>
+ {(showPassword) =>
showPassword && (
{
)}
-
- )
-
- expect(getByTestId('name')).toBeDefined()
- expect(getByTestId('name').value).toBe('erikras')
- fireEvent.focus(getByTestId('name'))
- expect(getByTestId('name').value).toBe('erikras')
- expect(queryByTestId('password')).toBe(null)
- fireEvent.click(getByText('Toggle'))
- expect(getByTestId('password').value).toBe('f1nal-f0rm-RULEZ')
- fireEvent.focus(getByTestId('password'))
- expect(getByTestId('password').value).toBe('f1nal-f0rm-RULEZ')
- })
-})
+ ,
+ );
+
+ expect(getByTestId("name")).toBeDefined();
+ expect(getByTestId("name").value).toBe("erikras");
+ fireEvent.focus(getByTestId("name"));
+ expect(getByTestId("name").value).toBe("erikras");
+ expect(queryByTestId("password")).toBe(null);
+ fireEvent.click(getByText("Toggle"));
+ expect(getByTestId("password").value).toBe("f1nal-f0rm-RULEZ");
+ fireEvent.focus(getByTestId("password"));
+ expect(getByTestId("password").value).toBe("f1nal-f0rm-RULEZ");
+ });
+});
diff --git a/src/context.js b/src/context.js
index 896f5733..cd0ab9d4 100644
--- a/src/context.js
+++ b/src/context.js
@@ -1,5 +1,5 @@
// @flow
-import * as React from 'react'
-import type { FormApi } from 'final-form'
+import * as React from "react";
+import type { FormApi } from "final-form";
-export default React.createContext>()
+export default React.createContext>();
diff --git a/src/getValue.js b/src/getValue.js
index 6de52120..7f779795 100644
--- a/src/getValue.js
+++ b/src/getValue.js
@@ -1,67 +1,69 @@
// @flow
-const getSelectedValues = options => {
- const result = []
+const getSelectedValues = (options) => {
+ const result = [];
if (options) {
for (let index = 0; index < options.length; index++) {
- const option = options[index]
+ const option = options[index];
if (option.selected) {
- result.push(option.value)
+ result.push(option.value);
}
}
}
- return result
-}
+ return result;
+};
const getValue = (
event: SyntheticInputEvent<*>,
currentValue: any,
valueProp: any,
- isReactNative: boolean
+ isReactNative: boolean,
) => {
if (
!isReactNative &&
event.nativeEvent &&
(event.nativeEvent: Object).text !== undefined
) {
- return (event.nativeEvent: Object).text
+ return (event.nativeEvent: Object).text;
}
if (isReactNative && event.nativeEvent) {
- return (event.nativeEvent: any).text
+ return (event.nativeEvent: any).text;
}
- const detypedEvent: any = event
- const { target: { type, value, checked } } = detypedEvent
+ const detypedEvent: any = event;
+ const {
+ target: { type, value, checked },
+ } = detypedEvent;
switch (type) {
- case 'checkbox':
+ case "checkbox":
if (valueProp !== undefined) {
// we are maintaining an array, not just a boolean
if (checked) {
// add value to current array value
return Array.isArray(currentValue)
? currentValue.concat(valueProp)
- : [valueProp]
+ : [valueProp];
} else {
// remove value from current array value
if (!Array.isArray(currentValue)) {
- return currentValue
+ return currentValue;
}
- const index = currentValue.indexOf(valueProp)
+ const index = currentValue.indexOf(valueProp);
if (index < 0) {
- return currentValue
+ return currentValue;
} else {
return currentValue
.slice(0, index)
- .concat(currentValue.slice(index + 1))
+ .concat(currentValue.slice(index + 1));
}
}
} else {
// it's just a boolean
- return !!checked
+ return !!checked;
}
- case 'select-multiple':
- return getSelectedValues((event.target: any).options)
+ case "select-multiple":
+ return getSelectedValues((event.target: any).options);
default:
- return value
+ return value;
}
-}
+};
-export default getValue
+export default getValue;
diff --git a/src/getValue.test.js b/src/getValue.test.js
index 6809d762..80a21377 100644
--- a/src/getValue.test.js
+++ b/src/getValue.test.js
@@ -1,286 +1,286 @@
-import { noop } from 'lodash'
-import getValue from './getValue'
+import { noop } from "lodash";
+import getValue from "./getValue";
-describe('getValue', () => {
- it('should return event.nativeEvent.text if defined and not react-native', () => {
+describe("getValue", () => {
+ it("should return event.nativeEvent.text if defined and not react-native", () => {
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
nativeEvent: {
- text: 'foo'
- }
+ text: "foo",
+ },
},
undefined,
undefined,
- false
- )
- ).toBe('foo')
- })
+ false,
+ ),
+ ).toBe("foo");
+ });
- it('should return event.nativeEvent.text if react-native', () => {
+ it("should return event.nativeEvent.text if react-native", () => {
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
nativeEvent: {
- text: 'foo'
- }
+ text: "foo",
+ },
},
undefined,
undefined,
- true
- )
- ).toBe('foo')
+ true,
+ ),
+ ).toBe("foo");
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
nativeEvent: {
- text: undefined
- }
+ text: undefined,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(undefined)
+ true,
+ ),
+ ).toBe(undefined);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
nativeEvent: {
- text: null
- }
+ text: null,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(null)
- })
+ true,
+ ),
+ ).toBe(null);
+ });
- it('should return event.target.checked if checkbox with no value parameter', () => {
+ it("should return event.target.checked if checkbox with no value parameter", () => {
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: true
- }
+ type: "checkbox",
+ checked: true,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(true)
+ true,
+ ),
+ ).toBe(true);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: true
- }
+ type: "checkbox",
+ checked: true,
+ },
},
undefined,
undefined,
- false
- )
- ).toBe(true)
+ false,
+ ),
+ ).toBe(true);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: undefined
- }
+ type: "checkbox",
+ checked: undefined,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(false)
+ true,
+ ),
+ ).toBe(false);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: undefined
- }
+ type: "checkbox",
+ checked: undefined,
+ },
},
undefined,
undefined,
- false
- )
- ).toBe(false)
- })
+ false,
+ ),
+ ).toBe(false);
+ });
- it('should return add or remove the value to an array if checkbox with value parameter', () => {
+ it("should return add or remove the value to an array if checkbox with value parameter", () => {
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: true
- }
+ type: "checkbox",
+ checked: true,
+ },
},
undefined,
- 'foo',
- false
- )
- ).toEqual(['foo'])
+ "foo",
+ false,
+ ),
+ ).toEqual(["foo"]);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: true
- }
+ type: "checkbox",
+ checked: true,
+ },
},
- ['A', 'B'],
- 'C',
- false
- )
- ).toEqual(['A', 'B', 'C'])
+ ["A", "B"],
+ "C",
+ false,
+ ),
+ ).toEqual(["A", "B", "C"]);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: false
- }
+ type: "checkbox",
+ checked: false,
+ },
},
- ['foo'],
- 'foo',
- false
- )
- ).toEqual([])
+ ["foo"],
+ "foo",
+ false,
+ ),
+ ).toEqual([]);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: false
- }
+ type: "checkbox",
+ checked: false,
+ },
},
- ['A', 'B', 'C'],
- 'B',
- false
- )
- ).toEqual(['A', 'C'])
+ ["A", "B", "C"],
+ "B",
+ false,
+ ),
+ ).toEqual(["A", "C"]);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: false
- }
+ type: "checkbox",
+ checked: false,
+ },
},
- ['A', 'B', 'C'],
- 'F',
- false
- )
- ).toEqual(['A', 'B', 'C'])
+ ["A", "B", "C"],
+ "F",
+ false,
+ ),
+ ).toEqual(["A", "B", "C"]);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'checkbox',
- checked: false
- }
+ type: "checkbox",
+ checked: false,
+ },
},
undefined,
- 'F',
- false
- )
- ).toBeUndefined()
- })
+ "F",
+ false,
+ ),
+ ).toBeUndefined();
+ });
- it('should return a number type for numeric inputs, when a value is set', () => {
+ it("should return a number type for numeric inputs, when a value is set", () => {
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'number',
- value: '3.1415'
- }
+ type: "number",
+ value: "3.1415",
+ },
},
undefined,
undefined,
- true
- )
- ).toBe('3.1415')
+ true,
+ ),
+ ).toBe("3.1415");
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'range',
- value: '2.71828'
- }
+ type: "range",
+ value: "2.71828",
+ },
},
undefined,
undefined,
- true
- )
- ).toBe('2.71828')
+ true,
+ ),
+ ).toBe("2.71828");
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'number',
- value: '3'
- }
+ type: "number",
+ value: "3",
+ },
},
undefined,
undefined,
- false
- )
- ).toBe('3')
+ false,
+ ),
+ ).toBe("3");
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'range',
- value: '3.1415'
- }
+ type: "range",
+ value: "3.1415",
+ },
},
undefined,
undefined,
- false
- )
- ).toBe('3.1415')
+ false,
+ ),
+ ).toBe("3.1415");
expect(
getValue(
@@ -288,240 +288,240 @@ describe('getValue', () => {
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'range',
- value: ''
- }
+ type: "range",
+ value: "",
+ },
},
undefined,
undefined,
- false
- )
- ).toBe('')
- })
+ false,
+ ),
+ ).toBe("");
+ });
- it('should return selected options if is a multiselect', () => {
+ it("should return selected options if is a multiselect", () => {
const options = [
- { selected: true, value: 'foo' },
- { selected: true, value: 'bar' },
- { selected: false, value: 'baz' }
- ]
+ { selected: true, value: "foo" },
+ { selected: true, value: "bar" },
+ { selected: false, value: "baz" },
+ ];
const expected = options
- .filter(option => option.selected)
- .map(option => option.value)
+ .filter((option) => option.selected)
+ .map((option) => option.value);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'select-multiple',
- options
- }
+ type: "select-multiple",
+ options,
+ },
},
undefined,
undefined,
- true
- )
- ).toEqual(expected)
+ true,
+ ),
+ ).toEqual(expected);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'select-multiple'
- }
+ type: "select-multiple",
+ },
},
undefined,
undefined,
- false
- )
- ).toEqual([]) // no options specified
+ false,
+ ),
+ ).toEqual([]); // no options specified
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- type: 'select-multiple',
- options
- }
+ type: "select-multiple",
+ options,
+ },
},
undefined,
undefined,
- false
- )
- ).toEqual(expected)
- })
+ false,
+ ),
+ ).toEqual(expected);
+ });
- it('should return event.target.value if not file or checkbox', () => {
+ it("should return event.target.value if not file or checkbox", () => {
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: undefined
- }
+ value: undefined,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(undefined)
+ true,
+ ),
+ ).toBe(undefined);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: undefined
- }
+ value: undefined,
+ },
},
undefined,
undefined,
- false
- )
- ).toBe(undefined)
+ false,
+ ),
+ ).toBe(undefined);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: null
- }
+ value: null,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(null)
+ true,
+ ),
+ ).toBe(null);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: null
- }
+ value: null,
+ },
},
undefined,
undefined,
- false
- )
- ).toBe(null)
+ false,
+ ),
+ ).toBe(null);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: true
- }
+ value: true,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(true)
+ true,
+ ),
+ ).toBe(true);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: true
- }
+ value: true,
+ },
},
undefined,
undefined,
- false
- )
- ).toBe(true)
+ false,
+ ),
+ ).toBe(true);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: false
- }
+ value: false,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(false)
+ true,
+ ),
+ ).toBe(false);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: false
- }
+ value: false,
+ },
},
undefined,
undefined,
- false
- )
- ).toBe(false)
+ false,
+ ),
+ ).toBe(false);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: 42
- }
+ value: 42,
+ },
},
undefined,
undefined,
- true
- )
- ).toBe(42)
+ true,
+ ),
+ ).toBe(42);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: 42
- }
+ value: 42,
+ },
},
undefined,
undefined,
- false
- )
- ).toBe(42)
+ false,
+ ),
+ ).toBe(42);
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: 'foo'
- }
+ value: "foo",
+ },
},
undefined,
undefined,
- true
- )
- ).toBe('foo')
+ true,
+ ),
+ ).toBe("foo");
expect(
getValue(
{
preventDefault: noop,
stopPropagation: noop,
target: {
- value: 'foo'
- }
+ value: "foo",
+ },
},
undefined,
undefined,
- false
- )
- ).toBe('foo')
- })
-})
+ false,
+ ),
+ ).toBe("foo");
+ });
+});
diff --git a/src/getters.js b/src/getters.js
index 9f81c2d0..16cf55ba 100644
--- a/src/getters.js
+++ b/src/getters.js
@@ -1,61 +1,61 @@
-import type { FormState, FieldState } from 'final-form'
+import type { FormState, FieldState } from "final-form";
const addLazyState = (dest: Object, state: Object, keys: string[]): void => {
- keys.forEach(key => {
+ keys.forEach((key) => {
Object.defineProperty(dest, key, {
get: () => state[key],
- enumerable: true
- })
- })
-}
+ enumerable: true,
+ });
+ });
+};
export const addLazyFormState = (dest: Object, state: FormState): void =>
addLazyState(dest, state, [
- 'active',
- 'dirty',
- 'dirtyFields',
- 'dirtySinceLastSubmit',
- 'dirtyFieldsSinceLastSubmit',
- 'error',
- 'errors',
- 'hasSubmitErrors',
- 'hasValidationErrors',
- 'initialValues',
- 'invalid',
- 'modified',
- 'modifiedSinceLastSubmit',
- 'pristine',
- 'submitError',
- 'submitErrors',
- 'submitFailed',
- 'submitSucceeded',
- 'submitting',
- 'touched',
- 'valid',
- 'validating',
- 'values',
- 'visited'
- ])
+ "active",
+ "dirty",
+ "dirtyFields",
+ "dirtySinceLastSubmit",
+ "dirtyFieldsSinceLastSubmit",
+ "error",
+ "errors",
+ "hasSubmitErrors",
+ "hasValidationErrors",
+ "initialValues",
+ "invalid",
+ "modified",
+ "modifiedSinceLastSubmit",
+ "pristine",
+ "submitError",
+ "submitErrors",
+ "submitFailed",
+ "submitSucceeded",
+ "submitting",
+ "touched",
+ "valid",
+ "validating",
+ "values",
+ "visited",
+ ]);
export const addLazyFieldMetaState = (dest: Object, state: FieldState): void =>
addLazyState(dest, state, [
- 'active',
- 'data',
- 'dirty',
- 'dirtySinceLastSubmit',
- 'error',
- 'initial',
- 'invalid',
- 'length',
- 'modified',
- 'modifiedSinceLastSubmit',
- 'pristine',
- 'submitError',
- 'submitFailed',
- 'submitSucceeded',
- 'submitting',
- 'touched',
- 'valid',
- 'validating',
- 'visited'
- ])
+ "active",
+ "data",
+ "dirty",
+ "dirtySinceLastSubmit",
+ "error",
+ "initial",
+ "invalid",
+ "length",
+ "modified",
+ "modifiedSinceLastSubmit",
+ "pristine",
+ "submitError",
+ "submitFailed",
+ "submitSucceeded",
+ "submitting",
+ "touched",
+ "valid",
+ "validating",
+ "visited",
+ ]);
diff --git a/src/index.js b/src/index.js
index eac6a858..11511b77 100644
--- a/src/index.js
+++ b/src/index.js
@@ -1,12 +1,12 @@
// @flow
-import Form from './ReactFinalForm'
-import FormSpy from './FormSpy'
-export { default as Field } from './Field'
-export { default as Form, version } from './ReactFinalForm'
-export { default as FormSpy } from './FormSpy'
-export { default as useField } from './useField'
-export { default as useFormState } from './useFormState'
-export { default as useForm } from './useForm'
+import Form from "./ReactFinalForm";
+import FormSpy from "./FormSpy";
+export { default as Field } from "./Field";
+export { default as Form, version } from "./ReactFinalForm";
+export { default as FormSpy } from "./FormSpy";
+export { default as useField } from "./useField";
+export { default as useFormState } from "./useFormState";
+export { default as useForm } from "./useForm";
export function withTypes() {
- return { Form, FormSpy }
+ return { Form, FormSpy };
}
diff --git a/src/index.js.flow b/src/index.js.flow
index 1b89f370..9e29bab1 100644
--- a/src/index.js.flow
+++ b/src/index.js.flow
@@ -1,14 +1,14 @@
// @flow
-import * as React from 'react'
-import type { FormApi, FormState, FormValuesShape } from 'final-form'
+import * as React from "react";
+import type { FormApi, FormState, FormValuesShape } from "final-form";
import type {
FieldProps,
FieldRenderProps,
FormProps,
FormSpyProps,
UseFieldConfig,
- UseFormStateParams
-} from './types'
+ UseFormStateParams,
+} from "./types";
export type {
FieldProps,
@@ -16,24 +16,24 @@ export type {
FormProps,
FormRenderProps,
FormSpyProps,
- FormSpyRenderProps
-} from './types'
+ FormSpyRenderProps,
+} from "./types";
-declare export var Field: React.ComponentType
-declare export var Form: React.ComponentType>
-declare export var FormSpy: React.ComponentType>
+declare export var Field: React.ComponentType;
+declare export var Form: React.ComponentType>;
+declare export var FormSpy: React.ComponentType>;
declare export function useForm(
- componentName?: string
-): FormApi
+ componentName?: string,
+): FormApi;
declare export function useFormState(
- params?: UseFormStateParams
-): FormState
+ params?: UseFormStateParams,
+): FormState;
declare export var useField: (
name: string,
- config?: UseFieldConfig
-) => FieldRenderProps
+ config?: UseFieldConfig,
+) => FieldRenderProps;
declare export function withTypes(): {
Form: React.ComponentType>,
- FormSpy: React.ComponentType>
-}
-declare export var version: string
+ FormSpy: React.ComponentType>,
+};
+declare export var version: string;
diff --git a/src/isReactNative.js b/src/isReactNative.js
index 95634d95..beba770e 100644
--- a/src/isReactNative.js
+++ b/src/isReactNative.js
@@ -1,8 +1,8 @@
// @flow
const isReactNative =
- typeof window !== 'undefined' &&
+ typeof window !== "undefined" &&
window.navigator &&
window.navigator.product &&
- window.navigator.product === 'ReactNative'
+ window.navigator.product === "ReactNative";
-export default isReactNative
+export default isReactNative;
diff --git a/src/isSyntheticEvent.js b/src/isSyntheticEvent.js
index 34998512..25670cbe 100644
--- a/src/isSyntheticEvent.js
+++ b/src/isSyntheticEvent.js
@@ -1,5 +1,5 @@
// @flow
const isSyntheticEvent = (candidate: any): boolean =>
- !!(candidate && typeof candidate.stopPropagation === 'function')
+ !!(candidate && typeof candidate.stopPropagation === "function");
-export default isSyntheticEvent
+export default isSyntheticEvent;
diff --git a/src/renderComponent.js b/src/renderComponent.js
index b2a57764..6120cb03 100644
--- a/src/renderComponent.js
+++ b/src/renderComponent.js
@@ -1,36 +1,36 @@
// @flow
-import * as React from 'react'
-import type { RenderableProps } from './types'
+import * as React from "react";
+import type { RenderableProps } from "./types";
// shared logic between components that use either render prop,
// children render function, or component prop
export default function renderComponent(
props: RenderableProps & T,
lazyProps: Object,
- name: string
+ name: string,
): React.Node {
- const { render, children, component, ...rest } = props
+ const { render, children, component, ...rest } = props;
if (component) {
return React.createElement(
component,
Object.assign(lazyProps, rest, {
children,
- render
- })
- )
+ render,
+ }),
+ );
}
if (render) {
return render(
children === undefined
? Object.assign(lazyProps, rest)
: // inject children back in
- Object.assign(lazyProps, rest, { children })
- )
+ Object.assign(lazyProps, rest, { children }),
+ );
}
- if (typeof children !== 'function') {
+ if (typeof children !== "function") {
throw new Error(
- `Must specify either a render prop, a render function as children, or a component prop to ${name}`
- )
+ `Must specify either a render prop, a render function as children, or a component prop to ${name}`,
+ );
}
- return children(Object.assign(lazyProps, rest))
+ return children(Object.assign(lazyProps, rest));
}
diff --git a/src/renderComponent.test.js b/src/renderComponent.test.js
index 74a1267c..b584b7ae 100644
--- a/src/renderComponent.test.js
+++ b/src/renderComponent.test.js
@@ -1,72 +1,72 @@
-import renderComponent from './renderComponent'
+import renderComponent from "./renderComponent";
-describe('renderComponent', () => {
- it('should pass both render and children prop', () => {
- const children = 'some children'
- const render = () => {}
+describe("renderComponent", () => {
+ it("should pass both render and children prop", () => {
+ const children = "some children";
+ const render = () => {};
const props = {
component: () => null,
children,
- render
- }
- const name = 'TestComponent'
- const result = renderComponent(props, {}, name)
- expect(result.props).toEqual({ children, render })
- })
+ render,
+ };
+ const name = "TestComponent";
+ const result = renderComponent(props, {}, name);
+ expect(result.props).toEqual({ children, render });
+ });
- it('should include children when rendering with render', () => {
- const children = 'some children'
- const render = jest.fn()
+ it("should include children when rendering with render", () => {
+ const children = "some children";
+ const render = jest.fn();
const props = {
children,
- render
- }
- const name = 'TestComponent'
- renderComponent(props, {}, name)
- expect(render).toHaveBeenCalled()
- expect(render).toHaveBeenCalledTimes(1)
- expect(render.mock.calls[0][0].children).toBe(children)
- })
+ render,
+ };
+ const name = "TestComponent";
+ renderComponent(props, {}, name);
+ expect(render).toHaveBeenCalled();
+ expect(render).toHaveBeenCalledTimes(1);
+ expect(render.mock.calls[0][0].children).toBe(children);
+ });
- it('should throw error if no render strategy is provided', () => {
- const children = 'some children'
+ it("should throw error if no render strategy is provided", () => {
+ const children = "some children";
const props = {
- children
- }
- const name = 'TestComponent'
+ children,
+ };
+ const name = "TestComponent";
expect(() => renderComponent(props, {}, name)).toThrow(
- `Must specify either a render prop, a render function as children, or a component prop to ${name}`
- )
- })
+ `Must specify either a render prop, a render function as children, or a component prop to ${name}`,
+ );
+ });
- it('should not evaluate any of the keys given in the second argument', () => {
- const children = 'some children'
- const render = jest.fn()
+ it("should not evaluate any of the keys given in the second argument", () => {
+ const children = "some children";
+ const render = jest.fn();
const props = {
children,
- render
- }
- const getA = jest.fn()
- const getB = jest.fn()
- const name = 'TestComponent'
+ render,
+ };
+ const getA = jest.fn();
+ const getB = jest.fn();
+ const name = "TestComponent";
renderComponent(
props,
{
get a() {
- getA()
- return 1
+ getA();
+ return 1;
},
get b() {
- getB()
- return 2
- }
+ getB();
+ return 2;
+ },
},
- name
- )
- expect(render).toHaveBeenCalled()
- expect(render).toHaveBeenCalledTimes(1)
- expect(render.mock.calls[0][0].children).toBe(children)
- expect(getA).not.toHaveBeenCalled()
- expect(getB).not.toHaveBeenCalled()
- })
-})
+ name,
+ );
+ expect(render).toHaveBeenCalled();
+ expect(render).toHaveBeenCalledTimes(1);
+ expect(render.mock.calls[0][0].children).toBe(children);
+ expect(getA).not.toHaveBeenCalled();
+ expect(getB).not.toHaveBeenCalled();
+ });
+});
diff --git a/src/shallowEqual.js b/src/shallowEqual.js
index 2b22bf16..27f4d824 100644
--- a/src/shallowEqual.js
+++ b/src/shallowEqual.js
@@ -1,24 +1,24 @@
// @flow
const shallowEqual = (a: any, b: any): boolean => {
if (a === b) {
- return true
+ return true;
}
- if (typeof a !== 'object' || !a || typeof b !== 'object' || !b) {
- return false
+ if (typeof a !== "object" || !a || typeof b !== "object" || !b) {
+ return false;
}
- var keysA = Object.keys(a)
- var keysB = Object.keys(b)
+ var keysA = Object.keys(a);
+ var keysB = Object.keys(b);
if (keysA.length !== keysB.length) {
- return false
+ return false;
}
- var bHasOwnProperty = Object.prototype.hasOwnProperty.bind(b)
+ var bHasOwnProperty = Object.prototype.hasOwnProperty.bind(b);
for (var idx = 0; idx < keysA.length; idx++) {
- var key = keysA[idx]
+ var key = keysA[idx];
if (!bHasOwnProperty(key) || a[key] !== b[key]) {
- return false
+ return false;
}
}
- return true
-}
+ return true;
+};
-export default shallowEqual
+export default shallowEqual;
diff --git a/src/shallowEqual.test.js b/src/shallowEqual.test.js
index f22382f7..3f5587d9 100644
--- a/src/shallowEqual.test.js
+++ b/src/shallowEqual.test.js
@@ -1,60 +1,60 @@
-import shallowEqual from './shallowEqual'
+import shallowEqual from "./shallowEqual";
-describe('shallowEqual', () => {
- it('returns false if either argument is null', () => {
- expect(shallowEqual(null, {})).toBe(false)
- expect(shallowEqual({}, null)).toBe(false)
- })
+describe("shallowEqual", () => {
+ it("returns false if either argument is null", () => {
+ expect(shallowEqual(null, {})).toBe(false);
+ expect(shallowEqual({}, null)).toBe(false);
+ });
- it('returns true if both arguments are null or undefined', () => {
- expect(shallowEqual(null, null)).toBe(true)
- expect(shallowEqual(undefined, undefined)).toBe(true)
- })
+ it("returns true if both arguments are null or undefined", () => {
+ expect(shallowEqual(null, null)).toBe(true);
+ expect(shallowEqual(undefined, undefined)).toBe(true);
+ });
- it('returns true if arguments are shallow equal', () => {
- expect(shallowEqual({ a: 1, b: 2, c: 3 }, { a: 1, b: 2, c: 3 })).toBe(true)
- })
+ it("returns true if arguments are shallow equal", () => {
+ expect(shallowEqual({ a: 1, b: 2, c: 3 }, { a: 1, b: 2, c: 3 })).toBe(true);
+ });
- it('returns false if arguments are not objects and not equal', () => {
- expect(shallowEqual(1, 2)).toBe(false)
- })
+ it("returns false if arguments are not objects and not equal", () => {
+ expect(shallowEqual(1, 2)).toBe(false);
+ });
- it('returns false if only one argument is not an object', () => {
- expect(shallowEqual(1, {})).toBe(false)
- })
+ it("returns false if only one argument is not an object", () => {
+ expect(shallowEqual(1, {})).toBe(false);
+ });
- it('returns false if first argument has too many keys', () => {
- expect(shallowEqual({ a: 1, b: 2, c: 3 }, { a: 1, b: 2 })).toBe(false)
- })
+ it("returns false if first argument has too many keys", () => {
+ expect(shallowEqual({ a: 1, b: 2, c: 3 }, { a: 1, b: 2 })).toBe(false);
+ });
- it('returns false if second argument has too many keys', () => {
- expect(shallowEqual({ a: 1, b: 2 }, { a: 1, b: 2, c: 3 })).toBe(false)
- })
+ it("returns false if second argument has too many keys", () => {
+ expect(shallowEqual({ a: 1, b: 2 }, { a: 1, b: 2, c: 3 })).toBe(false);
+ });
- it('returns true if values are not primitives but are ===', () => {
- let obj = {}
+ it("returns true if values are not primitives but are ===", () => {
+ let obj = {};
expect(shallowEqual({ a: 1, b: 2, c: obj }, { a: 1, b: 2, c: obj })).toBe(
- true
- )
- })
+ true,
+ );
+ });
- it('returns false if arguments are not shallow equal', () => {
+ it("returns false if arguments are not shallow equal", () => {
expect(shallowEqual({ a: 1, b: 2, c: {} }, { a: 1, b: 2, c: {} })).toBe(
- false
- )
- })
+ false,
+ );
+ });
- it('should treat objects created by `Object.create(null)` like any other plain object', () => {
+ it("should treat objects created by `Object.create(null)` like any other plain object", () => {
function Foo() {
- this.a = 1
+ this.a = 1;
}
- Foo.prototype.constructor = null
+ Foo.prototype.constructor = null;
- const object2 = { a: 1 }
- expect(shallowEqual(new Foo(), object2)).toBe(true)
+ const object2 = { a: 1 };
+ expect(shallowEqual(new Foo(), object2)).toBe(true);
- const object1 = Object.create(null)
- object1.a = 1
- expect(shallowEqual(object1, object2)).toBe(true)
- })
-})
+ const object1 = Object.create(null);
+ object1.a = 1;
+ expect(shallowEqual(object1, object2)).toBe(true);
+ });
+});
diff --git a/src/testUtils.js b/src/testUtils.js
index dae4c426..acef18ba 100644
--- a/src/testUtils.js
+++ b/src/testUtils.js
@@ -1,27 +1,29 @@
-import React from 'react'
+import React from "react";
-export const wrapWith = (mock, fn) => (...args) => {
- mock(...args)
- return fn(...args)
-}
+export const wrapWith =
+ (mock, fn) =>
+ (...args) => {
+ mock(...args);
+ return fn(...args);
+ };
/** A simple container component that allows boolean to be toggled with a button */
export function Toggle({ children }) {
- const [on, setOn] = React.useState(false)
+ const [on, setOn] = React.useState(false);
return (
{children(on)}
setOn(!on)}>Toggle
- )
+ );
}
export class ErrorBoundary extends React.Component {
componentDidCatch(error) {
- this.props.spy(error)
+ this.props.spy(error);
}
render() {
- return this.props.children
+ return this.props.children;
}
}
diff --git a/src/types.js.flow b/src/types.js.flow
index de029393..9c404ed2 100644
--- a/src/types.js.flow
+++ b/src/types.js.flow
@@ -1,5 +1,5 @@
// @flow
-import * as React from 'react'
+import * as React from "react";
import type {
FormApi,
Config,
@@ -8,14 +8,14 @@ import type {
FormSubscription,
FormValuesShape,
FieldSubscription,
- FieldValidator
-} from 'final-form'
+ FieldValidator,
+} from "final-form";
-type SupportedInputs = 'input' | 'select' | 'textarea'
+type SupportedInputs = "input" | "select" | "textarea";
export type ReactContext = {
- reactFinalForm: FormApi
-}
+ reactFinalForm: FormApi,
+};
export type FieldInputProps = {
name: string,
@@ -25,8 +25,8 @@ export type FieldInputProps = {
value: any,
type?: string,
checked?: boolean,
- multiple?: boolean
-}
+ multiple?: boolean,
+};
export type FieldRenderProps = {
input: FieldInputProps,
meta: {
@@ -48,47 +48,47 @@ export type FieldRenderProps = {
touched?: boolean,
valid?: boolean,
validating?: boolean,
- visited?: boolean
- }
-}
+ visited?: boolean,
+ },
+};
export type SubmitEvent = {
- preventDefault?: $PropertyType, 'preventDefault'>,
+ preventDefault?: $PropertyType, "preventDefault">,
stopPropagation?: $PropertyType<
SyntheticEvent,
- 'stopPropagation'
- >
-}
+ "stopPropagation",
+ >,
+};
export type FormRenderProps = {
handleSubmit: (?SubmitEvent) => ?Promise,
- form: FormApi
-} & FormState
+ form: FormApi,
+} & FormState;
export type FormSpyRenderProps = {
- form: FormApi
-} & FormState
+ form: FormApi,
+} & FormState;
export type RenderableProps = {
component?: React.ComponentType<*> | SupportedInputs,
children?: ((props: T) => React.Node) | React.Node,
- render?: (props: T) => React.Node
-}
+ render?: (props: T) => React.Node,
+};
export type FormProps = {
subscription?: FormSubscription,
decorators?: Decorator[],
form?: FormApi,
- initialValuesEqual?: (?Object, ?Object) => boolean
+ initialValuesEqual?: (?Object, ?Object) => boolean,
} & Config &
- RenderableProps>
+ RenderableProps>;
export type UseFieldAutoConfig = {
afterSubmit?: () => void,
allowNull?: boolean,
beforeSubmit?: () => void | false,
- children?: $PropertyType, 'children'>,
- component?: $PropertyType, 'component'>,
+ children?: $PropertyType, "children">,
+ component?: $PropertyType, "component">,
data?: Object,
defaultValue?: any,
format?: (value: any, name: string) => any,
@@ -100,27 +100,26 @@ export type UseFieldAutoConfig = {
type?: string,
validate?: FieldValidator,
validateFields?: string[],
- value?: any
-}
+ value?: any,
+};
export type UseFieldConfig = {
- subscription?: FieldSubscription
-} & UseFieldAutoConfig
+ subscription?: FieldSubscription,
+} & UseFieldAutoConfig;
export type FieldProps = UseFieldConfig & {
- name: string
-} & RenderableProps
+ name: string,
+} & RenderableProps;
export type UseFormStateParams = {
onChange?: (formState: FormState) => void,
- subscription?: FormSubscription
-}
+ subscription?: FormSubscription,
+};
-export type FormSpyProps<
- FormValues: FormValuesShape
-> = UseFormStateParams &
- RenderableProps>
+export type FormSpyProps =
+ UseFormStateParams &
+ RenderableProps>;
export type FormSpyPropsWithForm = {
- reactFinalForm: FormApi
-} & FormSpyProps
+ reactFinalForm: FormApi,
+} & FormSpyProps;
diff --git a/src/useConstant.js b/src/useConstant.js
index 929be24b..e71931ee 100644
--- a/src/useConstant.js
+++ b/src/useConstant.js
@@ -1,5 +1,5 @@
// @flow
-import React from 'react'
+import React from "react";
/**
* A simple hook to create a constant value that lives for
@@ -14,9 +14,9 @@ import React from 'react'
* @param {Function} init - A function to generate the value
*/
export default function useConstant(init: () => T): T {
- const ref = React.useRef()
+ const ref = React.useRef();
if (!ref.current) {
- ref.current = init()
+ ref.current = init();
}
- return ref.current
+ return ref.current;
}
diff --git a/src/useConstantCallback.js b/src/useConstantCallback.js
index 0651f8ab..24c0463c 100644
--- a/src/useConstantCallback.js
+++ b/src/useConstantCallback.js
@@ -1,5 +1,5 @@
// @flow
-import * as React from 'react'
+import * as React from "react";
/**
* Creates a callback, even with closures, that will be
@@ -8,9 +8,9 @@ import * as React from 'react'
* closures.
*/
export default function useConstantCallback(callback) {
- const ref = React.useRef(callback)
+ const ref = React.useRef(callback);
React.useEffect(() => {
- ref.current = callback
- })
- return React.useCallback((...args) => ref.current.apply(null, args), [])
+ ref.current = callback;
+ });
+ return React.useCallback((...args) => ref.current.apply(null, args), []);
}
diff --git a/src/useConstantCallback.test.js b/src/useConstantCallback.test.js
index b8c9bc86..bc8ca8e5 100644
--- a/src/useConstantCallback.test.js
+++ b/src/useConstantCallback.test.js
@@ -1,25 +1,25 @@
// @flow
-import * as React from 'react'
-import { render, cleanup } from '@testing-library/react'
-import '@testing-library/jest-dom/extend-expect'
-import useConstantCallback from './useConstantCallback'
+import * as React from "react";
+import { render, cleanup } from "@testing-library/react";
+import "@testing-library/jest-dom/extend-expect";
+import useConstantCallback from "./useConstantCallback";
-describe('useConstantCallback', () => {
- afterEach(cleanup)
+describe("useConstantCallback", () => {
+ afterEach(cleanup);
- it('should give the same instance on every render, even as params/deps change', () => {
- const callback = jest.fn()
+ it("should give the same instance on every render, even as params/deps change", () => {
+ const callback = jest.fn();
const MyComponent = () => {
- const [name, setName] = React.useState('John')
- const [age, setAge] = React.useState(20)
- const [isAdmin, setAdmin] = React.useState(false)
- const constantCallback = useConstantCallback(time => {
- expect(typeof time).toBe('number')
- callback(name, age, isAdmin)
- })
- const callbackRef = React.useRef(constantCallback)
- expect(callbackRef.current).toBe(constantCallback)
+ const [name, setName] = React.useState("John");
+ const [age, setAge] = React.useState(20);
+ const [isAdmin, setAdmin] = React.useState(false);
+ const constantCallback = useConstantCallback((time) => {
+ expect(typeof time).toBe("number");
+ callback(name, age, isAdmin);
+ });
+ const callbackRef = React.useRef(constantCallback);
+ expect(callbackRef.current).toBe(constantCallback);
return (
{
>
Call
- setName('Paul')}>
+ setName("Paul")}>
{name}
setAge(25)}>
{age}
setAdmin(true)}>
- {isAdmin ? 'Yes' : 'No'}
+ {isAdmin ? "Yes" : "No"}
- )
- }
- const { getByTestId } = render( )
- const call = getByTestId('call')
- const changeName = getByTestId('changeName')
- const changeAge = getByTestId('changeAge')
- const changeAdmin = getByTestId('changeAdmin')
-
- expect(changeName).toHaveTextContent('John')
- expect(changeAge).toHaveTextContent(20)
- expect(changeAdmin).toHaveTextContent('No')
- expect(callback).not.toHaveBeenCalled()
-
- call.click()
-
- expect(callback).toHaveBeenCalled()
- expect(callback).toHaveBeenCalledTimes(1)
- expect(callback.mock.calls[0][0]).toBe('John')
- expect(callback.mock.calls[0][1]).toBe(20)
- expect(callback.mock.calls[0][2]).toBe(false)
- expect(changeName).toHaveTextContent('John')
- expect(changeAge).toHaveTextContent(20)
- expect(changeAdmin).toHaveTextContent('No')
-
- changeName.click()
-
- expect(callback).toHaveBeenCalledTimes(1)
- expect(changeName).toHaveTextContent('Paul')
- expect(changeAge).toHaveTextContent(20)
- expect(changeAdmin).toHaveTextContent('No')
-
- call.click()
-
- expect(callback).toHaveBeenCalledTimes(2)
- expect(callback.mock.calls[1][0]).toBe('Paul')
- expect(callback.mock.calls[1][1]).toBe(20)
- expect(callback.mock.calls[1][2]).toBe(false)
-
- changeAge.click()
-
- expect(callback).toHaveBeenCalledTimes(2)
- expect(changeName).toHaveTextContent('Paul')
- expect(changeAge).toHaveTextContent(25)
- expect(changeAdmin).toHaveTextContent('No')
-
- call.click()
-
- expect(callback).toHaveBeenCalledTimes(3)
- expect(callback.mock.calls[2][0]).toBe('Paul')
- expect(callback.mock.calls[2][1]).toBe(25)
- expect(callback.mock.calls[2][2]).toBe(false)
-
- changeAdmin.click()
-
- expect(callback).toHaveBeenCalledTimes(3)
- expect(changeName).toHaveTextContent('Paul')
- expect(changeAge).toHaveTextContent(25)
- expect(changeAdmin).toHaveTextContent('Yes')
-
- call.click()
-
- expect(callback).toHaveBeenCalledTimes(4)
- expect(callback.mock.calls[3][0]).toBe('Paul')
- expect(callback.mock.calls[3][1]).toBe(25)
- expect(callback.mock.calls[3][2]).toBe(true)
- })
-})
+ );
+ };
+ const { getByTestId } = render( );
+ const call = getByTestId("call");
+ const changeName = getByTestId("changeName");
+ const changeAge = getByTestId("changeAge");
+ const changeAdmin = getByTestId("changeAdmin");
+
+ expect(changeName).toHaveTextContent("John");
+ expect(changeAge).toHaveTextContent(20);
+ expect(changeAdmin).toHaveTextContent("No");
+ expect(callback).not.toHaveBeenCalled();
+
+ call.click();
+
+ expect(callback).toHaveBeenCalled();
+ expect(callback).toHaveBeenCalledTimes(1);
+ expect(callback.mock.calls[0][0]).toBe("John");
+ expect(callback.mock.calls[0][1]).toBe(20);
+ expect(callback.mock.calls[0][2]).toBe(false);
+ expect(changeName).toHaveTextContent("John");
+ expect(changeAge).toHaveTextContent(20);
+ expect(changeAdmin).toHaveTextContent("No");
+
+ changeName.click();
+
+ expect(callback).toHaveBeenCalledTimes(1);
+ expect(changeName).toHaveTextContent("Paul");
+ expect(changeAge).toHaveTextContent(20);
+ expect(changeAdmin).toHaveTextContent("No");
+
+ call.click();
+
+ expect(callback).toHaveBeenCalledTimes(2);
+ expect(callback.mock.calls[1][0]).toBe("Paul");
+ expect(callback.mock.calls[1][1]).toBe(20);
+ expect(callback.mock.calls[1][2]).toBe(false);
+
+ changeAge.click();
+
+ expect(callback).toHaveBeenCalledTimes(2);
+ expect(changeName).toHaveTextContent("Paul");
+ expect(changeAge).toHaveTextContent(25);
+ expect(changeAdmin).toHaveTextContent("No");
+
+ call.click();
+
+ expect(callback).toHaveBeenCalledTimes(3);
+ expect(callback.mock.calls[2][0]).toBe("Paul");
+ expect(callback.mock.calls[2][1]).toBe(25);
+ expect(callback.mock.calls[2][2]).toBe(false);
+
+ changeAdmin.click();
+
+ expect(callback).toHaveBeenCalledTimes(3);
+ expect(changeName).toHaveTextContent("Paul");
+ expect(changeAge).toHaveTextContent(25);
+ expect(changeAdmin).toHaveTextContent("Yes");
+
+ call.click();
+
+ expect(callback).toHaveBeenCalledTimes(4);
+ expect(callback.mock.calls[3][0]).toBe("Paul");
+ expect(callback.mock.calls[3][1]).toBe(25);
+ expect(callback.mock.calls[3][2]).toBe(true);
+ });
+});
diff --git a/src/useField.js b/src/useField.js
index 9f66363d..c6ffa686 100644
--- a/src/useField.js
+++ b/src/useField.js
@@ -1,35 +1,39 @@
// @flow
-import * as React from 'react'
-import { fieldSubscriptionItems } from 'final-form'
+import * as React from "react";
+import { fieldSubscriptionItems } from "final-form";
import type {
FieldSubscription,
FieldState,
FormApi,
- FormValuesShape
-} from 'final-form'
-import type { UseFieldConfig, FieldInputProps, FieldRenderProps } from './types'
-import isReactNative from './isReactNative'
-import getValue from './getValue'
-import useForm from './useForm'
-import useLatest from './useLatest'
-import { addLazyFieldMetaState } from './getters'
-import useConstantCallback from './useConstantCallback'
+ FormValuesShape,
+} from "final-form";
+import type {
+ UseFieldConfig,
+ FieldInputProps,
+ FieldRenderProps,
+} from "./types";
+import isReactNative from "./isReactNative";
+import getValue from "./getValue";
+import useForm from "./useForm";
+import useLatest from "./useLatest";
+import { addLazyFieldMetaState } from "./getters";
+import useConstantCallback from "./useConstantCallback";
const all: FieldSubscription = fieldSubscriptionItems.reduce((result, key) => {
- result[key] = true
- return result
-}, {})
+ result[key] = true;
+ return result;
+}, {});
const defaultFormat = (value: ?any, name: string) =>
- value === undefined ? '' : value
+ value === undefined ? "" : value;
const defaultParse = (value: ?any, name: string) =>
- value === '' ? undefined : value
+ value === "" ? undefined : value;
-const defaultIsEqual = (a: any, b: any): boolean => a === b
+const defaultIsEqual = (a: any, b: any): boolean => a === b;
function useField(
name: string,
- config: UseFieldConfig = {}
+ config: UseFieldConfig = {},
): FieldRenderProps {
const {
afterSubmit,
@@ -45,13 +49,13 @@ function useField(
subscription = all,
type,
validateFields,
- value: _value
- } = config
- const form: FormApi = useForm('useField')
+ value: _value,
+ } = config;
+ const form: FormApi = useForm("useField");
- const configRef = useLatest(config)
+ const configRef = useLatest(config);
- const register = (callback: FieldState => void, silent: boolean) =>
+ const register = (callback: (FieldState) => void, silent: boolean) =>
// avoid using `state` const in any closures created inside `register`
// because they would refer `state` from current execution context
// whereas actual `state` would defined in the subsequent `useField` hook
@@ -63,19 +67,19 @@ function useField(
const {
beforeSubmit,
formatOnBlur,
- format = defaultFormat
- } = configRef.current
+ format = defaultFormat,
+ } = configRef.current;
if (formatOnBlur) {
- const { value } = ((form.getFieldState(name): any): FieldState)
- const formatted = format(value, name)
+ const { value } = ((form.getFieldState(name): any): FieldState);
+ const formatted = format(value, name);
if (formatted !== value) {
- form.change(name, formatted)
+ form.change(name, formatted);
}
}
- return beforeSubmit && beforeSubmit()
+ return beforeSubmit && beforeSubmit();
},
data,
defaultValue,
@@ -83,36 +87,36 @@ function useField(
initialValue,
isEqual: (a, b) => (configRef.current.isEqual || defaultIsEqual)(a, b),
silent,
- validateFields
- })
+ validateFields,
+ });
- const firstRender = React.useRef(true)
+ const firstRender = React.useRef(true);
// synchronously register and unregister to query field state for our subscription on first render
const [state, setState] = React.useState((): FieldState => {
- let initialState: FieldState = {}
+ let initialState: FieldState = {};
// temporarily disable destroyOnUnregister
- const destroyOnUnregister = form.destroyOnUnregister
- form.destroyOnUnregister = false
+ const destroyOnUnregister = form.destroyOnUnregister;
+ form.destroyOnUnregister = false;
- register(state => {
- initialState = state
- }, true)()
+ register((state) => {
+ initialState = state;
+ }, true)();
// return destroyOnUnregister to its original value
- form.destroyOnUnregister = destroyOnUnregister
+ form.destroyOnUnregister = destroyOnUnregister;
- return initialState
- })
+ return initialState;
+ });
React.useEffect(
() =>
- register(state => {
+ register((state) => {
if (firstRender.current) {
- firstRender.current = false
+ firstRender.current = false;
} else {
- setState(state)
+ setState(state);
}
}, false),
// eslint-disable-next-line react-hooks/exhaustive-deps
@@ -123,53 +127,53 @@ function useField(
// If we want to allow inline fat-arrow field-level validation functions, we
// cannot reregister field every time validate function !==.
// validate,
- initialValue
+ initialValue,
// The validateFields array is often passed as validateFields={[]}, creating
// a !== new array every time. If it needs to be changed, a rerender/reregister
// can be forced by changing the key prop
// validateFields
- ]
- )
+ ],
+ );
- const meta = {}
- addLazyFieldMetaState(meta, state)
+ const meta = {};
+ addLazyFieldMetaState(meta, state);
const input: FieldInputProps = {
name,
get value() {
- let value = state.value
+ let value = state.value;
if (formatOnBlur) {
- if (component === 'input') {
- value = defaultFormat(value, name)
+ if (component === "input") {
+ value = defaultFormat(value, name);
}
} else {
- value = format(value, name)
+ value = format(value, name);
}
if (value === null && !allowNull) {
- value = ''
+ value = "";
}
- if (type === 'checkbox' || type === 'radio') {
- return _value
- } else if (component === 'select' && multiple) {
- return value || []
+ if (type === "checkbox" || type === "radio") {
+ return _value;
+ } else if (component === "select" && multiple) {
+ return value || [];
}
- return value
+ return value;
},
get checked() {
- let value = state.value
- if (type === 'checkbox') {
- value = format(value, name)
+ let value = state.value;
+ if (type === "checkbox") {
+ value = format(value, name);
if (_value === undefined) {
- return !!value
+ return !!value;
} else {
- return !!(Array.isArray(value) && ~value.indexOf(_value))
+ return !!(Array.isArray(value) && ~value.indexOf(_value));
}
- } else if (type === 'radio') {
- return format(value, name) === _value
+ } else if (type === "radio") {
+ return format(value, name) === _value;
}
- return undefined
+ return undefined;
},
onBlur: useConstantCallback((event: ?SyntheticFocusEvent<*>) => {
- state.blur()
+ state.blur();
if (formatOnBlur) {
/**
* Here we must fetch the value directly from Final Form because we cannot
@@ -178,54 +182,54 @@ function useField(
* before calling `onBlur()`, but before the field has had a chance to receive
* the value update from Final Form.
*/
- const fieldState: any = form.getFieldState(state.name)
- state.change(format(fieldState.value, state.name))
+ const fieldState: any = form.getFieldState(state.name);
+ state.change(format(fieldState.value, state.name));
}
}),
onChange: useConstantCallback((event: SyntheticInputEvent<*> | any) => {
// istanbul ignore next
- if (process.env.NODE_ENV !== 'production' && event && event.target) {
- const targetType = event.target.type
+ if (process.env.NODE_ENV !== "production" && event && event.target) {
+ const targetType = event.target.type;
const unknown =
- ~['checkbox', 'radio', 'select-multiple'].indexOf(targetType) &&
+ ~["checkbox", "radio", "select-multiple"].indexOf(targetType) &&
!type &&
- component !== 'select'
+ component !== "select";
const value: any =
- targetType === 'select-multiple' ? state.value : _value
+ targetType === "select-multiple" ? state.value : _value;
if (unknown) {
console.error(
`You must pass \`type="${
- targetType === 'select-multiple' ? 'select' : targetType
+ targetType === "select-multiple" ? "select" : targetType
}"\` prop to your Field(${name}) component.\n` +
`Without it we don't know how to unpack your \`value\` prop - ${
Array.isArray(value) ? `[${value}]` : `"${value}"`
- }.`
- )
+ }.`,
+ );
}
}
const value: any =
event && event.target
? getValue(event, state.value, _value, isReactNative)
- : event
- state.change(parse(value, name))
+ : event;
+ state.change(parse(value, name));
}),
onFocus: useConstantCallback((event: ?SyntheticFocusEvent<*>) =>
- state.focus()
- )
- }
+ state.focus(),
+ ),
+ };
if (multiple) {
- input.multiple = multiple
+ input.multiple = multiple;
}
if (type !== undefined) {
- input.type = type
+ input.type = type;
}
- const renderProps: FieldRenderProps = { input, meta } // assign to force Flow check
- return renderProps
+ const renderProps: FieldRenderProps = { input, meta }; // assign to force Flow check
+ return renderProps;
}
-export default useField
+export default useField;
diff --git a/src/useField.test.js b/src/useField.test.js
index 5d87737f..1277f1ce 100644
--- a/src/useField.test.js
+++ b/src/useField.test.js
@@ -1,67 +1,67 @@
// @flow
-import * as React from 'react'
-import { render, fireEvent, cleanup } from '@testing-library/react'
-import '@testing-library/jest-dom/extend-expect'
-import { ErrorBoundary } from './testUtils'
-import Form from './ReactFinalForm'
-import Field from './Field'
-import { useField } from './index'
-import { act } from 'react-dom/test-utils'
+import * as React from "react";
+import { render, fireEvent, cleanup } from "@testing-library/react";
+import "@testing-library/jest-dom/extend-expect";
+import { ErrorBoundary } from "./testUtils";
+import Form from "./ReactFinalForm";
+import Field from "./Field";
+import { useField } from "./index";
+import { act } from "react-dom/test-utils";
-const onSubmitMock = values => {}
+const onSubmitMock = (values) => {};
-describe('useField', () => {
- afterEach(cleanup)
+describe("useField", () => {
+ afterEach(cleanup);
// Most of the functionality of useField is tested in Field.test.js
// This file is only for testing its use as a hook in other components
- it('should warn if not used inside a form', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ it("should warn if not used inside a form", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
const MyFieldComponent = () => {
- useField('name')
- return
- }
+ useField("name");
+ return
;
+ };
render(
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'useField must be used inside of a component'
- )
- console.error.mockRestore()
- })
+ "useField must be used inside of a component",
+ );
+ console.error.mockRestore();
+ });
- it('should subscribe to all by default', () => {
+ it("should subscribe to all by default", () => {
const MyFieldListener = () => {
- const { input, meta } = useField('name')
- expect(meta.active).toBe(false)
- expect(meta.data).toEqual({})
- expect(meta.dirty).toBe(false)
- expect(meta.dirtySinceLastSubmit).toBe(false)
- expect(meta.error).toBeUndefined()
- expect(meta.initial).toBeUndefined()
- expect(meta.invalid).toBe(false)
- expect(meta.length).toBeUndefined()
- expect(meta.modified).toBe(false)
- expect(meta.modifiedSinceLastSubmit).toBe(false)
- expect(meta.pristine).toBe(true)
- expect(meta.submitError).toBeUndefined()
- expect(meta.submitFailed).toBe(false)
- expect(meta.submitSucceeded).toBe(false)
- expect(meta.submitting).toBe(false)
- expect(meta.touched).toBe(false)
- expect(meta.valid).toBe(true)
- expect(meta.validating).toBe(false)
- expect(meta.visited).toBe(false)
- expect(input.value).toBe('')
- return null
- }
+ const { input, meta } = useField("name");
+ expect(meta.active).toBe(false);
+ expect(meta.data).toEqual({});
+ expect(meta.dirty).toBe(false);
+ expect(meta.dirtySinceLastSubmit).toBe(false);
+ expect(meta.error).toBeUndefined();
+ expect(meta.initial).toBeUndefined();
+ expect(meta.invalid).toBe(false);
+ expect(meta.length).toBeUndefined();
+ expect(meta.modified).toBe(false);
+ expect(meta.modifiedSinceLastSubmit).toBe(false);
+ expect(meta.pristine).toBe(true);
+ expect(meta.submitError).toBeUndefined();
+ expect(meta.submitFailed).toBe(false);
+ expect(meta.submitSucceeded).toBe(false);
+ expect(meta.submitting).toBe(false);
+ expect(meta.touched).toBe(false);
+ expect(meta.valid).toBe(true);
+ expect(meta.validating).toBe(false);
+ expect(meta.visited).toBe(false);
+ expect(input.value).toBe("");
+ return null;
+ };
render(
{() => (
@@ -70,16 +70,16 @@ describe('useField', () => {
)}
-
- )
- })
+ ,
+ );
+ });
- it('should track field state', () => {
- const spy = jest.fn()
+ it("should track field state", () => {
+ const spy = jest.fn();
const MyFieldListener = () => {
- spy(useField('name').input.value)
- return null
- }
+ spy(useField("name").input.value);
+ return null;
+ };
const { getByTestId } = render(
{() => (
@@ -88,31 +88,31 @@ describe('useField', () => {
)}
-
- )
- expect(getByTestId('name').value).toBe('')
+ ,
+ );
+ expect(getByTestId("name").value).toBe("");
// All forms without restricted subscriptions render twice at first because they
// need to update their validation and touched/modified/visited maps every time
// new fields are registered.
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[0][0]).toBe('')
- expect(spy.mock.calls[1][0]).toBe('')
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
- expect(getByTestId('name').value).toBe('erikras')
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0]).toBe('erikras')
- })
-
- it('should allow for creation of render-controlled components', () => {
- const spy = jest.fn()
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[0][0]).toBe("");
+ expect(spy.mock.calls[1][0]).toBe("");
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
+ expect(getByTestId("name").value).toBe("erikras");
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0]).toBe("erikras");
+ });
+
+ it("should allow for creation of render-controlled components", () => {
+ const spy = jest.fn();
const MemoizedDirtyDisplay = React.memo(({ dirty }) => {
- spy(dirty)
- return {dirty ? 'Dirty' : 'Pristine'}
- })
+ spy(dirty);
+ return {dirty ? "Dirty" : "Pristine"}
;
+ });
const MyFieldListener = () => {
- const field = useField('name', { subscription: { dirty: true } })
- return
- }
+ const field = useField("name", { subscription: { dirty: true } });
+ return ;
+ };
const { getByTestId } = render(
{() => (
@@ -121,40 +121,40 @@ describe('useField', () => {
)}
-
- )
- expect(getByTestId('name').value).toBe('')
- expect(getByTestId('dirty')).toHaveTextContent('Pristine')
- expect(spy).toHaveBeenCalledTimes(1)
- expect(spy.mock.calls[0][0]).toBe(false)
+ ,
+ );
+ expect(getByTestId("name").value).toBe("");
+ expect(getByTestId("dirty")).toHaveTextContent("Pristine");
+ expect(spy).toHaveBeenCalledTimes(1);
+ expect(spy.mock.calls[0][0]).toBe(false);
// simulate typing
- fireEvent.change(getByTestId('name'), { target: { value: 'e' } })
- expect(getByTestId('dirty')).toHaveTextContent('Dirty')
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(true)
- fireEvent.change(getByTestId('name'), { target: { value: 'er' } })
- fireEvent.change(getByTestId('name'), { target: { value: 'eri' } })
- fireEvent.change(getByTestId('name'), { target: { value: 'erik' } })
- fireEvent.change(getByTestId('name'), { target: { value: 'erikr' } })
- fireEvent.change(getByTestId('name'), { target: { value: 'erikra' } })
- fireEvent.change(getByTestId('name'), { target: { value: 'erikras' } })
+ fireEvent.change(getByTestId("name"), { target: { value: "e" } });
+ expect(getByTestId("dirty")).toHaveTextContent("Dirty");
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(true);
+ fireEvent.change(getByTestId("name"), { target: { value: "er" } });
+ fireEvent.change(getByTestId("name"), { target: { value: "eri" } });
+ fireEvent.change(getByTestId("name"), { target: { value: "erik" } });
+ fireEvent.change(getByTestId("name"), { target: { value: "erikr" } });
+ fireEvent.change(getByTestId("name"), { target: { value: "erikra" } });
+ fireEvent.change(getByTestId("name"), { target: { value: "erikras" } });
// dirty flag hasn't changed since the first character
- expect(spy).toHaveBeenCalledTimes(2)
+ expect(spy).toHaveBeenCalledTimes(2);
// make pristine again
- fireEvent.change(getByTestId('name'), { target: { value: '' } })
- expect(getByTestId('dirty')).toHaveTextContent('Pristine')
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0]).toBe(false)
- })
-
- it('should give same instance of handlers as value changes', () => {
- const spy = jest.fn()
+ fireEvent.change(getByTestId("name"), { target: { value: "" } });
+ expect(getByTestId("dirty")).toHaveTextContent("Pristine");
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0]).toBe(false);
+ });
+
+ it("should give same instance of handlers as value changes", () => {
+ const spy = jest.fn();
const MyField = ({ name }) => {
- const { input } = useField(name, { subscription: { value: true } })
- const { onChange, onFocus, onBlur } = input
- spy(onChange, onFocus, onBlur)
- return
- }
+ const { input } = useField(name, { subscription: { value: true } });
+ const { onChange, onFocus, onBlur } = input;
+ spy(onChange, onFocus, onBlur);
+ return ;
+ };
render(
{() => (
@@ -162,44 +162,44 @@ describe('useField', () => {
)}
-
- )
+ ,
+ );
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]) // onChange
- expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]) // onFocus
- expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]) // onBlur
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]); // onChange
+ expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]); // onFocus
+ expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]); // onBlur
- const [onChange, onFocus, onBlur] = spy.mock.calls[0]
- const setValue = value => {
+ const [onChange, onFocus, onBlur] = spy.mock.calls[0];
+ const setValue = (value) => {
act(() => {
- onFocus()
- onChange(value)
- onBlur()
- })
- }
-
- setValue('dog')
- expect(spy).toHaveBeenCalledTimes(3)
- expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]) // onChange
- expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]) // onFocus
- expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]) // onBlur
-
- setValue('cat')
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]) // onChange
- expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]) // onFocus
- expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]) // onBlur
- })
-
- it('should give same instance of handlers as name changes', () => {
- const spy = jest.fn()
+ onFocus();
+ onChange(value);
+ onBlur();
+ });
+ };
+
+ setValue("dog");
+ expect(spy).toHaveBeenCalledTimes(3);
+ expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]); // onChange
+ expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]); // onFocus
+ expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]); // onBlur
+
+ setValue("cat");
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]); // onChange
+ expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]); // onFocus
+ expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]); // onBlur
+ });
+
+ it("should give same instance of handlers as name changes", () => {
+ const spy = jest.fn();
const MyField = ({ name }) => {
- const { input } = useField(name, { subscription: { value: true } })
- const { onChange, onFocus, onBlur } = input
- spy(onChange, onFocus, onBlur)
- return
- }
+ const { input } = useField(name, { subscription: { value: true } });
+ const { onChange, onFocus, onBlur } = input;
+ spy(onChange, onFocus, onBlur);
+ return ;
+ };
const { rerender } = render(
{() => (
@@ -207,14 +207,13 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]) // onChange
- expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]) // onFocus
- expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]) // onBlur
+ ,
+ );
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]); // onChange
+ expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]); // onFocus
+ expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]); // onBlur
rerender(
@@ -223,26 +222,26 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]) // onChange
- expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]) // onFocus
- expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]) // onBlur
- expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]) // onChange
- expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]) // onFocus
- expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]) // onBlur
- })
-
- it('should give same instance of handlers as type changes', () => {
- const spy = jest.fn()
+ ,
+ );
+
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]); // onChange
+ expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]); // onFocus
+ expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]); // onBlur
+ expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]); // onChange
+ expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]); // onFocus
+ expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]); // onBlur
+ });
+
+ it("should give same instance of handlers as type changes", () => {
+ const spy = jest.fn();
const MyField = ({ name, type }) => {
- const { input } = useField(name, { subscription: { value: true }, type })
- const { onChange, onFocus, onBlur } = input
- spy(onChange, onFocus, onBlur)
- return
- }
+ const { input } = useField(name, { subscription: { value: true }, type });
+ const { onChange, onFocus, onBlur } = input;
+ spy(onChange, onFocus, onBlur);
+ return ;
+ };
const { rerender } = render(
{() => (
@@ -250,42 +249,44 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]) // onChange
- expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]) // onFocus
- expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]) // onBlur
+ ,
+ );
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]); // onChange
+ expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]); // onFocus
+ expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]); // onBlur
rerender(
{() => (
-
+
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]) // onChange
- expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]) // onFocus
- expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]) // onBlur
- expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]) // onChange
- expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]) // onFocus
- expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]) // onBlur
- })
-
- it('should give same instance of handlers as formatOnBlur changes', () => {
- const spy = jest.fn()
+ ,
+ );
+
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]); // onChange
+ expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]); // onFocus
+ expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]); // onBlur
+ expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]); // onChange
+ expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]); // onFocus
+ expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]); // onBlur
+ });
+
+ it("should give same instance of handlers as formatOnBlur changes", () => {
+ const spy = jest.fn();
const MyField = ({ name, formatOnBlur }) => {
- const { input } = useField(name, { subscription: { value: true }, formatOnBlur })
- const { onChange, onFocus, onBlur } = input
- spy(onChange, onFocus, onBlur)
- return
- }
+ const { input } = useField(name, {
+ subscription: { value: true },
+ formatOnBlur,
+ });
+ const { onChange, onFocus, onBlur } = input;
+ spy(onChange, onFocus, onBlur);
+ return ;
+ };
const { rerender } = render(
{() => (
@@ -293,14 +294,13 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]) // onChange
- expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]) // onFocus
- expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]) // onBlur
+ ,
+ );
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]); // onChange
+ expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]); // onFocus
+ expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]); // onBlur
rerender(
@@ -309,26 +309,29 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]) // onChange
- expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]) // onFocus
- expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]) // onBlur
- expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]) // onChange
- expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]) // onFocus
- expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]) // onBlur
- })
-
- it('should give same instance of handlers as parse changes', () => {
- const spy = jest.fn()
+ ,
+ );
+
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]); // onChange
+ expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]); // onFocus
+ expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]); // onBlur
+ expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]); // onChange
+ expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]); // onFocus
+ expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]); // onBlur
+ });
+
+ it("should give same instance of handlers as parse changes", () => {
+ const spy = jest.fn();
const MyField = ({ name, parse }) => {
- const { input } = useField(name, { subscription: { value: true }, parse })
- const { onChange, onFocus, onBlur } = input
- spy(onChange, onFocus, onBlur)
- return
- }
+ const { input } = useField(name, {
+ subscription: { value: true },
+ parse,
+ });
+ const { onChange, onFocus, onBlur } = input;
+ spy(onChange, onFocus, onBlur);
+ return ;
+ };
const { rerender } = render(
{() => (
@@ -336,14 +339,13 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]) // onChange
- expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]) // onFocus
- expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]) // onBlur
+ ,
+ );
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]); // onChange
+ expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]); // onFocus
+ expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]); // onBlur
rerender(
@@ -352,26 +354,29 @@ describe('useField', () => {
x} />
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]) // onChange
- expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]) // onFocus
- expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]) // onBlur
- expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]) // onChange
- expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]) // onFocus
- expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]) // onBlur
- })
-
- it('should give same instance of handlers as format changes', () => {
- const spy = jest.fn()
+ ,
+ );
+
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]); // onChange
+ expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]); // onFocus
+ expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]); // onBlur
+ expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]); // onChange
+ expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]); // onFocus
+ expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]); // onBlur
+ });
+
+ it("should give same instance of handlers as format changes", () => {
+ const spy = jest.fn();
const MyField = ({ name, format }) => {
- const { input } = useField(name, { subscription: { value: true }, format })
- const { onChange, onFocus, onBlur } = input
- spy(onChange, onFocus, onBlur)
- return
- }
+ const { input } = useField(name, {
+ subscription: { value: true },
+ format,
+ });
+ const { onChange, onFocus, onBlur } = input;
+ spy(onChange, onFocus, onBlur);
+ return ;
+ };
const { rerender } = render(
{() => (
@@ -379,14 +384,13 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]) // onChange
- expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]) // onFocus
- expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]) // onBlur
+ ,
+ );
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]); // onChange
+ expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]); // onFocus
+ expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]); // onBlur
rerender(
@@ -395,26 +399,29 @@ describe('useField', () => {
x} />
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]) // onChange
- expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]) // onFocus
- expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]) // onBlur
- expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]) // onChange
- expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]) // onFocus
- expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]) // onBlur
- })
-
- it('should give same instance of handlers as component changes', () => {
- const spy = jest.fn()
+ ,
+ );
+
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]); // onChange
+ expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]); // onFocus
+ expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]); // onBlur
+ expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]); // onChange
+ expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]); // onFocus
+ expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]); // onBlur
+ });
+
+ it("should give same instance of handlers as component changes", () => {
+ const spy = jest.fn();
const MyField = ({ name, component }) => {
- const { input } = useField(name, { subscription: { value: true }, component })
- const { onChange, onFocus, onBlur } = input
- spy(onChange, onFocus, onBlur)
- return
- }
+ const { input } = useField(name, {
+ subscription: { value: true },
+ component,
+ });
+ const { onChange, onFocus, onBlur } = input;
+ spy(onChange, onFocus, onBlur);
+ return ;
+ };
const { rerender } = render(
{() => (
@@ -422,14 +429,13 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(2)
- expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]) // onChange
- expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]) // onFocus
- expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]) // onBlur
+ ,
+ );
+ expect(spy).toHaveBeenCalledTimes(2);
+ expect(spy.mock.calls[1][0]).toBe(spy.mock.calls[0][0]); // onChange
+ expect(spy.mock.calls[1][1]).toBe(spy.mock.calls[0][1]); // onFocus
+ expect(spy.mock.calls[1][2]).toBe(spy.mock.calls[0][2]); // onBlur
rerender(
@@ -438,15 +444,15 @@ describe('useField', () => {
)}
-
- )
-
- expect(spy).toHaveBeenCalledTimes(4)
- expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]) // onChange
- expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]) // onFocus
- expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]) // onBlur
- expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]) // onChange
- expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]) // onFocus
- expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]) // onBlur
- })
-})
+ ,
+ );
+
+ expect(spy).toHaveBeenCalledTimes(4);
+ expect(spy.mock.calls[2][0]).toBe(spy.mock.calls[1][0]); // onChange
+ expect(spy.mock.calls[2][1]).toBe(spy.mock.calls[1][1]); // onFocus
+ expect(spy.mock.calls[2][2]).toBe(spy.mock.calls[1][2]); // onBlur
+ expect(spy.mock.calls[3][0]).toBe(spy.mock.calls[2][0]); // onChange
+ expect(spy.mock.calls[3][1]).toBe(spy.mock.calls[2][1]); // onFocus
+ expect(spy.mock.calls[3][2]).toBe(spy.mock.calls[2][2]); // onBlur
+ });
+});
diff --git a/src/useForm.js b/src/useForm.js
index 6a1251df..9e319012 100644
--- a/src/useForm.js
+++ b/src/useForm.js
@@ -1,18 +1,18 @@
// @flow
-import * as React from 'react'
-import type { FormApi, FormValuesShape } from 'final-form'
-import ReactFinalFormContext from './context'
+import * as React from "react";
+import type { FormApi, FormValuesShape } from "final-form";
+import ReactFinalFormContext from "./context";
function useForm(
- componentName?: string
+ componentName?: string,
): FormApi {
- const form: ?FormApi = React.useContext(ReactFinalFormContext)
+ const form: ?FormApi = React.useContext(ReactFinalFormContext);
if (!form) {
throw new Error(
- `${componentName || 'useForm'} must be used inside of a component`
- )
+ `${componentName || "useForm"} must be used inside of a component`,
+ );
}
- return form
+ return form;
}
-export default useForm
+export default useForm;
diff --git a/src/useForm.test.js b/src/useForm.test.js
index 73e9591e..262a7f89 100644
--- a/src/useForm.test.js
+++ b/src/useForm.test.js
@@ -1,70 +1,70 @@
// @flow
-import React from 'react'
-import { render, cleanup } from '@testing-library/react'
-import '@testing-library/jest-dom/extend-expect'
-import { ErrorBoundary } from './testUtils'
-import Form from './ReactFinalForm'
-import { useForm } from './index'
+import React from "react";
+import { render, cleanup } from "@testing-library/react";
+import "@testing-library/jest-dom/extend-expect";
+import { ErrorBoundary } from "./testUtils";
+import Form from "./ReactFinalForm";
+import { useForm } from "./index";
-const onSubmitMock = values => {}
+const onSubmitMock = (values) => {};
-describe('useForm', () => {
- afterEach(cleanup)
+describe("useForm", () => {
+ afterEach(cleanup);
- it('should warn if not used inside a form', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ it("should warn if not used inside a form", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
const MyFormConsumer = () => {
- useForm()
- return
- }
+ useForm();
+ return
;
+ };
render(
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'useForm must be used inside of a component'
- )
- console.error.mockRestore()
- })
+ "useForm must be used inside of a component",
+ );
+ console.error.mockRestore();
+ });
- it('should warn with component name if not used inside a form', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ it("should warn with component name if not used inside a form", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
const MyFormConsumer = () => {
- useForm('MyFormConsumer')
- return
- }
+ useForm("MyFormConsumer");
+ return
;
+ };
render(
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'MyFormConsumer must be used inside of a component'
- )
- console.error.mockRestore()
- })
+ "MyFormConsumer must be used inside of a component",
+ );
+ console.error.mockRestore();
+ });
- it('should produce form if used inside ', () => {
+ it("should produce form if used inside ", () => {
const MyFormConsumer = () => {
- const form = useForm()
- expect(form).toBeDefined()
- expect(typeof form.change).toBe('function')
- expect(typeof form.reset).toBe('function')
+ const form = useForm();
+ expect(form).toBeDefined();
+ expect(typeof form.change).toBe("function");
+ expect(typeof form.reset).toBe("function");
return (
- {form ? 'Got a form!' : 'No form!'}
- )
- }
+ {form ? "Got a form!" : "No form!"}
+ );
+ };
const { getByTestId } = render(
- {() => }
- )
- expect(getByTestId('formCheck')).toHaveTextContent('Got a form!')
- })
-})
+ {() => } ,
+ );
+ expect(getByTestId("formCheck")).toHaveTextContent("Got a form!");
+ });
+});
diff --git a/src/useFormState.js b/src/useFormState.js
index 405a379e..3602d62e 100644
--- a/src/useFormState.js
+++ b/src/useFormState.js
@@ -1,52 +1,52 @@
// @flow
-import * as React from 'react'
-import type { UseFormStateParams } from './types'
-import type { FormState, FormApi, FormValuesShape } from 'final-form'
-import { all } from './ReactFinalForm'
-import useForm from './useForm'
-import { addLazyFormState } from './getters'
+import * as React from "react";
+import type { UseFormStateParams } from "./types";
+import type { FormState, FormApi, FormValuesShape } from "final-form";
+import { all } from "./ReactFinalForm";
+import useForm from "./useForm";
+import { addLazyFormState } from "./getters";
function useFormState({
onChange,
- subscription = all
+ subscription = all,
}: UseFormStateParams = {}): FormState {
- const form: FormApi = useForm('useFormState')
- const firstRender = React.useRef(true)
- const onChangeRef = React.useRef(onChange)
- onChangeRef.current = onChange
+ const form: FormApi = useForm("useFormState");
+ const firstRender = React.useRef(true);
+ const onChangeRef = React.useRef(onChange);
+ onChangeRef.current = onChange;
// synchronously register and unregister to query field state for our subscription on first render
const [state, setState] = React.useState>(
(): FormState => {
- let initialState: FormState = {}
- form.subscribe(state => {
- initialState = state
- }, subscription)()
+ let initialState: FormState = {};
+ form.subscribe((state) => {
+ initialState = state;
+ }, subscription)();
if (onChange) {
- onChange(initialState)
+ onChange(initialState);
}
- return initialState
- }
- )
+ return initialState;
+ },
+ );
React.useEffect(
() =>
- form.subscribe(newState => {
+ form.subscribe((newState) => {
if (firstRender.current) {
- firstRender.current = false
+ firstRender.current = false;
} else {
- setState(newState)
+ setState(newState);
if (onChangeRef.current) {
- onChangeRef.current(newState)
+ onChangeRef.current(newState);
}
}
}, subscription),
// eslint-disable-next-line react-hooks/exhaustive-deps
- []
- )
- const lazyState = {}
- addLazyFormState(lazyState, state)
- return lazyState
+ [],
+ );
+ const lazyState = {};
+ addLazyFormState(lazyState, state);
+ return lazyState;
}
-export default useFormState
+export default useFormState;
diff --git a/src/useFormState.test.js b/src/useFormState.test.js
index 2a3fa88d..71eb6226 100644
--- a/src/useFormState.test.js
+++ b/src/useFormState.test.js
@@ -1,42 +1,42 @@
// @flow
-import React from 'react'
-import { render, cleanup } from '@testing-library/react'
-import { ErrorBoundary } from './testUtils'
-import { useFormState, Form } from './index'
+import React from "react";
+import { render, cleanup } from "@testing-library/react";
+import { ErrorBoundary } from "./testUtils";
+import { useFormState, Form } from "./index";
-describe('useField', () => {
- afterEach(cleanup)
+describe("useField", () => {
+ afterEach(cleanup);
// Most of the functionality of useFormState is tested in FormSpy.test.js
// This file is only for testing its use as a hook in other components
- it('should warn if not used inside a form', () => {
- jest.spyOn(console, 'error').mockImplementation(() => {})
- const errorSpy = jest.fn()
+ it("should warn if not used inside a form", () => {
+ jest.spyOn(console, "error").mockImplementation(() => {});
+ const errorSpy = jest.fn();
const MyFormStateComponent = () => {
- useFormState()
- return
- }
+ useFormState();
+ return
;
+ };
render(
-
- )
- expect(errorSpy).toHaveBeenCalled()
- expect(errorSpy).toHaveBeenCalledTimes(1)
+ ,
+ );
+ expect(errorSpy).toHaveBeenCalled();
+ expect(errorSpy).toHaveBeenCalledTimes(1);
expect(errorSpy.mock.calls[0][0].message).toBe(
- 'useFormState must be used inside of a component'
- )
- console.error.mockRestore()
- })
+ "useFormState must be used inside of a component",
+ );
+ console.error.mockRestore();
+ });
- it('state should be enumerable', () => {
+ it("state should be enumerable", () => {
const Test = () => {
- const state = useFormState()
- expect(Object.keys(state).length > 0).toBe(true)
- return It worked
- }
+ const state = useFormState();
+ expect(Object.keys(state).length > 0).toBe(true);
+ return It worked
;
+ };
render(
{}}>
{({ handleSubmit }) => (
@@ -44,7 +44,7 @@ describe('useField', () => {
)}
-
- )
- })
-})
+ ,
+ );
+ });
+});
diff --git a/src/useLatest.js b/src/useLatest.js
index c39e27a8..05d1854a 100644
--- a/src/useLatest.js
+++ b/src/useLatest.js
@@ -1,12 +1,12 @@
// @flow
-import React from 'react'
+import React from "react";
export default function useLatest(value: T): { +current: T } {
- const ref = React.useRef(value)
+ const ref = React.useRef(value);
React.useEffect(() => {
- ref.current = value
- })
+ ref.current = value;
+ });
- return ref
+ return ref;
}
diff --git a/src/useWhenValueChanges.js b/src/useWhenValueChanges.js
index 812e24ef..471af8a7 100644
--- a/src/useWhenValueChanges.js
+++ b/src/useWhenValueChanges.js
@@ -1,16 +1,16 @@
// @flow
-import React from 'react'
+import React from "react";
export default function useWhenValueChanges(
value: any,
callback: () => void,
- isEqual: (any, any) => boolean = (a, b) => a === b
+ isEqual: (any, any) => boolean = (a, b) => a === b,
) {
- const previous = React.useRef(value)
+ const previous = React.useRef(value);
React.useEffect(() => {
if (!isEqual(value, previous.current)) {
- callback()
- previous.current = value
+ callback();
+ previous.current = value;
}
- })
+ });
}
diff --git a/typescript/Field.test.tsx b/typescript/Field.test.tsx
index d0270e29..5f12bf37 100644
--- a/typescript/Field.test.tsx
+++ b/typescript/Field.test.tsx
@@ -1,6 +1,6 @@
/* eslint-disable @typescript-eslint/no-unused-vars */
-import * as React from 'react';
-import { FieldRenderProps } from 'react-final-form';
+import * as React from "react";
+import { FieldRenderProps } from "react-final-form";
function FormText1({ input }: FieldRenderProps) {
// renders OK because of the used generic
diff --git a/typescript/FormSpy.test.tsx b/typescript/FormSpy.test.tsx
index 8e2e43e1..d3eb81af 100644
--- a/typescript/FormSpy.test.tsx
+++ b/typescript/FormSpy.test.tsx
@@ -1,6 +1,6 @@
/* eslint-disable @typescript-eslint/no-unused-vars */
-import * as React from 'react';
-import { FormSpy } from 'react-final-form';
+import * as React from "react";
+import { FormSpy } from "react-final-form";
function submitButtonSpy() {
return (
diff --git a/typescript/ReactFinalForm.test.tsx b/typescript/ReactFinalForm.test.tsx
index 8144e4e9..bbe7445c 100644
--- a/typescript/ReactFinalForm.test.tsx
+++ b/typescript/ReactFinalForm.test.tsx
@@ -1,9 +1,9 @@
/* tslint:disable: no-shadowed-variable */
/* eslint-disable @typescript-eslint/no-unused-vars */
-import { Decorator, Mutator } from 'final-form';
-import * as React from 'react';
-import { Field, Form } from 'react-final-form';
+import { Decorator, Mutator } from "final-form";
+import * as React from "react";
+import { Field, Form } from "react-final-form";
const noop = () => {};
// missing required props
@@ -77,7 +77,7 @@ function simpleSubscription() {
subscription={{
pristine: true,
submitting: true,
- values: true
+ values: true,
}}
>
{({ handleSubmit, form, submitting, pristine, values }) => (
@@ -97,7 +97,7 @@ function simpleSubscription() {
}
const setValue: Mutator = ([name, newValue], state, { changeValue }) => {
- changeValue(state, name, value => newValue);
+ changeValue(state, name, (value) => newValue);
};
function mutated() {
@@ -106,11 +106,11 @@ function mutated() {
{({
handleSubmit,
form: {
- mutators: { setValue }
+ mutators: { setValue },
},
submitting,
pristine,
- values
+ values,
}) => (
setValue('firstName', 'Kevin')}
+ onClick={(e) => setValue("firstName", "Kevin")}
disabled={submitting || pristine}
>
Reset
@@ -165,9 +165,9 @@ function withTypedFormData() {
);
}
-const decorator: Decorator = form => {
+const decorator: Decorator = (form) => {
return form.subscribe(({ values: { firstName } }) => firstName, {
- values: true
+ values: true,
});
};
@@ -179,7 +179,7 @@ function withTypedDecorator() {
// with wrong typed decorator
function withWrongTypedDecorator() {
return (
- >
+ >
// $ExpectError
decorators={[decorator]}
onSubmit={noop}
diff --git a/typescript/index.d.ts b/typescript/index.d.ts
index 055f1365..1a78c99d 100644
--- a/typescript/index.d.ts
+++ b/typescript/index.d.ts
@@ -1,4 +1,4 @@
-import * as React from 'react';
+import * as React from "react";
import {
FormApi,
Config,
@@ -7,12 +7,15 @@ import {
FormSubscription,
FieldState,
FieldSubscription,
- FieldValidator
-} from 'final-form';
+ FieldValidator,
+} from "final-form";
-type SupportedInputs = 'input' | 'select' | 'textarea';
+type SupportedInputs = "input" | "select" | "textarea";
-export interface ReactContext, InitialFormValues = Partial> {
+export interface ReactContext<
+ FormValues = Record,
+ InitialFormValues = Partial,
+> {
reactFinalForm: FormApi;
}
@@ -20,7 +23,7 @@ export type FieldMetaState = Pick<
FieldState,
Exclude<
keyof FieldState,
- 'blur' | 'change' | 'focus' | 'name' | 'value'
+ "blur" | "change" | "focus" | "name" | "value"
>
>;
@@ -42,26 +45,30 @@ interface AnyObject {
export interface FieldRenderProps<
FieldValue,
- T extends HTMLElement = HTMLElement
+ T extends HTMLElement = HTMLElement,
> {
input: FieldInputProps;
meta: FieldMetaState;
[otherProp: string]: any;
}
-export interface FormRenderProps, InitialFormValues = Partial>
- extends FormState,
+export interface FormRenderProps<
+ FormValues = Record,
+ InitialFormValues = Partial,
+> extends FormState,
RenderableProps> {
form: FormApi;
handleSubmit: (
event?: Partial<
- Pick
- >
+ Pick
+ >,
) => Promise | undefined;
}
-export interface FormSpyRenderProps, InitialFormValues = Partial>
- extends FormState {
+export interface FormSpyRenderProps<
+ FormValues = Record,
+ InitialFormValues = Partial,
+> extends FormState {
form: FormApi;
}
@@ -71,8 +78,10 @@ export interface RenderableProps {
render?: (props: T) => React.ReactNode;
}
-export interface FormProps, InitialFormValues = Partial>
- extends Config,
+export interface FormProps<
+ FormValues = Record,
+ InitialFormValues = Partial,
+> extends Config,
RenderableProps> {
subscription?: FormSubscription;
decorators?: Array>;
@@ -103,19 +112,25 @@ export interface UseFieldConfig {
export interface FieldProps<
FieldValue,
RP extends FieldRenderProps,
- T extends HTMLElement = HTMLElement
-> extends UseFieldConfig, RenderableProps {
+ T extends HTMLElement = HTMLElement,
+> extends UseFieldConfig,
+ RenderableProps {
name: string;
[otherProp: string]: any;
}
-export interface UseFormStateParams, InitialFormValues = Partial> {
+export interface UseFormStateParams<
+ FormValues = Record,
+ InitialFormValues = Partial,
+> {
onChange?: (formState: FormState) => void;
subscription?: FormSubscription;
}
-export interface FormSpyProps, InitialFormValues = Partial>
- extends UseFormStateParams,
+export interface FormSpyProps<
+ FormValues = Record,
+ InitialFormValues = Partial,
+> extends UseFormStateParams,
RenderableProps> {}
export const Field: <
@@ -124,27 +139,40 @@ export const Field: <
FieldValue,
HTMLElement
>,
- T extends HTMLElement = HTMLElement
+ T extends HTMLElement = HTMLElement,
>(
- props: FieldProps
+ props: FieldProps,
) => React.ReactElement;
-export const Form: , InitialFormValues = Partial>(
- props: FormProps
+export const Form: <
+ FormValues = Record,
+ InitialFormValues = Partial,
+>(
+ props: FormProps,
) => React.ReactElement;
-export const FormSpy: , InitialFormValues = Partial>(
- props: FormSpyProps
+export const FormSpy: <
+ FormValues = Record,
+ InitialFormValues = Partial,
+>(
+ props: FormSpyProps,
) => React.ReactElement;
export function useField(
name: string,
- config?: UseFieldConfig
+ config?: UseFieldConfig,
): FieldRenderProps;
-export function useForm, InitialFormValues = Partial>(
- componentName?: string
-): FormApi;
-export function useFormState, InitialFormValues = Partial>(
- params?: UseFormStateParams
+export function useForm<
+ FormValues = Record,
+ InitialFormValues = Partial,
+>(componentName?: string): FormApi;
+export function useFormState<
+ FormValues = Record,
+ InitialFormValues = Partial,
+>(
+ params?: UseFormStateParams,
): FormState;
-export function withTypes, InitialFormValues = Partial>(): {
+export function withTypes<
+ FormValues = Record,
+ InitialFormValues = Partial,
+>(): {
Form: React.FC>;
FormSpy: React.FC>;
};
diff --git a/typescript/useFormState.test.tsx b/typescript/useFormState.test.tsx
index 3e333bc7..703d9c6b 100644
--- a/typescript/useFormState.test.tsx
+++ b/typescript/useFormState.test.tsx
@@ -1,7 +1,7 @@
/* eslint-disable @typescript-eslint/no-unused-vars */
-import { useFormState } from 'react-final-form';
+import { useFormState } from "react-final-form";
-const submittingToLabel = (submitting: boolean) => (submitting ? 'Yes' : 'No');
+const submittingToLabel = (submitting: boolean) => (submitting ? "Yes" : "No");
function Comp1() {
const { submitting } = useFormState();
From 480847e5a702aba981ce54ae0de4483c86f0f049 Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 12:35:05 +0200
Subject: [PATCH 5/8] Node version
---
package.json | 3 +++
1 file changed, 3 insertions(+)
diff --git a/package.json b/package.json
index fd2db392..f36432c5 100644
--- a/package.json
+++ b/package.json
@@ -28,6 +28,9 @@
"url": "https://github.com/final-form/react-final-form/issues"
},
"homepage": "https://github.com/final-form/react-final-form#readme",
+ "engines": {
+ "node": "14"
+ },
"devDependencies": {
"@babel/core": "^7.15.5",
"@babel/plugin-proposal-class-properties": "^7.14.5",
From 953ecce098c17a307360c5c457d37b85f8552c44 Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 12:40:14 +0200
Subject: [PATCH 6/8] Travis and Github actions
---
.github/ci.yml | 2 ++
.travis.yml | 8 ++++----
2 files changed, 6 insertions(+), 4 deletions(-)
diff --git a/.github/ci.yml b/.github/ci.yml
index d93f882f..9b1f7f9e 100644
--- a/.github/ci.yml
+++ b/.github/ci.yml
@@ -47,3 +47,5 @@ jobs:
run: yarn install --ignore-scripts --frozen-lockfile
- name: Run unit tests
run: yarn start test
+ - name: Run code coverage
+ run: npx codecov
diff --git a/.travis.yml b/.travis.yml
index 93ba8100..a446f863 100644
--- a/.travis.yml
+++ b/.travis.yml
@@ -8,13 +8,13 @@ cache:
notifications:
email: false
node_js:
- - '10'
- - '12'
- - '14'
+ - "10"
+ - "12"
+ - "14"
script:
- npm start validate
after_success:
- npx codecov
branches:
only:
- - master
+ - main
From 3a2a32bbd844fe9362892633fb34cafdd1997760 Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 13:00:44 +0200
Subject: [PATCH 7/8] Removed glow dep
---
package.json | 1 -
1 file changed, 1 deletion(-)
diff --git a/package.json b/package.json
index f36432c5..a421cb8c 100644
--- a/package.json
+++ b/package.json
@@ -70,7 +70,6 @@
"fast-deep-equal": "^3.1.3",
"final-form": "4.20.2",
"flow-bin": "^0.160.2",
- "glow": "^1.2.2",
"husky": "^4.3.0",
"jest": "^27.2.1",
"jest-mock-console": "^1.1.0",
From d73a998cf3873c9ac2676b2bf2fba721fbd2cb96 Mon Sep 17 00:00:00 2001
From: Erik Rasmussen
Date: Sat, 25 Sep 2021 13:01:42 +0200
Subject: [PATCH 8/8] Maybe fix CS CI bug?
---
.codesandbox/ci.json | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/.codesandbox/ci.json b/.codesandbox/ci.json
index d47c20ea..bbaf85e8 100644
--- a/.codesandbox/ci.json
+++ b/.codesandbox/ci.json
@@ -5,5 +5,6 @@
"/examples/field-level-validation",
"/examples/submission-errors",
"/examples/subscriptions"
- ]
+ ],
+ "node": "14"
}