@energycap/components 0.42.0 → 0.42.2-ECAP-26676-readonly-attribute-suppresses-help-popovers-on-checkboxes.20250122-1358

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 (41) hide show
  1. package/esm2022/lib/components.module.mjs +10 -5
  2. package/esm2022/lib/controls/calendar/calendar-item.component.mjs +46 -14
  3. package/esm2022/lib/controls/calendar/calendar.component.mjs +169 -121
  4. package/esm2022/lib/controls/calendar/calendar.types.mjs +2 -4
  5. package/esm2022/lib/controls/checkbox/checkbox.component.mjs +2 -2
  6. package/esm2022/lib/controls/date-input/date-input-selection-strategies/date-input-selection-strategy-base.mjs +57 -0
  7. package/esm2022/lib/controls/date-input/date-input-selection-strategies/day-selection-strategy.mjs +62 -0
  8. package/esm2022/lib/controls/date-input/date-input-selection-strategies/last-28-days-selection-strategy.mjs +100 -0
  9. package/esm2022/lib/controls/date-input/date-input-selection-strategies/last-7-days-selection-strategy.mjs +101 -0
  10. package/esm2022/lib/controls/date-input/date-input-selection-strategies/month-selection-strategy.mjs +76 -0
  11. package/esm2022/lib/controls/date-input/date-input-selection-strategies/quarter-selection-strategy.mjs +79 -0
  12. package/esm2022/lib/controls/date-input/date-input-selection-strategies/range-selection-strategy.mjs +210 -0
  13. package/esm2022/lib/controls/date-input/date-input-selection-strategies/year-selection-strategy.mjs +81 -0
  14. package/esm2022/lib/controls/date-input/date-input.component.mjs +322 -113
  15. package/esm2022/lib/controls/date-input/date-input.types.mjs +44 -0
  16. package/esm2022/lib/controls/file-upload/file-upload.component.mjs +1 -1
  17. package/esm2022/lib/controls/form-control/form-control.component.mjs +6 -12
  18. package/esm2022/lib/core/date-time-helper.mjs +10 -2
  19. package/esm2022/lib/shared/directives/keyboard-nav-container/keyboard-nav-container.directive.mjs +100 -0
  20. package/esm2022/public-api.mjs +63 -61
  21. package/fesm2022/energycap-components.mjs +1668 -509
  22. package/fesm2022/energycap-components.mjs.map +1 -1
  23. package/lib/components.module.d.ts +9 -8
  24. package/lib/controls/calendar/calendar-item.component.d.ts +11 -6
  25. package/lib/controls/calendar/calendar.component.d.ts +21 -23
  26. package/lib/controls/calendar/calendar.types.d.ts +11 -7
  27. package/lib/controls/date-input/date-input-selection-strategies/date-input-selection-strategy-base.d.ts +42 -0
  28. package/lib/controls/date-input/date-input-selection-strategies/day-selection-strategy.d.ts +21 -0
  29. package/lib/controls/date-input/date-input-selection-strategies/last-28-days-selection-strategy.d.ts +21 -0
  30. package/lib/controls/date-input/date-input-selection-strategies/last-7-days-selection-strategy.d.ts +21 -0
  31. package/lib/controls/date-input/date-input-selection-strategies/month-selection-strategy.d.ts +18 -0
  32. package/lib/controls/date-input/date-input-selection-strategies/quarter-selection-strategy.d.ts +18 -0
  33. package/lib/controls/date-input/date-input-selection-strategies/range-selection-strategy.d.ts +21 -0
  34. package/lib/controls/date-input/date-input-selection-strategies/year-selection-strategy.d.ts +20 -0
  35. package/lib/controls/date-input/date-input.component.d.ts +63 -28
  36. package/lib/controls/date-input/date-input.types.d.ts +62 -0
  37. package/lib/controls/form-control/form-control.component.d.ts +4 -6
  38. package/lib/shared/directives/keyboard-nav-container/keyboard-nav-container.directive.d.ts +23 -0
  39. package/package.json +1 -1
  40. package/public-api.d.ts +62 -60
  41. package/src/assets/locales/en_US.json +9 -1
@@ -122,11 +122,11 @@ export class CheckboxComponent extends FormControlBase {
122
122
  }
123
123
  }
