The formwatcher is a tool to easily improve forms with JavaScript with following goals in mind:
- Be completely unobtrusive (your forms still work without JS)
- Provide sensible defaults so you won't have to configure much
- Provide the best user experience possible
It is written in CoffeeScript and built with (and depends on):
Simply install with ender:
ender build formwatcher
or
ender add formwatcher
You can also just download the lib/
files, and install the dependencies manually, but I don't recommend it.
The main features include:
- Validation ¹
- Decorators ¹ (Turn a simple select input field into an image selector, or a font selector or display a nice hint)
- AJAX conversion (Turn a form into an AJAX call automatically)
- Simple html attribute configuration that is W3C valid.
- It's very easy to create your own validators / decorators.
- ...plenty more.
¹ Formwatcher is built to be easily extendible, so writing your own validators and decorators is both easy and recommended
You can configure formwatcher imperatively or by defining specific html attributes on your form
Although most of the time, the simplest way to configure Formwatcher is by attribute configuration, there are a few use cases where the imperative configuration is necessary or more appropriate:
Formwatcher.watch('form-id');
Instead of configuring your forms with JS, you can simply add the data-fw
attribute to your elements so Formwatcher knows how to handle them. The best
part about it: it doesn't break your HTML markup; it still validates fine.
<form action="" data-fw="">
<!-- Input fields -->
</form>
When Formwatcher is included, it will scan all forms in the document as soon as
it's loaded to see if there is any form with the data-fw
attribute, and attaches
itself to it.
The data-fw
attribute is just a JSON object, that will be passed to the new
Watcher
instance and serves as options
object.
A Formwatcher configuration could look like this:
<form action="" data-fw='{ "ajax": true, "validate": false }'>
</form>
NOTE: The
data-fw
content is pure JSON. All names and strings have to be in double quotes, so you have to put thedata-fw
value itself in single quotes.
Since it's JSON, not JS, you won't be able to directly define your callback functions here.
You can directly configure decorators or validators inside this JSON object:
<form action="" data-fw='{
"validate": true,
"Hint": { "auto": true },
"Float": { "decimalMark": "," }
}'>
</form>
Adding a validator is very easy. You just push a new instance of Formwatcher.Validator
to the Formwatcher.validators
list.
As an example, the required
validator:
# ## Required validator
#
# If it's a checkbox, it has to be checked. Otherwise the value can't be 0 or an empty string (whitespace is trimmed)
Formwatcher.validators.push class extends Formwatcher.Validator
name: "Required"
description: "Makes sure the value is not blank (nothing or spaces)."
classNames: [ "required" ]
validate: (value, input) ->
return "Can not be blank." if (input.attr("type") is "checkbox" and not input.is(":checked")) or not trim value
true
If the validate()
funciton returns a string, the validation failed, and the string is used as error message.
If true
is returned, the validation passed.
Writing decorators is a bit more complex, but not difficult neither. The basic concept is again pushing instances of
Formwatcher.Decorator
to Formwatcher.decorators
:
Formwatcher.decorators.push class extends Formwatcher.Decorator
name: "SomeDecorator"
description: "The description of the decorator"
nodeNames: [ "INPUT", "TEXTAREA" ]
defaultOptions:
myOption: "test"
accepts: (input) ->
if super input # The default implementation checks for classes and node names.
# Return true if this decorator should decorate this input field.
return true
false
decorate: (input) ->
# The decorate function has to return an object with all fields that have been generated and that should be updated
# with `.focus`, `.validated`, etc... classes.
# The `elements` object HAS TO contain at least `input` which is the field that will hold the actual value to be
# transmitted. (It can be changed to a hidden field)
elements = { input: input }
# Here is your code to nicely wrap (or replace) the input field.
return elements
Take a look at the built in hint decorator for a full example.