|
| 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