124
124
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CheckboxComponent, deps: [{ token: i1.ValidationMessageService }, { token: i2.FormGroupHelper }], target: i0.ɵɵFactoryTarget.Component }); }
125
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: CheckboxComponent, selector: "ec-checkbox", inputs: { name: "name", dependentCheckboxesGroup: "dependentCheckboxesGroup", ignoreDisabledDependents: "ignoreDisabledDependents" }, viewQueries: [{ propertyName: "inputElement", first: true, predicate: ["checkboxInput"], descendants: true, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div class=\"control\">\r\n <label class=\"checkbox\"\r\n [ngClass]=\"{'is-disabled': formModel.disabled, 'no-label': !label, 'is-readonly': readonly}\">\r\n <input id=\"{{id}}_input\"\r\n #checkboxInput\r\n class='input'\r\n [class.indeterminate]=\"indeterminate\"\r\n [attr.id]=\"inputId\"\r\n [attr.name]=\"name\"\r\n type=\"checkbox\"\r\n tabindex=\"{{tabindex}}\"\r\n [formControl]=\"formModel\"\r\n [attr.cdkFocusInitial]=\"autofocus || null\">\r\n <i class=\"ec-icon icon-check\"></i>\r\n\r\n <span id=\"{{id}}_label\"\r\n *ngIf=\"label\"\r\n class=\"label\">\r\n <span [innerHtml]=\"label | translate\"></span>\r\n <ec-help-popover id=\"{{id}}_helpPopover\"\r\n *ngIf=\"helpPopover\"\r\n class=\"d-inline-block my-n2 mx-n1\"\r\n text=\"{{helpPopover | translate}}\"\r\n contentPosition=\"{{helpPopoverPosition}}\">\r\n </ec-help-popover>\r\n </span>\r\n </label>\r\n</div>", styles: [":host{color:var(--ec-form-control-color);font-size:var(--ec-form-control-font-size);display:block;margin-bottom:1rem;width:100%}:host :host-context(.form-condensed){margin-bottom:.5rem}:host .control{width:100%;display:flex;flex-direction:column}:host .control.control-label-bottom{flex-direction:column-reverse}:host .control.control-label-left{flex-direction:row}:host .control.control-label-left label{margin-right:.25rem}:host .control.control-label-right{flex-direction:row-reverse}:host .control.control-label-right label{margin-left:.25rem}:host .control.control-label-left,:host .control.control-label-right{align-items:center}:host .control.control-label-left label,:host .control.control-label-right label{flex:1 1;margin-top:0;margin-bottom:0}:host .control.control-label-left .control-input,:host .control.control-label-right .control-input{flex:2 2}:host .control.is-readonly input,:host .control.is-readonly select,:host .control.is-readonly textarea{border-color:var(--ec-form-control-border-color-readonly);background-color:var(--ec-form-control-background-color-readonly);background-clip:border-box;background-image:none;color:var(--ec-form-control-color-readonly);opacity:1;-webkit-user-select:none;user-select:none;pointer-events:none;overflow:hidden;white-space:nowrap}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid{background-color:var(--ec-form-control-background-color-invalid);border-color:var(--ec-form-control-border-color-invalid);background-repeat:no-repeat;background-position:.5rem center;background-size:1rem,1rem;padding-left:1.75rem;background-image:none}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid:focus,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid:focus{border-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-invalid{display:inline-flex;position:absolute;left:.5rem;top:.5rem;z-index:1}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-required,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-required{display:none}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button{background-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button:not(:focus){border-color:var(--ec-form-control-border-color-invalid)}:host .textbox-group{display:flex;position:relative}:host textarea:focus,:host input:focus,:host select:focus{outline:none}:host(.w-auto){width:auto}.checkbox{cursor:pointer;display:inline-flex;margin-bottom:0;position:relative}.checkbox.is-disabled{cursor:default}.input{margin-top:.5rem;opacity:0;position:absolute;z-index:-1}.input:not(:checked)+.ec-icon{color:var(--ec-form-control-border-color)}.input:not(:checked)+.ec-icon:before{display:none}.input.indeterminate+.ec-icon,.input:indeterminate+.ec-icon{color:var(--ec-color-interactive)}.input.indeterminate+.ec-icon:before,.input:indeterminate+.ec-icon:before{content:\"\";background-color:currentColor;display:block;width:10px;height:3px}.input:focus+.ec-icon{color:var(--ec-color-interactive);box-shadow:var(--ec-form-control-box-shadow-focus);border-color:var(--ec-form-control-border-color-focus)}.input:disabled+.ec-icon{color:var(--ec-form-control-color-disabled);background-color:var(--ec-form-control-background-color-disabled);border-color:var(--ec-form-control-border-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.input:disabled~.label{color:var(--ec-form-control-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.ec-icon{background-color:var(--ec-form-control-background-color);background-clip:padding-box;border:1px solid currentColor;color:var(--ec-color-interactive);margin-top:.5rem;flex:none;pointer-events:none;display:inline-flex;align-items:center;justify-content:center;height:1em;width:1em;border-radius:.125rem}.ec-icon:before{font-size:.6875em}.label{line-height:1.25rem;padding:.375rem 0;margin-left:.5rem;min-height:2rem;height:auto}.checkbox.is-readonly{pointer-events:none}.checkbox.is-readonly .input{opacity:0}.checkbox.is-readonly .ec-icon{background-color:var(--ec-form-control-background-color-readonly);border-color:var(--ec-form-control-border-color-readonly)}.checkbox.is-readonly .label,.checkbox.is-readonly .ec-icon{opacity:1;color:var(--ec-form-control-color)}.no-label .input,.no-label .icon-check{margin-top:0}\n"], dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: i5.HelpPopoverComponent, selector: "ec-help-popover", inputs: ["id", "text", "contentPosition", "maxWidth"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] }); }
125
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: CheckboxComponent, selector: "ec-checkbox", inputs: { name: "name", dependentCheckboxesGroup: "dependentCheckboxesGroup", ignoreDisabledDependents: "ignoreDisabledDependents" }, viewQueries: [{ propertyName: "inputElement", first: true, predicate: ["checkboxInput"], descendants: true, static: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: "<div class=\"control\">\r\n <label class=\"checkbox\"\r\n [ngClass]=\"{'is-disabled': formModel.disabled, 'no-label': !label, 'is-readonly': readonly}\">\r\n <input id=\"{{id}}_input\"\r\n #checkboxInput\r\n class='input'\r\n [class.indeterminate]=\"indeterminate\"\r\n [attr.id]=\"inputId\"\r\n [attr.name]=\"name\"\r\n type=\"checkbox\"\r\n tabindex=\"{{tabindex}}\"\r\n [formControl]=\"formModel\"\r\n [attr.cdkFocusInitial]=\"autofocus || null\">\r\n <i class=\"ec-icon icon-check\"></i>\r\n\r\n <span id=\"{{id}}_label\"\r\n *ngIf=\"label\"\r\n class=\"label\">\r\n <span [innerHtml]=\"label | translate\"></span>\r\n <ec-help-popover id=\"{{id}}_helpPopover\"\r\n *ngIf=\"helpPopover\"\r\n class=\"d-inline-block my-n2 mx-n1\"\r\n text=\"{{helpPopover | translate}}\"\r\n contentPosition=\"{{helpPopoverPosition}}\">\r\n </ec-help-popover>\r\n </span>\r\n </label>\r\n</div>", styles: [":host{color:var(--ec-form-control-color);font-size:var(--ec-form-control-font-size);display:block;margin-bottom:1rem;width:100%}:host :host-context(.form-condensed){margin-bottom:.5rem}:host .control{width:100%;display:flex;flex-direction:column}:host .control.control-label-bottom{flex-direction:column-reverse}:host .control.control-label-left{flex-direction:row}:host .control.control-label-left label{margin-right:.25rem}:host .control.control-label-right{flex-direction:row-reverse}:host .control.control-label-right label{margin-left:.25rem}:host .control.control-label-left,:host .control.control-label-right{align-items:center}:host .control.control-label-left label,:host .control.control-label-right label{flex:1 1;margin-top:0;margin-bottom:0}:host .control.control-label-left .control-input,:host .control.control-label-right .control-input{flex:2 2}:host .control.is-readonly input,:host .control.is-readonly select,:host .control.is-readonly textarea{border-color:var(--ec-form-control-border-color-readonly);background-color:var(--ec-form-control-background-color-readonly);background-clip:border-box;background-image:none;color:var(--ec-form-control-color-readonly);opacity:1;-webkit-user-select:none;user-select:none;pointer-events:none;overflow:hidden;white-space:nowrap}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid{background-color:var(--ec-form-control-background-color-invalid);border-color:var(--ec-form-control-border-color-invalid);background-repeat:no-repeat;background-position:.5rem center;background-size:1rem,1rem;padding-left:1.75rem;background-image:none}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid:focus,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid:focus{border-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-invalid{display:inline-flex;position:absolute;left:.5rem;top:.5rem;z-index:1}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-required,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-required{display:none}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button{background-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button:not(:focus){border-color:var(--ec-form-control-border-color-invalid)}:host .textbox-group{display:flex;position:relative}:host textarea:focus,:host input:focus,:host select:focus{outline:none}:host(.w-auto){width:auto}.checkbox{cursor:pointer;display:inline-flex;margin-bottom:0;position:relative}.checkbox.is-disabled{cursor:default}.input{margin-top:.5rem;opacity:0;position:absolute;z-index:-1}.input:not(:checked)+.ec-icon{color:var(--ec-form-control-border-color)}.input:not(:checked)+.ec-icon:before{display:none}.input.indeterminate+.ec-icon,.input:indeterminate+.ec-icon{color:var(--ec-color-interactive)}.input.indeterminate+.ec-icon:before,.input:indeterminate+.ec-icon:before{content:\"\";background-color:currentColor;display:block;width:10px;height:3px}.input:focus+.ec-icon{color:var(--ec-color-interactive);box-shadow:var(--ec-form-control-box-shadow-focus);border-color:var(--ec-form-control-border-color-focus)}.input:disabled+.ec-icon{color:var(--ec-form-control-color-disabled);background-color:var(--ec-form-control-background-color-disabled);border-color:var(--ec-form-control-border-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.input:disabled~.label{color:var(--ec-form-control-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.ec-icon{background-color:var(--ec-form-control-background-color);background-clip:padding-box;border:1px solid currentColor;color:var(--ec-color-interactive);margin-top:.5rem;flex:none;pointer-events:none;display:inline-flex;align-items:center;justify-content:center;height:1em;width:1em;border-radius:.125rem}.ec-icon:before{font-size:.6875em}.label{line-height:1.25rem;padding:.375rem 0;margin-left:.5rem;min-height:2rem;height:auto}.checkbox.is-readonly{pointer-events:none}.checkbox.is-readonly .input{opacity:0}.checkbox.is-readonly .ec-icon{background-color:var(--ec-form-control-background-color-readonly);border-color:var(--ec-form-control-border-color-readonly)}.checkbox.is-readonly .label,.checkbox.is-readonly .ec-icon{opacity:1;color:var(--ec-form-control-color)}.checkbox.is-readonly ec-help-popover{pointer-events:auto}.no-label .input,.no-label .icon-check{margin-top:0}\n"], dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.CheckboxControlValueAccessor, selector: "input[type=checkbox][formControlName],input[type=checkbox][formControl],input[type=checkbox][ngModel]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: i5.HelpPopoverComponent, selector: "ec-help-popover", inputs: ["id", "text", "contentPosition", "maxWidth"] }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] }); }
126
126
  }
127
127
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: CheckboxComponent, decorators: [{
128
128
  type: Component,
129
- args: [{ selector: 'ec-checkbox', template: "<div class=\"control\">\r\n <label class=\"checkbox\"\r\n [ngClass]=\"{'is-disabled': formModel.disabled, 'no-label': !label, 'is-readonly': readonly}\">\r\n <input id=\"{{id}}_input\"\r\n #checkboxInput\r\n class='input'\r\n [class.indeterminate]=\"indeterminate\"\r\n [attr.id]=\"inputId\"\r\n [attr.name]=\"name\"\r\n type=\"checkbox\"\r\n tabindex=\"{{tabindex}}\"\r\n [formControl]=\"formModel\"\r\n [attr.cdkFocusInitial]=\"autofocus || null\">\r\n <i class=\"ec-icon icon-check\"></i>\r\n\r\n <span id=\"{{id}}_label\"\r\n *ngIf=\"label\"\r\n class=\"label\">\r\n <span [innerHtml]=\"label | translate\"></span>\r\n <ec-help-popover id=\"{{id}}_helpPopover\"\r\n *ngIf=\"helpPopover\"\r\n class=\"d-inline-block my-n2 mx-n1\"\r\n text=\"{{helpPopover | translate}}\"\r\n contentPosition=\"{{helpPopoverPosition}}\">\r\n </ec-help-popover>\r\n </span>\r\n </label>\r\n</div>", styles: [":host{color:var(--ec-form-control-color);font-size:var(--ec-form-control-font-size);display:block;margin-bottom:1rem;width:100%}:host :host-context(.form-condensed){margin-bottom:.5rem}:host .control{width:100%;display:flex;flex-direction:column}:host .control.control-label-bottom{flex-direction:column-reverse}:host .control.control-label-left{flex-direction:row}:host .control.control-label-left label{margin-right:.25rem}:host .control.control-label-right{flex-direction:row-reverse}:host .control.control-label-right label{margin-left:.25rem}:host .control.control-label-left,:host .control.control-label-right{align-items:center}:host .control.control-label-left label,:host .control.control-label-right label{flex:1 1;margin-top:0;margin-bottom:0}:host .control.control-label-left .control-input,:host .control.control-label-right .control-input{flex:2 2}:host .control.is-readonly input,:host .control.is-readonly select,:host .control.is-readonly textarea{border-color:var(--ec-form-control-border-color-readonly);background-color:var(--ec-form-control-background-color-readonly);background-clip:border-box;background-image:none;color:var(--ec-form-control-color-readonly);opacity:1;-webkit-user-select:none;user-select:none;pointer-events:none;overflow:hidden;white-space:nowrap}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid{background-color:var(--ec-form-control-background-color-invalid);border-color:var(--ec-form-control-border-color-invalid);background-repeat:no-repeat;background-position:.5rem center;background-size:1rem,1rem;padding-left:1.75rem;background-image:none}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid:focus,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid:focus{border-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-invalid{display:inline-flex;position:absolute;left:.5rem;top:.5rem;z-index:1}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-required,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-required{display:none}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button{background-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button:not(:focus){border-color:var(--ec-form-control-border-color-invalid)}:host .textbox-group{display:flex;position:relative}:host textarea:focus,:host input:focus,:host select:focus{outline:none}:host(.w-auto){width:auto}.checkbox{cursor:pointer;display:inline-flex;margin-bottom:0;position:relative}.checkbox.is-disabled{cursor:default}.input{margin-top:.5rem;opacity:0;position:absolute;z-index:-1}.input:not(:checked)+.ec-icon{color:var(--ec-form-control-border-color)}.input:not(:checked)+.ec-icon:before{display:none}.input.indeterminate+.ec-icon,.input:indeterminate+.ec-icon{color:var(--ec-color-interactive)}.input.indeterminate+.ec-icon:before,.input:indeterminate+.ec-icon:before{content:\"\";background-color:currentColor;display:block;width:10px;height:3px}.input:focus+.ec-icon{color:var(--ec-color-interactive);box-shadow:var(--ec-form-control-box-shadow-focus);border-color:var(--ec-form-control-border-color-focus)}.input:disabled+.ec-icon{color:var(--ec-form-control-color-disabled);background-color:var(--ec-form-control-background-color-disabled);border-color:var(--ec-form-control-border-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.input:disabled~.label{color:var(--ec-form-control-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.ec-icon{background-color:var(--ec-form-control-background-color);background-clip:padding-box;border:1px solid currentColor;color:var(--ec-color-interactive);margin-top:.5rem;flex:none;pointer-events:none;display:inline-flex;align-items:center;justify-content:center;height:1em;width:1em;border-radius:.125rem}.ec-icon:before{font-size:.6875em}.label{line-height:1.25rem;padding:.375rem 0;margin-left:.5rem;min-height:2rem;height:auto}.checkbox.is-readonly{pointer-events:none}.checkbox.is-readonly .input{opacity:0}.checkbox.is-readonly .ec-icon{background-color:var(--ec-form-control-background-color-readonly);border-color:var(--ec-form-control-border-color-readonly)}.checkbox.is-readonly .label,.checkbox.is-readonly .ec-icon{opacity:1;color:var(--ec-form-control-color)}.no-label .input,.no-label .icon-check{margin-top:0}\n"] }]
129
+ args: [{ selector: 'ec-checkbox', template: "<div class=\"control\">\r\n <label class=\"checkbox\"\r\n [ngClass]=\"{'is-disabled': formModel.disabled, 'no-label': !label, 'is-readonly': readonly}\">\r\n <input id=\"{{id}}_input\"\r\n #checkboxInput\r\n class='input'\r\n [class.indeterminate]=\"indeterminate\"\r\n [attr.id]=\"inputId\"\r\n [attr.name]=\"name\"\r\n type=\"checkbox\"\r\n tabindex=\"{{tabindex}}\"\r\n [formControl]=\"formModel\"\r\n [attr.cdkFocusInitial]=\"autofocus || null\">\r\n <i class=\"ec-icon icon-check\"></i>\r\n\r\n <span id=\"{{id}}_label\"\r\n *ngIf=\"label\"\r\n class=\"label\">\r\n <span [innerHtml]=\"label | translate\"></span>\r\n <ec-help-popover id=\"{{id}}_helpPopover\"\r\n *ngIf=\"helpPopover\"\r\n class=\"d-inline-block my-n2 mx-n1\"\r\n text=\"{{helpPopover | translate}}\"\r\n contentPosition=\"{{helpPopoverPosition}}\">\r\n </ec-help-popover>\r\n </span>\r\n </label>\r\n</div>", styles: [":host{color:var(--ec-form-control-color);font-size:var(--ec-form-control-font-size);display:block;margin-bottom:1rem;width:100%}:host :host-context(.form-condensed){margin-bottom:.5rem}:host .control{width:100%;display:flex;flex-direction:column}:host .control.control-label-bottom{flex-direction:column-reverse}:host .control.control-label-left{flex-direction:row}:host .control.control-label-left label{margin-right:.25rem}:host .control.control-label-right{flex-direction:row-reverse}:host .control.control-label-right label{margin-left:.25rem}:host .control.control-label-left,:host .control.control-label-right{align-items:center}:host .control.control-label-left label,:host .control.control-label-right label{flex:1 1;margin-top:0;margin-bottom:0}:host .control.control-label-left .control-input,:host .control.control-label-right .control-input{flex:2 2}:host .control.is-readonly input,:host .control.is-readonly select,:host .control.is-readonly textarea{border-color:var(--ec-form-control-border-color-readonly);background-color:var(--ec-form-control-background-color-readonly);background-clip:border-box;background-image:none;color:var(--ec-form-control-color-readonly);opacity:1;-webkit-user-select:none;user-select:none;pointer-events:none;overflow:hidden;white-space:nowrap}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid{background-color:var(--ec-form-control-background-color-invalid);border-color:var(--ec-form-control-border-color-invalid);background-repeat:no-repeat;background-position:.5rem center;background-size:1rem,1rem;padding-left:1.75rem;background-image:none}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid:focus,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid:focus{border-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-invalid,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-invalid{display:inline-flex;position:absolute;left:.5rem;top:.5rem;z-index:1}:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-invalid~.icon-required,:host .control.invalid .textbox-group-input ::ng-deep .control input.ng-valid~.icon-required{display:none}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button{background-color:var(--ec-form-control-background-color-invalid)}:host .control.invalid:not(.open) .textbox-group-btn-right ::ng-deep button:not(:focus){border-color:var(--ec-form-control-border-color-invalid)}:host .textbox-group{display:flex;position:relative}:host textarea:focus,:host input:focus,:host select:focus{outline:none}:host(.w-auto){width:auto}.checkbox{cursor:pointer;display:inline-flex;margin-bottom:0;position:relative}.checkbox.is-disabled{cursor:default}.input{margin-top:.5rem;opacity:0;position:absolute;z-index:-1}.input:not(:checked)+.ec-icon{color:var(--ec-form-control-border-color)}.input:not(:checked)+.ec-icon:before{display:none}.input.indeterminate+.ec-icon,.input:indeterminate+.ec-icon{color:var(--ec-color-interactive)}.input.indeterminate+.ec-icon:before,.input:indeterminate+.ec-icon:before{content:\"\";background-color:currentColor;display:block;width:10px;height:3px}.input:focus+.ec-icon{color:var(--ec-color-interactive);box-shadow:var(--ec-form-control-box-shadow-focus);border-color:var(--ec-form-control-border-color-focus)}.input:disabled+.ec-icon{color:var(--ec-form-control-color-disabled);background-color:var(--ec-form-control-background-color-disabled);border-color:var(--ec-form-control-border-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.input:disabled~.label{color:var(--ec-form-control-color-disabled);opacity:var(--ec-form-control-opacity-disabled)}.ec-icon{background-color:var(--ec-form-control-background-color);background-clip:padding-box;border:1px solid currentColor;color:var(--ec-color-interactive);margin-top:.5rem;flex:none;pointer-events:none;display:inline-flex;align-items:center;justify-content:center;height:1em;width:1em;border-radius:.125rem}.ec-icon:before{font-size:.6875em}.label{line-height:1.25rem;padding:.375rem 0;margin-left:.5rem;min-height:2rem;height:auto}.checkbox.is-readonly{pointer-events:none}.checkbox.is-readonly .input{opacity:0}.checkbox.is-readonly .ec-icon{background-color:var(--ec-form-control-background-color-readonly);border-color:var(--ec-form-control-border-color-readonly)}.checkbox.is-readonly .label,.checkbox.is-readonly .ec-icon{opacity:1;color:var(--ec-form-control-color)}.checkbox.is-readonly ec-help-popover{pointer-events:auto}.no-label .input,.no-label .icon-check{margin-top:0}\n"] }]
130
130
  }], ctorParameters: () => [{ type: i1.ValidationMessageService }, { type: i2.FormGroupHelper }], propDecorators: { name: [{
131
131
  type: Input
132
132
  }], dependentCheckboxesGroup: [{
@@ -0,0 +1,57 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ /**
4
+ * Contains the logic for selecting, formatting, and parsing date dates for a specific selection mode.
5
+ */
6
+ export class DateInputSelectionStrategyBase {
7
+ /** Parses the string into a date using the provided parse formats. */
8
+ parseString(value, parseFormats, opts) {
9
+ if (!value) {
10
+ return null;
11
+ }
12
+ const date = moment(value, parseFormats);
13
+ if (date.isValid()) {
14
+ // If we're in a temporary state (when the user is typing in a textbox), we need to set the year to the current year if it's before the min year
15
+ // This prevents the calendars from showing unhelpful dates that are in the distant past before the user finishes typing in the date.
16
+ // We don't do this modification when the real selection happens on textbox blur.
17
+ if (opts?.shiftToCurrentYearIfBelow && date.isBefore(opts.shiftToCurrentYearIfBelow)) {
18
+ date.set('year', moment().year());
19
+ }
20
+ return date.toDate();
21
+ }
22
+ return null;
23
+ }
24
+ /** Returns a new selection for the provided date. */
25
+ getSelectionForQuickSelectDate(date, existingSelection) {
26
+ return this.getSelectionFromDate(date, existingSelection);
27
+ }
28
+ /** Returns the view for the primary calendar for the selection and selection mode */
29
+ getPrimaryCalendarView(selection, minDate, maxDate, currentView) {
30
+ let date;
31
+ // If we have a range, use the end date, defaulting to the start date or today
32
+ if (DateInput.isSelectionRange(selection)) {
33
+ date = selection.end || selection.start || new Date();
34
+ // If we have a single date, use it, defaulting to today
35
+ }
36
+ else {
37
+ date = selection || new Date();
38
+ }
39
+ // Ensure the date is within the min and max dates
40
+ date = date < minDate ? minDate : date;
41
+ date = date > maxDate ? maxDate : date;
42
+ // If the current calendar view mode is valid for the selection mode, use it, otherwise use the selection view mode
43
+ // This prevents the calendar from changing view modes as the user types/blurs the textboxes as long as the view mode is valid
44
+ const viewMode = this.validViewModes.includes(currentView.mode) ? currentView.mode : this.selectionViewMode;
45
+ return { mode: viewMode, date };
46
+ }
47
+ ;
48
+ /**
49
+ * Returns the view for the secondary calendar for the selection and selection mode.
50
+ * This is is only used by the range selection strategy. All others just return the current date.
51
+ */
52
+ getSecondaryCalendarView(selection, minDate, maxDate, currentView) {
53
+ return { mode: this.selectionViewMode, date: new Date() };
54
+ }
55
+ ;
56
+ }
57
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"date-input-selection-strategy-base.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/controls/date-input/date-input-selection-strategies/date-input-selection-strategy-base.ts"],"names":[],"mappings":"AAAA,OAAO,MAAM,MAAM,QAAQ,CAAC;AAE5B,OAAO,EAAE,SAAS,EAAE,MAAM,qBAAqB,CAAC;AAEhD;;GAEG;AACH,MAAM,OAAgB,8BAA8B;IASlD,sEAAsE;IACtE,WAAW,CAAC,KAAgC,EAAE,YAAsB,EAAE,IAA+B;QACnG,IAAI,CAAC,KAAK,EAAE,CAAC;YACX,OAAO,IAAI,CAAC;QACd,CAAC;QACD,MAAM,IAAI,GAAG,MAAM,CAAC,KAAK,EAAE,YAAY,CAAC,CAAC;QAEzC,IAAI,IAAI,CAAC,OAAO,EAAE,EAAE,CAAC;YACnB,gJAAgJ;YAChJ,qIAAqI;YACrI,iFAAiF;YACjF,IAAI,IAAI,EAAE,yBAAyB,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,yBAAyB,CAAC,EAAE,CAAC;gBACrF,IAAI,CAAC,GAAG,CAAC,MAAM,EAAE,MAAM,EAAE,CAAC,IAAI,EAAE,CAAC,CAAC;YACpC,CAAC;YAED,OAAO,IAAI,CAAC,MAAM,EAAE,CAAC;QACvB,CAAC;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAaD,qDAAqD;IACrD,8BAA8B,CAAC,IAAU,EAAE,iBAA8C;QACvF,OAAO,IAAI,CAAC,oBAAoB,CAAC,IAAI,EAAE,iBAAiB,CAAC,CAAC;IAC5D,CAAC;IAMD,qFAAqF;IACrF,sBAAsB,CAAC,SAAqC,EAAE,OAAa,EAAE,OAAa,EAAE,WAA0B;QACpH,IAAI,IAAU,CAAC;QAEf,8EAA8E;QAC9E,IAAI,SAAS,CAAC,gBAAgB,CAAC,SAAS,CAAC,EAAE,CAAC;YAC1C,IAAI,GAAG,SAAS,CAAC,GAAG,IAAI,SAAS,CAAC,KAAK,IAAI,IAAI,IAAI,EAAE,CAAC;YACxD,wDAAwD;QACxD,CAAC;aAAM,CAAC;YACN,IAAI,GAAG,SAAS,IAAI,IAAI,IAAI,EAAE,CAAC;QACjC,CAAC;QAED,kDAAkD;QAClD,IAAI,GAAG,IAAI,GAAG,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC;QACvC,IAAI,GAAG,IAAI,GAAG,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC;QAEvC,mHAAmH;QACnH,8HAA8H;QAC9H,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;QAE5G,OAAO,EAAE,IAAI,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC;IAClC,CAAC;IAAA,CAAC;IAEF;;;OAGG;IACH,wBAAwB,CAAC,SAAqC,EAAE,OAAa,EAAE,OAAa,EAAE,WAA0B;QACtH,OAAO,EAAE,IAAI,EAAE,IAAI,CAAC,iBAAiB,EAAE,IAAI,EAAE,IAAI,IAAI,EAAE,EAAE,CAAA;IAC3D,CAAC;IAAA,CAAC;CACH","sourcesContent":["import moment from \"moment\";\r\nimport { Calendar } from \"../../calendar/calendar.types\";\r\nimport { DateInput } from \"../date-input.types\";\r\n\r\n/**\r\n * Contains the logic for selecting, formatting, and parsing date dates for a specific selection mode.\r\n */\r\nexport abstract class DateInputSelectionStrategyBase {\r\n  /** The calendar view mode where selections occur for this mode. This is used to zoom the calendar into the selection */\r\n  abstract readonly selectionViewMode: Calendar.ViewMode;\r\n  /** The list of view modes that are accessible by this selection mode. Used to determine whether the calendar view needs to be changed or not on selection change. */\r\n  abstract readonly validViewModes: Calendar.ViewMode[];\r\n  /** Whether the secondary textbox should be shown for this selection mode */\r\n  abstract readonly showSecondaryTextbox: boolean;\r\n  /** Whether the secondary calendar should be shown for this selection mode */\r\n  abstract readonly showSecondaryCalendar: boolean;\r\n  /** Parses the string into a date using the provided parse formats. */\r\n  parseString(value: string | null | undefined, parseFormats: string[], opts?: DateInput.ParsingOptions): Date | null {\r\n    if (!value) {\r\n      return null;\r\n    }\r\n    const date = moment(value, parseFormats);\r\n\r\n    if (date.isValid()) {\r\n      // If we're in a temporary state (when the user is typing in a textbox), we need to set the year to the current year if it's before the min year\r\n      // This prevents the calendars from showing unhelpful dates that are in the distant past before the user finishes typing in the date.\r\n      // We don't do this modification when the real selection happens on textbox blur.\r\n      if (opts?.shiftToCurrentYearIfBelow && date.isBefore(opts.shiftToCurrentYearIfBelow)) {\r\n        date.set('year', moment().year());\r\n      }\r\n\r\n      return date.toDate();\r\n    }\r\n\r\n    return null;\r\n  }\r\n  /** Formats a date input selection into values to be displayed in the date input's textboxes. */\r\n  abstract formatSelection(selection?: DateInput.Selection | null): DateInput.TextboxGroup['value'];\r\n  /** \r\n   * Parses the date input's values into a date selection to display in the calendars.\r\n   * `parseFromEnd` controls whether the fixed range modes (last7days, last28days) will key off the end date instead of the start date.\r\n   * This is used when the user is typing in the end date textbox.\r\n   */\r\n  abstract parseTextboxValues(value: DateInput.TextboxGroup['value'], parseFormats: string[], opts?: DateInput.ParsingOptions): DateInput.Selection | null;\r\n  /** Returns a selection given a date and an existing selection (in the case of date ranges). Called when a date is clicked in the calendars. */\r\n  abstract getSelectionFromDate(date: Date, existingSelection?: DateInput.Selection | null): DateInput.Selection | null;\r\n  /** Returns a new selection that best matches the given previous selection from a different selection mode */\r\n  abstract getNewSelectionFromExisting(previousSelection?: DateInput.Selection | null): DateInput.Selection | null;\r\n  /** Returns a new selection for the provided date. */\r\n  getSelectionForQuickSelectDate(date: Date, existingSelection?: DateInput.Selection | null): DateInput.Selection | null {\r\n    return this.getSelectionFromDate(date, existingSelection);\r\n  }\r\n  /** Returns the next selection. Called by the date input steppers to move the range. */\r\n  abstract getNextSelection(selection: DateInput.Selection | null): DateInput.Selection | null;\r\n  /** Returns the previous selection. Called by the date input steppers to move the range. */\r\n  abstract getPreviousSelection(selection: DateInput.Selection | null): DateInput.Selection | null;\r\n  \r\n  /** Returns the view for the primary calendar for the selection and selection mode */\r\n  getPrimaryCalendarView(selection: DateInput.Selection | null, minDate: Date, maxDate: Date, currentView: Calendar.View): Calendar.View {\r\n    let date: Date;\r\n\r\n    // If we have a range, use the end date, defaulting to the start date or today\r\n    if (DateInput.isSelectionRange(selection)) {\r\n      date = selection.end || selection.start || new Date();\r\n    // If we have a single date, use it, defaulting to today\r\n    } else {\r\n      date = selection || new Date();\r\n    }\r\n\r\n    // Ensure the date is within the min and max dates\r\n    date = date < minDate ? minDate : date;\r\n    date = date > maxDate ? maxDate : date;\r\n\r\n    // If the current calendar view mode is valid for the selection mode, use it, otherwise use the selection view mode\r\n    // This prevents the calendar from changing view modes as the user types/blurs the textboxes as long as the view mode is valid\r\n    const viewMode = this.validViewModes.includes(currentView.mode) ? currentView.mode : this.selectionViewMode;\r\n\r\n    return { mode: viewMode, date };\r\n  };\r\n  \r\n  /**\r\n   * Returns the view for the secondary calendar for the selection and selection mode.\r\n   * This is is only used by the range selection strategy. All others just return the current date.\r\n   */\r\n  getSecondaryCalendarView(selection: DateInput.Selection | null, minDate: Date, maxDate: Date, currentView: Calendar.View): Calendar.View {\r\n    return { mode: this.selectionViewMode, date: new Date() }\r\n  };\r\n}\r\n"]}
@@ -0,0 +1,62 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ import { DateInputSelectionStrategyBase } from "./date-input-selection-strategy-base";
4
+ /**
5
+ * Selection strategy for the 'day' selection mode.
6
+ */
7
+ export class DaySelectionStrategy extends DateInputSelectionStrategyBase {
8
+ constructor(dateDisplayPipe) {
9
+ super();
10
+ this.dateDisplayPipe = dateDisplayPipe;
11
+ this.selectionViewMode = 'day';
12
+ this.validViewModes = ['day', 'month', 'year'];
13
+ this.showSecondaryTextbox = false;
14
+ this.showSecondaryCalendar = false;
15
+ }
16
+ formatSelection(selection) {
17
+ // If the selection is a single date, format it into the first textbox.
18
+ if (DateInput.isSelectionSingleDate(selection)) {
19
+ return { textbox: this.dateDisplayPipe.transform(selection, true), textbox2: null };
20
+ // Otherwise, clear both textboxes. This shouldn't happen based on current date input flows,
21
+ // and even if it did there's not really a good way to handle it.
22
+ }
23
+ else {
24
+ return { textbox: null, textbox2: null };
25
+ }
26
+ }
27
+ parseTextboxValues(value, parseFormats, opts) {
28
+ return this.parseString(value.textbox, parseFormats, opts);
29
+ }
30
+ getSelectionFromDate(date) {
31
+ // Nice and simple
32
+ return date;
33
+ }
34
+ getNewSelectionFromExisting(previousSelection) {
35
+ // If the previous selection was a range, return the end date and default to the start date if there isn't one.
36
+ if (DateInput.isSelectionRange(previousSelection)) {
37
+ return previousSelection.end || previousSelection.start;
38
+ // If the previous selection was a single date, return it.
39
+ }
40
+ else if (DateInput.isSelectionSingleDate(previousSelection)) {
41
+ return previousSelection;
42
+ }
43
+ return null;
44
+ }
45
+ getNextSelection(selection) {
46
+ // Add a single day to the selection
47
+ if (DateInput.isSelectionSingleDate(selection)) {
48
+ return moment(selection).add(1, 'day').toDate();
49
+ }
50
+ // This shouldn't happen based on current date input flows, but if it does just return the selection as-is.
51
+ return selection;
52
+ }
53
+ getPreviousSelection(selection) {
54
+ // Subtract a single day from the selection
55
+ if (DateInput.isSelectionSingleDate(selection)) {
56
+ return moment(selection).subtract(1, 'day').toDate();
57
+ }
58
+ // This shouldn't happen based on current date input flows, but if it does just return the selection as-is.
59
+ return selection;
60
+ }
61
+ }
62
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF5LXNlbGVjdGlvbi1zdHJhdGVneS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NvbXBvbmVudHMvc3JjL2xpYi9jb250cm9scy9kYXRlLWlucHV0L2RhdGUtaW5wdXQtc2VsZWN0aW9uLXN0cmF0ZWdpZXMvZGF5LXNlbGVjdGlvbi1zdHJhdGVneS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLE1BQU0sTUFBTSxRQUFRLENBQUM7QUFHNUIsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ2hELE9BQU8sRUFBRSw4QkFBOEIsRUFBRSxNQUFNLHNDQUFzQyxDQUFDO0FBRXRGOztHQUVHO0FBQ0gsTUFBTSxPQUFPLG9CQUFxQixTQUFRLDhCQUE4QjtJQU10RSxZQUNVLGVBQWdDO1FBRXhDLEtBQUssRUFBRSxDQUFDO1FBRkEsb0JBQWUsR0FBZixlQUFlLENBQWlCO1FBTjFCLHNCQUFpQixHQUFzQixLQUFLLENBQUM7UUFDN0MsbUJBQWMsR0FBd0IsQ0FBQyxLQUFLLEVBQUUsT0FBTyxFQUFFLE1BQU0sQ0FBQyxDQUFDO1FBQy9ELHlCQUFvQixHQUFHLEtBQUssQ0FBQztRQUM3QiwwQkFBcUIsR0FBRyxLQUFLLENBQUM7SUFNOUMsQ0FBQztJQUVELGVBQWUsQ0FBQyxTQUFzQztRQUNwRCx1RUFBdUU7UUFDdkUsSUFBSSxTQUFTLENBQUMscUJBQXFCLENBQUMsU0FBUyxDQUFDLEVBQUUsQ0FBQztZQUMvQyxPQUFPLEVBQUUsT0FBTyxFQUFFLElBQUksQ0FBQyxlQUFlLENBQUMsU0FBUyxDQUFDLFNBQVMsRUFBRSxJQUFJLENBQUMsRUFBRSxRQUFRLEVBQUUsSUFBSSxFQUFFLENBQUM7WUFDdEYsNEZBQTRGO1lBQzVGLGlFQUFpRTtRQUNqRSxDQUFDO2FBQU0sQ0FBQztZQUNOLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUUsQ0FBQztRQUMzQyxDQUFDO0lBQ0gsQ0FBQztJQUVELGtCQUFrQixDQUFDLEtBQXNDLEVBQUUsWUFBc0IsRUFBRSxJQUErQjtRQUNoSCxPQUFPLElBQUksQ0FBQyxXQUFXLENBQUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxZQUFZLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDN0QsQ0FBQztJQUVELG9CQUFvQixDQUFDLElBQVU7UUFDN0Isa0JBQWtCO1FBQ2xCLE9BQU8sSUFBSSxDQUFDO0lBQ2QsQ0FBQztJQUVELDJCQUEyQixDQUFDLGlCQUE4QztRQUN4RSwrR0FBK0c7UUFDL0csSUFBSSxTQUFTLENBQUMsZ0JBQWdCLENBQUMsaUJBQWlCLENBQUMsRUFBRSxDQUFDO1lBQ2xELE9BQU8saUJBQWlCLENBQUMsR0FBRyxJQUFJLGlCQUFpQixDQUFDLEtBQUssQ0FBQztZQUMxRCwwREFBMEQ7UUFDMUQsQ0FBQzthQUFNLElBQUksU0FBUyxDQUFDLHFCQUFxQixDQUFDLGlCQUFpQixDQUFDLEVBQUUsQ0FBQztZQUM5RCxPQUFPLGlCQUFpQixDQUFDO1FBQzNCLENBQUM7UUFFRCxPQUFPLElBQUksQ0FBQztJQUNkLENBQUM7SUFFRCxnQkFBZ0IsQ0FBQyxTQUFxQztRQUNwRCxvQ0FBb0M7UUFDcEMsSUFBSSxTQUFTLENBQUMscUJBQXFCLENBQUMsU0FBUyxDQUFDLEVBQUUsQ0FBQztZQUMvQyxPQUFPLE1BQU0sQ0FBQyxTQUFTLENBQUMsQ0FBQyxHQUFHLENBQUMsQ0FBQyxFQUFFLEtBQUssQ0FBQyxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBQ2xELENBQUM7UUFFRCwyR0FBMkc7UUFDM0csT0FBTyxTQUFTLENBQUM7SUFDbkIsQ0FBQztJQUVELG9CQUFvQixDQUFDLFNBQXFDO1FBQ3hELDJDQUEyQztRQUMzQyxJQUFJLFNBQVMsQ0FBQyxxQkFBcUIsQ0FBQyxTQUFTLENBQUMsRUFBRSxDQUFDO1lBQy9DLE9BQU8sTUFBTSxDQUFDLFNBQVMsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxDQUFDLEVBQUUsS0FBSyxDQUFDLENBQUMsTUFBTSxFQUFFLENBQUM7UUFDdkQsQ0FBQztRQUVELDJHQUEyRztRQUMzRyxPQUFPLFNBQVMsQ0FBQztJQUNuQixDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgbW9tZW50IGZyb20gXCJtb21lbnRcIjtcclxuaW1wb3J0IHsgRGF0ZURpc3BsYXlQaXBlIH0gZnJvbSBcIi4uLy4uLy4uL3NoYXJlZC9kaXNwbGF5L3BpcGVzL2RhdGUtZGlzcGxheS5waXBlXCI7XHJcbmltcG9ydCB7IENhbGVuZGFyIH0gZnJvbSBcIi4uLy4uL2NhbGVuZGFyL2NhbGVuZGFyLnR5cGVzXCI7XHJcbmltcG9ydCB7IERhdGVJbnB1dCB9IGZyb20gXCIuLi9kYXRlLWlucHV0LnR5cGVzXCI7XHJcbmltcG9ydCB7IERhdGVJbnB1dFNlbGVjdGlvblN0cmF0ZWd5QmFzZSB9IGZyb20gXCIuL2RhdGUtaW5wdXQtc2VsZWN0aW9uLXN0cmF0ZWd5LWJhc2VcIjtcclxuXHJcbi8qKlxyXG4gKiBTZWxlY3Rpb24gc3RyYXRlZ3kgZm9yIHRoZSAnZGF5JyBzZWxlY3Rpb24gbW9kZS5cclxuICovXHJcbmV4cG9ydCBjbGFzcyBEYXlTZWxlY3Rpb25TdHJhdGVneSBleHRlbmRzIERhdGVJbnB1dFNlbGVjdGlvblN0cmF0ZWd5QmFzZSB7XHJcbiAgcHVibGljIHJlYWRvbmx5IHNlbGVjdGlvblZpZXdNb2RlOiBDYWxlbmRhci5WaWV3TW9kZSA9ICdkYXknO1xyXG4gIHB1YmxpYyByZWFkb25seSB2YWxpZFZpZXdNb2RlczogQ2FsZW5kYXIuVmlld01vZGVbXSA9IFsnZGF5JywgJ21vbnRoJywgJ3llYXInXTtcclxuICBwdWJsaWMgcmVhZG9ubHkgc2hvd1NlY29uZGFyeVRleHRib3ggPSBmYWxzZTtcclxuICBwdWJsaWMgcmVhZG9ubHkgc2hvd1NlY29uZGFyeUNhbGVuZGFyID0gZmFsc2U7XHJcblxyXG4gIGNvbnN0cnVjdG9yKFxyXG4gICAgcHJpdmF0ZSBkYXRlRGlzcGxheVBpcGU6IERhdGVEaXNwbGF5UGlwZVxyXG4gICkge1xyXG4gICAgc3VwZXIoKTtcclxuICB9XHJcblxyXG4gIGZvcm1hdFNlbGVjdGlvbihzZWxlY3Rpb24/OiBEYXRlSW5wdXQuU2VsZWN0aW9uIHwgbnVsbCk6IERhdGVJbnB1dC5UZXh0Ym94R3JvdXBbJ3ZhbHVlJ10ge1xyXG4gICAgLy8gSWYgdGhlIHNlbGVjdGlvbiBpcyBhIHNpbmdsZSBkYXRlLCBmb3JtYXQgaXQgaW50byB0aGUgZmlyc3QgdGV4dGJveC5cclxuICAgIGlmIChEYXRlSW5wdXQuaXNTZWxlY3Rpb25TaW5nbGVEYXRlKHNlbGVjdGlvbikpIHtcclxuICAgICAgcmV0dXJuIHsgdGV4dGJveDogdGhpcy5kYXRlRGlzcGxheVBpcGUudHJhbnNmb3JtKHNlbGVjdGlvbiwgdHJ1ZSksIHRleHRib3gyOiBudWxsIH07XHJcbiAgICAvLyBPdGhlcndpc2UsIGNsZWFyIGJvdGggdGV4dGJveGVzLiBUaGlzIHNob3VsZG4ndCBoYXBwZW4gYmFzZWQgb24gY3VycmVudCBkYXRlIGlucHV0IGZsb3dzLFxyXG4gICAgLy8gYW5kIGV2ZW4gaWYgaXQgZGlkIHRoZXJlJ3Mgbm90IHJlYWxseSBhIGdvb2Qgd2F5IHRvIGhhbmRsZSBpdC5cclxuICAgIH0gZWxzZSB7XHJcbiAgICAgIHJldHVybiB7IHRleHRib3g6IG51bGwsIHRleHRib3gyOiBudWxsIH07XHJcbiAgICB9XHJcbiAgfVxyXG5cclxuICBwYXJzZVRleHRib3hWYWx1ZXModmFsdWU6IERhdGVJbnB1dC5UZXh0Ym94R3JvdXBbJ3ZhbHVlJ10sIHBhcnNlRm9ybWF0czogc3RyaW5nW10sIG9wdHM/OiBEYXRlSW5wdXQuUGFyc2luZ09wdGlvbnMpOiBEYXRlSW5wdXQuU2VsZWN0aW9uIHwgbnVsbCB7XHJcbiAgICByZXR1cm4gdGhpcy5wYXJzZVN0cmluZyh2YWx1ZS50ZXh0Ym94LCBwYXJzZUZvcm1hdHMsIG9wdHMpO1xyXG4gIH1cclxuXHJcbiAgZ2V0U2VsZWN0aW9uRnJvbURhdGUoZGF0ZTogRGF0ZSk6IERhdGVJbnB1dC5TZWxlY3Rpb24gfCBudWxsIHtcclxuICAgIC8vIE5pY2UgYW5kIHNpbXBsZVxyXG4gICAgcmV0dXJuIGRhdGU7XHJcbiAgfVxyXG5cclxuICBnZXROZXdTZWxlY3Rpb25Gcm9tRXhpc3RpbmcocHJldmlvdXNTZWxlY3Rpb24/OiBEYXRlSW5wdXQuU2VsZWN0aW9uIHwgbnVsbCk6IERhdGVJbnB1dC5TZWxlY3Rpb24gfCBudWxsIHtcclxuICAgIC8vIElmIHRoZSBwcmV2aW91cyBzZWxlY3Rpb24gd2FzIGEgcmFuZ2UsIHJldHVybiB0aGUgZW5kIGRhdGUgYW5kIGRlZmF1bHQgdG8gdGhlIHN0YXJ0IGRhdGUgaWYgdGhlcmUgaXNuJ3Qgb25lLlxyXG4gICAgaWYgKERhdGVJbnB1dC5pc1NlbGVjdGlvblJhbmdlKHByZXZpb3VzU2VsZWN0aW9uKSkge1xyXG4gICAgICByZXR1cm4gcHJldmlvdXNTZWxlY3Rpb24uZW5kIHx8IHByZXZpb3VzU2VsZWN0aW9uLnN0YXJ0O1xyXG4gICAgLy8gSWYgdGhlIHByZXZpb3VzIHNlbGVjdGlvbiB3YXMgYSBzaW5nbGUgZGF0ZSwgcmV0dXJuIGl0LlxyXG4gICAgfSBlbHNlIGlmIChEYXRlSW5wdXQuaXNTZWxlY3Rpb25TaW5nbGVEYXRlKHByZXZpb3VzU2VsZWN0aW9uKSkge1xyXG4gICAgICByZXR1cm4gcHJldmlvdXNTZWxlY3Rpb247XHJcbiAgICB9XHJcblxyXG4gICAgcmV0dXJuIG51bGw7XHJcbiAgfVxyXG5cclxuICBnZXROZXh0U2VsZWN0aW9uKHNlbGVjdGlvbjogRGF0ZUlucHV0LlNlbGVjdGlvbiB8IG51bGwpOiBEYXRlSW5wdXQuU2VsZWN0aW9uIHwgbnVsbCB7XHJcbiAgICAvLyBBZGQgYSBzaW5nbGUgZGF5IHRvIHRoZSBzZWxlY3Rpb25cclxuICAgIGlmIChEYXRlSW5wdXQuaXNTZWxlY3Rpb25TaW5nbGVEYXRlKHNlbGVjdGlvbikpIHtcclxuICAgICAgcmV0dXJuIG1vbWVudChzZWxlY3Rpb24pLmFkZCgxLCAnZGF5JykudG9EYXRlKCk7XHJcbiAgICB9XHJcblxyXG4gICAgLy8gVGhpcyBzaG91bGRuJ3QgaGFwcGVuIGJhc2VkIG9uIGN1cnJlbnQgZGF0ZSBpbnB1dCBmbG93cywgYnV0IGlmIGl0IGRvZXMganVzdCByZXR1cm4gdGhlIHNlbGVjdGlvbiBhcy1pcy5cclxuICAgIHJldHVybiBzZWxlY3Rpb247XHJcbiAgfVxyXG5cclxuICBnZXRQcmV2aW91c1NlbGVjdGlvbihzZWxlY3Rpb246IERhdGVJbnB1dC5TZWxlY3Rpb24gfCBudWxsKTogRGF0ZUlucHV0LlNlbGVjdGlvbiB8IG51bGwge1xyXG4gICAgLy8gU3VidHJhY3QgYSBzaW5nbGUgZGF5IGZyb20gdGhlIHNlbGVjdGlvblxyXG4gICAgaWYgKERhdGVJbnB1dC5pc1NlbGVjdGlvblNpbmdsZURhdGUoc2VsZWN0aW9uKSkge1xyXG4gICAgICByZXR1cm4gbW9tZW50KHNlbGVjdGlvbikuc3VidHJhY3QoMSwgJ2RheScpLnRvRGF0ZSgpO1xyXG4gICAgfVxyXG5cclxuICAgIC8vIFRoaXMgc2hvdWxkbid0IGhhcHBlbiBiYXNlZCBvbiBjdXJyZW50IGRhdGUgaW5wdXQgZmxvd3MsIGJ1dCBpZiBpdCBkb2VzIGp1c3QgcmV0dXJuIHRoZSBzZWxlY3Rpb24gYXMtaXMuXHJcbiAgICByZXR1cm4gc2VsZWN0aW9uO1xyXG4gIH1cclxufVxyXG4iXX0=
@@ -0,0 +1,100 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ import { DateInputSelectionStrategyBase } from "./date-input-selection-strategy-base";
4
+ /**
5
+ * Selection strategy for the 'last 28 days' selection mode.
6
+ */
7
+ export class Last28DaysSelectionStrategy extends DateInputSelectionStrategyBase {
8
+ constructor(rangeStrategy) {
9
+ super();
10
+ this.rangeStrategy = rangeStrategy;
11
+ this.selectionViewMode = 'day';
12
+ this.validViewModes = ['day', 'month', 'year'];
13
+ this.showSecondaryTextbox = true;
14
+ this.showSecondaryCalendar = false;
15
+ }
16
+ formatSelection(value) {
17
+ // Delegate to the range strategy for formatting since the logic is the same
18
+ return this.rangeStrategy.formatSelection(value);
19
+ }
20
+ parseTextboxValues(value, parseFormats, opts) {
21
+ let start = this.parseString(value.textbox, parseFormats, opts);
22
+ let end = this.parseString(value.textbox2, parseFormats, opts);
23
+ if (start && end) {
24
+ // Even if both dates are provided, it's possible that they don't match a 28-day range
25
+ // Update the start and end dates to match a 28-day range, keying off the end date if parseFromEnd is true
26
+ // parseFromEnd is true when a date is being entered from the end date textbox
27
+ if (opts?.parseFromEnd) {
28
+ start = moment(end).subtract(27, 'days').toDate();
29
+ }
30
+ else {
31
+ end = moment(start).add(27, 'days').toDate();
32
+ }
33
+ }
34
+ // If the start date is not provided, we'll use the end date to calculate the start date
35
+ if (!start && end && !opts?.preventAutoComplete) {
36
+ start = moment(end).subtract(27, 'days').toDate();
37
+ }
38
+ // If the end date is not provided, we'll use the start date to calculate the end date
39
+ if (!end && start && !opts?.preventAutoComplete) {
40
+ end = moment(start).add(27, 'days').toDate();
41
+ }
42
+ if (!start && !end) {
43
+ return null;
44
+ }
45
+ else {
46
+ return { start, end };
47
+ }
48
+ }
49
+ getSelectionFromDate(date) {
50
+ return {
51
+ start: moment(date).subtract(27, 'days').toDate(),
52
+ end: date
53
+ };
54
+ }
55
+ getNewSelectionFromExisting(previousSelection) {
56
+ if (DateInput.isSelectionRange(previousSelection)) {
57
+ // If the range only has a start date, make the 28-day range start on that date
58
+ if (!previousSelection.end && previousSelection.start) {
59
+ return {
60
+ start: previousSelection.start,
61
+ end: moment(previousSelection.start).add(27, 'days').toDate()
62
+ };
63
+ // If the range only has an end date, make the 28-day range end on that date
64
+ }
65
+ else if (previousSelection.end) {
66
+ return this.getSelectionFromDate(previousSelection.end);
67
+ }
68
+ // If the previous selection was a single date, make the range the 28 days ending on that date
69
+ }
70
+ else if (DateInput.isSelectionSingleDate(previousSelection)) {
71
+ return this.getSelectionFromDate(previousSelection);
72
+ }
73
+ return null;
74
+ }
75
+ getNextSelection(selection) {
76
+ // Move the selection to the next 28 days
77
+ if (DateInput.isSelectionRange(selection) && selection.end) {
78
+ return {
79
+ start: moment(selection.end).add(1, 'day').toDate(),
80
+ end: moment(selection.end).add(28, 'days').toDate()
81
+ };
82
+ }
83
+ // This shouldn't happen since the next stepper is disabled if the selection only has one date.
84
+ // Just return the selection as-is in this case.
85
+ return selection;
86
+ }
87
+ getPreviousSelection(selection) {
88
+ // Move the selection to the previous 28 days
89
+ if (DateInput.isSelectionRange(selection) && selection.start) {
90
+ return {
91
+ start: moment(selection.start).subtract(28, 'days').toDate(),
92
+ end: moment(selection.start).subtract(1, 'day').toDate()
93
+ };
94
+ }
95
+ // This shouldn't happen since the previous stepper is disabled if the selection only has one date.
96
+ // Just return the selection as-is in this case.
97
+ return selection;
98
+ }
99
+ }
100
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"last-28-days-selection-strategy.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/controls/date-input/date-input-selection-strategies/last-28-days-selection-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from \"moment\";\r\nimport { Calendar } from \"../../calendar/calendar.types\";\r\nimport { DateInput } from \"../date-input.types\";\r\nimport { DateInputSelectionStrategyBase } from \"./date-input-selection-strategy-base\";\r\nimport { RangeSelectionStrategy } from \"./range-selection-strategy\";\r\n\r\n/**\r\n * Selection strategy for the 'last 28 days' selection mode.\r\n */\r\nexport class Last28DaysSelectionStrategy extends DateInputSelectionStrategyBase {\r\n  public readonly selectionViewMode: Calendar.ViewMode = 'day';\r\n  public readonly validViewModes: Calendar.ViewMode[] = ['day', 'month', 'year'];\r\n  public readonly showSecondaryTextbox = true;\r\n  public readonly showSecondaryCalendar = false;\r\n\r\n  constructor(\r\n    private rangeStrategy: RangeSelectionStrategy\r\n  ) {\r\n    super();\r\n  }\r\n\r\n  formatSelection(value?: DateInput.Selection | null): DateInput.TextboxGroup['value'] {\r\n    // Delegate to the range strategy for formatting since the logic is the same\r\n    return this.rangeStrategy.formatSelection(value);\r\n  }\r\n\r\n  parseTextboxValues(value: DateInput.TextboxGroup['value'], parseFormats: string[], opts?: DateInput.ParsingOptions): DateInput.Selection | null {\r\n    let start = this.parseString(value.textbox, parseFormats, opts);\r\n    let end = this.parseString(value.textbox2, parseFormats, opts);\r\n\r\n    if (start && end) {\r\n      // Even if both dates are provided, it's possible that they don't match a 28-day range\r\n      // Update the start and end dates to match a 28-day range, keying off the end date if parseFromEnd is true\r\n      // parseFromEnd is true when a date is being entered from the end date textbox\r\n      if (opts?.parseFromEnd) {\r\n        start = moment(end).subtract(27, 'days').toDate();\r\n      } else {\r\n        end = moment(start).add(27, 'days').toDate();\r\n      }\r\n    }\r\n\r\n    // If the start date is not provided, we'll use the end date to calculate the start date\r\n    if (!start && end && !opts?.preventAutoComplete) {\r\n      start = moment(end).subtract(27, 'days').toDate();\r\n    }\r\n\r\n    // If the end date is not provided, we'll use the start date to calculate the end date\r\n    if (!end && start && !opts?.preventAutoComplete) {\r\n      end = moment(start).add(27, 'days').toDate();\r\n    }\r\n\r\n    if (!start && !end) {\r\n      return null;\r\n    } else {\r\n      return { start, end };\r\n    }\r\n  }\r\n\r\n  getSelectionFromDate(date: Date): DateInput.Selection | null {\r\n    return { \r\n      start: moment(date).subtract(27, 'days').toDate(),\r\n      end: date\r\n    };\r\n  }\r\n\r\n  getNewSelectionFromExisting(previousSelection?: DateInput.Selection): DateInput.Selection | null {\r\n    if (DateInput.isSelectionRange(previousSelection)) {\r\n      // If the range only has a start date, make the 28-day range start on that date\r\n      if (!previousSelection.end && previousSelection.start) {\r\n        return {\r\n          start: previousSelection.start,\r\n          end: moment(previousSelection.start).add(27, 'days').toDate()\r\n        };\r\n      // If the range only has an end date, make the 28-day range end on that date\r\n      } else if (previousSelection.end) {\r\n        return this.getSelectionFromDate(previousSelection.end);\r\n      }\r\n    // If the previous selection was a single date, make the range the 28 days ending on that date\r\n    } else if (DateInput.isSelectionSingleDate(previousSelection)) {\r\n      return this.getSelectionFromDate(previousSelection);\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  getNextSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Move the selection to the next 28 days\r\n    if (DateInput.isSelectionRange(selection) && selection.end) {\r\n      return {\r\n        start: moment(selection.end).add(1, 'day').toDate(),\r\n        end: moment(selection.end).add(28, 'days').toDate()\r\n      };\r\n    }\r\n\r\n    // This shouldn't happen since the next stepper is disabled if the selection only has one date.\r\n    // Just return the selection as-is in this case.\r\n    return selection;\r\n  }\r\n\r\n  getPreviousSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Move the selection to the previous 28 days\r\n    if (DateInput.isSelectionRange(selection) && selection.start) {\r\n      return {\r\n        start: moment(selection.start).subtract(28, 'days').toDate(),\r\n        end: moment(selection.start).subtract(1, 'day').toDate()\r\n      };\r\n    }\r\n\r\n    // This shouldn't happen since the previous stepper is disabled if the selection only has one date.\r\n    // Just return the selection as-is in this case.\r\n    return selection;\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,101 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ import { DateInputSelectionStrategyBase } from "./date-input-selection-strategy-base";
4
+ /**
5
+ * Selection strategy for the 'last 7 days' selection mode.
6
+ */
7
+ export class Last7DaysSelectionStrategy extends DateInputSelectionStrategyBase {
8
+ constructor(rangeStrategy) {
9
+ super();
10
+ this.rangeStrategy = rangeStrategy;
11
+ this.selectionViewMode = 'day';
12
+ this.validViewModes = ['day', 'month', 'year'];
13
+ this.showSecondaryTextbox = true;
14
+ this.showSecondaryCalendar = false;
15
+ }
16
+ formatSelection(value) {
17
+ // Delegate to the range strategy for formatting since the logic is the same
18
+ return this.rangeStrategy.formatSelection(value);
19
+ }
20
+ parseTextboxValues(value, parseFormats, opts) {
21
+ let start = this.parseString(value.textbox, parseFormats, opts);
22
+ let end = this.parseString(value.textbox2, parseFormats, opts);
23
+ if (start && end) {
24
+ // Even if both dates are provided, it's possible that they don't match a 7-day range
25
+ // Update the start and end dates to match a 7-day range, keying off the end date if parseFromEnd is true
26
+ // parseFromEnd is true when a date is being entered from the end date textbox
27
+ if (opts?.parseFromEnd) {
28
+ start = moment(end).subtract(6, 'days').toDate();
29
+ }
30
+ else {
31
+ end = moment(start).add(6, 'days').toDate();
32
+ }
33
+ }
34
+ // If the start date is not provided, we'll use the end date to calculate the start date
35
+ if (!start && end && !opts?.preventAutoComplete) {
36
+ start = moment(end).subtract(6, 'days').toDate();
37
+ }
38
+ // If the end date is not provided, we'll use the start date to calculate the end date
39
+ if (!end && start && !opts?.preventAutoComplete) {
40
+ end = moment(start).add(6, 'days').toDate();
41
+ }
42
+ if (!start && !end) {
43
+ return null;
44
+ }
45
+ else {
46
+ return { start, end };
47
+ }
48
+ }
49
+ getSelectionFromDate(date) {
50
+ // Base the selection on the given date, moving back 6 days
51
+ return {
52
+ start: moment(date).subtract(6, 'days').toDate(),
53
+ end: date
54
+ };
55
+ }
56
+ getNewSelectionFromExisting(previousSelection) {
57
+ if (DateInput.isSelectionRange(previousSelection)) {
58
+ // If we don't have an end date, create the new selection based on the start date forwards
59
+ if (!previousSelection.end && previousSelection.start) {
60
+ return {
61
+ start: previousSelection.start,
62
+ end: moment(previousSelection.start).add(6, 'days').toDate()
63
+ };
64
+ // Otherwise, create the new selection based on the end date backwards
65
+ }
66
+ else if (previousSelection.end) {
67
+ return this.getSelectionFromDate(previousSelection.end);
68
+ }
69
+ // If we only have a single date, create the new selection based on that date backwards
70
+ }
71
+ else if (DateInput.isSelectionSingleDate(previousSelection)) {
72
+ return this.getSelectionFromDate(previousSelection);
73
+ }
74
+ return null;
75
+ }
76
+ getNextSelection(selection) {
77
+ // Shift the range forward by 7 days
78
+ if (DateInput.isSelectionRange(selection) && selection.end) {
79
+ return {
80
+ start: moment(selection.end).add(1, 'day').toDate(),
81
+ end: moment(selection.end).add(7, 'days').toDate()
82
+ };
83
+ }
84
+ // This shouldn't happen since the next stepper is disabled if the selection only has one date.
85
+ // Just return the selection as-is in this case.
86
+ return selection;
87
+ }
88
+ getPreviousSelection(selection) {
89
+ // Shift the range back 7 days
90
+ if (DateInput.isSelectionRange(selection) && selection.start) {
91
+ return {
92
+ start: moment(selection.start).subtract(7, 'days').toDate(),
93
+ end: moment(selection.start).subtract(1, 'day').toDate()
94
+ };
95
+ }
96
+ // This shouldn't happen since the previous stepper is disabled if the selection only has one date.
97
+ // Just return the selection as-is in this case.
98
+ return selection;
99
+ }
100
+ }
101
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"last-7-days-selection-strategy.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/controls/date-input/date-input-selection-strategies/last-7-days-selection-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from \"moment\";\r\nimport { Calendar } from \"../../calendar/calendar.types\";\r\nimport { DateInput } from \"../date-input.types\";\r\nimport { DateInputSelectionStrategyBase } from \"./date-input-selection-strategy-base\";\r\nimport { RangeSelectionStrategy } from \"./range-selection-strategy\";\r\n\r\n/**\r\n * Selection strategy for the 'last 7 days' selection mode.\r\n */\r\nexport class Last7DaysSelectionStrategy extends DateInputSelectionStrategyBase {\r\n  public readonly selectionViewMode: Calendar.ViewMode = 'day';\r\n  public readonly validViewModes: Calendar.ViewMode[] = ['day', 'month', 'year'];\r\n  public readonly showSecondaryTextbox = true;\r\n  public readonly showSecondaryCalendar = false;\r\n\r\n  constructor(\r\n    private rangeStrategy: RangeSelectionStrategy\r\n  ) {\r\n    super();\r\n  }\r\n\r\n  formatSelection(value?: DateInput.Selection | null): DateInput.TextboxGroup['value'] {\r\n    // Delegate to the range strategy for formatting since the logic is the same\r\n    return this.rangeStrategy.formatSelection(value);\r\n  }\r\n\r\n  parseTextboxValues(value: DateInput.TextboxGroup['value'], parseFormats: string[], opts?: DateInput.ParsingOptions): DateInput.Selection | null {\r\n    let start = this.parseString(value.textbox, parseFormats, opts);\r\n    let end = this.parseString(value.textbox2, parseFormats, opts);\r\n\r\n    if (start && end) {\r\n      // Even if both dates are provided, it's possible that they don't match a 7-day range\r\n      // Update the start and end dates to match a 7-day range, keying off the end date if parseFromEnd is true\r\n      // parseFromEnd is true when a date is being entered from the end date textbox\r\n      if (opts?.parseFromEnd) {\r\n        start = moment(end).subtract(6, 'days').toDate();\r\n      } else {\r\n        end = moment(start).add(6, 'days').toDate();\r\n      }\r\n    }\r\n\r\n    // If the start date is not provided, we'll use the end date to calculate the start date\r\n    if (!start && end && !opts?.preventAutoComplete) {\r\n      start = moment(end).subtract(6, 'days').toDate();\r\n    }\r\n\r\n    // If the end date is not provided, we'll use the start date to calculate the end date\r\n    if (!end && start && !opts?.preventAutoComplete) {\r\n      end = moment(start).add(6, 'days').toDate();\r\n    }\r\n\r\n    if (!start && !end) {\r\n      return null;\r\n    } else {\r\n      return { start, end };\r\n    }\r\n  }\r\n\r\n  getSelectionFromDate(date: Date): DateInput.Selection | null {\r\n    // Base the selection on the given date, moving back 6 days\r\n    return {\r\n      start: moment(date).subtract(6, 'days').toDate(),\r\n      end: date\r\n    };\r\n  }\r\n\r\n  getNewSelectionFromExisting(previousSelection?: DateInput.Selection | null): DateInput.Selection | null {\r\n    if (DateInput.isSelectionRange(previousSelection)) {\r\n      // If we don't have an end date, create the new selection based on the start date forwards\r\n      if (!previousSelection.end && previousSelection.start) {\r\n        return {\r\n          start: previousSelection.start,\r\n          end: moment(previousSelection.start).add(6, 'days').toDate()\r\n        };\r\n      // Otherwise, create the new selection based on the end date backwards\r\n      } else if (previousSelection.end) {\r\n        return this.getSelectionFromDate(previousSelection.end);\r\n      }\r\n    // If we only have a single date, create the new selection based on that date backwards\r\n    } else if (DateInput.isSelectionSingleDate(previousSelection)) {\r\n      return this.getSelectionFromDate(previousSelection);\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  getNextSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Shift the range forward by 7 days\r\n    if (DateInput.isSelectionRange(selection) && selection.end) {\r\n      return {\r\n        start: moment(selection.end).add(1, 'day').toDate(),\r\n        end: moment(selection.end).add(7, 'days').toDate()\r\n      };\r\n    }\r\n\r\n    // This shouldn't happen since the next stepper is disabled if the selection only has one date.\r\n    // Just return the selection as-is in this case.\r\n    return selection;\r\n  }\r\n\r\n  getPreviousSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Shift the range back 7 days\r\n    if (DateInput.isSelectionRange(selection) && selection.start) {\r\n      return {\r\n        start: moment(selection.start).subtract(7, 'days').toDate(),\r\n        end: moment(selection.start).subtract(1, 'day').toDate()\r\n      };\r\n    }\r\n\r\n    // This shouldn't happen since the previous stepper is disabled if the selection only has one date.\r\n    // Just return the selection as-is in this case.\r\n    return selection;\r\n  }\r\n}\r\n"]}
@@ -0,0 +1,76 @@
1
+ import moment from "moment";
2
+ import { DateInput } from "../date-input.types";
3
+ import { DateInputSelectionStrategyBase } from "./date-input-selection-strategy-base";
4
+ /**
5
+ * Selection strategy for the 'month' selection mode.
6
+ */
7
+ export class MonthSelectionStrategy extends DateInputSelectionStrategyBase {
8
+ constructor() {
9
+ super(...arguments);
10
+ this.selectionViewMode = 'month';
11
+ this.validViewModes = ['month', 'year'];
12
+ this.showSecondaryTextbox = false;
13
+ this.showSecondaryCalendar = false;
14
+ }
15
+ formatSelection(selection) {
16
+ if (!selection) {
17
+ return { textbox: null, textbox2: null };
18
+ }
19
+ // We only need one date in the range to format since it only spans one month
20
+ // A single date selection shouldn't happen in this mode but it's simple enough to handle
21
+ const date = DateInput.isSelectionRange(selection) ? selection.start : selection;
22
+ return {
23
+ textbox: moment(date).format('MMM, YYYY'),
24
+ textbox2: null
25
+ };
26
+ }
27
+ parseTextboxValues(value, parseFormats, opts) {
28
+ // Only one textbox is used in this mode
29
+ const date = this.parseString(value.textbox, parseFormats, opts);
30
+ if (!date) {
31
+ return null;
32
+ }
33
+ return this.getSelectionFromDate(date);
34
+ }
35
+ getSelectionFromDate(date) {
36
+ return {
37
+ start: moment(date).startOf('month').toDate(),
38
+ end: moment(date).endOf('month').toDate()
39
+ };
40
+ }
41
+ getNewSelectionFromExisting(previousSelection) {
42
+ if (DateInput.isSelectionRange(previousSelection)) {
43
+ // We'll use the existing selection's end date and fall back to the start date if it's not available
44
+ const date = previousSelection.end || previousSelection.start;
45
+ if (date) {
46
+ return this.getSelectionFromDate(date);
47
+ }
48
+ }
49
+ else if (DateInput.isSelectionSingleDate(previousSelection)) {
50
+ // If the previous selection was a single date, make the range that month
51
+ return this.getSelectionFromDate(previousSelection);
52
+ }
53
+ return null;
54
+ }
55
+ getNextSelection(selection) {
56
+ // Move the selection to the next month
57
+ if (DateInput.isSelectionRange(selection) && selection.start) {
58
+ const date = moment(selection.start).add(1, 'month');
59
+ return this.getSelectionFromDate(date.toDate());
60
+ }
61
+ // This shouldn't happen since the next stepper is disabled if the selection only has one date.
62
+ // If it does, just return the selection as-is.
63
+ return selection;
64
+ }
65
+ getPreviousSelection(selection) {
66
+ // Move the selection to the previous month
67
+ if (DateInput.isSelectionRange(selection) && selection.start) {
68
+ const date = moment(selection.start).subtract(1, 'month');
69
+ return this.getSelectionFromDate(date.toDate());
70
+ }
71
+ // This shouldn't happen since the previous stepper is disabled if the selection only has one date.
72
+ // If it does, just return the selection as-is.
73
+ return selection;
74
+ }
75
+ }
76
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"month-selection-strategy.js","sourceRoot":"","sources":["../../../../../../../projects/components/src/lib/controls/date-input/date-input-selection-strategies/month-selection-strategy.ts"],"names":[],"mappings":"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","sourcesContent":["import moment from \"moment\";\r\nimport { Calendar } from \"../../calendar/calendar.types\";\r\nimport { DateInput } from \"../date-input.types\";\r\nimport { DateInputSelectionStrategyBase } from \"./date-input-selection-strategy-base\";\r\n\r\n/**\r\n * Selection strategy for the 'month' selection mode.\r\n */\r\nexport class MonthSelectionStrategy extends DateInputSelectionStrategyBase {\r\n  public readonly selectionViewMode: Calendar.ViewMode = 'month';\r\n  public readonly validViewModes: Calendar.ViewMode[] = ['month', 'year'];\r\n  public readonly showSecondaryTextbox = false;\r\n  public readonly showSecondaryCalendar = false;\r\n\r\n  formatSelection(selection?: DateInput.Selection | null): DateInput.TextboxGroup['value'] {\r\n    if (!selection) {\r\n      return { textbox: null, textbox2: null };\r\n    }\r\n\r\n    // We only need one date in the range to format since it only spans one month\r\n    // A single date selection shouldn't happen in this mode but it's simple enough to handle\r\n    const date = DateInput.isSelectionRange(selection) ? selection.start : selection;\r\n    return {\r\n      textbox: moment(date).format('MMM, YYYY'),\r\n      textbox2: null\r\n    };\r\n  }\r\n\r\n  parseTextboxValues(value: DateInput.TextboxGroup['value'], parseFormats: string[], opts?: DateInput.ParsingOptions): DateInput.Selection | null {\r\n    // Only one textbox is used in this mode\r\n    const date = this.parseString(value.textbox, parseFormats, opts);\r\n\r\n    if (!date) {\r\n      return null;\r\n    }\r\n\r\n    return this.getSelectionFromDate(date);\r\n  }\r\n\r\n  getSelectionFromDate(date: Date): DateInput.Selection | null {\r\n    return {\r\n      start: moment(date).startOf('month').toDate(),\r\n      end: moment(date).endOf('month').toDate()\r\n    };\r\n  }\r\n\r\n  getNewSelectionFromExisting(previousSelection?: DateInput.Selection): DateInput.Selection | null {\r\n    if (DateInput.isSelectionRange(previousSelection)) {\r\n      // We'll use the existing selection's end date and fall back to the start date if it's not available\r\n      const date = previousSelection.end || previousSelection.start;\r\n      if (date) {\r\n        return this.getSelectionFromDate(date);\r\n      }\r\n    } else if (DateInput.isSelectionSingleDate(previousSelection)) {\r\n      // If the previous selection was a single date, make the range that month\r\n      return this.getSelectionFromDate(previousSelection);\r\n    }\r\n\r\n    return null;\r\n  }\r\n\r\n  getNextSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Move the selection to the next month\r\n    if (DateInput.isSelectionRange(selection) && selection.start) { \r\n      const date = moment(selection.start).add(1, 'month');\r\n      return this.getSelectionFromDate(date.toDate());\r\n    }\r\n\r\n    // This shouldn't happen since the next stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n\r\n  getPreviousSelection(selection: DateInput.Selection | null): DateInput.Selection | null {\r\n    // Move the selection to the previous month\r\n    if (DateInput.isSelectionRange(selection) && selection.start) {\r\n      const date = moment(selection.start).subtract(1, 'month');\r\n      return this.getSelectionFromDate(date.toDate());\r\n    }\r\n\r\n    // This shouldn't happen since the previous stepper is disabled if the selection only has one date.\r\n    // If it does, just return the selection as-is.\r\n    return selection;\r\n  }\r\n}\r\n"]}