UNPKG

gentics-ui-core

Version:

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

284 lines 31.9 kB
import { Attribute, ChangeDetectorRef, Component, Directive, EventEmitter, HostListener, Input, Optional, Output, forwardRef } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { KeyCode } from '../../common/keycodes'; import { coerceToBoolean } from '../../common/coerce-to-boolean'; import * as i0 from "@angular/core"; const GTX_RADIO_GROUP_VALUE_ACCESSOR = { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => RadioGroup), multi: true }; const GTX_RADIO_BUTTON_VALUE_ACCESSOR = { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => RadioButton), multi: true }; /** * RadioGroup groups multiple {@link RadioButton} elements together. * Use ngModel to connect it to a form model. */ export class RadioGroup { constructor() { this.radioButtons = []; this.onTouched = () => { }; this.onChange = (_) => { }; this.groupID = RadioGroup.instanceCounter++; } get uniqueName() { return 'group-' + this.groupID; } add(radio) { if (this.radioButtons.indexOf(radio) < 0) { this.radioButtons.push(radio); } } remove(radio) { let pos = this.radioButtons.indexOf(radio); if (pos >= 0) { this.radioButtons.splice(pos, 1); } } radioSelected(selected) { for (let radio of this.radioButtons) { if (radio != selected) { radio.writeValue(selected ? selected.value : null); } } // setTimeout because this method is invoked from a child component (RadioButton), which is the wrong direction // for change propagation (which should normally always be parent -> child). If we synchronously now update the // ngModel value, we will cause "changed after checked" errors in dev mode. setTimeout(() => { this.onChange(selected ? selected.value : null); }); } writeValue(value) { for (let radio of this.radioButtons) { radio.writeValue(value); } } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } } RadioGroup.instanceCounter = 0; /** @nocollapse */ RadioGroup.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: RadioGroup, deps: [], target: i0.ɵɵFactoryTarget.Directive }); /** @nocollapse */ RadioGroup.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.3.8", type: RadioGroup, selector: "gtx-radio-group, [gtx-radio-group]", providers: [GTX_RADIO_GROUP_VALUE_ACCESSOR], ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: RadioGroup, decorators: [{ type: Directive, args: [{ selector: 'gtx-radio-group, [gtx-radio-group]', providers: [GTX_RADIO_GROUP_VALUE_ACCESSOR] }] }], ctorParameters: function () { return []; } }); /** * RadioButton wraps the native `<input type="radio">` form element. * To connect multiple radio buttons with a form via ngModel, * wrap them in a {@link RadioGroup} (`<gtx-radio-group>`). * * ```html * <gtx-radio-button [(ngModel)]="val" value="A" label="A"></gtx-radio-button> * <gtx-radio-button [(ngModel)]="val" value="B" label="B"></gtx-radio-button> * <gtx-radio-button [(ngModel)]="val" value="C" label="C"></gtx-radio-button> * ``` * * ## Stateless Mode * By default, the RadioButton keeps track of its own internal checked state. This makes sense * for most use cases, such as when used in a form bound to ngModel. * * However, in some cases we want to explicitly set the state from outside. This is done by binding * to the <code>checked</code> attribute. When this attribute is bound, the checked state of the * RadioButton will *only* change when the value of the binding changes. Clicking on the RadioButton * will have no effect other than to emit an event which the parent can use to update the binding. * * Here is a basic example of a stateless RadioButton where the parent component manages the state: * * ```html * <gtx-radio-button [checked]="isChecked"></gtx-checkbox> * ``` */ export class RadioButton { constructor(group, modelAttrib, changeDetector) { this.group = group; this.changeDetector = changeDetector; /** * Sets the radio button to be auto-focused. Handled by `AutofocusDirective`. */ this.autofocus = false; /** * The disabled state of the control */ this.disabled = false; /** * ID of the control */ this.id = randomID(); /** * Label for the radio button */ this.label = ''; /** * Sets the required state */ this.required = false; /** * Value associated with this input */ this.value = ''; /** * Blur event */ this.blur = new EventEmitter(true); /** * Focus event */ this.focus = new EventEmitter(true); /** * Change event */ this.change = new EventEmitter(true); this.tabbedFocus = false; this.inputChecked = false; /** * See note above on stateless mode. */ this.statelessMode = false; this.onChange = (_) => { }; this.onTouched = () => { }; // Pre-set a common input name for grouped input elements if (group) { this.name = group.uniqueName; } else if (modelAttrib) { this.name = modelAttrib; } } /** * The checked state of the control. When set, the RadioButton will be * in stateless mode. */ get checked() { return this.inputChecked; } set checked(val) { this.statelessMode = true; if (val != this.inputChecked) { this.inputChecked = coerceToBoolean(val); if (val && this.group) { this.group.radioSelected(this); } if (val) { this.onChange(this.value); } else if (val === false) { if (this.group) { this.group.radioSelected(null); } this.onChange(false); } this.change.emit(this.value); } } onBlur() { this.blur.emit(this.checked); this.onTouched(); this.tabbedFocus = false; } onFocus() { this.focus.emit(this.checked); } focusHandler(e) { if (e.keyCode === KeyCode.Tab) { if (!this.tabbedFocus) { this.tabbedFocus = true; } } } writeValue(value) { this.inputChecked = (value === this.value); } ngOnInit() { if (this.inputChecked) { this.onChange(this.value); } if (this.group) { this.group.add(this); if (this.inputChecked) { this.group.radioSelected(this); } } } ngOnDestroy() { if (this.group) { this.group.remove(this); } } onInputChecked(e, input) { if (e) { e.stopPropagation(); } if (this.statelessMode) { let newState = input.checked; if (input.checked !== this.inputChecked) { input.checked = !!this.inputChecked; } this.change.emit(newState); return false; } this.inputChecked = true; this.onChange(this.value); this.change.emit(this.value); if (this.group) { this.group.radioSelected(this); } return true; } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { this.onTouched = fn; } setDisabledState(disabled) { this.disabled = disabled; this.changeDetector.markForCheck(); } } /** @nocollapse */ RadioButton.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: RadioButton, deps: [{ token: RadioGroup, optional: true }, { token: 'ngModel', attribute: true }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); /** @nocollapse */ RadioButton.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: RadioButton, selector: "gtx-radio-button", inputs: { autofocus: "autofocus", checked: "checked", disabled: "disabled", id: "id", label: "label", name: "name", required: "required", value: "value" }, outputs: { blur: "blur", focus: "focus", change: "change" }, host: { listeners: { "keyup": "focusHandler($event)" } }, providers: [GTX_RADIO_BUTTON_VALUE_ACCESSOR], ngImport: i0, template: "<div>\n <input type=\"radio\"\n [attr.id]=\"id\"\n [attr.name]=\"name\"\n [checked]=\"checked\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [value]=\"value\"\n\n (blur)=\"onBlur()\"\n (focus)=\"onFocus()\"\n (change)=\"onInputChecked($event, input)\"\n\n [class.tabbed]=\"tabbedFocus\"\n\n #input\n >\n <label [attr.for]=\"id\" (click)=\"input.focus()\">{{ label }}</label>\n</div>\n" }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: RadioButton, decorators: [{ type: Component, args: [{ selector: 'gtx-radio-button', providers: [GTX_RADIO_BUTTON_VALUE_ACCESSOR], template: "<div>\n <input type=\"radio\"\n [attr.id]=\"id\"\n [attr.name]=\"name\"\n [checked]=\"checked\"\n [disabled]=\"disabled\"\n [required]=\"required\"\n [value]=\"value\"\n\n (blur)=\"onBlur()\"\n (focus)=\"onFocus()\"\n (change)=\"onInputChecked($event, input)\"\n\n [class.tabbed]=\"tabbedFocus\"\n\n #input\n >\n <label [attr.for]=\"id\" (click)=\"input.focus()\">{{ label }}</label>\n</div>\n" }] }], ctorParameters: function () { return [{ type: RadioGroup, decorators: [{ type: Optional }] }, { type: undefined, decorators: [{ type: Attribute, args: ['ngModel'] }] }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { autofocus: [{ type: Input }], checked: [{ type: Input }], disabled: [{ type: Input }], id: [{ type: Input }], label: [{ type: Input }], name: [{ type: Input }], required: [{ type: Input }], value: [{ type: Input }], blur: [{ type: Output }], focus: [{ type: Output }], change: [{ type: Output }], focusHandler: [{ type: HostListener, args: ['keyup', ['$event']] }] } }); function randomID() { return 'radio-' + Math.random().toString(36).substr(2); } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radio-button.component.js","sourceRoot":"","sources":["../../../../../src/components/radio-button/radio-button.component.ts","../../../../../src/components/radio-button/radio-button.tpl.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Attribute,\n    ChangeDetectorRef,\n    Component,\n    Directive,\n    EventEmitter,\n    HostListener,\n    Input,\n    OnDestroy,\n    OnInit,\n    Optional,\n    Output,\n    forwardRef\n} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\nimport {KeyCode} from '../../common/keycodes';\nimport {coerceToBoolean} from '../../common/coerce-to-boolean';\n\n\nconst GTX_RADIO_GROUP_VALUE_ACCESSOR = {\n    provide: NG_VALUE_ACCESSOR,\n    useExisting: forwardRef(() => RadioGroup),\n    multi: true\n};\n\nconst GTX_RADIO_BUTTON_VALUE_ACCESSOR = {\n    provide: NG_VALUE_ACCESSOR,\n    useExisting: forwardRef(() => RadioButton),\n    multi: true\n};\n\n/**\n * RadioGroup groups multiple {@link RadioButton} elements together.\n * Use ngModel to connect it to a form model.\n */\n@Directive({\n    selector: 'gtx-radio-group, [gtx-radio-group]',\n    providers: [GTX_RADIO_GROUP_VALUE_ACCESSOR]\n})\nexport class RadioGroup implements ControlValueAccessor {\n\n    private static instanceCounter: number = 0;\n\n    private radioButtons: RadioButton[] = [];\n    private groupID: number;\n\n    get uniqueName(): string {\n        return 'group-' + this.groupID;\n    }\n\n    constructor() {\n        this.groupID = RadioGroup.instanceCounter++;\n    }\n\n    add(radio: RadioButton): void {\n        if (this.radioButtons.indexOf(radio) < 0) {\n            this.radioButtons.push(radio);\n        }\n    }\n\n    remove(radio: RadioButton): void {\n        let pos: number = this.radioButtons.indexOf(radio);\n        if (pos >= 0) {\n            this.radioButtons.splice(pos, 1);\n        }\n    }\n\n    radioSelected(selected?: RadioButton): void {\n        for (let radio of this.radioButtons) {\n            if (radio != selected) {\n                radio.writeValue(selected ? selected.value : null);\n            }\n        }\n        // setTimeout because this method is invoked from a child component (RadioButton), which is the wrong direction\n        // for change propagation (which should normally always be parent -> child). If we synchronously now update the\n        // ngModel value, we will cause \"changed after checked\" errors in dev mode.\n        setTimeout(() => {\n            this.onChange(selected ? selected.value : null);\n        });\n    }\n\n    writeValue(value: any): void {\n        for (let radio of this.radioButtons) {\n            radio.writeValue(value);\n        }\n    }\n\n    registerOnChange(fn: Function): void {\n        this.onChange = fn;\n    }\n\n    registerOnTouched(fn: Function): void {\n        this.onTouched = fn;\n    }\n\n    private onTouched: Function = () => {};\n    private onChange: Function = (_: any) => {};\n}\n\n\n/**\n * RadioButton wraps the native `<input type=\"radio\">` form element.\n * To connect multiple radio buttons with a form via ngModel,\n * wrap them in a {@link RadioGroup} (`<gtx-radio-group>`).\n *\n * ```html\n * <gtx-radio-button [(ngModel)]=\"val\" value=\"A\" label=\"A\"></gtx-radio-button>\n * <gtx-radio-button [(ngModel)]=\"val\" value=\"B\" label=\"B\"></gtx-radio-button>\n * <gtx-radio-button [(ngModel)]=\"val\" value=\"C\" label=\"C\"></gtx-radio-button>\n * ```\n *\n * ## Stateless Mode\n * By default, the RadioButton keeps track of its own internal checked state. This makes sense\n * for most use cases, such as when used in a form bound to ngModel.\n *\n * However, in some cases we want to explicitly set the state from outside. This is done by binding\n * to the <code>checked</code> attribute. When this attribute is bound, the checked state of the\n * RadioButton will *only* change when the value of the binding changes. Clicking on the RadioButton\n * will have no effect other than to emit an event which the parent can use to update the binding.\n *\n * Here is a basic example of a stateless RadioButton where the parent component manages the state:\n *\n * ```html\n * <gtx-radio-button [checked]=\"isChecked\"></gtx-checkbox>\n * ```\n */\n@Component({\n    selector: 'gtx-radio-button',\n    templateUrl: './radio-button.tpl.html',\n    providers: [GTX_RADIO_BUTTON_VALUE_ACCESSOR]\n})\nexport class RadioButton implements ControlValueAccessor, OnInit, OnDestroy {\n    /**\n     * Sets the radio button to be auto-focused. Handled by `AutofocusDirective`.\n     */\n    @Input() autofocus: boolean = false;\n\n    /**\n     * The checked state of the control. When set, the RadioButton will be\n     * in stateless mode.\n     */\n    @Input() get checked(): boolean {\n        return this.inputChecked;\n    }\n    set checked(val: boolean) {\n        this.statelessMode = true;\n        if (val != this.inputChecked) {\n            this.inputChecked = coerceToBoolean(val);\n            if (val && this.group) {\n                this.group.radioSelected(this);\n            }\n            if (val) {\n                this.onChange(this.value);\n            } else if (val === false) {\n                if (this.group) {\n                    this.group.radioSelected(null);\n                }\n                this.onChange(false);\n            }\n            this.change.emit(this.value);\n        }\n    }\n\n    /**\n     * The disabled state of the control\n     */\n    @Input() disabled: boolean = false;\n\n    /**\n     * ID of the control\n     */\n    @Input() id: string = randomID();\n\n    /**\n     * Label for the radio button\n     */\n    @Input() label: string = '';\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     * Value associated with this input\n     */\n    @Input() value: any = '';\n\n    /**\n     * Blur event\n     */\n    @Output() blur = new EventEmitter<boolean>(true);\n\n    /**\n     * Focus event\n     */\n    @Output() focus = new EventEmitter<boolean>(true);\n\n    /**\n     * Change event\n     */\n    @Output() change = new EventEmitter<any>(true);\n\n    tabbedFocus: boolean = false;\n    private inputChecked: boolean = false;\n    /**\n     * See note above on stateless mode.\n     */\n    private statelessMode: boolean = false;\n\n    constructor(@Optional() private group: RadioGroup,\n                @Attribute('ngModel') modelAttrib: string,\n                private changeDetector: ChangeDetectorRef) {\n\n        // Pre-set a common input name for grouped input elements\n        if (group) {\n            this.name = group.uniqueName;\n        } else if (modelAttrib) {\n            this.name = modelAttrib;\n        }\n    }\n\n    onBlur(): void {\n        this.blur.emit(this.checked);\n        this.onTouched();\n        this.tabbedFocus = false;\n    }\n\n    onFocus(): void {\n        this.focus.emit(this.checked);\n    }\n\n    @HostListener('keyup', ['$event'])\n      focusHandler(e: KeyboardEvent): void {\n          if (e.keyCode === KeyCode.Tab) {\n              if (!this.tabbedFocus) {\n                  this.tabbedFocus = true;\n              }\n          }\n      }\n\n    writeValue(value: any): void {\n        this.inputChecked = (value === this.value);\n    }\n\n    ngOnInit(): void {\n        if (this.inputChecked) {\n            this.onChange(this.value);\n        }\n\n        if (this.group) {\n            this.group.add(this);\n            if (this.inputChecked) {\n                this.group.radioSelected(this);\n            }\n        }\n    }\n\n    ngOnDestroy(): void {\n        if (this.group) {\n            this.group.remove(this);\n        }\n    }\n\n    onInputChecked(e: Event, input: HTMLInputElement): boolean {\n        if (e) {\n            e.stopPropagation();\n        }\n        if (this.statelessMode) {\n            let newState = input.checked;\n            if (input.checked !== this.inputChecked) {\n                input.checked = !!this.inputChecked;\n            }\n            this.change.emit(newState);\n            return false;\n        }\n\n        this.inputChecked = true;\n        this.onChange(this.value);\n        this.change.emit(this.value);\n        if (this.group) {\n            this.group.radioSelected(this);\n        }\n        return true;\n    }\n\n    registerOnChange(fn: Function): void { this.onChange = fn; }\n    registerOnTouched(fn: Function): void { this.onTouched = fn; }\n    setDisabledState(disabled: boolean): void {\n        this.disabled = disabled;\n        this.changeDetector.markForCheck();\n    }\n\n    private onChange: Function = (_: any) => {};\n    private onTouched: Function = () => {};\n}\n\nfunction randomID(): string {\n    return 'radio-' + Math.random().toString(36).substr(2);\n}\n","<div>\n    <input type=\"radio\"\n           [attr.id]=\"id\"\n           [attr.name]=\"name\"\n           [checked]=\"checked\"\n           [disabled]=\"disabled\"\n           [required]=\"required\"\n           [value]=\"value\"\n\n           (blur)=\"onBlur()\"\n           (focus)=\"onFocus()\"\n           (change)=\"onInputChecked($event, input)\"\n\n           [class.tabbed]=\"tabbedFocus\"\n\n           #input\n    >\n    <label [attr.for]=\"id\" (click)=\"input.focus()\">{{ label }}</label>\n</div>\n"]}