@nakedobjects/gemini
Version:
Single Page Application client for a Naked Objects application.
129 lines • 17.5 kB
JavaScript
import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core';
import { DateTime } from 'luxon';
import { debounceTime } from 'rxjs/operators';
import { focus, safeUnsubscribe } from '../helpers-components';
import { BehaviorSubject } from 'rxjs';
import { defaultShortTimeFormat, defaultTimeFormat, shortTimeFormat } from '@nakedobjects/services';
import * as i0 from "@angular/core";
import * as i1 from "@angular/forms";
import * as i2 from "../clear.directive";
export class TimePickerComponent {
inputEvents;
outputEvents;
id;
inputField;
constructor() {
this.outputEvents = new EventEmitter();
}
timeValue = null;
modelValue = '';
eventsSub;
bSubject;
sub;
set model(s) {
this.modelValue = s;
if (this.bSubject) {
this.bSubject.next(s);
}
}
get model() {
return this.modelValue;
}
get time() {
return this.timeValue;
}
set time(time) {
if (time && time.isValid) {
this.timeValue = time;
this.outputEvents.emit({ type: 'timeChanged', data: time.toFormat(defaultTimeFormat) });
}
}
validInputFormats = [defaultTimeFormat, defaultShortTimeFormat, shortTimeFormat];
validateTime(newValue) {
let dt = DateTime.now();
for (const f of this.validInputFormats) {
dt = DateTime.fromFormat(newValue, f);
if (dt.isValid) {
break;
}
}
return dt;
}
sameTime(t1, t2) {
return t2 &&
t1.hour === t2.hour &&
t1.minute === t2.minute &&
t1.second === t2.second;
}
setTimeIfChanged(newTime) {
if (!this.sameTime(newTime, this.time)) {
this.time = newTime;
setTimeout(() => this.model = newTime.toFormat(defaultShortTimeFormat));
}
}
setTime(newValue) {
if (newValue === '' || newValue == null) {
this.timeValue = null;
this.outputEvents.emit({ type: 'timeCleared', data: '' });
}
else {
const dt = this.validateTime(newValue);
if (dt.isValid) {
this.setTimeIfChanged(dt);
}
else {
this.timeValue = null;
this.outputEvents.emit({ type: 'timeInvalid', data: newValue });
}
}
}
inputChanged(newValue) {
this.setTime(newValue);
}
ngOnInit() {
if (this.inputEvents) {
this.eventsSub = this.inputEvents.subscribe((e) => {
if (e.type === 'setTime') {
this.setTime(e.data);
}
});
}
}
clear() {
this.modelValue = '';
this.setTime('');
}
get subject() {
if (!this.bSubject) {
const initialValue = this.model;
this.bSubject = new BehaviorSubject(initialValue);
this.sub = this.bSubject.pipe(debounceTime(200)).subscribe((data) => this.inputChanged(data));
}
return this.bSubject;
}
ngOnDestroy() {
safeUnsubscribe(this.sub);
safeUnsubscribe(this.eventsSub);
}
focus() {
return focus(this.inputField);
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: TimePickerComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: TimePickerComponent, selector: "nof-time-picker", inputs: { inputEvents: "inputEvents", id: "id" }, outputs: { outputEvents: "outputEvents" }, viewQueries: [{ propertyName: "inputField", first: true, predicate: ["focus"], descendants: true }], ngImport: i0, template: "<input #focus type=\"text\" [id]=\"id\" [(ngModel)]=\"model\" [nofClear]=\"subject\" (clear)=\"clear()\">\n", styles: ["input{width:var(--field-value-width)}.ng-clearable{background-image:var(--clearable-image);background-repeat:no-repeat;background-position:right -10px;background-size:8px}.ng-clearable[class*=link-color]{background-image:var(--clearable-link-image)}.ng-clearable.ng-x{background-position:right 2px center}.ng-clearable.ng-onX{cursor:pointer}input::-ms-clear{display:none}\n"], dependencies: [{ kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i2.ClearDirective, selector: "[nofClear]", inputs: ["nofClear"], outputs: ["clear"] }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: TimePickerComponent, decorators: [{
type: Component,
args: [{ selector: 'nof-time-picker', template: "<input #focus type=\"text\" [id]=\"id\" [(ngModel)]=\"model\" [nofClear]=\"subject\" (clear)=\"clear()\">\n", styles: ["input{width:var(--field-value-width)}.ng-clearable{background-image:var(--clearable-image);background-repeat:no-repeat;background-position:right -10px;background-size:8px}.ng-clearable[class*=link-color]{background-image:var(--clearable-link-image)}.ng-clearable.ng-x{background-position:right 2px center}.ng-clearable.ng-onX{cursor:pointer}input::-ms-clear{display:none}\n"] }]
}], ctorParameters: () => [], propDecorators: { inputEvents: [{
type: Input,
args: [{ required: true }]
}], outputEvents: [{
type: Output
}], id: [{
type: Input,
args: [{ required: true }]
}], inputField: [{
type: ViewChild,
args: ['focus', { static: false }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"time-picker.component.js","sourceRoot":"","sources":["../../../../gemini/src/time-picker/time-picker.component.ts","../../../../gemini/src/time-picker/time-picker.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    Component,\n    ElementRef,\n    EventEmitter,\n    Input,\n    OnDestroy,\n    OnInit,\n    Output,\n    ViewChild\n} from '@angular/core';\nimport { DateTime } from 'luxon';\nimport { debounceTime } from 'rxjs/operators';\nimport { focus, safeUnsubscribe } from '../helpers-components';\nimport { BehaviorSubject, SubscriptionLike as ISubscription } from 'rxjs';\nimport { defaultShortTimeFormat, defaultTimeFormat, shortTimeFormat } from '@nakedobjects/services';\n\nexport interface ITimePickerOutputEvent {\n    type: 'timeChanged' | 'timeCleared' | 'timeInvalid';\n    data: string;\n}\n\nexport interface ITimePickerInputEvent {\n    type: 'setTime';\n    data: string;\n}\n\n@Component({\n    selector: 'nof-time-picker',\n    templateUrl: 'time-picker.component.html',\n    styleUrls: ['time-picker.component.css']\n})\nexport class TimePickerComponent implements OnInit, OnDestroy {\n\n    @Input({required : true})\n    inputEvents!: EventEmitter<ITimePickerInputEvent>;\n\n    @Output()\n    outputEvents: EventEmitter<ITimePickerOutputEvent>;\n\n    @Input({required : true})\n    id!: string;\n\n    @ViewChild('focus', {static: false})\n    inputField?: ElementRef;\n\n    constructor() {\n        this.outputEvents = new EventEmitter<ITimePickerOutputEvent>();\n    }\n\n    private timeValue: DateTime | null = null;\n    private modelValue = '';\n    private eventsSub?: ISubscription;\n    private bSubject?: BehaviorSubject<string>;\n    private sub?: ISubscription;\n\n    set model(s: string) {\n        this.modelValue = s;\n\n        if (this.bSubject) {\n            this.bSubject.next(s);\n        }\n    }\n\n    get model(): string {\n        return this.modelValue;\n    }\n\n    get time(): DateTime | null {\n        return this.timeValue;\n    }\n\n    set time(time: DateTime | null) {\n        if (time && time.isValid) {\n            this.timeValue = time;\n            this.outputEvents.emit({ type: 'timeChanged', data: time.toFormat(defaultTimeFormat) });\n        }\n    }\n\n    private validInputFormats = [defaultTimeFormat, defaultShortTimeFormat, shortTimeFormat];\n\n    private validateTime(newValue: string) {\n        let dt : DateTime<true> | DateTime<false> = DateTime.now();\n\n        for (const f of this.validInputFormats) {\n            dt = DateTime.fromFormat(newValue, f);\n            if (dt.isValid) {\n                break;\n            }\n        }\n\n        return dt;\n    }\n\n    private sameTime(t1 : DateTime, t2 : DateTime | null) {\n        return t2 &&\n               t1.hour === t2.hour &&\n               t1.minute === t2.minute &&\n               t1.second === t2.second;\n    }\n\n    setTimeIfChanged(newTime: DateTime) {\n        if (!this.sameTime(newTime, this.time)) {\n            this.time = newTime;\n            setTimeout(() => this.model = newTime.toFormat(defaultShortTimeFormat));\n        }\n    }\n\n    setTime(newValue: string) {\n\n        if (newValue === '' || newValue == null) {\n            this.timeValue = null;\n            this.outputEvents.emit({ type: 'timeCleared', data: '' });\n        } else {\n            const dt = this.validateTime(newValue);\n\n            if (dt.isValid) {\n                this.setTimeIfChanged(dt);\n            } else {\n                this.timeValue = null;\n                this.outputEvents.emit({ type: 'timeInvalid', data: newValue });\n            }\n        }\n    }\n\n    inputChanged(newValue: string) {\n        this.setTime(newValue);\n    }\n\n    ngOnInit() {\n\n        if (this.inputEvents) {\n            this.eventsSub = this.inputEvents.subscribe((e: ITimePickerInputEvent) => {\n                if (e.type === 'setTime') {\n                    this.setTime(e.data);\n                }\n            });\n        }\n    }\n\n    clear() {\n        this.modelValue = '';\n        this.setTime('');\n    }\n\n    get subject() {\n        if (!this.bSubject) {\n            const initialValue = this.model;\n            this.bSubject = new BehaviorSubject(initialValue);\n\n            this.sub = this.bSubject.pipe(debounceTime(200)).subscribe((data: string) => this.inputChanged(data));\n        }\n\n        return this.bSubject;\n    }\n\n    ngOnDestroy(): void {\n        safeUnsubscribe(this.sub);\n        safeUnsubscribe(this.eventsSub);\n    }\n\n    focus() {\n        return focus(this.inputField);\n    }\n}\n","<input #focus type=\"text\" [id]=\"id\" [(ngModel)]=\"model\" [nofClear]=\"subject\" (clear)=\"clear()\">\n"]}