Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

WIP: Formatic 3000 #124

Open
wants to merge 44 commits into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 1 commit
Commits
Show all changes
44 commits
Select commit Hold shift + click to select a range
230f394
Draft of futureland spec
jdeal Apr 6, 2019
8eec2ce
Add future demo page
jdeal Apr 6, 2019
cebd6fa
Add simple form example/test/code.
jdeal Apr 6, 2019
8f390f4
FormContainer works either as controlled/uncontrolled
jdeal Apr 6, 2019
2c827b3
Rename some files
jdeal Apr 6, 2019
c5d0bc8
Fix test descriptions
jdeal Apr 6, 2019
6739955
Add FieldContainer example/test/code
jdeal Apr 6, 2019
f23a600
Add a built-in TextInput
jdeal Apr 6, 2019
3fe0ad1
Add built-in TextField per the spec
jdeal Apr 6, 2019
f193215
Cleanup tests
jdeal Apr 6, 2019
af6e3e0
Auto-generate unique input ids
jdeal Apr 6, 2019
0fa226e
Clean up test descriptions
jdeal Apr 6, 2019
067a500
Add support for renderTag
jdeal Apr 9, 2019
8878def
Fix linting errors
jdeal Apr 9, 2019
0f4c5d5
Remove reliance on symbols for renderKeys
jdeal Apr 9, 2019
0c571f3
Add accessibilty to problems/goals
jdeal Apr 9, 2019
018b2b6
Add support for renderComponent
jdeal Apr 10, 2019
221b69c
Split up monolithic index
jdeal Apr 10, 2019
2476fc3
Support nested values
jdeal Apr 10, 2019
ace4df5
Simplify UncontrolledContainer
jdeal Apr 11, 2019
565da6f
Remove private components from API
jdeal Apr 11, 2019
be8de3c
Make sure uncontrolled demo forms do not rerender
jdeal Apr 20, 2019
7e5f9bd
Check if `onChange` is a function before calling it
jdeal Apr 20, 2019
abe9f0c
Add failing test for avoiding rerendering
jdeal Apr 20, 2019
737daa6
Add ReactiveValueContainer/useReactiveValue primitives
jdeal Apr 24, 2019
dfc5f54
Use ReactiveValue to implement more performant useField
jdeal Apr 24, 2019
9cc81ac
Add IntegerField/IntegerInput
jdeal Apr 25, 2019
9218c37
Remove unused fields
jdeal Apr 25, 2019
c84bc32
Add AutoFields. Move field-components to their own folder.
Apr 27, 2019
6aeaf24
Remove AutoFields default in FormContainer
Apr 29, 2019
7d7a3ad
Better tests
Apr 29, 2019
9b6362d
Use RenderContext instead of ReactiveValue to get initial form value
Apr 29, 2019
0185cdd
Rename files/dirs
Apr 29, 2019
089c4f3
Merge pull request #126 from zapier/add-autofields
jdeal May 4, 2019
06c06b4
Allow using the whole value
jdeal May 4, 2019
4bc8326
Just a naming tweak
jdeal May 4, 2019
65f7c08
Even more performance ReactiveValue with metadata
jdeal May 4, 2019
87660c8
Use useReactiveValueMeta for AutoFields
jdeal May 4, 2019
a4a0242
Update meta when root value changes
May 9, 2019
4d3981b
Bump zapier-scripts to current
May 9, 2019
302cf96
Add jest-dom and convert ReactiveValue tests
May 9, 2019
0b80b23
Add eslint override to allow react-create-class
May 9, 2019
a08c04a
Ensure we have meta when updating
May 10, 2019
5e9b175
Merge pull request #127 from zapier/notify-meta-on-value
stevelikesmusic May 10, 2019
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
17 changes: 11 additions & 6 deletions src/future/builtin-fields/AutoFields.js
Original file line number Diff line number Diff line change
@@ -1,17 +1,22 @@
import React, { useContext } from 'react';
import { RenderContext } from '@/src/future/context';
import React from 'react';
import { useReactiveValueMeta } from '@/src/future/ReactiveValue';
import { startCase } from '@/src/stringUtils';

const inferSchema = ([key, value]) => ({
const inferSchema = ({ key, type }) => ({
key,
label: startCase(key),
type: typeof value,
type,
});

export default function createAutoFields(defaultFieldComponents) {
return function AutoFields(props) {
const { initialValue: formValues } = useContext(RenderContext);
const schema = Object.entries(formValues).map(inferSchema);
const meta = useReactiveValueMeta();
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@stevelikesmusic I made an even more performant ReactiveValue. (Unfortunately, it did get a bit more complex, but it's still nicely isolated behind hooks.) And as part of that, I added a useReactiveValueMeta that I swapped out here. It gives you access to property types and only updates when those types change. This is also dynamic, so new properties that are added will generate new auto fields.

I think this may be roughly how validation errors will work once we get to that. Effectively, validation errors will just be extra metadata this sits alongside the values.

const schema = Object.keys(meta.propertyTypes).map(key =>
inferSchema({
key,
type: meta.propertyTypes[key],
})
);
const components = { ...defaultFieldComponents, ...props.components };

return schema.map(field => {
Expand Down
8 changes: 4 additions & 4 deletions src/future/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -24,10 +24,10 @@ export function FormContainer({
const [isControlled] = useState(savedDefaultValue === undefined);

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do we need to put this in state? I could see caching the initial value of props.defaultValue.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's effectively what I'm doing here. When you use useState without changing the value, it's effectively like using a constructor in a class-based component. I'm saving the value for later rather than mutating it every time. The value passed to useState is the initial value and will never change here.

const normalizedValue = isControlled ? value : savedDefaultValue;
// Keep context object pure.
const renderContext = useMemo(
() => ({ renderTag, renderComponent, initialValue: normalizedValue }),
[renderTag, renderComponent]
);
const renderContext = useMemo(() => ({ renderTag, renderComponent }), [
renderTag,
renderComponent,
]);
return (
<RenderContext.Provider value={renderContext}>
<ReactiveValueContainer onChange={onChange} value={normalizedValue}>
Expand Down