@nova-ui/bits
Version:
SolarWinds Nova Framework
293 lines • 47.5 kB
JavaScript
// © 2022 SolarWinds Worldwide, LLC. All rights reserved.
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files (the "Software"), to
// deal in the Software without restriction, including without limitation the
// rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
// sell copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions:
//
// The above copyright notice and this permission notice shall be included in
// all copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
// THE SOFTWARE.
import { ChangeDetectorRef, Component, ContentChildren, ElementRef, EventEmitter, forwardRef, Input, Optional, Output, QueryList, Renderer2, ViewChild, ViewContainerRef, ViewEncapsulation, } from "@angular/core";
import { NG_VALUE_ACCESSOR } from "@angular/forms";
import _isNil from "lodash/isNil";
import _isUndefined from "lodash/isUndefined";
import { DOCUMENT_CLICK_EVENT } from "../../constants/event.constants";
import { EventBusService } from "../../services/event-bus.service";
import { NuiFormFieldControl } from "../form-field/public-api";
import * as i0 from "@angular/core";
import * as i1 from "../../services/event-bus.service";
import * as i2 from "@angular/common";
/*
* <example-url>./../examples/index.html#/radio-group</example-url>
*/
export class RadioGroupComponent {
/**
* Stores the value from selected radio
*/
get value() {
return this._value;
}
set value(newValue) {
if (this._value !== newValue) {
// Set this before proceeding to ensure no circular loop occurs with selection.
this._value = newValue;
this.updateSelectedRadioFromValue();
this.checkSelectedRadioButton();
}
}
constructor(renderer) {
this.renderer = renderer;
/**
* Input to set aria label text
*/
this.ariaLabel = "";
/**
* Emits an event when the radio selection changes
*/
this.valueChange = new EventEmitter();
this._value = null;
this.selectedRadio = null;
this.subscriptions = new Array();
this.setChildDisabled = (child) => {
if (!_isUndefined(this.disabled)) {
child.disabled = this.disabled;
}
};
}
registerChild(child) {
this.renderer.setAttribute(child.inputViewContainer.element.nativeElement, "name", this.name);
this.subscriptions.push(this.subscribeToRadioEvent(child));
// timeout to prevent "expression changed after it has been checked" error
setTimeout(() => {
this.setChildDisabled(child);
});
}
ngAfterContentInit() {
this.children.toArray().forEach((child) => this.registerChild(child));
this.children.changes.subscribe((radioComponentQueryList) => {
this.subscriptions.forEach((sub) => sub.unsubscribe());
radioComponentQueryList
.toArray()
.forEach((child) => this.registerChild(child));
});
}
onChange(value) { }
onTouched() { }
writeValue(value) {
this.value = value;
}
registerOnChange(fn) {
this.onChange = fn;
}
registerOnTouched(fn) {
this.onTouched = fn;
}
setDisabledState(isDisabled) {
this.disabled = isDisabled;
if (this.children) {
this.children.toArray().forEach(this.setChildDisabled);
}
}
ngOnDestroy() {
this.subscriptions.forEach((sub) => sub.unsubscribe());
}
checkSelectedRadioButton() {
if (this.selectedRadio && !this.selectedRadio.checked) {
this.selectedRadio.checked = true;
}
}
updateSelectedRadioFromValue() {
// If the value already matches the selected radio, do nothing.
const isAlreadySelected = this.selectedRadio !== null &&
this.selectedRadio.value === this._value;
if (this.children && !isAlreadySelected) {
this.children.forEach((radio) => {
radio.checked = this.value === radio.value;
if (radio.checked) {
this.selectedRadio = radio;
}
});
}
}
subscribeToRadioEvent(radio) {
return radio.valueChange.subscribe((value) => {
this.value = value;
this.valueChange.emit(value);
if (!radio.keepFormPristine) {
this.onChange(this.value);
this.onTouched();
}
this.writeValue(this.value);
});
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RadioGroupComponent, deps: [{ token: i0.Renderer2 }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: RadioGroupComponent, selector: "nui-radio-group", inputs: { ariaLabel: "ariaLabel", name: "name", value: "value", disabled: "disabled" }, outputs: { valueChange: "valueChange" }, host: { attributes: { "role": "radiogroup" } }, providers: [
{
provide: NuiFormFieldControl,
useExisting: forwardRef(() => RadioGroupComponent),
multi: true,
},
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => RadioGroupComponent),
multi: true,
},
], queries: [{ propertyName: "children", predicate: i0.forwardRef(() => RadioComponent), descendants: true }], ngImport: i0, template: `<div class="nui-radio-group" [attr.aria-label]="ariaLabel">
<ng-content></ng-content>
</div>`, isInline: true }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RadioGroupComponent, decorators: [{
type: Component,
args: [{
selector: "nui-radio-group",
template: `<div class="nui-radio-group" [attr.aria-label]="ariaLabel">
<ng-content></ng-content>
</div>`,
providers: [
{
provide: NuiFormFieldControl,
useExisting: forwardRef(() => RadioGroupComponent),
multi: true,
},
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => RadioGroupComponent),
multi: true,
},
],
host: { role: "radiogroup" },
}]
}], ctorParameters: () => [{ type: i0.Renderer2 }], propDecorators: { ariaLabel: [{
type: Input
}], name: [{
type: Input
}], value: [{
type: Input
}], disabled: [{
type: Input
}], valueChange: [{
type: Output
}], children: [{
type: ContentChildren,
args: [forwardRef(() => RadioComponent), { descendants: true }]
}] } });
/**
* @ignore
* Should be used only within nui-radio-group.
*/
export class RadioComponent {
/**
* Sets whether the radio button is disabled
*/
get disabled() {
return this._disabled;
}
set disabled(val) {
if (this._disabled !== val) {
this.changeDetector.markForCheck();
}
this._disabled = val;
}
constructor(radioGroup, changeDetector, eventBusService) {
this.changeDetector = changeDetector;
this.eventBusService = eventBusService;
/**
* Emits an event when the value changes
*/
this.valueChange = new EventEmitter();
/**
* Sets whether the radio button is selected
*/
this.checked = false;
/**
* Input to set aria label text
*/
this.ariaLabel = "";
this.keepFormPristine = true;
this._disabled = false;
this.radioGroup = radioGroup;
}
ngOnInit() {
if (this.radioGroup !== null) {
if (this.radioGroup.value === this.value) {
this.checked = true;
}
}
if (this.checked) {
// TODO: remove timeout in v10 NUI-4843
// nui-radio-group should subscribe before event is emitted
this.timeoutId = setTimeout(() => {
this.valueChange.emit(this.value);
}, 0);
}
// Checks if user supplied any content as a label for radio button to adjust styles for radio buttons without labels
this.radioTranscludeIsEmpty = _isNil(this.radioTransclude.nativeElement.firstChild);
}
ngOnDestroy() {
clearTimeout(this.timeoutId);
}
hoverHandler() {
this.hovered = !this.hovered;
}
changeHandler() {
this.keepFormPristine = false;
this.valueChange.emit(this.value);
}
onInputClick(event) {
// We have to stop propagation for click events on the visual hidden input element.
// By default, when a user clicks on a label element, a generated click event will be
// dispatched on the associated input element. Since we are using a label element as our
// root container, the click event on the `radio-button` will be executed twice.
// The real click event will bubble up, and the generated click event also tries to bubble up.
// This will lead to multiple click events.
// Preventing bubbling for the second event will solve that issue.
event.stopPropagation();
this.eventBusService.getStream(DOCUMENT_CLICK_EVENT).next(event);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RadioComponent, deps: [{ token: RadioGroupComponent, optional: true }, { token: i0.ChangeDetectorRef }, { token: i1.EventBusService }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: RadioComponent, selector: "nui-radio", inputs: { value: "value", hovered: "hovered", checked: "checked", hint: "hint", disabled: "disabled", ariaLabel: "ariaLabel" }, outputs: { valueChange: "valueChange" }, host: { attributes: { "role": "radio" }, properties: { "class.nui-radio--hovered": "hovered", "class.nui-radio--checked": "checked" } }, queries: [{ propertyName: "contentHints", predicate: ["[nui-radio-hint]"] }], viewQueries: [{ propertyName: "inputViewContainer", first: true, predicate: ["inputViewContainer"], descendants: true, read: ViewContainerRef, static: true }, { propertyName: "radioTransclude", first: true, predicate: ["radioTransclude"], descendants: true, static: true }], ngImport: i0, template: "<div\n class=\"nui-radio\"\n [class.nui-radio--empty]=\"radioTranscludeIsEmpty && !hint\"\n>\n <label\n class=\"nui-radio__label\"\n [class.nui-radio__label--translucent]=\"disabled\"\n (mouseenter)=\"hoverHandler()\"\n (mouseleave)=\"hoverHandler()\"\n >\n <div class=\"nui-radio__content\">\n <input\n #inputViewContainer\n class=\"nui-radio__input\"\n type=\"radio\"\n [value]=\"value\"\n [checked]=\"checked\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n (change)=\"changeHandler()\"\n (click)=\"onInputClick($event)\"\n />\n\n <div\n class=\"nui-radio__mark\"\n [class.nui-radio__mark--disabled]=\"disabled\"\n >\n <div class=\"nui-radio__glyph\"></div>\n </div>\n\n <div #radioTransclude class=\"nui-radio__transclude\">\n <ng-content></ng-content>\n </div>\n </div>\n\n <div #hintDiv class=\"nui-radio__hint nui-help-hint\">\n <ng-content select=\"[hint]\"></ng-content>\n <ng-container *ngIf=\"!!hint\">\n {{ hint }}\n </ng-container>\n </div>\n </label>\n</div>\n", styles: [".nui-radio-group-inline>.nui-radio-group{display:inline-flex}.nui-radio-group-inline>.nui-radio-group .nui-radio{margin-left:5px}.nui .nui-radio{border-radius:3px;cursor:pointer;line-height:14px}.nui .nui-radio:hover{background-color:var(--nui-color-bg-transparent-hover,rgba(17, 17, 17, .05))}.nui .nui-radio__label{cursor:pointer;font-weight:400;margin:0;min-height:30px;min-width:30px;padding:7px;position:relative;width:100%}.nui .nui-radio__label--translucent{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none;user-select:none;color:var(--nui-color-text-disabled,rgba(17, 17, 17, .3));cursor:default}.nui .nui-radio__content{align-items:center;display:flex;flex-wrap:nowrap;justify-content:flex-start;min-height:16px}.nui .nui-radio__mark{background:var(--nui-color-bg-content,#fff);background-size:16px;border:solid 2px;border-color:var(--nui-color-active,#0079aa);border-radius:100%;height:16px;margin:0 7px 0 0;padding:3px;width:16px;flex-shrink:0;flex-grow:0}.nui .nui-radio__mark.nui-radio__mark--disabled{background-color:var(--nui-color-bg-content,#fff);border-color:var(--nui-color-disabled,#b3b3b3);cursor:not-allowed}.nui .nui-radio__glyph{background:var(--nui-color-bg-content,#fff);border-radius:100%;display:none;height:100%;width:100%}.nui .nui-radio__input{-webkit-appearance:none;-moz-appearance:none;appearance:none;border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;outline:0;overflow:hidden;padding:0;position:absolute;width:1px}.nui .nui-radio__input[disabled],.nui .nui-radio__input.disabled{cursor:not-allowed}.nui .nui-radio__input:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.nui .nui-radio__input:disabled+.nui-radio__mark{background-color:var(--nui-color-bg-content,#fff);border-color:var(--nui-color-disabled,#b3b3b3)}.nui .nui-radio.nui-radio--empty{display:inline-block}.nui .nui-radio.nui-radio--empty .nui-radio__label{justify-content:center;margin:auto;width:auto}.nui .nui-radio.nui-radio--empty .nui-radio__mark{margin:auto}.nui .nui-radio .nui-help-hint{line-height:14px;margin-left:23px;padding:0}.nui .nui-radio--checked .nui-radio__mark{background-color:var(--nui-color-active,#0079aa);border-color:var(--nui-color-active,#0079aa)}.nui .nui-radio--checked .nui-radio__mark>.nui-radio__glyph{display:block}.nui .nui-radio--checked .nui-radio__mark--disabled{background-color:var(--nui-color-off,#4d4d4d);border-color:var(--nui-color-off,#4d4d4d)}.nui .nui-radio--checked .nui-radio__input+.nui-radio__mark{background-color:var(--nui-color-active,#0079aa);border-color:var(--nui-color-active,#0079aa)}.nui .nui-radio--checked .nui-radio__input+.nui-radio__mark>.nui-radio__glyph{display:block}.nui .nui-radio--checked .nui-radio__input:disabled+.nui-radio__mark{background-color:var(--nui-color-disabled,#b3b3b3);border-color:var(--nui-color-disabled,#b3b3b3)}.nui-repeat-item__radio .nui-radio:hover{background:none}.nui-repeat-item__radio .nui-radio .nui-radio__hint{display:none}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: RadioComponent, decorators: [{
type: Component,
args: [{ selector: "nui-radio", encapsulation: ViewEncapsulation.None, host: {
"[class.nui-radio--hovered]": "hovered",
"[class.nui-radio--checked]": "checked",
role: "radio",
}, template: "<div\n class=\"nui-radio\"\n [class.nui-radio--empty]=\"radioTranscludeIsEmpty && !hint\"\n>\n <label\n class=\"nui-radio__label\"\n [class.nui-radio__label--translucent]=\"disabled\"\n (mouseenter)=\"hoverHandler()\"\n (mouseleave)=\"hoverHandler()\"\n >\n <div class=\"nui-radio__content\">\n <input\n #inputViewContainer\n class=\"nui-radio__input\"\n type=\"radio\"\n [value]=\"value\"\n [checked]=\"checked\"\n [disabled]=\"disabled\"\n [attr.aria-label]=\"ariaLabel\"\n (change)=\"changeHandler()\"\n (click)=\"onInputClick($event)\"\n />\n\n <div\n class=\"nui-radio__mark\"\n [class.nui-radio__mark--disabled]=\"disabled\"\n >\n <div class=\"nui-radio__glyph\"></div>\n </div>\n\n <div #radioTransclude class=\"nui-radio__transclude\">\n <ng-content></ng-content>\n </div>\n </div>\n\n <div #hintDiv class=\"nui-radio__hint nui-help-hint\">\n <ng-content select=\"[hint]\"></ng-content>\n <ng-container *ngIf=\"!!hint\">\n {{ hint }}\n </ng-container>\n </div>\n </label>\n</div>\n", styles: [".nui-radio-group-inline>.nui-radio-group{display:inline-flex}.nui-radio-group-inline>.nui-radio-group .nui-radio{margin-left:5px}.nui .nui-radio{border-radius:3px;cursor:pointer;line-height:14px}.nui .nui-radio:hover{background-color:var(--nui-color-bg-transparent-hover,rgba(17, 17, 17, .05))}.nui .nui-radio__label{cursor:pointer;font-weight:400;margin:0;min-height:30px;min-width:30px;padding:7px;position:relative;width:100%}.nui .nui-radio__label--translucent{-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;-o-user-select:none;user-select:none;color:var(--nui-color-text-disabled,rgba(17, 17, 17, .3));cursor:default}.nui .nui-radio__content{align-items:center;display:flex;flex-wrap:nowrap;justify-content:flex-start;min-height:16px}.nui .nui-radio__mark{background:var(--nui-color-bg-content,#fff);background-size:16px;border:solid 2px;border-color:var(--nui-color-active,#0079aa);border-radius:100%;height:16px;margin:0 7px 0 0;padding:3px;width:16px;flex-shrink:0;flex-grow:0}.nui .nui-radio__mark.nui-radio__mark--disabled{background-color:var(--nui-color-bg-content,#fff);border-color:var(--nui-color-disabled,#b3b3b3);cursor:not-allowed}.nui .nui-radio__glyph{background:var(--nui-color-bg-content,#fff);border-radius:100%;display:none;height:100%;width:100%}.nui .nui-radio__input{-webkit-appearance:none;-moz-appearance:none;appearance:none;border:0;clip:rect(0 0 0 0);height:1px;margin:-1px;outline:0;overflow:hidden;padding:0;position:absolute;width:1px}.nui .nui-radio__input[disabled],.nui .nui-radio__input.disabled{cursor:not-allowed}.nui .nui-radio__input:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.nui .nui-radio__input:disabled+.nui-radio__mark{background-color:var(--nui-color-bg-content,#fff);border-color:var(--nui-color-disabled,#b3b3b3)}.nui .nui-radio.nui-radio--empty{display:inline-block}.nui .nui-radio.nui-radio--empty .nui-radio__label{justify-content:center;margin:auto;width:auto}.nui .nui-radio.nui-radio--empty .nui-radio__mark{margin:auto}.nui .nui-radio .nui-help-hint{line-height:14px;margin-left:23px;padding:0}.nui .nui-radio--checked .nui-radio__mark{background-color:var(--nui-color-active,#0079aa);border-color:var(--nui-color-active,#0079aa)}.nui .nui-radio--checked .nui-radio__mark>.nui-radio__glyph{display:block}.nui .nui-radio--checked .nui-radio__mark--disabled{background-color:var(--nui-color-off,#4d4d4d);border-color:var(--nui-color-off,#4d4d4d)}.nui .nui-radio--checked .nui-radio__input+.nui-radio__mark{background-color:var(--nui-color-active,#0079aa);border-color:var(--nui-color-active,#0079aa)}.nui .nui-radio--checked .nui-radio__input+.nui-radio__mark>.nui-radio__glyph{display:block}.nui .nui-radio--checked .nui-radio__input:disabled+.nui-radio__mark{background-color:var(--nui-color-disabled,#b3b3b3);border-color:var(--nui-color-disabled,#b3b3b3)}.nui-repeat-item__radio .nui-radio:hover{background:none}.nui-repeat-item__radio .nui-radio .nui-radio__hint{display:none}\n"] }]
}], ctorParameters: () => [{ type: RadioGroupComponent, decorators: [{
type: Optional
}] }, { type: i0.ChangeDetectorRef }, { type: i1.EventBusService }], propDecorators: { value: [{
type: Input
}], valueChange: [{
type: Output
}], hovered: [{
type: Input
}], checked: [{
type: Input
}], hint: [{
type: Input
}], disabled: [{
type: Input
}], ariaLabel: [{
type: Input
}], inputViewContainer: [{
type: ViewChild,
args: ["inputViewContainer", { static: true, read: ViewContainerRef }]
}], radioTransclude: [{
type: ViewChild,
args: ["radioTransclude", { static: true }]
}], contentHints: [{
type: ContentChildren,
args: ["[nui-radio-hint]"]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"radio-group.component.js","sourceRoot":"","sources":["../../../../src/lib/radio/radio-group.component.ts","../../../../src/lib/radio/radio.component.html"],"names":[],"mappings":"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","sourcesContent":["// © 2022 SolarWinds Worldwide, LLC. All rights reserved.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n//  of this software and associated documentation files (the \"Software\"), to\n//  deal in the Software without restriction, including without limitation the\n//  rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n//  sell copies of the Software, and to permit persons to whom the Software is\n//  furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n//  all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n//  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n//  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n//  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n//  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n//  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n//  THE SOFTWARE.\n\nimport {\n    AfterContentInit,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    Input,\n    OnDestroy,\n    OnInit,\n    Optional,\n    Output,\n    QueryList,\n    Renderer2,\n    TemplateRef,\n    ViewChild,\n    ViewContainerRef,\n    ViewEncapsulation,\n} from \"@angular/core\";\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from \"@angular/forms\";\nimport _isNil from \"lodash/isNil\";\nimport _isUndefined from \"lodash/isUndefined\";\nimport { Subscription } from \"rxjs\";\n\nimport { DOCUMENT_CLICK_EVENT } from \"../../constants/event.constants\";\nimport { EventBusService } from \"../../services/event-bus.service\";\nimport { NuiFormFieldControl } from \"../form-field/public-api\";\n\n/*\n * <example-url>./../examples/index.html#/radio-group</example-url>\n */\n@Component({\n    selector: \"nui-radio-group\",\n    template: `<div class=\"nui-radio-group\" [attr.aria-label]=\"ariaLabel\">\n        <ng-content></ng-content>\n    </div>`,\n    providers: [\n        {\n            provide: NuiFormFieldControl,\n            useExisting: forwardRef(() => RadioGroupComponent),\n            multi: true,\n        },\n        {\n            provide: NG_VALUE_ACCESSOR,\n            useExisting: forwardRef(() => RadioGroupComponent),\n            multi: true,\n        },\n    ],\n    host: { role: \"radiogroup\" },\n})\nexport class RadioGroupComponent\n    implements AfterContentInit, OnDestroy, ControlValueAccessor\n{\n    /**\n     * Input to set aria label text\n     */\n    @Input() public ariaLabel: string = \"\";\n\n    /**\n     * Sets the \"name\" attribute for each radio button in the group\n     */\n    @Input() public name: string;\n\n    /**\n     * Stores the value from selected radio\n     */\n    @Input()\n    get value(): any {\n        return this._value;\n    }\n    set value(newValue: any) {\n        if (this._value !== newValue) {\n            // Set this before proceeding to ensure no circular loop occurs with selection.\n            this._value = newValue;\n\n            this.updateSelectedRadioFromValue();\n            this.checkSelectedRadioButton();\n        }\n    }\n\n    /**\n     * Sets whether the group is disabled\n     */\n    @Input() public disabled?: boolean;\n\n    /**\n     * Emits an event when the radio selection changes\n     */\n    @Output() public valueChange = new EventEmitter<any>();\n\n    @ContentChildren(forwardRef(() => RadioComponent), { descendants: true })\n    private children: QueryList<RadioComponent>;\n    private _value: any = null;\n    private selectedRadio: RadioComponent | null = null;\n    private subscriptions = new Array<Subscription>();\n    constructor(private renderer: Renderer2) {}\n\n    private registerChild(child: RadioComponent): void {\n        this.renderer.setAttribute(\n            child.inputViewContainer.element.nativeElement,\n            \"name\",\n            this.name\n        );\n        this.subscriptions.push(this.subscribeToRadioEvent(child));\n        // timeout to prevent \"expression changed after it has been checked\" error\n        setTimeout(() => {\n            this.setChildDisabled(child);\n        });\n    }\n\n    public ngAfterContentInit(): void {\n        this.children.toArray().forEach((child) => this.registerChild(child));\n        this.children.changes.subscribe(\n            (radioComponentQueryList: QueryList<RadioComponent>) => {\n                this.subscriptions.forEach((sub) => sub.unsubscribe());\n                radioComponentQueryList\n                    .toArray()\n                    .forEach((child) => this.registerChild(child));\n            }\n        );\n    }\n\n    public onChange(value: any): void {}\n\n    public onTouched(): void {}\n\n    public writeValue(value: any): void {\n        this.value = value;\n    }\n\n    public registerOnChange(fn: (value: any) => void): void {\n        this.onChange = fn;\n    }\n\n    public registerOnTouched(fn: () => void): void {\n        this.onTouched = fn;\n    }\n\n    public setDisabledState(isDisabled: boolean): void {\n        this.disabled = isDisabled;\n        if (this.children) {\n            this.children.toArray().forEach(this.setChildDisabled);\n        }\n    }\n\n    public ngOnDestroy(): void {\n        this.subscriptions.forEach((sub) => sub.unsubscribe());\n    }\n\n    private setChildDisabled = (child: RadioComponent) => {\n        if (!_isUndefined(this.disabled)) {\n            child.disabled = this.disabled;\n        }\n    };\n\n    private checkSelectedRadioButton() {\n        if (this.selectedRadio && !this.selectedRadio.checked) {\n            this.selectedRadio.checked = true;\n        }\n    }\n\n    private updateSelectedRadioFromValue(): void {\n        // If the value already matches the selected radio, do nothing.\n        const isAlreadySelected =\n            this.selectedRadio !== null &&\n            this.selectedRadio.value === this._value;\n        if (this.children && !isAlreadySelected) {\n            this.children.forEach((radio) => {\n                radio.checked = this.value === radio.value;\n                if (radio.checked) {\n                    this.selectedRadio = radio;\n                }\n            });\n        }\n    }\n\n    private subscribeToRadioEvent(radio: RadioComponent): Subscription {\n        return radio.valueChange.subscribe((value: any) => {\n            this.value = value;\n            this.valueChange.emit(value);\n            if (!radio.keepFormPristine) {\n                this.onChange(this.value);\n                this.onTouched();\n            }\n            this.writeValue(this.value);\n        });\n    }\n}\n\n/**\n * @ignore\n * Should be used only within nui-radio-group.\n */\n@Component({\n    selector: \"nui-radio\",\n    templateUrl: \"./radio.component.html\",\n    styleUrls: [\"./radio.component.less\"],\n    encapsulation: ViewEncapsulation.None,\n    host: {\n        \"[class.nui-radio--hovered]\": \"hovered\",\n        \"[class.nui-radio--checked]\": \"checked\",\n        role: \"radio\",\n    },\n})\nexport class RadioComponent implements OnInit, OnDestroy {\n    /**\n     * Sets the radio instance value\n     */\n    @Input() public value: any;\n\n    /**\n     * Emits an event when the value changes\n     */\n    @Output() public valueChange = new EventEmitter<any>();\n\n    @Input() public hovered: boolean;\n\n    /**\n     * Sets whether the radio button is selected\n     */\n    @Input() public checked = false;\n\n    /**\n     * Adds hint text under the radio button\n     */\n    @Input() public hint?: string;\n\n    /**\n     * Sets whether the radio button is disabled\n     */\n    @Input()\n    public get disabled(): boolean {\n        return this._disabled;\n    }\n    public set disabled(val: boolean) {\n        if (this._disabled !== val) {\n            this.changeDetector.markForCheck();\n        }\n        this._disabled = val;\n    }\n\n    /**\n     * Input to set aria label text\n     */\n    @Input() public ariaLabel: string = \"\";\n\n    @ViewChild(\"inputViewContainer\", { static: true, read: ViewContainerRef })\n    public inputViewContainer: ViewContainerRef;\n\n    @ViewChild(\"radioTransclude\", { static: true })\n    public radioTransclude: ElementRef;\n\n    public radioTranscludeIsEmpty: boolean;\n    public keepFormPristine: boolean = true;\n\n    private timeoutId: number;\n    private _disabled: boolean = false;\n\n    readonly radioGroup: RadioGroupComponent | null;\n\n    @ContentChildren(\"[nui-radio-hint]\")\n    protected contentHints: QueryList<TemplateRef<any>>;\n\n    constructor(\n        @Optional() radioGroup: RadioGroupComponent,\n        private changeDetector: ChangeDetectorRef,\n        private eventBusService: EventBusService\n    ) {\n        this.radioGroup = radioGroup;\n    }\n\n    public ngOnInit(): void {\n        if (this.radioGroup !== null) {\n            if (this.radioGroup.value === this.value) {\n                this.checked = true;\n            }\n        }\n\n        if (this.checked) {\n            // TODO: remove timeout in v10 NUI-4843\n            // nui-radio-group should subscribe before event is emitted\n            this.timeoutId = (<any>setTimeout(() => {\n                this.valueChange.emit(this.value);\n            }, 0)) as number;\n        }\n\n        // Checks if user supplied any content as a label for radio button to adjust styles for radio buttons without labels\n        this.radioTranscludeIsEmpty = _isNil(\n            this.radioTransclude.nativeElement.firstChild\n        );\n    }\n\n    public ngOnDestroy(): void {\n        clearTimeout(this.timeoutId);\n    }\n\n    public hoverHandler(): void {\n        this.hovered = !this.hovered;\n    }\n\n    public changeHandler(): void {\n        this.keepFormPristine = false;\n        this.valueChange.emit(this.value);\n    }\n\n    public onInputClick(event: MouseEvent): void {\n        // We have to stop propagation for click events on the visual hidden input element.\n        // By default, when a user clicks on a label element, a generated click event will be\n        // dispatched on the associated input element. Since we are using a label element as our\n        // root container, the click event on the `radio-button` will be executed twice.\n        // The real click event will bubble up, and the generated click event also tries to bubble up.\n        // This will lead to multiple click events.\n        // Preventing bubbling for the second event will solve that issue.\n        event.stopPropagation();\n        this.eventBusService.getStream(DOCUMENT_CLICK_EVENT).next(event);\n    }\n}\n","<div\n    class=\"nui-radio\"\n    [class.nui-radio--empty]=\"radioTranscludeIsEmpty && !hint\"\n>\n    <label\n        class=\"nui-radio__label\"\n        [class.nui-radio__label--translucent]=\"disabled\"\n        (mouseenter)=\"hoverHandler()\"\n        (mouseleave)=\"hoverHandler()\"\n    >\n        <div class=\"nui-radio__content\">\n            <input\n                #inputViewContainer\n                class=\"nui-radio__input\"\n                type=\"radio\"\n                [value]=\"value\"\n                [checked]=\"checked\"\n                [disabled]=\"disabled\"\n                [attr.aria-label]=\"ariaLabel\"\n                (change)=\"changeHandler()\"\n                (click)=\"onInputClick($event)\"\n            />\n\n            <div\n                class=\"nui-radio__mark\"\n                [class.nui-radio__mark--disabled]=\"disabled\"\n            >\n                <div class=\"nui-radio__glyph\"></div>\n            </div>\n\n            <div #radioTransclude class=\"nui-radio__transclude\">\n                <ng-content></ng-content>\n            </div>\n        </div>\n\n        <div #hintDiv class=\"nui-radio__hint nui-help-hint\">\n            <ng-content select=\"[hint]\"></ng-content>\n            <ng-container *ngIf=\"!!hint\">\n                {{ hint }}\n            </ng-container>\n        </div>\n    </label>\n</div>\n"]}