@nova-ui/bits
Version:
SolarWinds Nova Framework
130 lines • 24.9 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 { ChangeDetectionStrategy, ChangeDetectorRef, Component, EventEmitter, Input, Output, ViewEncapsulation, } from "@angular/core";
import _cloneDeep from "lodash/cloneDeep";
import moment from "moment/moment";
import { TimeframeService } from "./services/timeframe.service";
import * as i0 from "@angular/core";
import * as i1 from "./services/timeframe.service";
import * as i2 from "../date-time-picker/date-time-picker.component";
// <example-url>./../examples/index.html#/time-frame-picker</example-url>
export class TimeFramePickerComponent {
constructor(timeFrameService, changeDetector) {
this.timeFrameService = timeFrameService;
this.changeDetector = changeDetector;
/** callback to be invoked on model change */
this.changed = new EventEmitter();
}
ngOnChanges(changes) {
if (changes["startModel"]) {
this.model = TimeframeService.cloneTimeFrame(this.startModel);
this.validateCombination();
}
}
ngOnInit() {
if (this.startModel) {
this.model = TimeframeService.cloneTimeFrame(this.startModel);
this.validateCombination();
}
}
selectPreset(key, value) {
const timeframe = this.timeFrameService.getTimeframe(value.startDatetimePattern, value.endDatetimePattern);
timeframe.selectedPresetId = key;
this.model = timeframe;
}
isPresetSelected(key) {
return this.model && this.model.selectedPresetId === key;
}
onChangeInternalStart(event) {
this.model.startDatetime = event;
this.onChangeInternal();
}
onChangeInternalEnd(event) {
this.model.endDatetime = event;
this.onChangeInternal();
}
onBlurInternal() {
this.validateCombination();
if (!this.model.startDatetime || !this.model.endDatetime) {
this.model = _cloneDeep(this.modelDefault);
}
else {
this.modelDefault = _cloneDeep(this.model);
}
this.isFocused = false;
}
onFocusInternal() {
this.modelDefault = _cloneDeep(this.model);
this.isFocused = true;
}
validateCombination() {
if (this.model.startDatetime && this.model.endDatetime) {
if (this.model.startDatetime >= this.model.endDatetime) {
this.model.endDatetime = moment(this.model.startDatetime.valueOf() + this.distanceToEndDate);
this.model.selectedPresetId = undefined;
this.model.title = undefined;
}
else {
this.updateDistanceToEndDate(this.model.startDatetime, this.model.endDatetime);
}
}
}
updateDistanceToEndDate(newStartDatetime, newEndDatetime) {
const startMoment = moment(newStartDatetime);
const endMoment = moment(newEndDatetime);
if (startMoment.isValid() && endMoment.isValid()) {
this.distanceToEndDate =
newEndDatetime.valueOf() - newStartDatetime.valueOf();
}
}
onChangeInternal() {
if (!this.isFocused) {
this.validateCombination();
}
// clear 'selectedPresetId' and 'title' values
if (this.model.selectedPresetId) {
const timeFrame = this.timeFrameService.getTimeframeByPresetId(this.model.selectedPresetId);
if (!this.model.startDatetime ||
!this.model.endDatetime ||
!this.timeFrameService.isEqualDuration(this.model, timeFrame)) {
this.model.selectedPresetId = undefined;
this.model.title = undefined;
}
}
this.changed.emit(this.model);
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TimeFramePickerComponent, deps: [{ token: i1.TimeframeService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TimeFramePickerComponent, selector: "nui-time-frame-picker", inputs: { minDate: "minDate", maxDate: "maxDate", startModel: "startModel", appendToBody: "appendToBody" }, outputs: { changed: "changed" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"nui-time-frame-picker\">\n <div class=\"nui-time-frame-picker__date-times\">\n <label class=\"nui-time-frame-picker__label\" i18n>Start</label>\n <span\n class=\"nui-time-frame-picker__date-time nui-time-frame-picker__date-time_start\"\n >\n <nui-date-time-picker\n [minDate]=\"minDate\"\n [maxDate]=\"maxDate\"\n [model]=\"model.startDatetime\"\n (modelChanged)=\"onChangeInternalStart($event)\"\n (focus)=\"onFocusInternal()\"\n (blur)=\"onBlurInternal()\"\n [appendToBody]=\"appendToBody\"\n ></nui-date-time-picker>\n </span>\n <label class=\"nui-time-frame-picker__label\" i18n>End</label>\n <span\n class=\"nui-time-frame-picker__date-time nui-time-frame-picker__date-time_end\"\n >\n <nui-date-time-picker\n [minDate]=\"minDate\"\n [maxDate]=\"maxDate\"\n [model]=\"model.endDatetime\"\n (modelChanged)=\"onChangeInternalEnd($event)\"\n (focus)=\"onFocusInternal()\"\n (blur)=\"onBlurInternal()\"\n [appendToBody]=\"appendToBody\"\n ></nui-date-time-picker>\n </span>\n </div>\n</div>\n", styles: [".nui-time-frame-picker__date-time{display:flex}.nui-time-frame-picker__date-time_start{margin-bottom:15px}.nui-time-frame-picker__date-time_end{padding-bottom:10px}\n"], dependencies: [{ kind: "component", type: i2.DateTimePickerComponent, selector: "nui-date-time-picker", inputs: ["maxDate", "minDate", "displayMode", "isDisabled", "initEmpty", "handleTimezone", "appendToBody", "ariaLabel", "model"], outputs: ["modelChanged"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TimeFramePickerComponent, decorators: [{
type: Component,
args: [{ selector: "nui-time-frame-picker", encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"nui-time-frame-picker\">\n <div class=\"nui-time-frame-picker__date-times\">\n <label class=\"nui-time-frame-picker__label\" i18n>Start</label>\n <span\n class=\"nui-time-frame-picker__date-time nui-time-frame-picker__date-time_start\"\n >\n <nui-date-time-picker\n [minDate]=\"minDate\"\n [maxDate]=\"maxDate\"\n [model]=\"model.startDatetime\"\n (modelChanged)=\"onChangeInternalStart($event)\"\n (focus)=\"onFocusInternal()\"\n (blur)=\"onBlurInternal()\"\n [appendToBody]=\"appendToBody\"\n ></nui-date-time-picker>\n </span>\n <label class=\"nui-time-frame-picker__label\" i18n>End</label>\n <span\n class=\"nui-time-frame-picker__date-time nui-time-frame-picker__date-time_end\"\n >\n <nui-date-time-picker\n [minDate]=\"minDate\"\n [maxDate]=\"maxDate\"\n [model]=\"model.endDatetime\"\n (modelChanged)=\"onChangeInternalEnd($event)\"\n (focus)=\"onFocusInternal()\"\n (blur)=\"onBlurInternal()\"\n [appendToBody]=\"appendToBody\"\n ></nui-date-time-picker>\n </span>\n </div>\n</div>\n", styles: [".nui-time-frame-picker__date-time{display:flex}.nui-time-frame-picker__date-time_start{margin-bottom:15px}.nui-time-frame-picker__date-time_end{padding-bottom:10px}\n"] }]
}], ctorParameters: () => [{ type: i1.TimeframeService }, { type: i0.ChangeDetectorRef }], propDecorators: { minDate: [{
type: Input
}], maxDate: [{
type: Input
}], startModel: [{
type: Input
}], appendToBody: [{
type: Input
}], changed: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"time-frame-picker.component.js","sourceRoot":"","sources":["../../../../src/lib/time-frame-picker/time-frame-picker.component.ts","../../../../src/lib/time-frame-picker/time-frame-picker.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    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    Input,\n    OnChanges,\n    OnInit,\n    Output,\n    ViewEncapsulation,\n} from \"@angular/core\";\nimport _cloneDeep from \"lodash/cloneDeep\";\nimport { Moment } from \"moment/moment\";\nimport moment from \"moment/moment\";\n\nimport { ITimeframe, ITimeFramePreset } from \"./public-api\";\nimport { TimeframeService } from \"./services/timeframe.service\";\n\n// <example-url>./../examples/index.html#/time-frame-picker</example-url>\n\n@Component({\n    selector: \"nui-time-frame-picker\",\n    templateUrl: \"./time-frame-picker.component.html\",\n    styleUrls: [\"./time-frame-picker.component.less\"],\n    encapsulation: ViewEncapsulation.None,\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class TimeFramePickerComponent implements OnChanges, OnInit {\n    /**  earliest selectable date */\n    @Input() minDate: Moment;\n    /**  latest selectable date */\n    @Input() maxDate: Moment;\n    /** model of timepicker */\n    @Input() startModel: ITimeframe;\n    /** Allows popup box to be attached to document.body */\n    @Input() appendToBody: boolean;\n\n    public model: ITimeframe;\n\n    /** callback to be invoked on model change */\n    @Output() public changed = new EventEmitter<ITimeframe>();\n\n    public isFocused: boolean;\n    public modelDefault: any;\n\n    public distanceToEndDate: number; // to keep distance between start and end-date\n\n    constructor(\n        private timeFrameService: TimeframeService,\n        public changeDetector: ChangeDetectorRef\n    ) {}\n\n    public ngOnChanges(changes: any): void {\n        if (changes[\"startModel\"]) {\n            this.model = TimeframeService.cloneTimeFrame(this.startModel);\n            this.validateCombination();\n        }\n    }\n\n    public ngOnInit(): void {\n        if (this.startModel) {\n            this.model = TimeframeService.cloneTimeFrame(this.startModel);\n            this.validateCombination();\n        }\n    }\n\n    public selectPreset(key: string, value: ITimeFramePreset): void {\n        const timeframe = this.timeFrameService.getTimeframe(\n            value.startDatetimePattern,\n            value.endDatetimePattern\n        );\n        timeframe.selectedPresetId = key;\n        this.model = timeframe;\n    }\n\n    public isPresetSelected(key: string): boolean {\n        return this.model && this.model.selectedPresetId === key;\n    }\n\n    public onChangeInternalStart(event: any): void {\n        this.model.startDatetime = event;\n        this.onChangeInternal();\n    }\n\n    public onChangeInternalEnd(event: any): void {\n        this.model.endDatetime = event;\n        this.onChangeInternal();\n    }\n\n    public onBlurInternal(): void {\n        this.validateCombination();\n\n        if (!this.model.startDatetime || !this.model.endDatetime) {\n            this.model = _cloneDeep(this.modelDefault);\n        } else {\n            this.modelDefault = _cloneDeep(this.model);\n        }\n        this.isFocused = false;\n    }\n\n    public onFocusInternal(): void {\n        this.modelDefault = _cloneDeep(this.model);\n        this.isFocused = true;\n    }\n\n    private validateCombination() {\n        if (this.model.startDatetime && this.model.endDatetime) {\n            if (this.model.startDatetime >= this.model.endDatetime) {\n                this.model.endDatetime = moment(\n                    this.model.startDatetime.valueOf() + this.distanceToEndDate\n                );\n                this.model.selectedPresetId = undefined;\n                this.model.title = undefined;\n            } else {\n                this.updateDistanceToEndDate(\n                    this.model.startDatetime,\n                    this.model.endDatetime\n                );\n            }\n        }\n    }\n\n    private updateDistanceToEndDate(\n        newStartDatetime: Date | Moment,\n        newEndDatetime: Date | Moment\n    ) {\n        const startMoment = moment(newStartDatetime);\n        const endMoment = moment(newEndDatetime);\n\n        if (startMoment.isValid() && endMoment.isValid()) {\n            this.distanceToEndDate =\n                newEndDatetime.valueOf() - newStartDatetime.valueOf();\n        }\n    }\n\n    private onChangeInternal() {\n        if (!this.isFocused) {\n            this.validateCombination();\n        }\n        // clear 'selectedPresetId' and 'title' values\n        if (this.model.selectedPresetId) {\n            const timeFrame = this.timeFrameService.getTimeframeByPresetId(\n                this.model.selectedPresetId\n            );\n\n            if (\n                !this.model.startDatetime ||\n                !this.model.endDatetime ||\n                !this.timeFrameService.isEqualDuration(this.model, timeFrame)\n            ) {\n                this.model.selectedPresetId = undefined;\n                this.model.title = undefined;\n            }\n        }\n        this.changed.emit(this.model);\n    }\n}\n","<div class=\"nui-time-frame-picker\">\n    <div class=\"nui-time-frame-picker__date-times\">\n        <label class=\"nui-time-frame-picker__label\" i18n>Start</label>\n        <span\n            class=\"nui-time-frame-picker__date-time nui-time-frame-picker__date-time_start\"\n        >\n            <nui-date-time-picker\n                [minDate]=\"minDate\"\n                [maxDate]=\"maxDate\"\n                [model]=\"model.startDatetime\"\n                (modelChanged)=\"onChangeInternalStart($event)\"\n                (focus)=\"onFocusInternal()\"\n                (blur)=\"onBlurInternal()\"\n                [appendToBody]=\"appendToBody\"\n            ></nui-date-time-picker>\n        </span>\n        <label class=\"nui-time-frame-picker__label\" i18n>End</label>\n        <span\n            class=\"nui-time-frame-picker__date-time nui-time-frame-picker__date-time_end\"\n        >\n            <nui-date-time-picker\n                [minDate]=\"minDate\"\n                [maxDate]=\"maxDate\"\n                [model]=\"model.endDatetime\"\n                (modelChanged)=\"onChangeInternalEnd($event)\"\n                (focus)=\"onFocusInternal()\"\n                (blur)=\"onBlurInternal()\"\n                [appendToBody]=\"appendToBody\"\n            ></nui-date-time-picker>\n        </span>\n    </div>\n</div>\n"]}