UNPKG

gentics-ui-core

Version:

This is the common core framework for the Gentics CMS and Mesh UI, and other Angular applications.

192 lines 22.9 kB
import { ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, Input, Output, ViewChild } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { coerceToBoolean } from '../../common/coerce-to-boolean'; import * as i0 from "@angular/core"; const GTX_RANGE_VALUE_ACCESSOR = { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => Range), multi: true }; /** * The Range wraps the native `<input type="range">` form element. * * ```html * <gtx-range [(ngModel)]="latitude" step="5" min="-180" max="180"></gtx-range> * ``` */ export class Range { constructor(elementRef, changeDetector) { this.elementRef = elementRef; this.changeDetector = changeDetector; /** * Sets the input field to be auto-focused. Handled by `AutofocusDirective`. */ this.autofocus = false; /** * Sets the disabled state of the input. */ this.disabled = false; /** * Sets a label for the slider. */ this.label = ''; /** * Sets the required state. */ this.required = false; /** * Blur event */ this.blur = new EventEmitter(); /** * Focus event */ this.focus = new EventEmitter(); /** * Change event */ this.change = new EventEmitter(); this.active = false; this.thumbLeft = ''; this.showThumbLabel = true; // ValueAccessor members this.onChange = (value) => { }; this.onTouched = () => { }; } /** * Set to false to not show the thumb label. Defaults to true. */ get thumbLabel() { return this.showThumbLabel; } set thumbLabel(value) { this.showThumbLabel = coerceToBoolean(value); } get canModify() { return !this.disabled; } ngOnInit() { this.writeValue(this.value); } onBlur(e) { e.stopPropagation(); const value = this.getValueFromEvent(e); this.blur.emit(value); this.change.emit(value); } /** * IE11 only fires the 'change' event rather than the 'input' event as the range input value is changed. */ onChangeEvent(e) { e.stopPropagation(); if (this.canModify) { const value = this.currentValue = this.getValueFromEvent(e); this.onChange(value); this.change.emit(value); } } onFocus(e) { e.stopPropagation(); this.focus.emit(this.value); } /** * Browsers other than IE11 fire 'input' continuously as the range value is changed, and fires 'change' on mouseup. */ onInput(e) { e.stopPropagation(); if (this.canModify) { const value = this.currentValue = this.getValueFromEvent(e); this.onChange(value); this.change.emit(value); } } onMousedown(e) { if (this.canModify) { this.active = true; this.setThumbPosition(e); } } onMouseup() { this.active = false; } onMousemove(e) { if (this.canModify) { if (this.active) { this.setThumbPosition(e); } } } writeValue(value) { if (value !== this.currentValue) { this.currentValue = value; this.inputElement.nativeElement.value = this.currentValue; } } registerOnChange(fn) { this.onChange = (value) => fn(Number(value)); } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(disabled) { this.disabled = disabled; this.changeDetector.markForCheck(); } getValueFromEvent(e) { const target = e.target; return Number(target.value); } setThumbPosition(e) { const endMargin = 8; const rangeWrapper = this.elementRef.nativeElement.querySelector('.range-field'); const boundingRect = rangeWrapper.getBoundingClientRect(); const wrapperLeft = boundingRect.left; const wrapperWidth = boundingRect.width; let left = e.pageX - wrapperLeft; if (left < endMargin) { left = endMargin; } else if (left > wrapperWidth - endMargin) { left = wrapperWidth - endMargin; } this.thumbLeft = left + 'px'; } } /** @nocollapse */ Range.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Range, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ Range.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: Range, selector: "gtx-range", inputs: { autofocus: "autofocus", disabled: "disabled", label: "label", max: "max", min: "min", name: "name", required: "required", step: "step", value: "value", id: "id", thumbLabel: "thumbLabel" }, outputs: { blur: "blur", focus: "focus", change: "change" }, providers: [GTX_RANGE_VALUE_ACCESSOR], viewQueries: [{ propertyName: "inputElement", first: true, predicate: ["input"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"range-field\">\n <input type=\"range\"\n [disabled]=\"disabled\"\n [attr.max]=\"max\"\n [attr.min]=\"min\"\n [attr.name]=\"name\"\n [required]=\"required\"\n [attr.step]=\"step\"\n\n [attr.id]=\"id\"\n\n (blur)=\"onBlur($event)\"\n (change)=\"onChangeEvent($event)\"\n (focus)=\"onFocus($event)\"\n (input)=\"onInput($event)\"\n (mousedown)=\"onMousedown($event)\"\n (mouseup)=\"onMouseup()\"\n (mousemove)=\"onMousemove($event)\"\n\n #input\n >\n <span class=\"thumb\"\n [class.hidden]=\"!thumbLabel\"\n [class.active]=\"active\"\n [style.left]=\"thumbLeft\">\n <span class=\"value\">{{ currentValue }}</span>\n </span>\n</div>\n<label [attr.for]=\"id\">{{ label }}</label>\n" }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Range, decorators: [{ type: Component, args: [{ selector: 'gtx-range', providers: [GTX_RANGE_VALUE_ACCESSOR], template: "<div class=\"range-field\">\n <input type=\"range\"\n [disabled]=\"disabled\"\n [attr.max]=\"max\"\n [attr.min]=\"min\"\n [attr.name]=\"name\"\n [required]=\"required\"\n [attr.step]=\"step\"\n\n [attr.id]=\"id\"\n\n (blur)=\"onBlur($event)\"\n (change)=\"onChangeEvent($event)\"\n (focus)=\"onFocus($event)\"\n (input)=\"onInput($event)\"\n (mousedown)=\"onMousedown($event)\"\n (mouseup)=\"onMouseup()\"\n (mousemove)=\"onMousemove($event)\"\n\n #input\n >\n <span class=\"thumb\"\n [class.hidden]=\"!thumbLabel\"\n [class.active]=\"active\"\n [style.left]=\"thumbLeft\">\n <span class=\"value\">{{ currentValue }}</span>\n </span>\n</div>\n<label [attr.for]=\"id\">{{ label }}</label>\n" }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { autofocus: [{ type: Input }], disabled: [{ type: Input }], label: [{ type: Input }], max: [{ type: Input }], min: [{ type: Input }], name: [{ type: Input }], required: [{ type: Input }], step: [{ type: Input }], value: [{ type: Input }], id: [{ type: Input }], thumbLabel: [{ type: Input }], blur: [{ type: Output }], focus: [{ type: Output }], change: [{ type: Output }], inputElement: [{ type: ViewChild, args: ['input', { static: true }] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"range.component.js","sourceRoot":"","sources":["../../../../../src/components/range/range.component.ts","../../../../../src/components/range/range.tpl.html"],"names":[],"mappings":"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","sourcesContent":["import {ChangeDetectorRef, Component, ElementRef, EventEmitter, forwardRef, Input, Output, ViewChild} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\nimport {coerceToBoolean} from '../../common/coerce-to-boolean';\n\nconst GTX_RANGE_VALUE_ACCESSOR = {\n    provide: NG_VALUE_ACCESSOR,\n    useExisting: forwardRef(() => Range),\n    multi: true\n};\n\n/**\n * The Range wraps the native `<input type=\"range\">` form element.\n *\n * ```html\n * <gtx-range [(ngModel)]=\"latitude\" step=\"5\" min=\"-180\" max=\"180\"></gtx-range>\n * ```\n */\n@Component({\n    selector: 'gtx-range',\n    templateUrl: './range.tpl.html',\n    providers: [GTX_RANGE_VALUE_ACCESSOR]\n})\nexport class Range implements ControlValueAccessor {\n    /**\n     * Sets the input field to be auto-focused. Handled by `AutofocusDirective`.\n     */\n    @Input() autofocus: boolean = false;\n\n    /**\n     * Sets the disabled state of the input.\n     */\n    @Input() disabled: boolean = false;\n\n    /**\n     * Sets a label for the slider.\n     */\n    @Input() label: string = '';\n\n    /**\n     * Maximum allowed value.\n     */\n    @Input() max: number;\n\n    /**\n     * Minimum allowed value.\n     */\n    @Input() min: number;\n\n    /**\n     * Name of the input.\n     */\n    @Input() name: string;\n\n    /**\n     * Sets the required state.\n     */\n    @Input() required: boolean = false;\n\n    /**\n     * Amount to increment by when sliding.\n     */\n    @Input() step: number;\n\n    /**\n     * Sets the value of the slider.\n     */\n    @Input() value: number;\n\n    /**\n     * Sets an id for the slider.\n     */\n    @Input() id: string;\n\n    /**\n     * Set to false to not show the thumb label. Defaults to true.\n     */\n    @Input()\n    get thumbLabel(): boolean {\n        return this.showThumbLabel;\n    }\n    set thumbLabel(value: boolean) {\n        this.showThumbLabel = coerceToBoolean(value);\n    }\n\n    /**\n     * Blur event\n     */\n    @Output() blur = new EventEmitter<number>();\n\n    /**\n     * Focus event\n     */\n    @Output() focus = new EventEmitter<number>();\n\n    /**\n     * Change event\n     */\n    @Output() change = new EventEmitter<number>();\n\n\n    active: boolean = false;\n    thumbLeft: string = '';\n    currentValue: number;\n    private showThumbLabel: boolean = true;\n\n\n    @ViewChild('input', { static: true }) private inputElement: ElementRef;\n\n    private get canModify(): boolean {\n        return !this.disabled;\n    }\n\n    // ValueAccessor members\n    onChange = (value: any): void => { };\n    onTouched = (): void => { };\n\n    constructor(private elementRef: ElementRef,\n                private changeDetector: ChangeDetectorRef) {}\n\n    ngOnInit(): void {\n        this.writeValue(this.value);\n    }\n\n    onBlur(e: FocusEvent): void {\n        e.stopPropagation();\n        const value = this.getValueFromEvent(e);\n        this.blur.emit(value);\n        this.change.emit(value);\n    }\n\n    /**\n     * IE11 only fires the 'change' event rather than the 'input' event as the range input value is changed.\n     */\n    onChangeEvent(e: Event): void | boolean {\n        e.stopPropagation();\n        if (this.canModify) {\n            const value = this.currentValue = this.getValueFromEvent(e);\n            this.onChange(value);\n            this.change.emit(value);\n        }\n    }\n\n    onFocus(e: FocusEvent): void {\n        e.stopPropagation();\n        this.focus.emit(this.value);\n    }\n\n    /**\n     * Browsers other than IE11 fire 'input' continuously as the range value is changed, and fires 'change' on mouseup.\n     */\n    onInput(e: Event): void {\n        e.stopPropagation();\n        if (this.canModify) {\n            const value = this.currentValue = this.getValueFromEvent(e);\n            this.onChange(value);\n            this.change.emit(value);\n        }\n    }\n\n    onMousedown(e: MouseEvent): void {\n        if (this.canModify) {\n            this.active = true;\n            this.setThumbPosition(e);\n        }\n    }\n\n    onMouseup(): void {\n        this.active = false;\n\n    }\n\n    onMousemove(e: MouseEvent): void {\n        if (this.canModify) {\n            if (this.active) {\n                this.setThumbPosition(e);\n            }\n        }\n    }\n\n    writeValue(value: any): void {\n        if (value !== this.currentValue) {\n            this.currentValue = value;\n            this.inputElement.nativeElement.value = this.currentValue;\n        }\n    }\n\n    registerOnChange(fn: (newValue: number) => void): void {\n        this.onChange = (value: any) => fn(Number(value));\n    }\n\n    registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    setDisabledState(disabled: boolean): void {\n        this.disabled = disabled;\n        this.changeDetector.markForCheck();\n    }\n\n    private getValueFromEvent(e: Event): number {\n        const target: HTMLInputElement = <HTMLInputElement> e.target;\n        return Number(target.value);\n    }\n\n    private setThumbPosition(e: MouseEvent): void {\n        const endMargin = 8;\n        const rangeWrapper = this.elementRef.nativeElement.querySelector('.range-field') as HTMLDivElement;\n        const boundingRect = rangeWrapper.getBoundingClientRect();\n        const wrapperLeft = boundingRect.left;\n        const wrapperWidth = boundingRect.width;\n        let left = e.pageX - wrapperLeft;\n        if (left < endMargin) {\n            left = endMargin;\n        } else if (left > wrapperWidth - endMargin) {\n            left = wrapperWidth - endMargin;\n        }\n        this.thumbLeft = left + 'px';\n    }\n}\n","<div class=\"range-field\">\n    <input type=\"range\"\n           [disabled]=\"disabled\"\n           [attr.max]=\"max\"\n           [attr.min]=\"min\"\n           [attr.name]=\"name\"\n           [required]=\"required\"\n           [attr.step]=\"step\"\n\n           [attr.id]=\"id\"\n\n           (blur)=\"onBlur($event)\"\n           (change)=\"onChangeEvent($event)\"\n           (focus)=\"onFocus($event)\"\n           (input)=\"onInput($event)\"\n           (mousedown)=\"onMousedown($event)\"\n           (mouseup)=\"onMouseup()\"\n           (mousemove)=\"onMousemove($event)\"\n\n           #input\n    >\n    <span class=\"thumb\"\n          [class.hidden]=\"!thumbLabel\"\n          [class.active]=\"active\"\n          [style.left]=\"thumbLeft\">\n        <span class=\"value\">{{ currentValue }}</span>\n    </span>\n</div>\n<label [attr.for]=\"id\">{{ label }}</label>\n"]}