A minimal scrolling bar with two sliders that can be used for range scrolling/selection.
Demo: https://welai.github.io/dual-range-bar
Dual range bar is something alike the timeline control in Adobe Audition/Premiere. It allows you to control a range variable via dragging operations on sliders and mouse wheel scrolling. The primary design purpose is an alternative scrollbar rather than a value input, though it can be used in both scenarios.
Dual range bar does not need any dependencies, and can work well by simply referring the script with a <script/>
tag in your webpage, while at the same time it fits for modern web development paradigm, providing modules (UMD) and developed with TypeScript.
You may have a quick look at the source of the demo page: index.html
Import dual range bar with a script reference declaration:
<script src="dist/dual-range-bar.min.js"></script>
The src
path varies according to the location you make copy of the script.
$ npm install dual-range-bar
And in you JavaScript or TypeScript code:
import { DualHRangeBar, DualVRangeBar } from 'dual-range-bar'
You can create new dual range bars in a fully-declarative manner, but you will have no access to the DualRangeBar
instance in JavaScript, which is usually unwanted. Anyway, initializing dual range bar with class names just works, and data properties are still accessible via data attributes of the container element.
<div class="drbar-container"></div>
This by default creates a horizontal dual range bar. Note that, initializations based on class names are executed only once, at window.onload
, and only <div/>
's with no child nodes will be initialized.
To create a vertical dual range bar, adding a drbar-vertical
class name to the container's class list.
<div class="drbar-container drbar-vertical"></div>
<div id="my-drbar-container"></div>
And in your JavaScript/TypeScript:
const drbar = new DualHRangeBar('my-drbar-container')
Or, this will create a vertical dual range bar:
const drbar = new DualVRangeBar('my-drbar-container')
Data interface of dual range bar is simple.
console.log(drbar.lower, drbar.upper, drbar.lowerBound, drbar.upperBound)
lower
property is the value represented by the left/top slider. upper
property is the value represented by the right/bottom slider.
lowerBound
is the minimum possible value of the ranges, i.e. the leftmost/topmost value on the bar. upperBound
is the maximum possible value of the ranges, i.e. the rightmost/bottommost value on the bar. A lowerBound
that is larger than the upperBound
is allowed.
console.log(drbar.minSpan, drbar.maxSpan)
minSpan
is the minimum range span, and maxSpan
is the maximum range span. Dual range bar will try to prevent the sliders from representing a range smaller than the minSpan
or larger than the maxSpan
. Here, the "span" is always upper - lower
.
All of the properties mentioned above are mutable. But you have to ensure that you have made them correct, or unexpected behaviors may occur. Manual changes to the data properties does not emit any events.
When you have no access to the DualRangeBar
instance, values are also accessible via data attributes of the container element.
console.log(parseFloat(document.getElementById('my-drbar-container').dataset.lowerBound))
Dual range bar provides multiple configuration options for specifying initial values and styling.
const drbar = new DualHRangeBar('my-drbar-container', {
minimizes: false, // Minimises the container when inactive
size: 'default', // Size of the dual range bar
lowerBound: 0, // Initial value for "lowerBound"
upperBound: 1, // Initial value for "upperBound"
minSpan: 0.2, // Initial value for "minSpan"
maxSpan: 1, // Initial value for "maxSpan"
lower: 0, // Initial value for "lower"
upper: 1, // Initial value for "upper"
sliderColor: '#1E88A8', // Color of the slider
sliderActive: '#08789b',// Slider color when active
rangeColor: '#7DB9DE', // Color of the range slider
rangeActive: '#5da8d6', // Range slider color when active
bgColor: '#aaaaaa44', // Color of the background
})
When minimizes
is true
, the dual range bar will narrow to a thin line when the mouse pointer is out of the container.
There are four available size
's for dual range bar, small
, default
, large
and huge
.
Dual range bar does not validate the color specification strings, the strings will be written directly into the CSS. When the sliderColor
and rangeColor
are specified, sliderActive
and rangeActive
will NOT change accordingly, you may have to specify the two colors manually. If you need to generate the darkened colors programmatically, chroma.js is a good choice.
The event interface is quite straight-forward. When data is updated at user input, an update
event is emitted from the container element. The DualRangeBar
instance inherits the container's EventTarget
interface, so listening to the DualRangeBar
instance works the same. The emitted event is a CustomEvent
object, with its detail
property referring the DualRangeBar
instance.
drbar.addEventListener('update', (e) => {
console.log(e.detail.lower, e.detail.upper)
})
// This works the same
document.getElementById('my-drbar-container').addEventListener('update',
(e) => { console.log(e.detail.lower, e.detail.upper) })
Though styling in the configuration options of the constructor is recommended, it is also possible to add custom style sheets to achieve detailed control to the style.
To override the default styles, you can select the elements via the container's id. Eg., the container's id is my-drbar-container
, to style the sliders, you need to query the #my-drbar-container .drbar-slider
CSS selector. This will override the default style specification implemented using the selector .drbar-container .drbar-slider
.
Most dimensions & colors are specified with CSS variables, overriding these variables will make changes to the dimensions & colors.
.drbar-container {
/* Colors */
--slider-color: #1E88A8;
--range-color: #7DB9DE;
--bg-color: #aaaaaa44;
--slider-active: #08789b;
--range-active: #5da8d6;
/* Sizes */
--slider-wh: 20px;
--range-thick: 15px;
--bg-thick: 10px;
--mini-thick: 4px;
--mini-ratio: calc(4/15);
}
Please refer to the src/style.css
when advanced styling is needed.
Pull requests are welcomed.