mis-crystal-design-system 2.9.9 → 3.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/bundles/mis-crystal-design-system-timepicker.umd.js +630 -0
  2. package/bundles/mis-crystal-design-system-timepicker.umd.js.map +1 -0
  3. package/bundles/mis-crystal-design-system-timepicker.umd.min.js +16 -0
  4. package/bundles/mis-crystal-design-system-timepicker.umd.min.js.map +1 -0
  5. package/bundles/mis-crystal-design-system-timerangepicker.umd.js +177 -0
  6. package/bundles/mis-crystal-design-system-timerangepicker.umd.js.map +1 -0
  7. package/bundles/mis-crystal-design-system-timerangepicker.umd.min.js +2 -0
  8. package/bundles/mis-crystal-design-system-timerangepicker.umd.min.js.map +1 -0
  9. package/esm2015/timepicker/index.js +2 -0
  10. package/esm2015/timepicker/mis-crystal-design-system-timepicker.js +6 -0
  11. package/esm2015/timepicker/public_api.js +3 -0
  12. package/esm2015/timepicker/time.namespace.js +2 -0
  13. package/esm2015/timepicker/timepicker.component.js +189 -0
  14. package/esm2015/timepicker/timepicker.directive.js +62 -0
  15. package/esm2015/timepicker/timepicker.module.js +20 -0
  16. package/esm2015/timerangepicker/index.js +2 -0
  17. package/esm2015/timerangepicker/mis-crystal-design-system-timerangepicker.js +5 -0
  18. package/esm2015/timerangepicker/public_api.js +3 -0
  19. package/esm2015/timerangepicker/timerange.namespace.js +2 -0
  20. package/esm2015/timerangepicker/timerangepicker.component.js +123 -0
  21. package/esm2015/timerangepicker/timerangepicker.module.js +18 -0
  22. package/fesm2015/mis-crystal-design-system-timepicker.js +272 -0
  23. package/fesm2015/mis-crystal-design-system-timepicker.js.map +1 -0
  24. package/fesm2015/mis-crystal-design-system-timerangepicker.js +147 -0
  25. package/fesm2015/mis-crystal-design-system-timerangepicker.js.map +1 -0
  26. package/package.json +1 -1
  27. package/timepicker/index.d.ts +1 -0
  28. package/timepicker/mis-crystal-design-system-timepicker.d.ts +5 -0
  29. package/timepicker/mis-crystal-design-system-timepicker.metadata.json +1 -0
  30. package/timepicker/package.json +11 -0
  31. package/timepicker/public_api.d.ts +2 -0
  32. package/timepicker/time.namespace.d.ts +4 -0
  33. package/timepicker/timepicker.component.d.ts +38 -0
  34. package/timepicker/timepicker.directive.d.ts +16 -0
  35. package/timepicker/timepicker.module.d.ts +4 -0
  36. package/timerangepicker/index.d.ts +1 -0
  37. package/timerangepicker/mis-crystal-design-system-timerangepicker.d.ts +4 -0
  38. package/timerangepicker/mis-crystal-design-system-timerangepicker.metadata.json +1 -0
  39. package/timerangepicker/package.json +11 -0
  40. package/timerangepicker/public_api.d.ts +2 -0
  41. package/timerangepicker/timerange.namespace.d.ts +9 -0
  42. package/timerangepicker/timerangepicker.component.d.ts +29 -0
  43. package/timerangepicker/timerangepicker.module.d.ts +4 -0
