Skip to content
This repository was archived by the owner on Mar 2, 2018. It is now read-only.

Commit 22f3901

Browse files
committed
the datetime picker cannot update when user modifying value
1 parent 8cdc947 commit 22f3901

File tree

1 file changed

+82
-76
lines changed

1 file changed

+82
-76
lines changed

src/datetime-picker.directive.ts

+82-76
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,9 @@ import {
33
Input, OnChanges, OnInit, Optional, Output,
44
SimpleChanges, SkipSelf, ViewContainerRef
55
} from '@angular/core';
6-
import {AbstractControl, ControlContainer, FormGroup, FormGroupDirective} from '@angular/forms';
7-
import {NguiDatetimePickerComponent} from './datetime-picker.component';
8-
import {NguiDatetime} from './datetime';
6+
import { AbstractControl, ControlContainer, FormGroup, FormGroupDirective } from '@angular/forms';
7+
import { NguiDatetimePickerComponent } from './datetime-picker.component';
8+
import { NguiDatetime } from './datetime';
99

1010
declare var moment: any;
1111

@@ -29,36 +29,36 @@ function isNaN(value) {
2929
* If the given string is not a valid date, it defaults back to today
3030
*/
3131
@Directive({
32-
selector : '[ngui-datetime-picker]',
32+
selector: '[ngui-datetime-picker]',
3333
providers: [NguiDatetime]
3434
})
3535
export class NguiDatetimePickerDirective implements OnInit, OnChanges {
36-
@Input('date-format') dateFormat: string;
37-
@Input('parse-format') parseFormat: string;
38-
@Input('date-only') dateOnly: boolean;
39-
@Input('time-only') timeOnly: boolean;
40-
@Input('close-on-select') closeOnSelect: boolean = true;
41-
@Input('default-value') defaultValue: Date | string;
42-
@Input('minute-step') minuteStep: number;
43-
@Input('min-date') minDate: Date | string;
44-
@Input('max-date') maxDate: Date | string;
45-
@Input('min-hour') minHour: Date | number;
46-
@Input('max-hour') maxHour: Date | number;
47-
@Input('disabled-dates') disabledDates: Date[];
48-
@Input('show-close-layer') showCloseLayer: boolean;
36+
@Input('date-format') dateFormat: string;
37+
@Input('parse-format') parseFormat: string;
38+
@Input('date-only') dateOnly: boolean;
39+
@Input('time-only') timeOnly: boolean;
40+
@Input('close-on-select') closeOnSelect: boolean = true;
41+
@Input('default-value') defaultValue: Date | string;
42+
@Input('minute-step') minuteStep: number;
43+
@Input('min-date') minDate: Date | string;
44+
@Input('max-date') maxDate: Date | string;
45+
@Input('min-hour') minHour: Date | number;
46+
@Input('max-hour') maxHour: Date | number;
47+
@Input('disabled-dates') disabledDates: Date[];
48+
@Input('show-close-layer') showCloseLayer: boolean;
4949
@Input('show-today-shortcut') showTodayShortcut: boolean = false;
5050
@Input('show-week-numbers') showWeekNumbers: boolean;
5151
@Input() formControlName: string;
52-
@Input('is-draggable') isDraggable: boolean = true;
52+
@Input('is-draggable') isDraggable: boolean = true;
5353

54-
@Input('ngModel') ngModel: any;
54+
@Input('ngModel') ngModel: any;
5555
@Output('ngModelChange') ngModelChange = new EventEmitter();
56-
@Output('valueChanged') valueChanged$ = new EventEmitter();
57-
@Output('popupClosed') popupClosed$ = new EventEmitter();
56+
@Output('valueChanged') valueChanged$ = new EventEmitter();
57+
@Output('popupClosed') popupClosed$ = new EventEmitter();
5858

5959
private el: HTMLInputElement; /* input element */
6060
private nguiDatetimePickerEl: HTMLElement; /* dropdown element */
61-
private componentRef:ComponentRef<NguiDatetimePickerComponent>; /* dropdown component reference */
61+
private componentRef: ComponentRef<NguiDatetimePickerComponent>; /* dropdown component reference */
6262
private ctrl: AbstractControl;
6363
private sub: any;
6464
// private justShown: boolean;
@@ -67,9 +67,9 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
6767
clickedDatetimePicker: boolean;
6868
userModifyingValue: boolean = false;
6969

70-
constructor (
71-
private resolver:ComponentFactoryResolver,
72-
private viewContainerRef:ViewContainerRef,
70+
constructor(
71+
private resolver: ComponentFactoryResolver,
72+
private viewContainerRef: ViewContainerRef,
7373
@Optional() @Host() @SkipSelf() private parent: ControlContainer
7474
) {
7575
this.el = this.viewContainerRef.element.nativeElement;
@@ -117,8 +117,8 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
117117
}
118118
}
119119

120-
ngOnInit ():void {
121-
if(this.parent && this.formControlName) {
120+
ngOnInit(): void {
121+
if (this.parent && this.formControlName) {
122122
if (this.parent["form"]) {
123123
this.ctrl = (<FormGroup>this.parent["form"]).get(this.formControlName);
124124
} else if (this.parent["path"]) {
@@ -138,19 +138,19 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
138138
this.normalizeInput();
139139

140140
//wrap this element with a <div> tag, so that we can position dynamic element correctly
141-
let wrapper = document.createElement("div");
142-
wrapper.className = 'ngui-datetime-picker-wrapper';
141+
let wrapper = document.createElement("div");
142+
wrapper.className = 'ngui-datetime-picker-wrapper';
143143
this.el.parentElement.insertBefore(wrapper, this.el.nextSibling);
144144
wrapper.appendChild(this.el);
145145

146146
if (this.ngModel && this.ngModel.getTime) { // if it is a Date object given, set dateValue and toString method
147147
this.ngModel.toString = () => NguiDatetime.formatDate(this.ngModel, this.dateFormat, this.dateOnly);
148148
}
149-
setTimeout( () => { // after [(ngModel)] is applied
149+
setTimeout(() => { // after [(ngModel)] is applied
150150
if (this.el.tagName === 'INPUT') {
151151
this.inputElValueChanged(this.el.value); //set this.el.dateValue and reformat this.el.value
152152
}
153-
if(this.ctrl) {
153+
if (this.ctrl) {
154154
this.ctrl.markAsPristine();
155155
}
156156
});
@@ -160,7 +160,7 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
160160
// if this element is not an input tag, move dropdown after input tag
161161
// so that it displays correctly
162162
this.inputEl = this.el.tagName === "INPUT" ?
163-
<HTMLInputElement>this.el : <HTMLInputElement>this.el.querySelector("input");
163+
<HTMLInputElement>this.el : <HTMLInputElement>this.el.querySelector("input");
164164

165165
if (this.inputEl) {
166166
this.inputEl.addEventListener('focus', this.showDatetimePicker);
@@ -176,7 +176,7 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
176176

177177
ngOnChanges(changes: SimpleChanges) {
178178
let date;
179-
if(changes && changes['ngModel']) {
179+
if (changes && changes['ngModel']) {
180180
date = changes['ngModel'].currentValue;
181181

182182
if (date && typeof date !== 'string') {
@@ -186,22 +186,28 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
186186
} else if (date && typeof date === 'string') {
187187
/** if program assigns a string value, then format to date later */
188188
if (!this.userModifyingValue) {
189-
setTimeout( () => {
189+
setTimeout(() => {
190190
let dt = this.getDate(date);
191191
dt.toString = () => NguiDatetime.formatDate(dt, this.dateFormat, this.dateOnly);
192192
this.ngModel = dt;
193-
this.inputEl.value = ''+dt;
193+
this.inputEl.value = '' + dt;
194194
})
195+
} else {
196+
let changeDate: any = new Date(date);
197+
if (changeDate.toString() !== "Invalid Date") {
198+
this.setInputElDateValue(date);
199+
this.updateDatepicker();
200+
}
195201
}
196202
}
197-
}
203+
}
198204
this.userModifyingValue = false;
199205
}
200206

201207
updateDatepicker() {
202-
if(this.componentRef) {
208+
if (this.componentRef) {
203209
let component = this.componentRef.instance;
204-
component.defaultValue = <Date>this.el['dateValue'];
210+
component.defaultValue = <Date>this.el['dateValue'];
205211
}
206212
}
207213

@@ -214,13 +220,13 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
214220
this.el['dateValue'] = null;
215221
}
216222

217-
if(this.ctrl) {
223+
if (this.ctrl) {
218224
this.ctrl.markAsDirty();
219225
}
220226
}
221227

222-
ngOnDestroy ():void {
223-
if(this.sub) {
228+
ngOnDestroy(): void {
229+
if (this.sub) {
224230
this.sub.unsubscribe();
225231
}
226232
}
@@ -229,7 +235,7 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
229235
inputElValueChanged = (date: string | Date): void => {
230236
this.setInputElDateValue(date);
231237
this.el.value = date.toString();
232-
if(this.ctrl) {
238+
if (this.ctrl) {
233239
this.ctrl.patchValue(this.el.value);
234240
}
235241
this.ngModel = this.el['dateValue'];
@@ -240,14 +246,14 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
240246
};
241247

242248
//show datetimePicker element below the current element
243-
showDatetimePicker = (event?): void => {
249+
showDatetimePicker = (event?): void => {
244250
if (this.componentRef) { /* if already shown, do nothing */
245251
return;
246252
}
247253

248254
let factory = this.resolver.resolveComponentFactory(NguiDatetimePickerComponent);
249255

250-
this.componentRef = this.viewContainerRef.createComponent(factory);
256+
this.componentRef = this.viewContainerRef.createComponent(factory);
251257
this.nguiDatetimePickerEl = this.componentRef.location.nativeElement;
252258
this.nguiDatetimePickerEl.setAttribute('tabindex', '32767');
253259
this.nguiDatetimePickerEl.setAttribute('draggable', String(this.isDraggable));
@@ -264,21 +270,21 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
264270
this.nguiDatetimePickerEl.addEventListener('blur', (event) => {
265271
this.hideDatetimePicker();
266272
});
267-
this.nguiDatetimePickerEl.addEventListener('dragstart',this.drag_start,false);
268-
document.body.addEventListener('dragover',this.drag_over,false);
269-
document.body.addEventListener('drop',this.drop,false);
273+
this.nguiDatetimePickerEl.addEventListener('dragstart', this.drag_start, false);
274+
document.body.addEventListener('dragover', this.drag_over, false);
275+
document.body.addEventListener('drop', this.drop, false);
270276

271277
let component = this.componentRef.instance;
272-
component.defaultValue = <Date>this.defaultValue || <Date>this.el['dateValue'];
273-
component.dateFormat = this.dateFormat;
274-
component.dateOnly = this.dateOnly;
275-
component.timeOnly = this.timeOnly;
276-
component.minuteStep = this.minuteStep;
277-
component.minDate = <Date>this.minDate;
278-
component.maxDate = <Date>this.maxDate;
279-
component.minHour = <number>this.minHour;
280-
component.maxHour = <number>this.maxHour;
281-
component.disabledDates = this.disabledDates;
278+
component.defaultValue = <Date>this.defaultValue || <Date>this.el['dateValue'];
279+
component.dateFormat = this.dateFormat;
280+
component.dateOnly = this.dateOnly;
281+
component.timeOnly = this.timeOnly;
282+
component.minuteStep = this.minuteStep;
283+
component.minDate = <Date>this.minDate;
284+
component.maxDate = <Date>this.maxDate;
285+
component.minHour = <number>this.minHour;
286+
component.maxHour = <number>this.maxHour;
287+
component.disabledDates = this.disabledDates;
282288
component.showCloseButton = this.closeOnSelect === false;
283289
component.showCloseLayer = this.showCloseLayer;
284290
component.showTodayShortcut = this.showTodayShortcut;
@@ -290,7 +296,7 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
290296
component.closing$.subscribe(() => {
291297
this.hideDatetimePicker();
292298
});
293-
299+
294300
//Hack not to fire tab keyup event
295301
// this.justShown = true;
296302
// setTimeout(() => this.justShown = false, 100);
@@ -321,26 +327,26 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
321327
event && event.stopPropagation();
322328
};
323329

324-
private elementIn (el:Node, containerEl:Node):boolean {
330+
private elementIn(el: Node, containerEl: Node): boolean {
325331
while (el = el.parentNode) {
326332
if (el === containerEl) return true;
327333
}
328334
return false;
329335
}
330336

331-
private styleDatetimePicker () {
337+
private styleDatetimePicker() {
332338
// setting position, width, and height of auto complete dropdown
333-
let thisElBCR = this.el.getBoundingClientRect();
339+
let thisElBCR = this.el.getBoundingClientRect();
334340
// this.nguiDatetimePickerEl.style.minWidth = thisElBCR.width + 'px';
335-
this.nguiDatetimePickerEl.style.position = 'absolute';
336-
this.nguiDatetimePickerEl.style.zIndex = '1000';
337-
this.nguiDatetimePickerEl.style.left = '0';
341+
this.nguiDatetimePickerEl.style.position = 'absolute';
342+
this.nguiDatetimePickerEl.style.zIndex = '1000';
343+
this.nguiDatetimePickerEl.style.left = '0';
338344
this.nguiDatetimePickerEl.style.transition = 'height 0.3s ease-in';
339345

340346
this.nguiDatetimePickerEl.style.visibility = 'hidden';
341347

342348
setTimeout(() => {
343-
let thisElBcr = this.el.getBoundingClientRect();
349+
let thisElBcr = this.el.getBoundingClientRect();
344350
let nguiDatetimePickerElBcr = this.nguiDatetimePickerEl.getBoundingClientRect();
345351

346352
if (thisElBcr.bottom + nguiDatetimePickerElBcr.height > window.innerHeight) {
@@ -355,38 +361,38 @@ export class NguiDatetimePickerDirective implements OnInit, OnChanges {
355361
});
356362
};
357363

358-
private getDate = (arg: any): Date => {
364+
private getDate = (arg: any): Date => {
359365
let date: Date = <Date>arg;
360366
if (typeof arg === 'string') {
361-
date = NguiDatetime.parseDate(arg, this.parseFormat, this.dateFormat);
367+
date = NguiDatetime.parseDate(arg, this.parseFormat, this.dateFormat);
362368
}
363369
return date;
364370
}
365371

366372
private drag_start = (event) => {
367-
if (document.activeElement.tagName == 'INPUT') {
373+
if (document.activeElement.tagName == 'INPUT') {
368374
event.preventDefault();
369375
return false; // block dragging
370-
}
376+
}
371377
var style = window.getComputedStyle(event.target, null);
372378
event.dataTransfer.setData("text/plain",
373-
(parseInt(style.getPropertyValue("left"),10) - event.clientX)
374-
+ ','
375-
+ (parseInt(style.getPropertyValue("top"),10) - event.clientY)
379+
(parseInt(style.getPropertyValue("left"), 10) - event.clientX)
380+
+ ','
381+
+ (parseInt(style.getPropertyValue("top"), 10) - event.clientY)
376382
);
377383
}
378384

379385
private drag_over(event) {
380386
event.preventDefault();
381387
return false;
382-
}
388+
}
383389

384390
private drop = (event) => {
385391
var offset = event.dataTransfer.getData("text/plain").split(',');
386-
this.nguiDatetimePickerEl.style.left = (event.clientX + parseInt(offset[0],10)) + 'px';
387-
this.nguiDatetimePickerEl.style.top = (event.clientY + parseInt(offset[1],10)) + 'px';
392+
this.nguiDatetimePickerEl.style.left = (event.clientX + parseInt(offset[0], 10)) + 'px';
393+
this.nguiDatetimePickerEl.style.top = (event.clientY + parseInt(offset[1], 10)) + 'px';
388394
this.nguiDatetimePickerEl.style.bottom = '';
389395
event.preventDefault();
390396
return false;
391-
}
397+
}
392398
}

0 commit comments

Comments
 (0)