@getdrip/much-select-elm
v0.18.3
Published
A fancy selector web component written (mostly) in elm.
Downloads
40
Keywords
Readme
<much-select>
A web component - powered by Elm - that will create a powerful select menu.
Prior Art, Inspiration, and Goals
The project draws heavy inspiration from the jquery based selectize.js.
The need for this project is that we want to use selectize.js however we need the over all app to be built in Elm. Elm needs to "own" the DOM and selectize is built in a way that's not compatible with that.
The goal for this project to achieve near feature parity with selectize using web components. The API will be different, so it will not be a drop in replacement, but hopefully it will not be too hard to replace one with the other.
Other Similar Projects
Installation
npm i much-select-elm
Usage
The npm package provides the class MuchSelect
(which inherits from HTMLElement
). To use it, use something like the following to define a custom element.
import MuchSelect from "@getdrip/much-select-elm";
if (!customElements.get("much-select")) {
// Putting guard rails around this because browsers do not like
// having the same custom element defined more than once.
window.customElements.define("much-select", MuchSelect);
}
Development
Pre-requisites
asdf
This project manages its Node version with asdf
asdf-node-js
You'll need asdf
installed as well as the asdf nodejs plugin
(gpg
is a dependency used by asdf to verify plugin asset validity)
brew install gpg
brew install asdf
asdf plugin add nodejs https://github.com/asdf-vm/asdf-nodejs.git
soupault
We use a static site generator called Soupault to make the sandbox. There several ways to install it. I recommend using the OPAM method.
watchexec
The development build scripts depend on watchexec. You can install it anyway that makes sense on your machine as long as it's in the PATH
but I recommend using homebrew.
brew install watchexec
Initial Setup
To work on this project, clone the repo to your machine then run the following:
asdf install
npm install
npx elm-tooling install
This should install the correct version of node (via asdf
). All the npm packages, and all the elm-tools.
Watch and Develop
To run a webpack development server with the sandbox/demo page:
npm start
Now you can visit http://localhost:1234
Production Build
To do a production build run
npm run build
API
DOM
Attributes
allow-custom-options
The allow-custom-options
attribute will allow the user to add new options to the <much-select>
, not just one of given ones. Of course, you will want to know when that's happened. You might be able to just look at the value, but you might want to know when a custom option has been added too. For that there's the customValueSelected
event.
This is a boolean attribute, so it's presents means it's true. It's absence means it's false. So the default value for allowing custom options is false.
disabled
The disabled
attribute prevents the <much-select>
from responding to user input. It is supposed to work like disabling any of the standard form elements.
This is a boolean attribute, so it's presents means it's true. It's absence means it's false. So the default is for a <much-select>
to not be disabled.
events-only
The events-only
attribute prevents the <much-select>
from automatically updating its "light DOM". This is important for using <much-select>
inside of an Elm app (and possibly other situations too).
This is a boolean attribute, so if the events-only
attribute is present means it's true. It's absence means it's false. So the default is for a <much-select>
to not be in events only mode, and it will try to update the light DOM to reflect it's internal state.
max-dropdown-items
The max-dropdown-items
attribute sets the maximum number of items that <much-select>
will attempt to render in the dropdown.
The default value is 1000.
The max-dropdown-items
attribute needs a positive integer for its value.
This is also not enforced when a <much-select>
is using the datalist
output-style
.
If you set the value of max-dropdown-items
to 0
there will be no limit. Would anyone want that? Should it even be allowed?.
multi-select
The multi-select
attribute puts the <much-select>
into multi select mode, which allows the user to select multiple values.
This is a boolean attribute, so if the mulit-select
attribute is present, the <much-select>
will be in multi select mode. If not, it will be in the default single select mode.
multi-select-single-item-removal
The multi-select-single-item-removal
attribute adds buttons to remove individual selected options. This only works if the <much-select>
is in multi select mode.
This is a boolean attribute, so if the multi-select-single-item-removal
is present the <much-select>
will have the selected option removal buttons visible. If not, the default behavior is to not show them.
option-sorting
The option-sorting
attribute allows you to specify what order the options should appear in the dropdown.
The options for this are:
no-sorting
(default) respect the value the options are in alreadyby-option-label
sort the options alphabetically by their label
output-style
The output-style
attribute allows you to change the way a <much-select>
is rendered, which give it a different look and feel. Hopefully you can find one that matches the user experience that best fits your circumstances.
The options for this are:
custom-html
(default)datalist
custom-html
give you a lot of power to style the <much-select>
but a lot of the markup in hand rolled and an attempt to feel like the user is expecting but even our best effort here will probably fall short.
datalist
uses the datalist feature of the <input>
element to render the dropdown part. This can work well if you want to allow the user to type in a text field in a more free form way, and you want to make it easy for them to select a value they might want. It also uses more native widgets. Multi select mode can take up a lot of vertical space though.
placeholder
The placeholder
attribute is used to set the placeholder in the text input of the <much-select>
. Just like in the <input type="text">
it should only show up if the input is empty.
search-string-minimum-length
The search-string-minimum-length
attribute is used to manage how many characters a user needs to type in the #input-filter
before the options in the dropdown start being filtered.
selected-option-goes-to-top
The selected-option-goes-to-top
attribute lets you change the behavior of <much-select>
, so that when an option is selected it appears at the top of the dropdown menu. This only works when its output-mode
is custom-html
and it's in single select mode.
selected-value
The selected-value
attribute is used to set the value of the <much-select>
.
You should not use this in combination with using selected
attributes on the <option>
tags in the select-input
slot. You should just pick one place in the DOM to track the selected value.
selected-value-encoding
The selected-value-encoding
attribute is used to change how the values work in the selected-value
attribute.
The options for this attribute are:
comma
(default) the selected options are comma delegatedjson
the selected options are encoded in json andencodeURIComponent()
comma
is the most straight forward but if you allow values that have commas in them things are not going to go well. As far as I can tell json
can handle any kinds of text values, however it is a bit harder to just look at and know what the selected value is.
json
encoding is recommended.
show-dropdown-footer
The show-dropdown-footer
attribute is used to show the footer in the dropdown of the <much-select>
. The footer contains potentially useful information about how many options are really available but might not be visible because the results are being filter.
Slots
no-options
If there are no options to display, show this message.
no-options
If there are no options to display, show this message.
no-filtered-options
If the user has typed in a search filter that just does not have any good matches show this message.
Options
Events
valueChanged
This event fires any time the value changes.
valueCleared
This event fires if the <much-select>
is cleared.
optionSelected
This event fires if the <much-select>
is in single or multi select mode, but it's mostly for multi select mode. It will just have the newly selected option in it (not all the selected options like the valueChanged
event).
optionDeselected
This event fires if the <much-select>
is in single or multi select mode, but it's mostly for multi select mode. It will just have the newly deselected option in it. This is kinda of the inverse of the optionSelected
event.
inputKeyUp
This event fires every time a user types in the #input-filter
for filtering the options.
inputKeyUpDebounced
This event fires every time a user types in the #input-filter
but is debounced by half a second. The idea here is if you want to hook a <much-select>
up to an API you can use this event to kick off your API calls to add additional options based on what the user is has "searched" for.