Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/choonkeat/nativeform

Working with form in Elm without storing raw form state in Elm. Demo at https://nativeform.netlify.app
https://github.com/choonkeat/nativeform

elm form forms

Last synced: about 1 month ago
JSON representation

Working with form in Elm without storing raw form state in Elm. Demo at https://nativeform.netlify.app

Awesome Lists containing this project

README

        

# NativeForm

Working with `form` in Elm without storing raw form state in Elm, aka "Use the [Platform](https://developer.mozilla.org/en-US/docs/Web/API/Document/forms)".

Before

1. Use `value someValue` to set form field value when we render a form (e.g. edit form)
- we are now obliged to keep `someValue` in sync with user's input
1. Use `onInput` msg to update `someValue`
- store this value in model
- this value may not be the desired type, e.g. storing raw `String` on behalf of a `Time.Posix` model state

After

1. Use `defaultValue someValue` to set form field value when we render a form (e.g. edit form)
- no longer obliged to keep `someValue` in sync with user's input
1. Use `NativeForm.decoder` anytime to retrieve the form values from browser [`document.forms`](https://developer.mozilla.org/en-US/docs/Web/API/Document/forms)
- No need for `onInput` nor a `msg` for every form field; only do so for fields that really need handling
- But forms must be given the `id` attribute and form fields must be given a `name` attribute

## Demo

Checkout the demo at https://nativeform.netlify.app and its source code at [example](https://github.com/choonkeat/nativeform/blob/main/example/src/Main.elm) subdirectory.

## Getting started

1. Pass the browser `document.forms` into your Elm app through [Flags](https://guide.elm-lang.org/interop/flags.html)

```diff

var app = Elm.Main.init({
node: document.getElementById('myapp'),
flags: {
documentForms: document.forms // <-- important!
}
})

```
1. Store the `documentForms : Json.Encode.Value` from your `Flags` in your `Model`
1. Wire up any (or many) events, e.g. `form [ on "change" OnFormChange ]` or `input [ onInput (always OnFormChange) ]`
1. And call `Json.Decode.decodeValue (NativeForm.decoder formId) model.documentForms` to get the list of form field name and values.

## Notes

1. No matter how many fields you have, you only need one `Msg`
1. Always give your form an `id` value
1. Always give your form fields a `name` value
1. Does not handle `input[type=file]` in a useful manner; manage file uploads by other means.
1. All user input values are either `OneValue String` or `ManyValues (List String)`, including `input[type=number]`; do your own conversion to `Int` or `Time.Posix` etc.
1. When using this library, your form id and field names are "stringly typed". You should use a custom type to manage a fixed set of form fields, but this library does not require you to do so.
1. Do not deal with the `List (String, Value)` returned from `NativeForm.decoder`. Instead, you should parse them into a value of your desired type. e.g.

```
parseDontValidate :
List ( String, NativeForm.Value )
-> Result String UserInfo
```

See a [sample `parseDontValidate` implementation](https://github.com/choonkeat/nativeform/blob/main/example/src/Main.elm#L529-L544)