Accessible HTML <select>
element replacement with customizable UI. This
package provides integration of the
szn-select project for
Vue.js projects.
First install the package:
npm install --save @seznam/szn-select-vue
The <szn-select>
(as the SznSelect
component) element can be then used the
same way as you would use a <select>
element in your project. The following
example shows various usage options:
<template>
<form action="/submit-form" method="post">
<div class="inline-form">
<label for="mySelect">
Choose one
</label>
<!-- use the <SznSelect> component as if it was an ordinary <select> element -->
<szn-select name="singleOption" id="mySelect" @change="onFirstChanged">
<option value="1">first</option>
<option value="2" selected>second</option>
<optgroup label="this is a group">
<option value="3">option groups are supported as well</option>
</optgroup>
</szn-select>
</div>
<div class="inline-form">
<label for="anotherSelect">
Choose any
</label>
<szn-select name="manyOptions" id="anotherSelect" multiple>
<option value="foo">foo</option>
<option value="foo" title="multiple options may have the same value">foo 2</option>
<option value="bar" selected>bar</option>
<option value="baz" disabled>baz</option>
</szn-select>
</div>
</form>
</template>
<script>
import SznSelect from '@seznam/szn-select-vue'
export default {
components: {
SznSelect,
},
props: {
onFirstChanged: Function,
},
}
</script>
Notice that the SznSelect
component is
registered locally.
This is necessary, as the SznSelect
component registers the <szn-select>
element with Vue as an ignored element.
The registration as ignored element is necessary, otherwise Vue will not allow
the SznSelect
element create an actual native <szn-select>
element, which
is necessary to make this work properly.
If you want to register the SznSelect
component globally, use a different
element name, for example <szn-select-vue>
(it is best to do this in the
entry point of your application):
import SznSelect from '@seznam/szn-select-vue'
import Vue from 'vue'
Vue.component('szn-select-vue', SznSelect)
Now you can use the <szn-select-vue>
component in your application globally:
<template>
<szn-select-vue name="chosenOption">
<option value="1">first</option>
<option value="2" selected>second</option>
<optgroup label="this is a group">
<option value="3">option groups are supported as well</option>
</optgroup>
</szn-select-vue>
</template>
In order to improve the UX in case of extremely-slowly-loaded JS, legacy
browsers or situations where JS is disabled or fails, it is recommended to
include the following CSS file in your page to provide fallback styles for the
native <select>
element that would be where the <SznSelect>
component is
used:
<link rel="stylesheet" href="https://unpkg.com/@seznam/szn-select@<VERSION>/szn-select-nojs.css">
The component, by default, injects the szn-select
's
loader script
into the <head>
to load the latest compatible version of szn-select using
the latest compatible loader. This technique is used automatically provide
bugfixes and compatibility updates (compatibility with the assistive
technologies does shift in time, unfortunately).
Another thing the loader does for you is selecting the smallest, best-performing bundle for the current browser. This means that the modern browsers (e.g. Chrome) tend to download smaller bundles than the legacy browsers (e.g. Internet Explorer).
The downside of this is that the browser has to download two more JS files
from the unpkg.com's CDN (by default, can be
configured to use other CDNs, e.g. jsdelivr.com,
or self-hosted files) to enable all features of the <SznSelect>
component.
This behavior may be, if needed, modified using the loader options passed as
an object to the component in the loaderOptions
prop:
<template>
<szn-select :loaderOptions="{ /* loader options go here, see below for details */ }">
...
</szn-select>
</template>
Note that only the first <SznSelect>
instance activated by Vue will perform
the loading, therefore passing different options to multiple SznSelect
s on a
single page might result in inconsistent or unexpected results. To avoid this,
a common pattern used is using a higher-order component, as seen below:
<template>
<szn-select v-bind="$attrs" v-on="$listeners" :loaderOptions="loaderOptions">
<slot/>
</szn-select>
</template>
<script>
import SznSelect from '@seznam/szn-select-vue'
const LOADER_OPTIONS = {
// loaders options go here, see below for details
}
export default {
components: {
SznSelect,
},
data: { // the data property usually should be a function, but these are constants, so it's OK
loaderOptions: LOADER_OPTIONS,
},
}
</script>
The loader options are:
enable
- whether to use a loader at all. This defaults totrue
, setting this option tofalse
is useful when using a self-hosted deployment (see below). Use this option with care as it usually leads to getting no updates and bugfixes until you update the dependencies in your project manually.useEmbeddedLoader
- When set totrue
, the component will use a more light-weight loader to load theszn-element
's bundle. This saves a single (tiny) JavaScript file download. This usually leads to the same results as using a full-feature loader, however, if there are bugfixes available for the loader, you will might not receive them in certain situations. In most cases it is OK to set this option totrue
. Defaults tofalse
.useAsyncLoading
- when set totrue
, theasync
attribute will be set on the<script>
elements used to load theszn-select
's JavaScript logic. Since the loader is always triggered after theDOMContentLoaded
event, the scripts will be loaded asynchronously anyway regardless of this flag. This option is simply exposed from the underlying loader mechanics. Defaults totrue
.urls
- an object that provides URL overrides for the loader:-
package
- the URL at which the@seznam/szn-select
package's files are available. This can be overridden on per-file basis using the options below. Defaults tohttps://unpkg.com/@seznam/szn-select@<VERSION>/
, the trailing slash is optional.It is recommended to use a semver version range when using unpkg or a similar JS CDN, such as
1.x
to automatically receive the latest compatible version with the most recent bugfixes and compatibility updates. -
loader
- the URL override for the full-featured loader. This option has no effect when theuseEmbeddedLoader
is set totrue
. -
es3
- the URL override for the ES3-compatible implementation of theszn-select
element. This is used on legacy browsers if the implementation for theszn-tethered
helper element and the szn-elements runtime is already loaded. -
es2016
- the URL override for the ES2016-compatible implementation of theszn-select
element. This is used on modern browsers if the implementation for theszn-tethered
helper element and the szn-elements runtime is already loaded. -
bundle-elements.es3
- the URL override for the ES3-compatible implementation of theszn-select
andszn-tethered
elements. This is used on legacy browsers if the szn-elements runtime is already loaded. -
bundle-elements.es2016
- the URL override for the ES2016-compatible implementation of theszn-select
andszn-tethered
elements. This is used on modern browsers if the szn-elements runtime is already loaded. -
bundle-full.es3
- the URL override the ES3-compatible implementation of theszn-select
andszn-tethered
elements and the szn-elements runtime. This is used on legacy browsers if no dependencies are loaded yet. -
bundle-full.es2016
- the URL override the ES2016-compatible implementation of theszn-select
andszn-tethered
elements and the szn-elements runtime. This is used on modern browsers if no dependencies are loaded yet. -
bundle-full.ce
- the URL override the ES2016-compatible implementation of theszn-select
andszn-tethered
elements and the szn-elements runtime. This is used on modern browsers that support the custom elements API if no dependencies are loaded yet.
-
The downside of letting the <SznSelect>
component handle all the work is
that the implementation of the underlying <szn-select>
element is loaded
asynchronously. This may lead to a
FOUC or seeing the
native <select>
element with the fallback styles applied for a brief moment.
This can be improved by using synchronous loading of the <szn-select>
element by injecting the loader into the page ourselves (notice there is no
async
nor defer
attribute):
<script src="https://unpkg.com/@seznam/szn-select@<VERSION>/loader.min.js"></script>
Since the script would be executed synchronously, the loader will inject the
bundle using document.write()
, which will load the <szn-select>
element
before the DOMContentLoaded
event occurs. The <SznSelect>
component will
recognize that the szn-select
element has been already loaded and will not
inject the loader repeatedly.
The downside to the example above is that the script is loaded from another domain than your website, and Chrome will block the loading the of bundle on 2G connections. This can be resolved by hosting the files yourselves (see below).
For some reason or another it might be practical for you to host the bundle
files yourselves. First, start by installing the szn-select
package:
npm install --save @seznam/szn-select@<VERSION>
Next, make the files in the npm module available in your project over HTTP as static assets.
The final step depends on your preferred way of loading the bundle:
- if you leave the bundle loading to the
<SznSelect>
component, provide theloaderOptions
prop with theurls
object. Having theurls.package
option is usually enough, point it to your local URL at which you made theszn-select
's files available. - if you are using synchronous loading (see above), use
these data attributes
on the loader's
<script>
element.