Skip to content

Commit d3c0c38

Browse files
authored
Merge pull request #47 from qiniu/v3.x
v3
2 parents fb51187 + 2363187 commit d3c0c38

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

70 files changed

+22303
-11900
lines changed

.github/workflows/ci.yml

+5-3
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,11 @@ name: CI
33
on:
44
push:
55
branches:
6+
- v2.x
67
- master
78
pull_request:
89
branches:
10+
- v2.x
911
- master
1012

1113
jobs:
@@ -15,10 +17,10 @@ jobs:
1517

1618
strategy:
1719
matrix:
18-
node-version: [12.x, 14.x, 16.x]
20+
node-version: [16.x]
1921

2022
steps:
21-
- uses: actions/checkout@v1
23+
- uses: actions/checkout@v2
2224
- name: Use Node.js ${{ matrix.node-version }}
2325
uses: actions/setup-node@v1
2426
with:
@@ -30,7 +32,7 @@ jobs:
3032
env:
3133
CI: true
3234
- name: Collect coverage info to Coveralls
33-
uses: coverallsapp/github-action@v1.0.1
35+
uses: coverallsapp/github-action@1.1.3
3436
with:
3537
github-token: ${{ secrets.GITHUB_TOKEN }}
3638
path-to-lcov: ./coverage/lcov.info

.github/workflows/doc.yml

+10-8
Original file line numberDiff line numberDiff line change
@@ -15,11 +15,13 @@ jobs:
1515
- name: Use Node.js
1616
uses: actions/setup-node@v1
1717
with:
18-
node-version: 12.x
19-
- name: Generate and deploy document
20-
uses: JamesIves/[email protected]
21-
env:
22-
BRANCH: gh-pages
23-
FOLDER: docs
24-
BUILD_SCRIPT: npm ci && npm run build:doc
25-
ACCESS_TOKEN: ${{ secrets.ACCESS_TOKEN }}
18+
node-version: 16.x
19+
- name: Build docs
20+
run: |
21+
npm ci
22+
npm run build:doc
23+
- name: Deploy docs
24+
uses: JamesIves/[email protected]
25+
with:
26+
branch: gh-pages
27+
folder: dumi/dist/formstate-x

.github/workflows/tag.yml

+1
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ name: Tag
33
on:
44
push:
55
branches:
6+
- v2.x
67
- master
78

89
jobs:

.gitignore

+5-4
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
node_modules
2-
lib
3-
esm
4-
coverage
5-
docs
2+
/lib
3+
/esm
4+
/coverage
5+
.umi
6+
.vscode

.npmignore

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,3 @@
1-
docs
1+
dumi
22
jest.config.js
33
tsconfig.json

README.md

