UNPKG

quasar

Version:

Build high-performance VueJS user interfaces (SPA, PWA, SSR, Mobile and Desktop) in record time

231 lines (196 loc) 5.33 kB
import Vue from 'vue' import { getRatio, getModel, SliderMixin, keyCodes } from './slider-utils.js' import { between } from '../../utils/format.js' import { stopAndPrevent } from '../../utils/event.js' export default Vue.extend({ name: 'QSlider', mixins: [ SliderMixin ], props: { value: { required: true, default: null, validator: v => typeof v === 'number' || v === null }, labelValue: [String, Number] }, data () { return { model: this.value === null ? this.min : this.value, curRatio: 0 } }, watch: { value (v) { this.model = v === null ? 0 : between(v, this.min, this.max) }, min (v) { this.model = between(this.model, v, this.max) }, max (v) { this.model = between(this.model, this.min, v) } }, computed: { ratio () { return this.active === true ? this.curRatio : this.modelRatio }, modelRatio () { return (this.model - this.min) / (this.max - this.min) }, trackStyle () { return { width: (100 * this.ratio) + '%' } }, thumbStyle () { return { [this.horizProp]: (100 * this.ratio) + '%' } }, thumbClass () { return this.preventFocus === false && this.focus === true ? 'q-slider--focus' : null }, pinClass () { return 'q-slider__pin absolute flex flex-center' + (this.labelColor !== void 0 ? ` text-${this.labelColor}` : '') }, pinTextClass () { return 'q-slider__pin-value-marker-text' + (this.labelTextColor !== void 0 ? ` text-${this.labelTextColor}` : '') }, events () { if (this.editable === true) { return this.$q.platform.is.mobile === true ? { click: this.__mobileClick } : { mousedown: this.__activate, focus: this.__focus, blur: this.__blur, keydown: this.__keydown, keyup: this.__keyup } } }, computedLabel () { return this.labelValue !== void 0 ? this.labelValue : this.model } }, methods: { __updateValue (change) { if (this.model !== this.value) { this.$emit('input', this.model) } change === true && this.$emit('change', this.model) }, __getDragging () { return this.$el.getBoundingClientRect() }, __updatePosition (event, dragging = this.dragging) { const ratio = getRatio( event, dragging, this.$q.lang.rtl ) this.model = getModel(ratio, this.min, this.max, this.step, this.decimals) this.curRatio = this.snap !== true || this.step === 0 ? ratio : (this.model - this.min) / (this.max - this.min) }, __focus () { this.focus = true }, __keydown (evt) { if (!keyCodes.includes(evt.keyCode)) { return } stopAndPrevent(evt) const step = ([34, 33].includes(evt.keyCode) ? 10 : 1) * this.computedStep, offset = [34, 37, 40].includes(evt.keyCode) ? -step : step this.model = between( parseFloat((this.model + offset).toFixed(this.decimals)), this.min, this.max ) this.__updateValue() } }, render (h) { return h('div', { staticClass: this.value === null ? ' q-slider--no-value' : '', attrs: { role: 'slider', 'aria-valuemin': this.min, 'aria-valuemax': this.max, 'aria-valuenow': this.value, 'data-step': this.step, 'aria-disabled': this.disable, tabindex: this.computedTabindex }, class: this.classes, on: this.events, directives: this.editable ? [{ name: 'touch-pan', value: this.__pan, modifiers: { horizontal: true, prevent: true, stop: true, mouse: true, mouseAllDir: true } }] : null }, [ h('div', { staticClass: 'q-slider__track-container absolute overflow-hidden' }, [ h('div', { staticClass: 'q-slider__track absolute-full', style: this.trackStyle }), this.markers === true ? h('div', { staticClass: 'q-slider__track-markers absolute-full fit', style: this.markerStyle }) : null ]), h('div', { staticClass: 'q-slider__thumb-container absolute non-selectable', class: this.thumbClass, style: this.thumbStyle }, [ h('svg', { staticClass: 'q-slider__thumb absolute', attrs: { width: '21', height: '21' } }, [ h('circle', { attrs: { cx: '10.5', cy: '10.5', r: '7.875' } }) ]), this.label === true || this.labelAlways === true ? h('div', { class: this.pinClass }, [ h('div', { staticClass: 'q-slider__pin-value-marker' }, [ h('div', { staticClass: 'q-slider__pin-value-marker-bg' }), h('div', { class: this.pinTextClass }, [ this.computedLabel ]) ]) ]) : null, h('div', { staticClass: 'q-slider__focus-ring' }) ]) ]) } })