@@ -0,0 +1,189 @@
1
+ import { Component, EventEmitter, Input, Output, ViewChild, ViewChildren } from "@angular/core";
2
+ import * as moment from "moment-timezone";
3
+ import { TimepickerDirective } from "./timepicker.directive";
4
+ export class TimePickerComponent {
5
+ constructor() {
6
+ this.openStatus = false;
7
+ this.isHighlighted = 0;
8
+ this.isInvalid = false;
9
+ this.timeIntervals = [];
10
+ this.shouldScroll = false;
11
+ this.userInputFlag = false;
12
+ this.clockFormat = 12;
13
+ this.timezone = "Asia/Kolkata";
14
+ this.height = "max-content";
15
+ this.inputWidth = "100px";
16
+ this.interval = 15;
17
+ this.dateAsEpoch = moment().valueOf();
18
+ this.rangeValidity = true;
19
+ this.timeEmitter = new EventEmitter();
20
+ }
21
+ // gets all the li elements from the dropdown and scrolls to the highlighted element
22
+ set timeIntervalRefs(intervals) {
23
+ intervals.forEach(interval => {
24
+ if (interval.nativeElement.classList[0] === "highlight") {
25
+ const highlighted = interval.nativeElement;
26
+ setTimeout(() => highlighted.scrollIntoView({ behavior: "smooth", block: "center" }));
27
+ }
28
+ });
29
+ }
30
+ ngOnInit() {
31
+ this.timeFormat = this.clockFormat === 12 ? "hh:mm a" : "HH:mm";
32
+ this.populateDropdown();
33
+ this.chosenTime = this.timeIntervals[0];
34
+ this.calculateClosestInterval(this.chosenTime);
35
+ this.emitTime({
36
+ valid: !this.isInvalid,
37
+ time: this.chosenTime
38
+ });
39
+ }
40
+ ngOnChanges() {
41
+ moment.tz.setDefault(this.timezone);
42
+ this.currTime = moment().format();
43
+ if (this.timeFormat) {
44
+ // if the first interval is >= the chosen time
45
+ // then only update the value of chosen time
46
+ // else it remains the same
47
+ const chosenTimeMoment = moment(`${moment(this.dateAsEpoch).format("DD-MM-YYYY")} ${this.chosenTime}`, `'DD-MM-YYYY' ${this.timeFormat}`);
48
+ if (this.firstInterval >= chosenTimeMoment.valueOf() && this.rangeValidity && !this.userInputFlag) {
49
+ this.chosenTime = moment(this.firstInterval).format(this.timeFormat);
50
+ }
51
+ if (!this.userInputFlag)
52
+ this.populateDropdown();
53
+ this.userInputFlag = false;
54
+ this.isInvalid = !this.checkTimeValidity(this.chosenTime.trim());
55
+ this.emitTime({
56
+ valid: !this.isInvalid,
57
+ time: this.chosenTime
58
+ });
59
+ this.calculateClosestInterval(this.chosenTime);
60
+ }
61
+ }
62
+ emitTime(data) {
63
+ this.timeEmitter.emit(data);
64
+ }
65
+ // gets a boolean from overlay event to close the dropdown
66
+ closeDropdown(val) {
67
+ this.openStatus = val;
68
+ }
69
+ // toggle timepicker dropdown
70
+ openDropdown() {
71
+ this.openStatus = true;
72
+ }
73
+ checkTimeValidity(time) {
74
+ const RE12 = /^(([0-9][1-9]|1[0-2]):([0-5][0-9])( )?(am|pm|AM|PM))$/i;
75
+ const RE24 = /^([01][0-9]|2[0-3]):[0-5][0-9]$/;
76
+ const RE = this.clockFormat === 12 ? RE12 : RE24;
77
+ const timeMoment = moment(`${moment(this.dateAsEpoch).format("DD-MM-YYYY")} ${time}`, `'DD-MM-YYYY' ${this.timeFormat}`);
78
+ let flag = false;
79
+ // if the first interval is set to the start of the day
80
+ // then we don't check its validity against the current time
81
+ if (this.firstInterval &&
82
+ moment(this.firstInterval).format(this.timeFormat).valueOf() === moment().startOf("d").format(this.timeFormat).valueOf()) {
83
+ flag = time.match(RE) ? true : false;
84
+ }
85
+ else {
86
+ flag = time.match(RE) && timeMoment.isAfter(moment());
87
+ }
88
+ return flag;
89
+ }
90
+ // update chosen time as soon as the user clicks on an interval
91
+ onTimeSelect(time) {
92
+ this.isInvalid = !this.checkTimeValidity(time.trim());
93
+ if (!this.isInvalid) {
94
+ this.chosenTime = time;
95
+ this.calculateClosestInterval(this.chosenTime);
96
+ }
97
+ this.emitTime({
98
+ valid: !this.isInvalid,
99
+ time: time
100
+ });
101
+ this.openStatus = false;
102
+ if (this.timepickerDirective)
103
+ this.timepickerDirective.destroyOverlay();
104
+ }
105
+ // checks validity of time on input change and calculates the closest interval
106
+ onTimeChange(time) {
107
+ this.isInvalid = !this.checkTimeValidity(time.trim());
108
+ if (!this.isInvalid) {
109
+ this.userInputFlag = true;
110
+ this.calculateClosestInterval(time);
111
+ }
112
+ this.emitTime({
113
+ valid: !this.isInvalid,
114
+ time: time
115
+ });
116
+ }
117
+ calculateClosestInterval(time) {
118
+ const intervalMS = this.interval * 60 * 1000;
119
+ const chosenDate = moment(this.dateAsEpoch).format("DD-MM-YYYY");
120
+ // converting time passed as parameter to moment object and adding date
121
+ const parsedTimeWithDate = moment(`${chosenDate} ${time}`, `DD-MM-YYYY ${this.timeFormat}`);
122
+ // converting moment object to epoch so that calculations for rounding off are easier to do
123
+ const currEpoch = parsedTimeWithDate.valueOf();
124
+ const offset = currEpoch % intervalMS;
125
+ const roundedEpoch = offset >= intervalMS / 2 ? currEpoch + (intervalMS - offset) : currEpoch - offset;
126
+ // finding the index of element that needs to be highlighted
127
+ this.timeIntervals.forEach((interval, index, array) => {
128
+ const intervalObj = moment(`${chosenDate} ${interval}`, `DD-MM-YYYY ${this.timeFormat}`);
129
+ if (intervalObj.valueOf() === roundedEpoch)
130
+ this.isHighlighted = index;
131
+ if (array.length === 1)
132
+ this.isHighlighted = 0;
133
+ });
134
+ }
135
+ // populates the dropdown according to the first interval received
136
+ populateDropdown() {
137
+ this.timeIntervals = [];
138
+ // if picker is used as an individual component
139
+ if (!this.firstInterval) {
140
+ // firstInterval is initialised according to the current time
141
+ // if the date is same as the current date
142
+ if (moment(this.dateAsEpoch).format("DD-MM-YYYY") === moment().format("DD-MM-YYYY")) {
143
+ const offset = this.interval - (moment().minutes() % this.interval);
144
+ this.firstInterval = moment().add(offset, "m").valueOf();
145
+ }
146
+ // else the firstInterval is initialised as start of day
147
+ else {
148
+ this.firstInterval = moment().startOf("d").valueOf();
149
+ }
150
+ }
151
+ const start = moment(this.firstInterval);
152
+ const end = moment().endOf("d");
153
+ while (start.valueOf() < end.valueOf()) {
154
+ this.timeIntervals.push(start.format(this.timeFormat));
155
+ start.add(this.interval, "m");
156
+ }
157
+ // if the start time is equal to the interval just before midnight
158
+ // and the start date = end date
159
+ // push 11:59pm only
160
+ if (this.timeIntervals.length === 0) {
161
+ this.chosenTime = moment().endOf("d").format(this.timeFormat);
162
+ this.timeIntervals.push(moment().endOf("d").format(this.timeFormat));
163
+ }
164
+ }
165
+ }
166
+ TimePickerComponent.decorators = [
167
+ { type: Component, args: [{
168
+ selector: "mis-timepicker",
169
+ template: "<div class=\"timepicker-container\" [ngStyle]=\"{ height: height }\">\n <input\n type=\"text\"\n [(ngModel)]=\"chosenTime\"\n (ngModelChange)=\"onTimeChange(chosenTime)\"\n [ngClass]=\"{ invalid: isInvalid || !rangeValidity }\"\n [ngStyle]=\"{ width: inputWidth }\"\n (click)=\"openDropdown()\"\n misToolTip\n [showToolTip]=\"isInvalid || !rangeValidity\"\n [text]=\"'Invalid Time'\"\n [position]=\"'top'\"\n [showOnHover]=\"false\"\n #input\n cdkOverlayOrigin\n class=\"p\"\n />\n\n <ng-template #dropdownContainer libDropdownScroll libTimepicker [originEl]=\"input\" [openStatus]=\"openStatus\" (statusEmitter)=\"closeDropdown($event)\">\n <div *ngIf=\"openStatus\" class=\"timepicker-dropdown\" [ngStyle]=\"{ width: dropdownWidth || inputWidth }\">\n <ul #dropdown>\n <li #timeInterval (click)=\"onTimeSelect(interval)\" *ngFor=\"let interval of timeIntervals; index as i\" [ngClass]=\"{ highlight: i === isHighlighted }\">\n {{ interval }}\n <div class=\"ic-ui-check-24 selected-icon\" *ngIf=\"interval === chosenTime\"></div>\n </li>\n </ul>\n </div>\n </ng-template>\n</div>\n",
170
+ styles: [".h1{font-size:40px;line-height:48px}.h1,.h2{font-weight:400;letter-spacing:0}.h2{font-size:32px;line-height:40px}.h3{font-size:28px;line-height:36px}.h3,.h4{font-weight:400;letter-spacing:0}.h4{font-size:24px;line-height:32px}.h5-b{font-weight:700;letter-spacing:.25px}.h5,.h5-b{font-size:20px;line-height:28px}.h5{font-weight:400;letter-spacing:.15px}.h6-b{font-weight:700}.h6,.h6-b{font-size:16px;letter-spacing:0;line-height:24px}.h6,.p{font-weight:400}.p{font-size:16px;letter-spacing:0;line-height:180%}.h7-b{font-weight:700;letter-spacing:.25px}.h7,.h7-b{font-size:14px;line-height:20px}.h7{font-weight:400;letter-spacing:.2px}.h8-b{font-weight:700;letter-spacing:.25px}.h8,.h8-b{font-size:12px;line-height:18px}.h8{letter-spacing:.2px}.h8,.h9{font-weight:400}.h9{font-size:10px;letter-spacing:0;line-height:15px}.btn-lg-b{font-weight:700;letter-spacing:.5px}.btn-lg,.btn-lg-b{font-size:16px;line-height:24px}.btn-lg{font-weight:400;letter-spacing:.2px}.btn-sm{font-size:14px;font-weight:400;letter-spacing:.25px;line-height:20px}.btn-link{font-size:16px;line-height:24px}.btn-link,.display-1{font-weight:400;letter-spacing:0}.display-1{font-size:48px;line-height:56px}.display-2{font-size:14px;font-weight:400;letter-spacing:.5px;line-height:20px}*{box-sizing:border-box;font-family:Lato}.timepicker-container{display:inline-block;position:relative}input{text-align:center;border:none;border-bottom:1px solid #e0e0e0;outline:none;height:100%;width:100px;padding:4px 16px}input:hover{background:#f5f7fc}input:active,input:focus{background:#e6ebf7;border-bottom:1px solid #0937b2}.timepicker-dropdown{position:absolute;top:calc(100% + 4px);left:0;max-height:200px;overflow-y:auto;border:1px solid #e0e0e0;box-shadow:0 12px 24px rgba(0,0,0,.12),0 4px 8px rgba(0,0,0,.12);border-radius:4px;padding:8px;background:#fff}.timepicker-dropdown::-webkit-scrollbar{display:none}.timepicker-dropdown ul{margin:0;padding:0}.timepicker-dropdown li{text-align:start;list-style:none;padding:6px 12px;cursor:pointer;font-size:16px;color:#181f33;display:flex;justify-content:space-between;align-items:center}.timepicker-dropdown li:hover{background:#f5f7fc;border-radius:6px}.timepicker-dropdown li .selected-icon{font-weight:900}.highlight{background-color:#f5f7fc;border-radius:6px}.invalid{background:#fae1ea!important;border-bottom:1px solid #b00020!important}"]
171
+ },] }
172
+ ];
173
+ TimePickerComponent.ctorParameters = () => [];
174
+ TimePickerComponent.propDecorators = {
175
+ clockFormat: [{ type: Input }],
176
+ timezone: [{ type: Input }],
177
+ height: [{ type: Input }],
178
+ inputWidth: [{ type: Input }],
179
+ dropdownWidth: [{ type: Input }],
180
+ interval: [{ type: Input }],
181
+ dateAsEpoch: [{ type: Input }],
182
+ firstInterval: [{ type: Input }],
183
+ rangeValidity: [{ type: Input }],
184
+ timeEmitter: [{ type: Output }],
185
+ input: [{ type: ViewChild, args: ["input", { static: true },] }],
186
+ timepickerDirective: [{ type: ViewChild, args: [TimepickerDirective,] }],
187
+ timeIntervalRefs: [{ type: ViewChildren, args: ["timeInterval",] }]
188
+ };
189
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timepicker.component.js","sourceRoot":"","sources":["../../../../projects/mis-components/timepicker/timepicker.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, ElementRef, EventEmitter, Input, Output, QueryList, TemplateRef, ViewChild, ViewChildren } from \"@angular/core\";\nimport * as moment from \"moment-timezone\";\nimport { ITime } from \"./time.namespace\";\nimport { TimepickerDirective } from \"./timepicker.directive\";\n\n@Component({\n  selector: \"mis-timepicker\",\n  templateUrl: \"./timepicker.component.html\",\n  styleUrls: [\"./timepicker.component.scss\"]\n})\nexport class TimePickerComponent {\n  currTime!: string;\n  chosenTime: string;\n  openStatus: boolean = false;\n  isHighlighted: number = 0;\n  isInvalid: boolean = false;\n  timeIntervals: string[] = [];\n  shouldScroll: boolean = false;\n  userInputFlag: boolean = false;\n\n  @Input() clockFormat: number = 12;\n  timeFormat!: string;\n\n  @Input() timezone: string = \"Asia/Kolkata\";\n  @Input() height: string = \"max-content\";\n  @Input() inputWidth: string = \"100px\";\n  @Input() dropdownWidth?: string;\n  @Input() interval: number = 15;\n  @Input() dateAsEpoch: number = moment().valueOf();\n  @Input() firstInterval!: number;\n  @Input() rangeValidity: boolean = true;\n\n  @Output() timeEmitter = new EventEmitter<ITime>();\n  @ViewChild(\"input\", { static: true }) input: ElementRef;\n  @ViewChild(TimepickerDirective) timepickerDirective: TimepickerDirective;\n\n  // gets all the li elements from the dropdown and scrolls to the highlighted element\n  @ViewChildren(\"timeInterval\") set timeIntervalRefs(intervals) {\n    intervals.forEach(interval => {\n      if (interval.nativeElement.classList[0] === \"highlight\") {\n        const highlighted = interval.nativeElement;\n        setTimeout(() => highlighted.scrollIntoView({ behavior: \"smooth\", block: \"center\" }));\n      }\n    });\n  }\n\n  constructor() {}\n  ngOnInit(): void {\n    this.timeFormat = this.clockFormat === 12 ? \"hh:mm a\" : \"HH:mm\";\n    this.populateDropdown();\n    this.chosenTime = this.timeIntervals[0];\n    this.calculateClosestInterval(this.chosenTime);\n    this.emitTime({\n      valid: !this.isInvalid,\n      time: this.chosenTime\n    });\n  }\n\n  ngOnChanges(): void {\n    moment.tz.setDefault(this.timezone);\n    this.currTime = moment().format();\n    if (this.timeFormat) {\n      // if the first interval is >= the chosen time\n      // then only update the value of chosen time\n      // else it remains the same\n      const chosenTimeMoment = moment(`${moment(this.dateAsEpoch).format(\"DD-MM-YYYY\")} ${this.chosenTime}`, `'DD-MM-YYYY' ${this.timeFormat}`);\n\n      if (this.firstInterval >= chosenTimeMoment.valueOf() && this.rangeValidity && !this.userInputFlag) {\n        this.chosenTime = moment(this.firstInterval).format(this.timeFormat);\n      }\n      if (!this.userInputFlag) this.populateDropdown();\n      this.userInputFlag = false;\n      this.isInvalid = !this.checkTimeValidity(this.chosenTime.trim());\n      this.emitTime({\n        valid: !this.isInvalid,\n        time: this.chosenTime\n      });\n      this.calculateClosestInterval(this.chosenTime);\n    }\n  }\n\n  emitTime(data: ITime): void {\n    this.timeEmitter.emit(data);\n  }\n\n  // gets a boolean from overlay event to close the dropdown\n  closeDropdown(val: boolean) {\n    this.openStatus = val;\n  }\n\n  // toggle timepicker dropdown\n  openDropdown(): void {\n    this.openStatus = true;\n  }\n\n  checkTimeValidity(time: string): boolean {\n    const RE12 = /^(([0-9][1-9]|1[0-2]):([0-5][0-9])( )?(am|pm|AM|PM))$/i;\n    const RE24 = /^([01][0-9]|2[0-3]):[0-5][0-9]$/;\n    const RE = this.clockFormat === 12 ? RE12 : RE24;\n\n    const timeMoment = moment(`${moment(this.dateAsEpoch).format(\"DD-MM-YYYY\")} ${time}`, `'DD-MM-YYYY' ${this.timeFormat}`);\n    let flag: boolean = false;\n\n    // if the first interval is set to the start of the day\n    // then we don't check its validity against the current time\n    if (\n      this.firstInterval &&\n      moment(this.firstInterval).format(this.timeFormat).valueOf() === moment().startOf(\"d\").format(this.timeFormat).valueOf()\n    ) {\n      flag = time.match(RE) ? true : false;\n    } else {\n      flag = time.match(RE) && timeMoment.isAfter(moment());\n    }\n\n    return flag;\n  }\n\n  // update chosen time as soon as the user clicks on an interval\n  onTimeSelect(time: string): void {\n    this.isInvalid = !this.checkTimeValidity(time.trim());\n    if (!this.isInvalid) {\n      this.chosenTime = time;\n      this.calculateClosestInterval(this.chosenTime);\n    }\n    this.emitTime({\n      valid: !this.isInvalid,\n      time: time\n    });\n    this.openStatus = false;\n    if (this.timepickerDirective) this.timepickerDirective.destroyOverlay();\n  }\n\n  // checks validity of time on input change and calculates the closest interval\n  onTimeChange(time: string): void {\n    this.isInvalid = !this.checkTimeValidity(time.trim());\n    if (!this.isInvalid) {\n      this.userInputFlag = true;\n      this.calculateClosestInterval(time);\n    }\n    this.emitTime({\n      valid: !this.isInvalid,\n      time: time\n    });\n  }\n\n  calculateClosestInterval(time: string): void {\n    const intervalMS = this.interval * 60 * 1000;\n    const chosenDate = moment(this.dateAsEpoch).format(\"DD-MM-YYYY\");\n\n    // converting time passed as parameter to moment object and adding date\n    const parsedTimeWithDate = moment(`${chosenDate} ${time}`, `DD-MM-YYYY ${this.timeFormat}`);\n\n    // converting moment object to epoch so that calculations for rounding off are easier to do\n    const currEpoch = parsedTimeWithDate.valueOf();\n\n    const offset = currEpoch % intervalMS;\n    const roundedEpoch = offset >= intervalMS / 2 ? currEpoch + (intervalMS - offset) : currEpoch - offset;\n\n    // finding the index of element that needs to be highlighted\n    this.timeIntervals.forEach((interval, index, array) => {\n      const intervalObj = moment(`${chosenDate} ${interval}`, `DD-MM-YYYY ${this.timeFormat}`);\n      if (intervalObj.valueOf() === roundedEpoch) this.isHighlighted = index;\n      if (array.length === 1) this.isHighlighted = 0;\n    });\n  }\n\n  // populates the dropdown according to the first interval received\n  populateDropdown(): void {\n    this.timeIntervals = [];\n\n    // if picker is used as an individual component\n    if (!this.firstInterval) {\n      // firstInterval is initialised according to the current time\n      // if the date is same as the current date\n      if (moment(this.dateAsEpoch).format(\"DD-MM-YYYY\") === moment().format(\"DD-MM-YYYY\")) {\n        const offset = this.interval - (moment().minutes() % this.interval);\n        this.firstInterval = moment().add(offset, \"m\").valueOf();\n      }\n\n      // else the firstInterval is initialised as start of day\n      else {\n        this.firstInterval = moment().startOf(\"d\").valueOf();\n      }\n    }\n\n    const start = moment(this.firstInterval);\n    const end = moment().endOf(\"d\");\n\n    while (start.valueOf() < end.valueOf()) {\n      this.timeIntervals.push(start.format(this.timeFormat));\n      start.add(this.interval, \"m\");\n    }\n\n    // if the start time is equal to the interval just before midnight\n    // and the start date = end date\n    // push 11:59pm only\n    if (this.timeIntervals.length === 0) {\n      this.chosenTime = moment().endOf(\"d\").format(this.timeFormat);\n      this.timeIntervals.push(moment().endOf(\"d\").format(this.timeFormat));\n    }\n  }\n}\n"]}
@@ -0,0 +1,62 @@
1
+ import { Directive, Output, TemplateRef } from '@angular/core';
2
+ import { EventEmitter, Input, ViewContainerRef } from "@angular/core";
3
+ import { ConnectionPositionPair, Overlay, OverlayConfig } from "@angular/cdk/overlay";
4
+ import { TemplatePortal } from "@angular/cdk/portal";
5
+ export class TimepickerDirective {
6
+ constructor(templateRef, overlay, viewContainerRef) {
7
+ this.templateRef = templateRef;
8
+ this.overlay = overlay;
9
+ this.viewContainerRef = viewContainerRef;
10
+ this.openStatus = false;
11
+ this.statusEmitter = new EventEmitter();
12
+ }
13
+ set createOverlayOnInput(openStatus) {
14
+ this.openStatus = openStatus;
15
+ if (this.originEl && this.openStatus)
16
+ this.createOverlay(this.originEl);
17
+ }
18
+ ;
19
+ createOverlay(origin) {
20
+ const positions = [
21
+ new ConnectionPositionPair({ originX: "start", originY: "bottom" }, { overlayX: "start", overlayY: "top" }, 0, 4),
22
+ new ConnectionPositionPair({ originX: "end", originY: "bottom" }, { overlayX: "end", overlayY: "top" }, 0, 4)
23
+ ];
24
+ const overlayConfig = new OverlayConfig({
25
+ hasBackdrop: true,
26
+ backdropClass: "cdk-overlay-transparent-backdrop",
27
+ positionStrategy: this.overlay
28
+ .position()
29
+ //connecting the dropdown overlay to the input element
30
+ .flexibleConnectedTo(origin)
31
+ .withPositions([...positions])
32
+ .withPush(true)
33
+ });
34
+ this.overlayRef = this.overlay.create(overlayConfig);
35
+ const dropdownPortal = new TemplatePortal(this.templateRef, this.viewContainerRef);
36
+ this.overlayRef.attach(dropdownPortal);
37
+ this.overlayRef.backdropClick().subscribe(resp => {
38
+ this.openStatus = false;
39
+ this.statusEmitter.emit(false);
40
+ this.overlayRef.detach();
41
+ });
42
+ }
43
+ destroyOverlay() {
44
+ this.overlayRef.detach();
45
+ }
46
+ }
47
+ TimepickerDirective.decorators = [
48
+ { type: Directive, args: [{
49
+ selector: '[libTimepicker]'
50
+ },] }
51
+ ];
52
+ TimepickerDirective.ctorParameters = () => [
53
+ { type: TemplateRef },
54
+ { type: Overlay },
55
+ { type: ViewContainerRef }
56
+ ];
57
+ TimepickerDirective.propDecorators = {
58
+ originEl: [{ type: Input, args: ['originEl',] }],
59
+ statusEmitter: [{ type: Output }],
60
+ createOverlayOnInput: [{ type: Input, args: ['openStatus',] }]
61
+ };
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGltZXBpY2tlci5kaXJlY3RpdmUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9taXMtY29tcG9uZW50cy90aW1lcGlja2VyL3RpbWVwaWNrZXIuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsTUFBTSxFQUFFLFdBQVcsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMvRCxPQUFPLEVBQUUsWUFBWSxFQUFFLEtBQUssRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN0RSxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsT0FBTyxFQUFFLGFBQWEsRUFBYyxNQUFNLHNCQUFzQixDQUFDO0FBQ2xHLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxxQkFBcUIsQ0FBQztBQUtyRCxNQUFNLE9BQU8sbUJBQW1CO0lBVTlCLFlBQW9CLFdBQWlDLEVBQVUsT0FBZ0IsRUFBVSxnQkFBa0M7UUFBdkcsZ0JBQVcsR0FBWCxXQUFXLENBQXNCO1FBQVUsWUFBTyxHQUFQLE9BQU8sQ0FBUztRQUFVLHFCQUFnQixHQUFoQixnQkFBZ0IsQ0FBa0I7UUFUbkgsZUFBVSxHQUFZLEtBQUssQ0FBQztRQUUxQixrQkFBYSxHQUFHLElBQUksWUFBWSxFQUFXLENBQUM7SUFPd0UsQ0FBQztJQUwvSCxJQUEwQixvQkFBb0IsQ0FBQyxVQUFVO1FBQ3ZELElBQUksQ0FBQyxVQUFVLEdBQUcsVUFBVSxDQUFDO1FBQzdCLElBQUcsSUFBSSxDQUFDLFFBQVEsSUFBSSxJQUFJLENBQUMsVUFBVTtZQUFFLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ3pFLENBQUM7SUFBQSxDQUFDO0lBSUYsYUFBYSxDQUFDLE1BQVc7UUFDdkIsTUFBTSxTQUFTLEdBQUc7WUFDaEIsSUFBSSxzQkFBc0IsQ0FBQyxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsT0FBTyxFQUFFLFFBQVEsRUFBRSxFQUFFLEVBQUUsUUFBUSxFQUFFLE9BQU8sRUFBRSxRQUFRLEVBQUUsS0FBSyxFQUFFLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQztZQUNqSCxJQUFJLHNCQUFzQixDQUFDLEVBQUUsT0FBTyxFQUFFLEtBQUssRUFBRSxPQUFPLEVBQUUsUUFBUSxFQUFFLEVBQUUsRUFBRSxRQUFRLEVBQUUsS0FBSyxFQUFFLFFBQVEsRUFBRSxLQUFLLEVBQUUsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDO1NBQzlHLENBQUM7UUFFRixNQUFNLGFBQWEsR0FBRyxJQUFJLGFBQWEsQ0FBQztZQUN0QyxXQUFXLEVBQUUsSUFBSTtZQUNqQixhQUFhLEVBQUUsa0NBQWtDO1lBQ2pELGdCQUFnQixFQUFFLElBQUksQ0FBQyxPQUFPO2lCQUMzQixRQUFRLEVBQUU7Z0JBQ1gsc0RBQXNEO2lCQUNyRCxtQkFBbUIsQ0FBQyxNQUFNLENBQUM7aUJBQzNCLGFBQWEsQ0FBQyxDQUFDLEdBQUcsU0FBUyxDQUFDLENBQUM7aUJBQzdCLFFBQVEsQ0FBQyxJQUFJLENBQUM7U0FDbEIsQ0FBQyxDQUFDO1FBRUgsSUFBSSxDQUFDLFVBQVUsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxhQUFhLENBQUMsQ0FBQztRQUNyRCxNQUFNLGNBQWMsR0FBRyxJQUFJLGNBQWMsQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO1FBQ25GLElBQUksQ0FBQyxVQUFVLENBQUMsTUFBTSxDQUFDLGNBQWMsQ0FBQyxDQUFDO1FBQ3ZDLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxFQUFFLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxFQUFFO1lBQy9DLElBQUksQ0FBQyxVQUFVLEdBQUcsS0FBSyxDQUFDO1lBQ3hCLElBQUksQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1lBQy9CLElBQUksQ0FBQyxVQUFVLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDM0IsQ0FBQyxDQUFDLENBQUM7SUFDTCxDQUFDO0lBRUQsY0FBYztRQUNaLElBQUksQ0FBQyxVQUFVLENBQUMsTUFBTSxFQUFFLENBQUM7SUFDM0IsQ0FBQzs7O1lBNUNGLFNBQVMsU0FBQztnQkFDVCxRQUFRLEVBQUUsaUJBQWlCO2FBQzVCOzs7WUFQMkIsV0FBVztZQUVOLE9BQU87WUFEVixnQkFBZ0I7Ozt1QkFTM0MsS0FBSyxTQUFDLFVBQVU7NEJBQ2hCLE1BQU07bUNBRU4sS0FBSyxTQUFDLFlBQVkiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBEaXJlY3RpdmUsIE91dHB1dCwgVGVtcGxhdGVSZWYgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEV2ZW50RW1pdHRlciwgSW5wdXQsIFZpZXdDb250YWluZXJSZWYgfSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xuaW1wb3J0IHsgQ29ubmVjdGlvblBvc2l0aW9uUGFpciwgT3ZlcmxheSwgT3ZlcmxheUNvbmZpZywgT3ZlcmxheVJlZiB9IGZyb20gXCJAYW5ndWxhci9jZGsvb3ZlcmxheVwiO1xuaW1wb3J0IHsgVGVtcGxhdGVQb3J0YWwgfSBmcm9tIFwiQGFuZ3VsYXIvY2RrL3BvcnRhbFwiO1xuXG5ARGlyZWN0aXZlKHtcbiAgc2VsZWN0b3I6ICdbbGliVGltZXBpY2tlcl0nXG59KVxuZXhwb3J0IGNsYXNzIFRpbWVwaWNrZXJEaXJlY3RpdmUge1xuICBwcml2YXRlIG9wZW5TdGF0dXM6IGJvb2xlYW4gPSBmYWxzZTtcbiAgQElucHV0KCdvcmlnaW5FbCcpIG9yaWdpbkVsIDogYW55O1xuICBAT3V0cHV0KCkgc3RhdHVzRW1pdHRlciA9IG5ldyBFdmVudEVtaXR0ZXI8Ym9vbGVhbj4oKTsgXG5cbiAgQElucHV0KCdvcGVuU3RhdHVzJykgIHNldCBjcmVhdGVPdmVybGF5T25JbnB1dChvcGVuU3RhdHVzKXtcbiAgICB0aGlzLm9wZW5TdGF0dXMgPSBvcGVuU3RhdHVzO1xuICAgIGlmKHRoaXMub3JpZ2luRWwgJiYgdGhpcy5vcGVuU3RhdHVzKSB0aGlzLmNyZWF0ZU92ZXJsYXkodGhpcy5vcmlnaW5FbCk7XG4gIH07IFxuICBwcml2YXRlIG92ZXJsYXlSZWYhOiBPdmVybGF5UmVmO1xuICBjb25zdHJ1Y3Rvcihwcml2YXRlIHRlbXBsYXRlUmVmOiBUZW1wbGF0ZVJlZjxFbGVtZW50PiwgcHJpdmF0ZSBvdmVybGF5OiBPdmVybGF5LCBwcml2YXRlIHZpZXdDb250YWluZXJSZWY6IFZpZXdDb250YWluZXJSZWYpIHt9XG5cbiAgY3JlYXRlT3ZlcmxheShvcmlnaW46IGFueSk6IHZvaWQge1xuICAgIGNvbnN0IHBvc2l0aW9ucyA9IFtcbiAgICAgIG5ldyBDb25uZWN0aW9uUG9zaXRpb25QYWlyKHsgb3JpZ2luWDogXCJzdGFydFwiLCBvcmlnaW5ZOiBcImJvdHRvbVwiIH0sIHsgb3ZlcmxheVg6IFwic3RhcnRcIiwgb3ZlcmxheVk6IFwidG9wXCIgfSwgMCwgNCksXG4gICAgICBuZXcgQ29ubmVjdGlvblBvc2l0aW9uUGFpcih7IG9yaWdpblg6IFwiZW5kXCIsIG9yaWdpblk6IFwiYm90dG9tXCIgfSwgeyBvdmVybGF5WDogXCJlbmRcIiwgb3ZlcmxheVk6IFwidG9wXCIgfSwgMCwgNClcbiAgICBdO1xuXG4gICAgY29uc3Qgb3ZlcmxheUNvbmZpZyA9IG5ldyBPdmVybGF5Q29uZmlnKHtcbiAgICAgIGhhc0JhY2tkcm9wOiB0cnVlLFxuICAgICAgYmFja2Ryb3BDbGFzczogXCJjZGstb3ZlcmxheS10cmFuc3BhcmVudC1iYWNrZHJvcFwiLFxuICAgICAgcG9zaXRpb25TdHJhdGVneTogdGhpcy5vdmVybGF5XG4gICAgICAgIC5wb3NpdGlvbigpXG4gICAgICAgIC8vY29ubmVjdGluZyB0aGUgZHJvcGRvd24gb3ZlcmxheSB0byB0aGUgaW5wdXQgZWxlbWVudFxuICAgICAgICAuZmxleGlibGVDb25uZWN0ZWRUbyhvcmlnaW4pXG4gICAgICAgIC53aXRoUG9zaXRpb25zKFsuLi5wb3NpdGlvbnNdKVxuICAgICAgICAud2l0aFB1c2godHJ1ZSlcbiAgICB9KTtcblxuICAgIHRoaXMub3ZlcmxheVJlZiA9IHRoaXMub3ZlcmxheS5jcmVhdGUob3ZlcmxheUNvbmZpZyk7XG4gICAgY29uc3QgZHJvcGRvd25Qb3J0YWwgPSBuZXcgVGVtcGxhdGVQb3J0YWwodGhpcy50ZW1wbGF0ZVJlZiwgdGhpcy52aWV3Q29udGFpbmVyUmVmKTtcbiAgICB0aGlzLm92ZXJsYXlSZWYuYXR0YWNoKGRyb3Bkb3duUG9ydGFsKTtcbiAgICB0aGlzLm92ZXJsYXlSZWYuYmFja2Ryb3BDbGljaygpLnN1YnNjcmliZShyZXNwID0+IHtcbiAgICAgIHRoaXMub3BlblN0YXR1cyA9IGZhbHNlO1xuICAgICAgdGhpcy5zdGF0dXNFbWl0dGVyLmVtaXQoZmFsc2UpO1xuICAgICAgdGhpcy5vdmVybGF5UmVmLmRldGFjaCgpO1xuICAgIH0pO1xuICB9XG5cbiAgZGVzdHJveU92ZXJsYXkoKXtcbiAgICB0aGlzLm92ZXJsYXlSZWYuZGV0YWNoKCk7XG4gIH1cbiAgXG59XG4iXX0=
@@ -0,0 +1,20 @@
1
+ import { NgModule } from "@angular/core";
2
+ import { CommonModule } from '@angular/common';
3
+ import { FormsModule } from '@angular/forms';
4
+ import { TimePickerComponent } from "./timepicker.component";
5
+ import { ToolTipModule } from "mis-crystal-design-system/tooltip";
6
+ import { OverlayModule } from '@angular/cdk/overlay';
7
+ import { TimepickerDirective } from './timepicker.directive';
8
+ export class TimePickerModule {
9
+ static forRoot() {
10
+ return { ngModule: TimePickerModule, providers: [] };
11
+ }
12
+ }
13
+ TimePickerModule.decorators = [
14
+ { type: NgModule, args: [{
15
+ declarations: [TimePickerComponent, TimepickerDirective],
16
+ imports: [CommonModule, FormsModule, ToolTipModule, OverlayModule],
17
+ exports: [TimePickerComponent]
18
+ },] }
19
+ ];
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGltZXBpY2tlci5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9taXMtY29tcG9uZW50cy90aW1lcGlja2VyL3RpbWVwaWNrZXIubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQXVCLE1BQU0sZUFBZSxDQUFDO0FBQzlELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDN0MsT0FBTyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDN0QsT0FBTyxFQUFFLGFBQWEsRUFBRSxNQUFNLG1DQUFtQyxDQUFDO0FBQ2xFLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUNyRCxPQUFPLEVBQUUsbUJBQW1CLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQztBQU83RCxNQUFNLE9BQU8sZ0JBQWdCO0lBQzNCLE1BQU0sQ0FBQyxPQUFPO1FBQ1osT0FBTyxFQUFFLFFBQVEsRUFBRSxnQkFBZ0IsRUFBRSxTQUFTLEVBQUUsRUFBRSxFQUFFLENBQUM7SUFDdkQsQ0FBQzs7O1lBUkYsUUFBUSxTQUFDO2dCQUNSLFlBQVksRUFBRSxDQUFDLG1CQUFtQixFQUFFLG1CQUFtQixDQUFDO2dCQUN4RCxPQUFPLEVBQUUsQ0FBQyxZQUFZLEVBQUUsV0FBVyxFQUFFLGFBQWEsRUFBRSxhQUFhLENBQUM7Z0JBQ2xFLE9BQU8sRUFBRSxDQUFDLG1CQUFtQixDQUFDO2FBQy9CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUsIE1vZHVsZVdpdGhQcm92aWRlcnMgfSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEZvcm1zTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgVGltZVBpY2tlckNvbXBvbmVudCB9IGZyb20gXCIuL3RpbWVwaWNrZXIuY29tcG9uZW50XCI7XG5pbXBvcnQgeyBUb29sVGlwTW9kdWxlIH0gZnJvbSBcIm1pcy1jcnlzdGFsLWRlc2lnbi1zeXN0ZW0vdG9vbHRpcFwiO1xuaW1wb3J0IHsgT3ZlcmxheU1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2Nkay9vdmVybGF5JztcbmltcG9ydCB7IFRpbWVwaWNrZXJEaXJlY3RpdmUgfSBmcm9tICcuL3RpbWVwaWNrZXIuZGlyZWN0aXZlJztcblxuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbVGltZVBpY2tlckNvbXBvbmVudCwgVGltZXBpY2tlckRpcmVjdGl2ZV0sXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEZvcm1zTW9kdWxlLCBUb29sVGlwTW9kdWxlLCBPdmVybGF5TW9kdWxlXSxcbiAgZXhwb3J0czogW1RpbWVQaWNrZXJDb21wb25lbnRdXG59KVxuZXhwb3J0IGNsYXNzIFRpbWVQaWNrZXJNb2R1bGUge1xuICBzdGF0aWMgZm9yUm9vdCgpOiBNb2R1bGVXaXRoUHJvdmlkZXJzPFRpbWVQaWNrZXJNb2R1bGU+IHtcbiAgICByZXR1cm4geyBuZ01vZHVsZTogVGltZVBpY2tlck1vZHVsZSwgcHJvdmlkZXJzOiBbXSB9O1xuICB9XG59Il19
@@ -0,0 +1,2 @@
1
+ export * from "./public_api";
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9taXMtY29tcG9uZW50cy90aW1lcmFuZ2VwaWNrZXIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxjQUFjLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tIFwiLi9wdWJsaWNfYXBpXCI7Il19
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './index';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWlzLWNyeXN0YWwtZGVzaWduLXN5c3RlbS10aW1lcmFuZ2VwaWNrZXIuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9taXMtY29tcG9uZW50cy90aW1lcmFuZ2VwaWNrZXIvbWlzLWNyeXN0YWwtZGVzaWduLXN5c3RlbS10aW1lcmFuZ2VwaWNrZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLFNBQVMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9pbmRleCc7XG4iXX0=
@@ -0,0 +1,3 @@
1
+ export { TimeRangePickerComponent } from "./timerangepicker.component";
2
+ export { TimeRangePickerModule } from "./timerangepicker.module";
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL21pcy1jb21wb25lbnRzL3RpbWVyYW5nZXBpY2tlci9wdWJsaWNfYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLDZCQUE2QixDQUFDO0FBQ3ZFLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLDBCQUEwQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0IHsgVGltZVJhbmdlUGlja2VyQ29tcG9uZW50IH0gZnJvbSBcIi4vdGltZXJhbmdlcGlja2VyLmNvbXBvbmVudFwiO1xuZXhwb3J0IHsgVGltZVJhbmdlUGlja2VyTW9kdWxlIH0gZnJvbSBcIi4vdGltZXJhbmdlcGlja2VyLm1vZHVsZVwiIl19
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGltZXJhbmdlLm5hbWVzcGFjZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL21pcy1jb21wb25lbnRzL3RpbWVyYW5nZXBpY2tlci90aW1lcmFuZ2UubmFtZXNwYWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgaW50ZXJmYWNlIElUaW1lUmFuZ2V7XG4gIHZhbGlkOiBib29sZWFuLFxuICBzdGFydFRpbWU6IHN0cmluZyxcbiAgZW5kVGltZTogc3RyaW5nXG59XG5cbmV4cG9ydCBpbnRlcmZhY2UgSVRpbWV7XG4gIHZhbGlkOiBib29sZWFuLFxuICB0aW1lOiBzdHJpbmdcbn0iXX0=
@@ -0,0 +1,123 @@
1
+ import { Component, Input, ChangeDetectorRef, Output, EventEmitter } from "@angular/core";
2
+ import * as moment from "moment-timezone";
3
+ export class TimeRangePickerComponent {
4
+ constructor(cdr) {
5
+ this.cdr = cdr;
6
+ this.inputWidth = '100px';
7
+ this.height = '46px';
8
+ this.timezone = "Asia/Kolkata";
9
+ this.startDateEpoch = moment().tz(this.timezone).valueOf();
10
+ this.endDateEpoch = moment().tz(this.timezone).valueOf();
11
+ this.clockFormat = 12;
12
+ this.interval = 15;
13
+ this.timeRangeEmitter = new EventEmitter();
14
+ this.firstIntervalForStartPicker = moment().valueOf();
15
+ this.firstIntervalForEndPicker = moment().valueOf();
16
+ this.rangeValidity = true;
17
+ }
18
+ ngOnInit() {
19
+ console.log(this.timezone);
20
+ this.timeFormat = this.clockFormat === 12 ? "hh:mm a" : "HH:mm";
21
+ this.setFirstIntervals();
22
+ }
23
+ ngOnChanges() {
24
+ moment.tz.setDefault(this.timezone);
25
+ }
26
+ // calculate the first interval of the picker
27
+ setFirstIntervals() {
28
+ const startDate = moment(this.startDateEpoch).format("DD-MM-YYYY");
29
+ const endDate = moment(this.endDateEpoch).format("DD-MM-YYYY");
30
+ const minutes = moment().minutes();
31
+ const offset = this.interval - (minutes % this.interval);
32
+ // first interval for the start picker will always be the one closest to the current time
33
+ this.firstIntervalForStartPicker = moment().add(offset, "m").valueOf();
34
+ this.startTime = {
35
+ valid: true,
36
+ time: moment().add(offset, "m").format(this.timeFormat)
37
+ };
38
+ // for the end picker if the startDate and the endDate is same
39
+ // the first interval is set one interval ahead of the first interval for start picker
40
+ // else if the dates are different we set it to the start of the day
41
+ if (startDate === endDate) {
42
+ this.firstIntervalForEndPicker = moment().add(offset, "m").add(this.interval, "m").valueOf();
43
+ }
44
+ else {
45
+ this.firstIntervalForEndPicker = moment().startOf("d").valueOf();
46
+ }
47
+ this.endTime = {
48
+ valid: true,
49
+ time: moment(this.firstIntervalForEndPicker).format(this.timeFormat)
50
+ };
51
+ }
52
+ emitTimeRange(data) {
53
+ this.timeRangeEmitter.emit(data);
54
+ }
55
+ //handlers catch the emitted values and run validation
56
+ startPickerHandler(time) {
57
+ this.startTime = time;
58
+ // if the start time changes and the start date is the same as the end date
59
+ // and the start time >= end time
60
+ // update the first interval of end picker according to the time set in start picker
61
+ const startDate = moment(this.startDateEpoch).format("DD-MM-YYYY");
62
+ const endDate = moment(this.endDateEpoch).format("DD-MM-YYYY");
63
+ if (startDate === endDate) {
64
+ let minutes = moment(this.startTime.time, this.timeFormat).minutes();
65
+ let offset = this.interval - (minutes % this.interval);
66
+ this.firstIntervalForEndPicker = moment(`${startDate} ${this.startTime.time}`, `'DD-MM-YYYY' ${this.timeFormat}`).add(offset, "m").valueOf();
67
+ }
68
+ }
69
+ endPickerHandler(time) {
70
+ this.endTime = time;
71
+ const validity = this.checkTimeValidity(this.startTime.time.trim()) && this.checkTimeValidity(this.endTime.time.trim());
72
+ this.emitTimeRange({
73
+ valid: validity && this.rangeValidity,
74
+ startTime: this.startTime.time,
75
+ endTime: this.endTime.time
76
+ });
77
+ this.rangeValidation();
78
+ }
79
+ checkTimeValidity(time) {
80
+ const RE12 = /^(([0-9][1-9]|1[0-2]):([0-5][0-9])( )?(am|pm|AM|PM))$/i;
81
+ const RE24 = /^([01][0-9]|2[0-3]):[0-5][0-9]$/;
82
+ const RE = this.clockFormat === 12 ? RE12 : RE24;
83
+ const flag = time.match(RE) ? true : false;
84
+ return flag;
85
+ }
86
+ // validates end picker's input according to the start picker's input
87
+ rangeValidation() {
88
+ const startDate = moment(this.startDateEpoch).format("DD-MM-YYYY");
89
+ const endDate = moment(this.endDateEpoch).format("DD-MM-YYYY");
90
+ const validity = this.checkTimeValidity(this.startTime.time.trim()) && this.checkTimeValidity(this.endTime.time.trim());
91
+ if (validity && startDate === endDate) {
92
+ if (this.endTime.time === moment().endOf("d").add(1, "m").format(this.timeFormat)) {
93
+ this.endTime.time = moment().endOf("d").format(this.timeFormat);
94
+ }
95
+ const startMoment = moment(this.startTime.time, this.timeFormat);
96
+ const endMoment = moment(this.endTime.time, this.timeFormat);
97
+ this.rangeValidity = endMoment.diff(startMoment, "m") >= 1 ? true : false;
98
+ this.cdr.detectChanges();
99
+ }
100
+ }
101
+ }
102
+ TimeRangePickerComponent.decorators = [
103
+ { type: Component, args: [{
104
+ selector: "mis-timerangepicker",
105
+ template: "<div class=\"rangepicker-container\">\n <mis-timepicker\n [clockFormat]=\"clockFormat\"\n [interval]=\"interval\"\n [dateAsEpoch]=\"startDateEpoch\"\n [firstInterval]=\"firstIntervalForStartPicker\"\n (timeEmitter)=\"startPickerHandler($event)\"\n [timezone]=\"timezone\"\n [height]=\"height\"\n [inputWidth]=\"'200px'\"\n [dropdownWidth]=\"dropdownWidth\"\n ></mis-timepicker>\n <mis-timepicker\n [clockFormat]=\"clockFormat\"\n [interval]=\"interval\"\n [dateAsEpoch]=\"endDateEpoch\"\n [firstInterval]=\"firstIntervalForEndPicker\"\n (timeEmitter)=\"endPickerHandler($event)\"\n [rangeValidity]=\"rangeValidity\"\n [timezone]=\"timezone\"\n [height]=\"height\"\n [inputWidth]=\"'200px'\"\n [dropdownWidth]=\"dropdownWidth\"\n ></mis-timepicker>\n</div>\n",
106
+ styles: [".rangepicker-container{display:flex;gap:1rem}"]
107
+ },] }
108
+ ];
109
+ TimeRangePickerComponent.ctorParameters = () => [
110
+ { type: ChangeDetectorRef }
111
+ ];
112
+ TimeRangePickerComponent.propDecorators = {
113
+ inputWidth: [{ type: Input }],
114
+ dropdownWidth: [{ type: Input }],
115
+ height: [{ type: Input }],
116
+ timezone: [{ type: Input }],
117
+ startDateEpoch: [{ type: Input }],
118
+ endDateEpoch: [{ type: Input }],
119
+ clockFormat: [{ type: Input }],
120
+ interval: [{ type: Input }],
121
+ timeRangeEmitter: [{ type: Output }]
122
+ };
123
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timerangepicker.component.js","sourceRoot":"","sources":["../../../../projects/mis-components/timerangepicker/timerangepicker.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, ChangeDetectorRef, Output, EventEmitter } from \"@angular/core\";\nimport * as moment from \"moment-timezone\";\nimport { ITimeRange, ITime } from \"./timerange.namespace\";\n\n@Component({\n  selector: \"mis-timerangepicker\",\n  templateUrl: \"./timerangepicker.component.html\",\n  styleUrls: [\"./timerangepicker.component.scss\"]\n})\nexport class TimeRangePickerComponent {\n  @Input() inputWidth :string = '100px';\n  @Input() dropdownWidth?: string;\n  @Input() height: string = '46px';\n  @Input() timezone: string = \"Asia/Kolkata\";\n  @Input() startDateEpoch: number = moment().tz(this.timezone).valueOf();\n  @Input() endDateEpoch: number = moment().tz(this.timezone).valueOf();\n  @Input() clockFormat: number = 12;\n  @Input() interval: number = 15;\n  @Output() timeRangeEmitter = new EventEmitter<ITimeRange>();\n\n  timeFormat!: string;\n  firstIntervalForStartPicker: number = moment().valueOf();\n  firstIntervalForEndPicker: number = moment().valueOf();\n  startTime!: ITime;\n  endTime!: ITime;\n  rangeValidity: boolean = true;\n\n  constructor(private cdr: ChangeDetectorRef) {}\n\n  ngOnInit() {\n    console.log(this.timezone);\n    this.timeFormat = this.clockFormat === 12 ? \"hh:mm a\" : \"HH:mm\";\n    this.setFirstIntervals();\n  }\n\n  ngOnChanges(){\n    moment.tz.setDefault(this.timezone);\n  }\n\n  // calculate the first interval of the picker\n  setFirstIntervals(): void {\n    const startDate = moment(this.startDateEpoch).format(\"DD-MM-YYYY\");\n    const endDate = moment(this.endDateEpoch).format(\"DD-MM-YYYY\");\n    const minutes = moment().minutes();\n    const offset = this.interval - (minutes % this.interval);\n\n    // first interval for the start picker will always be the one closest to the current time\n    this.firstIntervalForStartPicker = moment().add(offset, \"m\").valueOf();\n    this.startTime = {\n      valid: true,\n      time: moment().add(offset, \"m\").format(this.timeFormat)\n    };\n\n    // for the end picker if the startDate and the endDate is same\n    // the first interval is set one interval ahead of the first interval for start picker\n    // else if the dates are different we set it to the start of the day\n    if (startDate === endDate) {\n      this.firstIntervalForEndPicker = moment().add(offset, \"m\").add(this.interval, \"m\").valueOf();\n    } else {\n      this.firstIntervalForEndPicker = moment().startOf(\"d\").valueOf();\n    }\n\n    this.endTime = {\n      valid: true,\n      time: moment(this.firstIntervalForEndPicker).format(this.timeFormat)\n    }\n  }\n\n  emitTimeRange(data: ITimeRange): void {\n    this.timeRangeEmitter.emit(data);\n  }\n\n  //handlers catch the emitted values and run validation\n  startPickerHandler(time: ITime): void {\n    this.startTime = time;\n\n    // if the start time changes and the start date is the same as the end date\n    // and the start time >=  end time\n    // update the first interval of end picker according to the time set in start picker\n    const startDate = moment(this.startDateEpoch).format(\"DD-MM-YYYY\");\n    const endDate = moment(this.endDateEpoch).format(\"DD-MM-YYYY\");\n\n    if (startDate === endDate) {\n      let minutes = moment(this.startTime.time, this.timeFormat).minutes();\n      let offset = this.interval - (minutes % this.interval);\n      this.firstIntervalForEndPicker = moment(`${startDate} ${this.startTime.time}`, `'DD-MM-YYYY' ${this.timeFormat}`).add(offset, \"m\").valueOf();\n    }\n  }\n\n  endPickerHandler(time: ITime): void {\n    this.endTime = time;\n\n    const validity = this.checkTimeValidity(this.startTime.time.trim()) && this.checkTimeValidity(this.endTime.time.trim());\n    this.emitTimeRange({\n      valid: validity && this.rangeValidity,\n      startTime: this.startTime.time,\n      endTime: this.endTime.time\n    });\n\n    this.rangeValidation();\n  }\n\n  checkTimeValidity(time: string): boolean {\n    const RE12 = /^(([0-9][1-9]|1[0-2]):([0-5][0-9])( )?(am|pm|AM|PM))$/i;\n    const RE24 = /^([01][0-9]|2[0-3]):[0-5][0-9]$/;\n    const RE = this.clockFormat === 12 ? RE12 : RE24;\n    const flag = time.match(RE) ? true : false;\n    return flag;\n  }\n\n  // validates end picker's input according to the start picker's input\n  rangeValidation() {\n    const startDate = moment(this.startDateEpoch).format(\"DD-MM-YYYY\");\n    const endDate = moment(this.endDateEpoch).format(\"DD-MM-YYYY\");\n    const validity = this.checkTimeValidity(this.startTime.time.trim()) && this.checkTimeValidity(this.endTime.time.trim());\n    if (validity && startDate === endDate) {\n      if (this.endTime.time === moment().endOf(\"d\").add(1, \"m\").format(this.timeFormat)) {\n        this.endTime.time = moment().endOf(\"d\").format(this.timeFormat);\n      }\n\n      const startMoment = moment(this.startTime.time, this.timeFormat);\n      const endMoment = moment(this.endTime.time, this.timeFormat);\n      this.rangeValidity = endMoment.diff(startMoment, \"m\") >= 1 ? true : false;\n      this.cdr.detectChanges();\n    }\n  }\n}\n"]}
@@ -0,0 +1,18 @@
1
+ import { NgModule } from "@angular/core";
2
+ import { CommonModule } from '@angular/common';
3
+ import { FormsModule } from '@angular/forms';
4
+ import { TimeRangePickerComponent } from "./timerangepicker.component";
5
+ import { TimePickerModule } from "mis-crystal-design-system/timepicker";
6
+ export class TimeRangePickerModule {
7
+ static forRoot() {
8
+ return { ngModule: TimeRangePickerModule, providers: [] };
9
+ }
10
+ }
11
+ TimeRangePickerModule.decorators = [
12
+ { type: NgModule, args: [{
13
+ declarations: [TimeRangePickerComponent],
14
+ imports: [CommonModule, FormsModule, TimePickerModule],
15
+ exports: [TimeRangePickerComponent]
16
+ },] }
17
+ ];
18
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGltZXJhbmdlcGlja2VyLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL21pcy1jb21wb25lbnRzL3RpbWVyYW5nZXBpY2tlci90aW1lcmFuZ2VwaWNrZXIubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQXVCLE1BQU0sZUFBZSxDQUFDO0FBQzlELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsV0FBVyxFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDN0MsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDdkUsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sc0NBQXNDLENBQUM7QUFPeEUsTUFBTSxPQUFPLHFCQUFxQjtJQUNoQyxNQUFNLENBQUMsT0FBTztRQUNaLE9BQU8sRUFBRSxRQUFRLEVBQUUscUJBQXFCLEVBQUUsU0FBUyxFQUFFLEVBQUUsRUFBRSxDQUFDO0lBQzVELENBQUM7OztZQVJGLFFBQVEsU0FBQztnQkFDUixZQUFZLEVBQUUsQ0FBQyx3QkFBd0IsQ0FBQztnQkFDeEMsT0FBTyxFQUFFLENBQUMsWUFBWSxFQUFFLFdBQVcsRUFBRSxnQkFBZ0IsQ0FBQztnQkFDdEQsT0FBTyxFQUFFLENBQUMsd0JBQXdCLENBQUM7YUFDcEMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSwgTW9kdWxlV2l0aFByb3ZpZGVycyB9IGZyb20gXCJAYW5ndWxhci9jb3JlXCI7XG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBUaW1lUmFuZ2VQaWNrZXJDb21wb25lbnQgfSBmcm9tIFwiLi90aW1lcmFuZ2VwaWNrZXIuY29tcG9uZW50XCI7XG5pbXBvcnQgeyBUaW1lUGlja2VyTW9kdWxlIH0gZnJvbSBcIm1pcy1jcnlzdGFsLWRlc2lnbi1zeXN0ZW0vdGltZXBpY2tlclwiO1xuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtUaW1lUmFuZ2VQaWNrZXJDb21wb25lbnRdLFxuICBpbXBvcnRzOiBbQ29tbW9uTW9kdWxlLCBGb3Jtc01vZHVsZSwgVGltZVBpY2tlck1vZHVsZV0sXG4gIGV4cG9ydHM6IFtUaW1lUmFuZ2VQaWNrZXJDb21wb25lbnRdXG59KVxuZXhwb3J0IGNsYXNzIFRpbWVSYW5nZVBpY2tlck1vZHVsZSB7XG4gIHN0YXRpYyBmb3JSb290KCk6IE1vZHVsZVdpdGhQcm92aWRlcnM8VGltZVJhbmdlUGlja2VyTW9kdWxlPiB7XG4gICAgcmV0dXJuIHsgbmdNb2R1bGU6IFRpbWVSYW5nZVBpY2tlck1vZHVsZSwgcHJvdmlkZXJzOiBbXSB9O1xuICB9XG59Il19