+18-49
Original file line numberDiff line numberDiff line change
@@ -5,62 +5,31 @@
55
[![](https://github.com/qiniu/formstate-x/workflows/Doc/badge.svg)](https://github.com/qiniu/formstate-x/actions?query=workflow%3ADoc+branch%3Amaster)
66
[![](https://github.com/qiniu/formstate-x/workflows/Publish/badge.svg)](https://github.com/qiniu/formstate-x/actions?query=workflow%3APublish+branch%3Amaster)
77

8-
Manage the state of your form with ease, inspired by awesome [formstate](https://github.com/formstate/formstate). formstate-x maintains and validates form state for you, totally automatically.
8+
formstate-x is a tool to help you manage form state, based on [MobX](https://mobx.js.org/). formstate-x provides:
99

10-
What we offer:
11-
12-
* **Type safety**: written in [Typescript](https://typescriptlang.org), you can compose complex form state without loss of type information
13-
14-
![Demo](./assets/demo.gif)
15-
16-
* **Reactive**: based on [MobX](https://mobx.js.org), every dependency's change causes reaction automatically and you can easily react to state's change
10+
* **Composability**: Forms are composition of inputs, complex inputs are composition of simpler inputs. With composability provided by formstate-x, you can build arbitrary complex forms or input components with maximal code reuse.
11+
* **Type safety**: With Typescript, no matter how complex or dynamic the form logic is, you can get type-safe result for value, error, validator, etc.
12+
* **Reactive validation**: With reactivity system of MobX, every dependency change triggers validation automatically and you can easily react to state change
1713
* **UI-independent**: formstate-x only deals with state / data, you can easily use it with any UI library you like
1814

19-
### Documents
20-
21-
Full documents [here](https://qiniu.github.io/formstate-x).
22-
23-
### Install
24-
25-
```shell
26-
npm i formstate-x
27-
// or
28-
yarn add formstate-x
29-
```
30-
31-
### Usage
15+
### Documentation
3216

33-
```javascript
34-
import { FieldState, FormState, bindInput } from 'formstate-x'
17+
You can find full documentation [here](https://qiniu.github.io/formstate-x/).
3518

36-
const foo = new FieldState('')
37-
const bar = new FieldState(0)
38-
const form = new FormState({ foo, bar })
19+
### Contributing
3920

40-
// handle submit
41-
async function handleSubmit(e) {
42-
e.preventDefault()
43-
const result = await form.validate()
44-
if (result.hasError) {
45-
alert('Invalid input!')
46-
return
47-
}
48-
// use the validated value
49-
await submitted = submit(result.value)
50-
}
21+
1. Fork the repo and clone the forked repo
22+
2. Install dependencies
5123

52-
// when render (with react)
53-
<form onSubmit={handleSubmit}>
54-
<FooInput {...bindInput(form.$.foo)}>
55-
<BarInput {...bindInput(form.$.bar)}>
56-
</form>
57-
```
24+
```shell
25+
npm i
26+
```
5827

59-
### Comparison with [formstate](https://github.com/formstate/formstate)
28+
3. Edit the code
29+
4. Do lint & unit test
6030

61-
formstate-x provides similar APIs with [formstate](https://github.com/formstate/formstate) because we like their simplicity. formstate has a very helpful document which we will recommend you to read. But there are some points of formstate that brought inconvenience to our development, and we got disagreement with formstate in related problems. That's why we build formstate-x:
31+
```shell
32+
npm run validate
33+
```
6234

63-
1. formstate uses MobX but not embracing it, which constrained its ability (related issue: [#11](https://github.com/formstate/formstate/issues/11)). formstate-x leverages MobX's power substantially, so we can easily track all dependencies when do validation, including dynamic values or dynamic valdiators. That's also why realizing cross-field validation is extremly easy with formstat-x.
64-
2. formstate mixes validated, safe value with readable value (`$`), in some cases it's not suitable to use either `$` or `value`. formstate-x provides `value` as readable value, `$` as validated and safe value and `_value` for UI binding only.
65-
3. formstate doesn't provide a good way to extract value from `FormState` ( related issues: [#25](https://github.com/formstate/formstate/issues/25) [#28](https://github.com/formstate/formstate/issues/28)). formstate-x provides `value` as `FormState`'s serialized value, with full type info.
66-
4. formstate dosen't provide a good way to disable part of a `FormState`. `FormStateLazy` is like a hack and very different concept from `FieldState` & `FormState`. formstate-x provides `disableValidationWhen` to let you realize conditional validation with a single method call.
35+
5. Commit and push, then create a pull request

dumi/.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
/src
2+
/dist

dumi/.umirc.ts

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import path from 'path'
2+
import { defineConfig } from 'dumi'
3+
4+
const repo = 'formstate-x'
5+
6+
export default defineConfig({
7+
title: repo,
8+
favicon: 'https://qiniu.staticfile.org/favicon.ico',
9+
logo: `/${repo}/logo.svg`,
10+
outputPath: `dist/${repo}`,
11+
mode: 'site',
12+
hash: true,
13+
// Because of using GitHub Pages
14+
base: `/${repo}/`,
15+
publicPath: `/${repo}/`,
16+
navs: [
17+
null,
18+
{
19+
title: 'GitHub',
20+
path: 'https://github.com/qiniu/formstate-x',
21+
},
22+
],
23+
alias: {
24+
'formstate-x': path.join(__dirname, 'src')
25+
},
26+
styles: [`.__dumi-default-navbar-logo { color: #454d64 !important; }`],
27+
mfsu: {}
28+
// more config: https://d.umijs.org/config
29+
})

dumi/docs/concepts/input/index.md

+104
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
---
2+
title: Input
3+
order: 1
4+
toc: menu
5+
---
6+
7+
### Input
8+
9+
A input is a part of an application, which collects info from user interaction. Typical inputs include:
10+
11+
* HTML `<input type="text" />` which collects a text string
12+
* [MUI `Date Picker`](https://mui.com/components/date-picker/) which collects a date
13+
* [Antd `Upload`](https://ant.design/components/upload/) which collects a file
14+
* Some `FullNameInput` which collects someone's full name
15+
* ...
16+
17+
### General Input API
18+
19+
We will introduce general inputs' API based on [React.js](https://reactjs.org/). It will not be very different in other UI frameworks like [Vue.js](https://vuejs.org/) or [Angular](https://angular.io/).
20+
21+
In a React.js application, a [(controlled) input component](https://reactjs.org/docs/forms.html#controlled-components) always provides an API like:
22+
23+
```ts
24+
type Props<T> = {
25+
value: T
26+
onChange: (event: ChangeEvent) => void
27+
}
28+
```
29+
30+
The prop `value` means the current input value, which will be displayed to the user.
31+
32+
The prop `onChange` is a handler for event `change`. When user interaction produces a new value, the input fires event `change`—the handler `onChange` will be called.
33+
34+
Typically the consumer of the input holds the value in a state. In the function `onChange`, it sets the state with the new value, which causes re-rendering, and the new value will be displayed.
35+
36+
An [uncontrolled input component](https://reactjs.org/docs/uncontrolled-components.html) may behave slightly different, but the flow are mostly the same.
37+
38+
A complex input may consist of multiple simpler inputs. MUI `Date Picker` includes a number input and a select-like date selector, A `FullNameInput` may includes a first-name input and a last-name input, etc. While no matter how complex it is, as consumer of the input, we do not need to know its implementation or UX details. All we need to do is making a convention of the input value type and then expect the input to collect it for us. That's the power of abstraction.
39+
40+
### Input API in formstate-x
41+
42+
While in forms of real applications, value of input is not the only thing we care. Users may make mistakes, we need to validate the input value and give users feedback.
43+
44+
The validation logic, which decides if a value valid, is always related with the logic of value composition and value collection—the logic of the input.
45+
46+
The feedback UI, which tells users if they made a mistake, is always placed beside the input UI, too.
47+
48+
Ideally we define inputs which extracts not only value-collection logic, but also validation and feedback logic. Like that the value can be accessed by the consumer, the validation result should be accessable for the consumer.
49+
50+
While the API above (`{ value, onChange }`) does not provide ability to encapsulate validation and feedback logic in inputs. That's why we introduce a new one:
51+
52+
```ts
53+
type Props = {
54+
state: State
55+
}
56+
```
57+
58+
`State` is a object including the input value and current validation info. For more details about it, check section [State](/concepts/state). By passing a state to an input component, information exchanges between the input and its consumer are built.
59+
60+
An important point is, the logic of (creating) state is expected to be provided by the input—that's how the input decides the validation logic. Apart from the component definition, the module of a input will also provide a state factoty, which makes the module like this:
61+
62+
```ts
63+
// the input state factory
64+
export function createState(): State {
65+
// create state with certain validation logic
66+
}
67+
68+
// the input component who accepts the state
69+
export default function Input({ state }: { state: State }) {
70+
// render input with value & validation info from `state`
71+
}
72+
```
73+
74+
The consumer of the input may access and imperatively control (if needed) value and
75+
validation info through `state`.
76+
77+
### Composability
78+
79+
Inputs are still composable. We can build a complex input based on simpler inputs. A `InputFooBar` which consists of `InputFoo` and `InputBar` may look like this:
80+
81+
_Below content is a pseudo-code sample. For more realistic example, you can check section [Composition](/guide/composition)._
82+
83+
```tsx | pure
84+
import InputFoo, * as inputFoo from './InputFoo'
85+
import InputBar, * as inputBar from './InputBar'
86+
87+
type State = Compose<inputFoo.State, inputBar.State>
88+
89+
export function createState(): State {
90+
return compose(
91+
inputFoo.createState(),
92+
inputBar.createState()
93+
)
94+
}
95+
96+
export default function InputFooBar({ state }: { state: State }) {
97+
return (
98+
<>
99+
<InputFoo state={state.foo} />
100+
<InputBar state={state.bar} />
101+
</>
102+
)
103+
}
104+
```

0 commit comments

Comments
 (0)