UNPKG

@blueprintjs/core

Version:
107 lines (105 loc) 16.4 kB
/* * Copyright 2016 Palantir Technologies, Inc. All rights reserved. * Licensed under the BSD-3 License as modified (the “License”); you may obtain a copy * of the license at https://github.com/palantir/blueprint/blob/master/LICENSE * and https://github.com/palantir/blueprint/blob/master/PATENTS */ "use strict"; var __extends = (this && this.__extends) || function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; function __() { this.constructor = d; } d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); }; var classNames = require("classnames"); var React = require("react"); var Classes = require("../../common/classes"); var Errors = require("../../common/errors"); var utils_1 = require("../../common/utils"); var coreSlider_1 = require("./coreSlider"); var handle_1 = require("./handle"); var RangeEnd; (function (RangeEnd) { RangeEnd[RangeEnd["LEFT"] = 0] = "LEFT"; RangeEnd[RangeEnd["RIGHT"] = 1] = "RIGHT"; })(RangeEnd || (RangeEnd = {})); var RangeSlider = (function (_super) { __extends(RangeSlider, _super); function RangeSlider() { var _this = this; _super.apply(this, arguments); this.displayName = "Blueprint.RangeSlider"; this.className = classNames(Classes.SLIDER, Classes.RANGE_SLIDER); this.handles = []; this.addHandleRef = function (ref) { if (ref != null) { _this.handles.push(ref); } }; this.getHandlerForIndex = function (index, callback) { return function (newValue) { if (utils_1.isFunction(callback)) { var _a = _this.props.value, leftValue = _a[0], rightValue = _a[1]; if (index === RangeEnd.LEFT) { callback([Math.min(newValue, rightValue), rightValue]); } else { callback([leftValue, Math.max(newValue, leftValue)]); } } }; }; this.handleChange = function (newValue) { var _a = _this.props.value, leftValue = _a[0], rightValue = _a[1]; var newLeftValue = newValue[0], newRightValue = newValue[1]; if ((leftValue !== newLeftValue || rightValue !== newRightValue) && utils_1.isFunction(_this.props.onChange)) { _this.props.onChange(newValue); } }; } RangeSlider.prototype.renderFill = function () { var _a = this.props.value, leftValue = _a[0], rightValue = _a[1]; if (leftValue === rightValue) { return undefined; } // expand by 1px in each direction so it sits under the handle border var left = Math.round((leftValue - this.props.min) * this.state.tickSize) - 1; var width = Math.round((rightValue - leftValue) * this.state.tickSize) + 2; if (width < 0) { left += width; width = Math.abs(width); } return React.createElement("div", {className: Classes.SLIDER + "-progress", style: { left: left, width: width }}); }; RangeSlider.prototype.renderHandles = function () { var _this = this; var _a = this.props, disabled = _a.disabled, max = _a.max, min = _a.min, onRelease = _a.onRelease, stepSize = _a.stepSize, value = _a.value; return value.map(function (val, index) { return (React.createElement(handle_1.Handle, {disabled: disabled, key: index, label: _this.formatLabel(val), max: max, min: min, onChange: _this.getHandlerForIndex(index, _this.handleChange), onRelease: _this.getHandlerForIndex(index, onRelease), ref: _this.addHandleRef, stepSize: stepSize, tickSize: _this.state.tickSize, value: val})); }); }; RangeSlider.prototype.handleTrackClick = function (event) { this.handles.reduce(function (min, handle) { // find closest handle to the mouse position var value = handle.clientToValue(event.clientX); if (Math.abs(value - handle.props.value) < Math.abs(value - min.props.value)) { return handle; } return min; }).beginHandleMovement(event); }; RangeSlider.prototype.validateProps = function (props) { var value = props.value; if (value == null || value[RangeEnd.LEFT] == null || value[RangeEnd.RIGHT] == null) { throw new Error(Errors.RANGESLIDER_NULL_VALUE); } }; RangeSlider.defaultProps = { disabled: false, labelStepSize: 1, max: 10, min: 0, showTrackFill: true, stepSize: 1, value: [0, 10], }; return RangeSlider; }(coreSlider_1.CoreSlider)); exports.RangeSlider = RangeSlider; exports.RangeSliderFactory = React.createFactory(RangeSlider); //# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/components/slider/rangeSlider.tsx"],"names":[],"mappings":"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","file":"components/slider/rangeSlider.js","sourcesContent":["/*\n * Copyright 2016 Palantir Technologies, Inc. All rights reserved.\n * Licensed under the BSD-3 License as modified (the “License”); you may obtain a copy\n * of the license at https://github.com/palantir/blueprint/blob/master/LICENSE\n * and https://github.com/palantir/blueprint/blob/master/PATENTS\n */\n\nimport * as classNames from \"classnames\";\nimport * as React from \"react\";\n\nimport * as Classes from \"../../common/classes\";\nimport * as Errors from \"../../common/errors\";\nimport { isFunction } from \"../../common/utils\";\nimport { CoreSlider, ICoreSliderProps } from \"./coreSlider\";\nimport { Handle } from \"./handle\";\n\nexport type NumberRange = [number, number];\n\nenum RangeEnd {\n    LEFT = 0,\n    RIGHT = 1,\n}\n\nexport interface IRangeSliderProps extends ICoreSliderProps {\n    /**\n     * Range value of slider. Handles will be rendered at each position in the range.\n     * @default [0, 10]\n     */\n    value?: NumberRange;\n\n    /** Callback invoked when the range value changes. */\n    onChange?(value: NumberRange): void;\n\n    /** Callback invoked when a handle is released. */\n    onRelease?(value: NumberRange): void;\n}\n\nexport class RangeSlider extends CoreSlider<IRangeSliderProps> {\n    public static defaultProps: IRangeSliderProps = {\n        disabled: false,\n        labelStepSize: 1,\n        max: 10,\n        min: 0,\n        showTrackFill: true,\n        stepSize: 1,\n        value: [0, 10],\n    };\n\n    public displayName = \"Blueprint.RangeSlider\";\n    public className = classNames(Classes.SLIDER, Classes.RANGE_SLIDER);\n\n    private handles: Handle[] = [];\n\n    protected renderFill() {\n        const [leftValue, rightValue] = this.props.value;\n        if (leftValue === rightValue) { return undefined; }\n        // expand by 1px in each direction so it sits under the handle border\n        let left = Math.round((leftValue - this.props.min) * this.state.tickSize) - 1;\n        let width = Math.round((rightValue - leftValue) * this.state.tickSize) + 2;\n        if (width < 0) {\n            left += width;\n            width = Math.abs(width);\n        }\n        return <div className={`${Classes.SLIDER}-progress`} style={{ left, width }} />;\n    }\n\n    protected renderHandles() {\n        const { disabled, max, min, onRelease, stepSize, value } = this.props;\n        return value.map((val, index) => (\n            <Handle\n                disabled={disabled}\n                key={index}\n                label={this.formatLabel(val)}\n                max={max}\n                min={min}\n                onChange={this.getHandlerForIndex(index, this.handleChange)}\n                onRelease={this.getHandlerForIndex(index, onRelease)}\n                ref={this.addHandleRef}\n                stepSize={stepSize}\n                tickSize={this.state.tickSize}\n                value={val}\n            />\n        ));\n    }\n\n    protected handleTrackClick(event: MouseEvent | React.MouseEvent<HTMLElement>) {\n        this.handles.reduce((min, handle) => {\n            // find closest handle to the mouse position\n            const value = handle.clientToValue(event.clientX);\n            if (Math.abs(value - handle.props.value) < Math.abs(value - min.props.value)) {\n                return handle;\n            }\n            return min;\n        }).beginHandleMovement(event);\n    }\n\n    protected validateProps(props: IRangeSliderProps) {\n        const { value } = props;\n        if (value == null || value[RangeEnd.LEFT] == null || value[RangeEnd.RIGHT] == null) {\n            throw new Error(Errors.RANGESLIDER_NULL_VALUE);\n        }\n    }\n\n    private addHandleRef = (ref: Handle) => {\n        if (ref != null) {\n            this.handles.push(ref);\n        }\n    }\n\n    private getHandlerForIndex = (index: RangeEnd, callback: (value: NumberRange) => any) => (newValue: number) => {\n        if (isFunction(callback)) {\n            const [leftValue, rightValue] = this.props.value;\n            if (index === RangeEnd.LEFT) {\n                callback([Math.min(newValue, rightValue), rightValue]);\n            } else {\n                callback([leftValue, Math.max(newValue, leftValue)]);\n            }\n        }\n    }\n\n    private handleChange = (newValue: NumberRange) => {\n        const [leftValue, rightValue] = this.props.value;\n        const [newLeftValue, newRightValue] = newValue;\n        if ((leftValue !== newLeftValue || rightValue !== newRightValue) && isFunction(this.props.onChange)) {\n            this.props.onChange(newValue);\n        }\n    }\n}\n\nexport const RangeSliderFactory = React.createFactory(RangeSlider);\n"],"sourceRoot":"/source/"}