@impartner/design-components 1.0.0 → 1.0.1
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.
- package/README.md +1 -1
- package/esm2020/lib/alert/alert.component.mjs +1 -1
- package/esm2020/lib/avatar/avatar.component.mjs +1 -1
- package/esm2020/lib/backdrop/backdrop.component.mjs +7 -1
- package/esm2020/lib/badge/badge.component.mjs +1 -1
- package/esm2020/lib/breadcrumb/breadcrumb.component.mjs +1 -1
- package/esm2020/lib/button/button.component.mjs +1 -1
- package/esm2020/lib/button-group/button-group.component.mjs +1 -1
- package/esm2020/lib/card-heading/card-heading.component.mjs +1 -1
- package/esm2020/lib/dropdown/components/dropdown-item/dropdown-item.component.mjs +1 -1
- package/esm2020/lib/dropdown/dropdown.component.mjs +1 -1
- package/esm2020/lib/file-upload/directives/file-drop.directive.mjs +1 -1
- package/esm2020/lib/file-upload/file-upload.component.mjs +1 -1
- package/esm2020/lib/form-field/controls/checkbox/checkbox.component.mjs +1 -1
- package/esm2020/lib/form-field/controls/input/input.directive.mjs +1 -1
- package/esm2020/lib/form-field/controls/radio/radio-button/radio-button.component.mjs +1 -1
- package/esm2020/lib/form-field/controls/radio/radio-group/radio-group.directive.mjs +2 -1
- package/esm2020/lib/form-field/controls/select/option/select-option.component.mjs +1 -1
- package/esm2020/lib/form-field/controls/select/select-model.mjs +2 -1
- package/esm2020/lib/form-field/controls/select/select.component.mjs +1 -1
- package/esm2020/lib/form-field/controls/shared/toggle/toggle.component.mjs +1 -1
- package/esm2020/lib/form-field/directives/error.directive.mjs +1 -1
- package/esm2020/lib/form-field/directives/hint.directive.mjs +1 -1
- package/esm2020/lib/form-field/shared/disabled.mixin.mjs +1 -1
- package/esm2020/lib/form-field/shared/error-state.mixin.mjs +1 -1
- package/esm2020/lib/icon/icon.component.mjs +14 -10
- package/esm2020/lib/modal/modal.component.mjs +51 -13
- package/esm2020/lib/pagination/pagination.component.mjs +1 -1
- package/esm2020/lib/progress-bar/progress-bar.component.mjs +1 -1
- package/esm2020/lib/scrollable/scrollable.component.mjs +14 -21
- package/esm2020/lib/select-icon/select-icon.component.mjs +1 -1
- package/esm2020/lib/spinner/spinner.component.mjs +1 -1
- package/esm2020/lib/table/table.component.mjs +1 -1
- package/esm2020/lib/text-highlight/text-highlight.component.mjs +1 -1
- package/esm2020/services/interaction.service.mjs +13 -3
- package/esm2020/utilities/sanitize.mjs +3 -3
- package/fesm2015/impartner-design-components.mjs +100 -45
- package/fesm2015/impartner-design-components.mjs.map +1 -1
- package/fesm2020/impartner-design-components.mjs +97 -44
- package/fesm2020/impartner-design-components.mjs.map +1 -1
- package/lib/backdrop/backdrop.component.d.ts +5 -3
- package/lib/icon/icon.component.d.ts +6 -4
- package/lib/modal/modal.component.d.ts +32 -7
- package/lib/scrollable/scrollable.component.d.ts +3 -1
- package/package.json +2 -2
- package/services/interaction.service.d.ts +1 -0
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
1
|
+
import { Component, EventEmitter, Input, Output, ViewChild } from '@angular/core';
|
|
2
2
|
import { ComponentSize } from '../../types';
|
|
3
3
|
import { ModalTheme } from './types';
|
|
4
4
|
import * as i0 from "@angular/core";
|
|
@@ -16,10 +16,7 @@ export class ModalComponent {
|
|
|
16
16
|
constructor(_interactionService) {
|
|
17
17
|
this._interactionService = _interactionService;
|
|
18
18
|
this._interactableId = 0;
|
|
19
|
-
|
|
20
|
-
* Determines if the Modal should be shown immediately.
|
|
21
|
-
*/
|
|
22
|
-
this.show = false;
|
|
19
|
+
this._show = false;
|
|
23
20
|
/**
|
|
24
21
|
* The theme of the Modal. Affects the layout of the Modal as well as some colors.
|
|
25
22
|
*/
|
|
@@ -37,6 +34,10 @@ export class ModalComponent {
|
|
|
37
34
|
* The subject text of the Modal.
|
|
38
35
|
*/
|
|
39
36
|
this.titleText = '';
|
|
37
|
+
/**
|
|
38
|
+
* The fallback button text to accept the Modal. This is bypassed if you content-project the accept button.
|
|
39
|
+
*/
|
|
40
|
+
this.acceptText = 'OK';
|
|
40
41
|
/**
|
|
41
42
|
* Determines if the Modal should fade in/out.
|
|
42
43
|
*/
|
|
@@ -59,6 +60,12 @@ export class ModalComponent {
|
|
|
59
60
|
this.accept = new EventEmitter();
|
|
60
61
|
this._interactableId = this._interactionService.add(this);
|
|
61
62
|
}
|
|
63
|
+
/**
|
|
64
|
+
* Determines if the Modal should be shown immediately.
|
|
65
|
+
*/
|
|
66
|
+
set show(show) {
|
|
67
|
+
this._show = show;
|
|
68
|
+
}
|
|
62
69
|
get interactableId() {
|
|
63
70
|
return this._interactableId;
|
|
64
71
|
}
|
|
@@ -66,7 +73,7 @@ export class ModalComponent {
|
|
|
66
73
|
return this.interactableId > 0;
|
|
67
74
|
}
|
|
68
75
|
get shown() {
|
|
69
|
-
return this.valid && this.
|
|
76
|
+
return this.valid && this._show;
|
|
70
77
|
}
|
|
71
78
|
get showWorkingIcon() {
|
|
72
79
|
return !!this.iconName && this.theme === ModalTheme.Working;
|
|
@@ -93,26 +100,46 @@ export class ModalComponent {
|
|
|
93
100
|
this._interactableId = 0;
|
|
94
101
|
}
|
|
95
102
|
handleDismiss(event) {
|
|
96
|
-
if (!this.dismissable
|
|
103
|
+
if (!this.dismissable ||
|
|
104
|
+
!!this.dismissEl?.nativeElement?.firstChild?.disabled) {
|
|
97
105
|
return;
|
|
98
106
|
}
|
|
99
|
-
this.
|
|
107
|
+
this._show = false;
|
|
100
108
|
this.dismiss.emit(event);
|
|
101
109
|
}
|
|
102
110
|
handleDeny(event) {
|
|
103
|
-
this.
|
|
111
|
+
if (!!this.denyEl?.nativeElement?.firstChild?.disabled) {
|
|
112
|
+
return;
|
|
113
|
+
}
|
|
114
|
+
this._show = false;
|
|
104
115
|
this.deny.emit(event);
|
|
105
116
|
}
|
|
106
117
|
handleAccept(event) {
|
|
107
|
-
this.
|
|
118
|
+
console.log('this.acceptEl', this.acceptEl);
|
|
119
|
+
if (!!this.acceptEl?.nativeElement?.firstChild?.disabled) {
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
this._show = false;
|
|
108
123
|
this.accept.emit(event);
|
|
109
124
|
}
|
|
125
|
+
/**
|
|
126
|
+
* Opens/shows the modal programmatically.
|
|
127
|
+
*/
|
|
128
|
+
open() {
|
|
129
|
+
this._show = true;
|
|
130
|
+
}
|
|
131
|
+
/**
|
|
132
|
+
* Closes/hides the modal programmatically.
|
|
133
|
+
*/
|
|
134
|
+
close() {
|
|
135
|
+
this._show = false;
|
|
136
|
+
}
|
|
110
137
|
}
|
|
111
138
|
ModalComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ModalComponent, deps: [{ token: i1.InteractionService }], target: i0.ɵɵFactoryTarget.Component });
|
|
112
|
-
ModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: ModalComponent, selector: "impdc-modal", inputs: { show: "show", theme: "theme", size: "size", iconName: "iconName", iconTheme: "iconTheme", titleText: "titleText", dismissable: "dismissable", fade: "fade", backdrop: "backdrop" }, outputs: { dismiss: "dismiss", deny: "deny", accept: "accept" }, ngImport: i0, template: "<div\n *ngIf=\"valid\"\n class=\"impdc modal\"\n [class.fade]=\"fade\"\n [class.in]=\"fade && shown\"\n [class.show]=\"shown\"\n [class.dismissable]=\"dismissable\"\n tabindex=\"-1\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.escape)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.enter)=\"handleAccept($event); $event.stopPropagation()\">\n <div\n class=\"modal-dialog modal-dialog-centered modal-dialog-scrollable modal-theme-{{\n theme\n }}\"\n [class.modal-sm]=\"size === 'sm'\"\n [class.modal-lg]=\"size === 'lg'\"\n [class.modal-xl]=\"size === 'xl'\"\n [class.modal-fullscreen]=\"size === 'fs'\"\n [class.modal-fullscreen-lg-down]=\"size !== 'fs'\"\n (click)=\"$event.stopPropagation()\">\n <div class=\"modal-content\">\n <div *ngIf=\"showDestructiveIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </div>\n <div #header class=\"modal-header\" [class.hide]=\"hideHeader\">\n <h3 class=\"modal-title\">\n <span *ngIf=\"showWorkingIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </span>\n <span>{{ titleText }}</span>\n </h3>\n <button\n impdcButton\n *ngIf=\"dismissable\"\n theme=\"close\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"></button>\n </div>\n <div #bodyEl class=\"modal-body\">\n <ng-content></ng-content>\n </div>\n <div #footer class=\"modal-footer\">\n <span\n #dismissEl\n *ngIf=\"dismissable\"\n [class.hide]=\"!dismissEl.children.length\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\">\n <ng-content select=\"[dismiss]\"></ng-content>\n </span>\n <span\n #denyEl\n [class.hide]=\"!denyEl.children.length\"\n (click)=\"handleDeny($event); $event.stopPropagation()\">\n <ng-content select=\"[deny]\"></ng-content>\n </span>\n <span\n #acceptEl\n [class.hide]=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <ng-content select=\"[accept]\"></ng-content>\n </span>\n <span\n #acceptFallbackEl\n *ngIf=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <button
|
|
139
|
+
ModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: ModalComponent, selector: "impdc-modal", inputs: { show: "show", theme: "theme", size: "size", iconName: "iconName", iconTheme: "iconTheme", titleText: "titleText", acceptText: "acceptText", dismissable: "dismissable", fade: "fade", backdrop: "backdrop" }, outputs: { dismiss: "dismiss", deny: "deny", accept: "accept" }, viewQueries: [{ propertyName: "dismissEl", first: true, predicate: ["dismissEl"], descendants: true }, { propertyName: "denyEl", first: true, predicate: ["denyEl"], descendants: true }, { propertyName: "acceptEl", first: true, predicate: ["acceptEl"], descendants: true }], ngImport: i0, template: "<div\n *ngIf=\"valid\"\n class=\"impdc modal\"\n [class.fade]=\"fade\"\n [class.in]=\"fade && shown\"\n [class.show]=\"shown\"\n [class.dismissable]=\"dismissable\"\n tabindex=\"-1\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.escape)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.enter)=\"handleAccept($event); $event.stopPropagation()\">\n <div\n class=\"modal-dialog modal-dialog-centered modal-dialog-scrollable modal-theme-{{\n theme\n }}\"\n [class.modal-sm]=\"size === 'sm'\"\n [class.modal-lg]=\"size === 'lg'\"\n [class.modal-xl]=\"size === 'xl'\"\n [class.modal-fullscreen]=\"size === 'fs'\"\n [class.modal-fullscreen-lg-down]=\"size !== 'fs'\"\n (click)=\"$event.stopPropagation()\">\n <div class=\"modal-content\">\n <div *ngIf=\"showDestructiveIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </div>\n <div #header class=\"modal-header\" [class.hide]=\"hideHeader\">\n <h3 class=\"modal-title\">\n <span *ngIf=\"showWorkingIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </span>\n <span>{{ titleText }}</span>\n </h3>\n <button\n impdcButton\n *ngIf=\"dismissable\"\n theme=\"close\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"></button>\n </div>\n <div #bodyEl class=\"modal-body\">\n <ng-content></ng-content>\n </div>\n <div #footer class=\"modal-footer\">\n <span\n #dismissEl\n *ngIf=\"dismissable\"\n [class.hide]=\"!dismissEl.children.length\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\">\n <ng-content select=\"[dismiss]\"></ng-content>\n </span>\n <span\n #denyEl\n [class.hide]=\"!denyEl.children.length\"\n (click)=\"handleDeny($event); $event.stopPropagation()\">\n <ng-content select=\"[deny]\"></ng-content>\n </span>\n <span\n #acceptEl\n [class.hide]=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <ng-content select=\"[accept]\"></ng-content>\n </span>\n <span\n #acceptFallbackEl\n *ngIf=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <button\n impdcButton\n [text]=\"acceptText\"\n [theme]=\"acceptFallbackTheme\"></button>\n </span>\n </div>\n </div>\n </div>\n</div>\n", styles: [".impdc.modal{-webkit-user-select:none;user-select:none}.impdc.modal .modal-icon{display:inline-flex;justify-content:center;align-items:center;height:40px;width:40px;color:var(--impd-color-gray-500);background:var(--impd-color-gray-100);border-radius:50%;flex-shrink:0}.impdc.modal .modal-icon .fa,.impdc.modal .modal-icon .fa-brands,.impdc.modal .modal-icon .fa-duotone,.impdc.modal .modal-icon .fa-light,.impdc.modal .modal-icon .fa-regular,.impdc.modal .modal-icon .fa-solid,.impdc.modal .modal-icon .fa-thin,.impdc.modal .modal-icon .fab,.impdc.modal .modal-icon .fad,.impdc.modal .modal-icon .fal,.impdc.modal .modal-icon .far,.impdc.modal .modal-icon .fas,.impdc.modal .modal-icon .fat{line-height:var(--impartner-hex-modal-title-line-height)}.impdc.modal.fade.in{background-color:transparent}.impdc.modal.show{display:block;opacity:1}.impdc.modal:not(.show),.impdc.modal.hide,.impdc.modal .hide{display:none}.impdc.modal.dismissable{cursor:pointer}.impdc.modal.dismissable .modal-dialog{cursor:default}.impdc.modal .modal-dialog{cursor:default;-webkit-user-select:auto;user-select:auto}.impdc.modal .modal-theme-simple .modal-header,.impdc.modal .modal-theme-destructive .modal-header{border:0;padding-bottom:0}.impdc.modal .modal-theme-simple .modal-body,.impdc.modal .modal-theme-destructive .modal-body{padding:calc(var(--impartner-hex-modal-padding) * .5) var(--impartner-hex-modal-padding) var(--impartner-hex-modal-padding)}.impdc.modal .modal-theme-simple .modal-footer,.impdc.modal .modal-theme-destructive .modal-footer{border:0;padding-top:0}.impdc.modal .modal-theme-working .modal-footer{padding:calc(var(--impartner-hex-modal-padding) * .5 - var(--impartner-hex-modal-footer-gap) * .5) calc(var(--impartner-hex-modal-padding) - var(--impartner-hex-modal-footer-gap) * .5)}.impdc.modal .modal-theme-destructive .modal-content{display:grid;grid-template-areas:\"modal-icon modal-header\" \"modal-icon modal-body\" \"modal-footer modal-footer\";grid-template-columns:auto 1fr;grid-template-rows:auto 1fr auto}.impdc.modal .modal-theme-destructive .modal-content>.modal-icon{grid-area:modal-icon;margin:var(--impartner-hex-modal-header-padding);margin-right:0;margin-bottom:0}.impdc.modal .modal-theme-destructive .modal-content>.modal-header{grid-area:modal-header;padding-left:16px;margin-top:8px;justify-content:unset}.impdc.modal .modal-theme-destructive .modal-content>.modal-body{grid-area:modal-body;padding-left:16px}.impdc.modal .modal-theme-destructive .modal-content>.modal-footer{grid-area:modal-footer}.impdc.modal .modal-theme-destructive .modal-icon{color:var(--impd-color-red-600);background:var(--impd-color-red-100)}\n"], dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.ButtonComponent, selector: "button[impdcButton]", inputs: ["type", "theme", "text", "disabled", "titleText", "ariaLabel"] }, { kind: "component", type: i4.IconComponent, selector: "impdc-icon, [impdc-icon]", inputs: ["name", "theme", "size"] }] });
|
|
113
140
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ModalComponent, decorators: [{
|
|
114
141
|
type: Component,
|
|
115
|
-
args: [{ selector: 'impdc-modal', template: "<div\n *ngIf=\"valid\"\n class=\"impdc modal\"\n [class.fade]=\"fade\"\n [class.in]=\"fade && shown\"\n [class.show]=\"shown\"\n [class.dismissable]=\"dismissable\"\n tabindex=\"-1\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.escape)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.enter)=\"handleAccept($event); $event.stopPropagation()\">\n <div\n class=\"modal-dialog modal-dialog-centered modal-dialog-scrollable modal-theme-{{\n theme\n }}\"\n [class.modal-sm]=\"size === 'sm'\"\n [class.modal-lg]=\"size === 'lg'\"\n [class.modal-xl]=\"size === 'xl'\"\n [class.modal-fullscreen]=\"size === 'fs'\"\n [class.modal-fullscreen-lg-down]=\"size !== 'fs'\"\n (click)=\"$event.stopPropagation()\">\n <div class=\"modal-content\">\n <div *ngIf=\"showDestructiveIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </div>\n <div #header class=\"modal-header\" [class.hide]=\"hideHeader\">\n <h3 class=\"modal-title\">\n <span *ngIf=\"showWorkingIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </span>\n <span>{{ titleText }}</span>\n </h3>\n <button\n impdcButton\n *ngIf=\"dismissable\"\n theme=\"close\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"></button>\n </div>\n <div #bodyEl class=\"modal-body\">\n <ng-content></ng-content>\n </div>\n <div #footer class=\"modal-footer\">\n <span\n #dismissEl\n *ngIf=\"dismissable\"\n [class.hide]=\"!dismissEl.children.length\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\">\n <ng-content select=\"[dismiss]\"></ng-content>\n </span>\n <span\n #denyEl\n [class.hide]=\"!denyEl.children.length\"\n (click)=\"handleDeny($event); $event.stopPropagation()\">\n <ng-content select=\"[deny]\"></ng-content>\n </span>\n <span\n #acceptEl\n [class.hide]=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <ng-content select=\"[accept]\"></ng-content>\n </span>\n <span\n #acceptFallbackEl\n *ngIf=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <button
|
|
142
|
+
args: [{ selector: 'impdc-modal', template: "<div\n *ngIf=\"valid\"\n class=\"impdc modal\"\n [class.fade]=\"fade\"\n [class.in]=\"fade && shown\"\n [class.show]=\"shown\"\n [class.dismissable]=\"dismissable\"\n tabindex=\"-1\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.escape)=\"handleDismiss($event); $event.stopPropagation()\"\n (keydown.enter)=\"handleAccept($event); $event.stopPropagation()\">\n <div\n class=\"modal-dialog modal-dialog-centered modal-dialog-scrollable modal-theme-{{\n theme\n }}\"\n [class.modal-sm]=\"size === 'sm'\"\n [class.modal-lg]=\"size === 'lg'\"\n [class.modal-xl]=\"size === 'xl'\"\n [class.modal-fullscreen]=\"size === 'fs'\"\n [class.modal-fullscreen-lg-down]=\"size !== 'fs'\"\n (click)=\"$event.stopPropagation()\">\n <div class=\"modal-content\">\n <div *ngIf=\"showDestructiveIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </div>\n <div #header class=\"modal-header\" [class.hide]=\"hideHeader\">\n <h3 class=\"modal-title\">\n <span *ngIf=\"showWorkingIcon\" class=\"modal-icon\">\n <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n </span>\n <span>{{ titleText }}</span>\n </h3>\n <button\n impdcButton\n *ngIf=\"dismissable\"\n theme=\"close\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\"></button>\n </div>\n <div #bodyEl class=\"modal-body\">\n <ng-content></ng-content>\n </div>\n <div #footer class=\"modal-footer\">\n <span\n #dismissEl\n *ngIf=\"dismissable\"\n [class.hide]=\"!dismissEl.children.length\"\n (click)=\"handleDismiss($event); $event.stopPropagation()\">\n <ng-content select=\"[dismiss]\"></ng-content>\n </span>\n <span\n #denyEl\n [class.hide]=\"!denyEl.children.length\"\n (click)=\"handleDeny($event); $event.stopPropagation()\">\n <ng-content select=\"[deny]\"></ng-content>\n </span>\n <span\n #acceptEl\n [class.hide]=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <ng-content select=\"[accept]\"></ng-content>\n </span>\n <span\n #acceptFallbackEl\n *ngIf=\"!acceptEl.children.length\"\n (click)=\"handleAccept($event); $event.stopPropagation()\">\n <button\n impdcButton\n [text]=\"acceptText\"\n [theme]=\"acceptFallbackTheme\"></button>\n </span>\n </div>\n </div>\n </div>\n</div>\n", styles: [".impdc.modal{-webkit-user-select:none;user-select:none}.impdc.modal .modal-icon{display:inline-flex;justify-content:center;align-items:center;height:40px;width:40px;color:var(--impd-color-gray-500);background:var(--impd-color-gray-100);border-radius:50%;flex-shrink:0}.impdc.modal .modal-icon .fa,.impdc.modal .modal-icon .fa-brands,.impdc.modal .modal-icon .fa-duotone,.impdc.modal .modal-icon .fa-light,.impdc.modal .modal-icon .fa-regular,.impdc.modal .modal-icon .fa-solid,.impdc.modal .modal-icon .fa-thin,.impdc.modal .modal-icon .fab,.impdc.modal .modal-icon .fad,.impdc.modal .modal-icon .fal,.impdc.modal .modal-icon .far,.impdc.modal .modal-icon .fas,.impdc.modal .modal-icon .fat{line-height:var(--impartner-hex-modal-title-line-height)}.impdc.modal.fade.in{background-color:transparent}.impdc.modal.show{display:block;opacity:1}.impdc.modal:not(.show),.impdc.modal.hide,.impdc.modal .hide{display:none}.impdc.modal.dismissable{cursor:pointer}.impdc.modal.dismissable .modal-dialog{cursor:default}.impdc.modal .modal-dialog{cursor:default;-webkit-user-select:auto;user-select:auto}.impdc.modal .modal-theme-simple .modal-header,.impdc.modal .modal-theme-destructive .modal-header{border:0;padding-bottom:0}.impdc.modal .modal-theme-simple .modal-body,.impdc.modal .modal-theme-destructive .modal-body{padding:calc(var(--impartner-hex-modal-padding) * .5) var(--impartner-hex-modal-padding) var(--impartner-hex-modal-padding)}.impdc.modal .modal-theme-simple .modal-footer,.impdc.modal .modal-theme-destructive .modal-footer{border:0;padding-top:0}.impdc.modal .modal-theme-working .modal-footer{padding:calc(var(--impartner-hex-modal-padding) * .5 - var(--impartner-hex-modal-footer-gap) * .5) calc(var(--impartner-hex-modal-padding) - var(--impartner-hex-modal-footer-gap) * .5)}.impdc.modal .modal-theme-destructive .modal-content{display:grid;grid-template-areas:\"modal-icon modal-header\" \"modal-icon modal-body\" \"modal-footer modal-footer\";grid-template-columns:auto 1fr;grid-template-rows:auto 1fr auto}.impdc.modal .modal-theme-destructive .modal-content>.modal-icon{grid-area:modal-icon;margin:var(--impartner-hex-modal-header-padding);margin-right:0;margin-bottom:0}.impdc.modal .modal-theme-destructive .modal-content>.modal-header{grid-area:modal-header;padding-left:16px;margin-top:8px;justify-content:unset}.impdc.modal .modal-theme-destructive .modal-content>.modal-body{grid-area:modal-body;padding-left:16px}.impdc.modal .modal-theme-destructive .modal-content>.modal-footer{grid-area:modal-footer}.impdc.modal .modal-theme-destructive .modal-icon{color:var(--impd-color-red-600);background:var(--impd-color-red-100)}\n"] }]
|
|
116
143
|
}], ctorParameters: function () { return [{ type: i1.InteractionService }]; }, propDecorators: { show: [{
|
|
117
144
|
type: Input
|
|
118
145
|
}], theme: [{
|
|
@@ -125,12 +152,23 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
125
152
|
type: Input
|
|
126
153
|
}], titleText: [{
|
|
127
154
|
type: Input
|
|
155
|
+
}], acceptText: [{
|
|
156
|
+
type: Input
|
|
128
157
|
}], dismissable: [{
|
|
129
158
|
type: Input
|
|
130
159
|
}], fade: [{
|
|
131
160
|
type: Input
|
|
132
161
|
}], backdrop: [{
|
|
133
162
|
type: Input
|
|
163
|
+
}], dismissEl: [{
|
|
164
|
+
type: ViewChild,
|
|
165
|
+
args: ['dismissEl', { static: false }]
|
|
166
|
+
}], denyEl: [{
|
|
167
|
+
type: ViewChild,
|
|
168
|
+
args: ['denyEl', { static: false }]
|
|
169
|
+
}], acceptEl: [{
|
|
170
|
+
type: ViewChild,
|
|
171
|
+
args: ['acceptEl', { static: false }]
|
|
134
172
|
}], dismiss: [{
|
|
135
173
|
type: Output
|
|
136
174
|
}], deny: [{
|
|
@@ -138,4 +176,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
138
176
|
}], accept: [{
|
|
139
177
|
type: Output
|
|
140
178
|
}] } });
|
|
141
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.component.js","sourceRoot":"","sources":["../../../../../../projects/design-components/src/lib/modal/modal.component.ts","../../../../../../projects/design-components/src/lib/modal/modal.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,YAAY,EACZ,KAAK,EAGL,MAAM,EACP,MAAM,eAAe,CAAC;AAGvB,OAAO,EAAE,aAAa,EAAE,MAAM,aAAa,CAAC;AAE5C,OAAO,EAAuB,UAAU,EAAe,MAAM,SAAS,CAAC;;;;;;AAEvE;;;;;GAKG;AAMH,MAAM,OAAO,cAAc;IAwGzB,YAAoB,mBAAuC;QAAvC,wBAAmB,GAAnB,mBAAmB,CAAoB;QAvGnD,oBAAe,GAAW,CAAC,CAAC;QAEpC;;WAEG;QAEH,SAAI,GAAY,KAAK,CAAC;QAEtB;;WAEG;QAEH,UAAK,GAAgB,UAAU,CAAC,MAAM,CAAC;QAEvC;;WAEG;QAEH,SAAI,GAAwB,aAAa,CAAC,MAAM,CAAC;QAQjD;;;WAGG;QAEH,cAAS,GAAyB,OAAO,CAAC;QAE1C;;WAEG;QAEH,cAAS,GAAW,EAAE,CAAC;QAQvB;;WAEG;QAEH,SAAI,GAAY,IAAI,CAAC;QAErB;;WAEG;QAEH,aAAQ,GAAY,IAAI,CAAC;QAEzB;;WAEG;QAEH,YAAO,GAAG,IAAI,YAAY,EAAsB,CAAC;QAEjD;;WAEG;QAEH,SAAI,GAAG,IAAI,YAAY,EAAsB,CAAC;QAE9C;;WAEG;QAEH,WAAM,GAAG,IAAI,YAAY,EAAsB,CAAC;QA+B9C,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,mBAAmB,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IAC5D,CAAC;IA9BD,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,cAAc,GAAG,CAAC,CAAC;IACjC,CAAC;IAED,IAAI,KAAK;QACP,OAAO,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,IAAI,CAAC;IACjC,CAAC;IAED,IAAI,eAAe;QACjB,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,KAAK,UAAU,CAAC,OAAO,CAAC;IAC9D,CAAC;IAED,IAAI,mBAAmB;QACrB,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,KAAK,UAAU,CAAC,WAAW,CAAC;IAClE,CAAC;IAED,IAAI,mBAAmB;QACrB,OAAO,IAAI,CAAC,KAAK,KAAK,UAAU,CAAC,WAAW,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS,CAAC;IACtE,CAAC;IAED,IAAI,UAAU;QACZ,OAAO,CAAC,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,WAAW,CAAC;IAC9C,CAAC;IAMD,QAAQ;QACN,IAAI,IAAI,CAAC,WAAW,KAAK,SAAS,IAAI,IAAI,CAAC,KAAK,KAAK,UAAU,CAAC,OAAO,EAAE;YACvE,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;SACzB;QAED,IAAI,IAAI,CAAC,QAAQ,KAAK,SAAS,IAAI,IAAI,CAAC,KAAK,KAAK,UAAU,CAAC,WAAW,EAAE;YACxE,IAAI,CAAC,QAAQ,GAAG,sBAAsB,CAAC;SACxC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC;QACrD,IAAI,CAAC,eAAe,GAAG,CAAC,CAAC;IAC3B,CAAC;IAED,aAAa,CAAC,KAAyB;QACrC,IAAI,CAAC,IAAI,CAAC,WAAW,EAAE;YACrB,OAAO;SACR;QACD,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC;QAClB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;IAED,UAAU,CAAC,KAAyB;QAClC,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC;QAClB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACxB,CAAC;IAED,YAAY,CAAC,KAAyB;QACpC,IAAI,CAAC,IAAI,GAAG,KAAK,CAAC;QAClB,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC1B,CAAC;;4GA3IU,cAAc;gGAAd,cAAc,kTCzB3B,gnFAuEA;4FD9Ca,cAAc;kBAL1B,SAAS;+BACE,aAAa;yGAWvB,IAAI;sBADH,KAAK;gBAON,KAAK;sBADJ,KAAK;gBAON,IAAI;sBADH,KAAK;gBAON,QAAQ;sBADP,KAAK;gBAQN,SAAS;sBADR,KAAK;gBAON,SAAS;sBADR,KAAK;gBAON,WAAW;sBADV,KAAK;gBAON,IAAI;sBADH,KAAK;gBAON,QAAQ;sBADP,KAAK;gBAON,OAAO;sBADN,MAAM;gBAOP,IAAI;sBADH,MAAM;gBAOP,MAAM;sBADL,MAAM","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output\n} from '@angular/core';\nimport { FontAwesomeIcon, FontAwesomeIconTheme } from '../../constants';\nimport { InteractionService } from '../../services';\nimport { ComponentSize } from '../../types';\nimport { ButtonTheme } from '../button';\nimport { ModalComponentSizes, ModalTheme, ModalThemes } from './types';\n\n/**\n * The `ModalComponent` (`<impdc-modal`) displays over the content of the page when a user action is required.\n * To use, import `ModalModule` or another module that imports and exports that module from `@impartner/design-components`.\n * `ModalModule` imports and exports BackdropModule, [ButtonModule](./?path=/docs/design-components-button),\n * and [IconModule](./?path=/docs/design-components-icon).\n */\n@Component({\n  selector: 'impdc-modal',\n  templateUrl: './modal.component.html',\n  styleUrls: ['./modal.component.scss']\n})\nexport class ModalComponent implements OnInit, OnDestroy {\n  private _interactableId: number = 0;\n\n  /**\n   * Determines if the Modal should be shown immediately.\n   */\n  @Input()\n  show: boolean = false;\n\n  /**\n   * The theme of the Modal. Affects the layout of the Modal as well as some colors.\n   */\n  @Input()\n  theme: ModalThemes = ModalTheme.Simple;\n\n  /**\n   * The size of the Modal.\n   */\n  @Input()\n  size: ModalComponentSizes = ComponentSize.Medium;\n\n  /**\n   * The Font Awesome 5 icon name. Only supported when the theme is Working or Destructive.\n   */\n  @Input()\n  iconName?: FontAwesomeIcon;\n\n  /**\n   * The Font Awesome 5 icon style. Fewer icons support the Regular theme and will fallback to Solid\n   * if not supported.\n   */\n  @Input()\n  iconTheme: FontAwesomeIconTheme = 'light';\n\n  /**\n   * The subject text of the Modal.\n   */\n  @Input()\n  titleText: string = '';\n\n  /**\n   * Allows the Modal to be dismissed. This is different from `accept` and `deny`. If a Modal is meant to be re-opened frequently it should be dismissable.\n   */\n  @Input()\n  dismissable?: boolean;\n\n  /**\n   * Determines if the Modal should fade in/out.\n   */\n  @Input()\n  fade: boolean = true;\n\n  /**\n   * Shows a backdrop behind the Modal.\n   */\n  @Input()\n  backdrop: boolean = true;\n\n  /**\n   * Event emitted when the Modal content is dismissed.\n   */\n  @Output()\n  dismiss = new EventEmitter<MouseEvent | Event>();\n\n  /**\n   * Event emitted when the Modal content is denied.\n   */\n  @Output()\n  deny = new EventEmitter<MouseEvent | Event>();\n\n  /**\n   * Event emitted when the Modal content is accepted.\n   */\n  @Output()\n  accept = new EventEmitter<MouseEvent | Event>();\n\n  get interactableId(): number {\n    return this._interactableId;\n  }\n\n  get valid(): boolean {\n    return this.interactableId > 0;\n  }\n\n  get shown(): boolean {\n    return this.valid && this.show;\n  }\n\n  get showWorkingIcon(): boolean {\n    return !!this.iconName && this.theme === ModalTheme.Working;\n  }\n\n  get showDestructiveIcon(): boolean {\n    return !!this.iconName && this.theme === ModalTheme.Destructive;\n  }\n\n  get acceptFallbackTheme(): ButtonTheme {\n    return this.theme === ModalTheme.Destructive ? 'danger' : 'primary';\n  }\n\n  get hideHeader(): boolean {\n    return !this.titleText && !this.dismissable;\n  }\n\n  constructor(private _interactionService: InteractionService) {\n    this._interactableId = this._interactionService.add(this);\n  }\n\n  ngOnInit(): void {\n    if (this.dismissable === undefined && this.theme === ModalTheme.Working) {\n      this.dismissable = true;\n    }\n\n    if (this.iconName === undefined && this.theme === ModalTheme.Destructive) {\n      this.iconName = 'exclamation-triangle';\n    }\n  }\n\n  ngOnDestroy(): void {\n    this._interactionService.remove(this.interactableId);\n    this._interactableId = 0;\n  }\n\n  handleDismiss(event: MouseEvent | Event): void {\n    if (!this.dismissable) {\n      return;\n    }\n    this.show = false;\n    this.dismiss.emit(event);\n  }\n\n  handleDeny(event: MouseEvent | Event): void {\n    this.show = false;\n    this.deny.emit(event);\n  }\n\n  handleAccept(event: MouseEvent | Event): void {\n    this.show = false;\n    this.accept.emit(event);\n  }\n}\n","<div\n  *ngIf=\"valid\"\n  class=\"impdc modal\"\n  [class.fade]=\"fade\"\n  [class.in]=\"fade && shown\"\n  [class.show]=\"shown\"\n  [class.dismissable]=\"dismissable\"\n  tabindex=\"-1\"\n  (click)=\"handleDismiss($event); $event.stopPropagation()\"\n  (keydown.escape)=\"handleDismiss($event); $event.stopPropagation()\"\n  (keydown.enter)=\"handleAccept($event); $event.stopPropagation()\">\n  <div\n    class=\"modal-dialog modal-dialog-centered modal-dialog-scrollable modal-theme-{{\n      theme\n    }}\"\n    [class.modal-sm]=\"size === 'sm'\"\n    [class.modal-lg]=\"size === 'lg'\"\n    [class.modal-xl]=\"size === 'xl'\"\n    [class.modal-fullscreen]=\"size === 'fs'\"\n    [class.modal-fullscreen-lg-down]=\"size !== 'fs'\"\n    (click)=\"$event.stopPropagation()\">\n    <div class=\"modal-content\">\n      <div *ngIf=\"showDestructiveIcon\" class=\"modal-icon\">\n        <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n      </div>\n      <div #header class=\"modal-header\" [class.hide]=\"hideHeader\">\n        <h3 class=\"modal-title\">\n          <span *ngIf=\"showWorkingIcon\" class=\"modal-icon\">\n            <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n          </span>\n          <span>{{ titleText }}</span>\n        </h3>\n        <button\n          impdcButton\n          *ngIf=\"dismissable\"\n          theme=\"close\"\n          (click)=\"handleDismiss($event); $event.stopPropagation()\"></button>\n      </div>\n      <div #bodyEl class=\"modal-body\">\n        <ng-content></ng-content>\n      </div>\n      <div #footer class=\"modal-footer\">\n        <span\n          #dismissEl\n          *ngIf=\"dismissable\"\n          [class.hide]=\"!dismissEl.children.length\"\n          (click)=\"handleDismiss($event); $event.stopPropagation()\">\n          <ng-content select=\"[dismiss]\"></ng-content>\n        </span>\n        <span\n          #denyEl\n          [class.hide]=\"!denyEl.children.length\"\n          (click)=\"handleDeny($event); $event.stopPropagation()\">\n          <ng-content select=\"[deny]\"></ng-content>\n        </span>\n        <span\n          #acceptEl\n          [class.hide]=\"!acceptEl.children.length\"\n          (click)=\"handleAccept($event); $event.stopPropagation()\">\n          <ng-content select=\"[accept]\"></ng-content>\n        </span>\n        <span\n          #acceptFallbackEl\n          *ngIf=\"!acceptEl.children.length\"\n          (click)=\"handleAccept($event); $event.stopPropagation()\">\n          <button impdcButton text=\"OK\" [theme]=\"acceptFallbackTheme\"></button>\n        </span>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
179
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal.component.js","sourceRoot":"","sources":["../../../../../../projects/design-components/src/lib/modal/modal.component.ts","../../../../../../projects/design-components/src/lib/modal/modal.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  OnInit,\n  Output,\n  ViewChild\n} from '@angular/core';\nimport { FontAwesomeIcon, FontAwesomeIconTheme } from '../../constants';\nimport { InteractionService } from '../../services';\nimport { ComponentSize } from '../../types';\nimport { ButtonTheme } from '../button';\nimport { ModalComponentSizes, ModalTheme, ModalThemes } from './types';\n\n/**\n * The `ModalComponent` (`<impdc-modal`) displays over the content of the page when a user action is required.\n * To use, import `ModalModule` or another module that imports and exports that module from `@impartner/design-components`.\n * `ModalModule` imports and exports BackdropModule, [ButtonModule](./?path=/docs/design-components-button),\n * and [IconModule](./?path=/docs/design-components-icon).\n */\n@Component({\n  selector: 'impdc-modal',\n  templateUrl: './modal.component.html',\n  styleUrls: ['./modal.component.scss']\n})\nexport class ModalComponent implements OnInit, OnDestroy {\n  private _interactableId = 0;\n  private _show = false;\n\n  /**\n   * Determines if the Modal should be shown immediately.\n   */\n  @Input()\n  set show(show: boolean) {\n    this._show = show;\n  }\n\n  /**\n   * The theme of the Modal. Affects the layout of the Modal as well as some colors.\n   */\n  @Input()\n  theme: ModalThemes = ModalTheme.Simple;\n\n  /**\n   * The size of the Modal.\n   */\n  @Input()\n  size: ModalComponentSizes = ComponentSize.Medium;\n\n  /**\n   * The Font Awesome 5 icon name. Only supported when the theme is Working or Destructive.\n   */\n  @Input()\n  iconName?: FontAwesomeIcon;\n\n  /**\n   * The Font Awesome 5 icon style. Fewer icons support the Regular theme and will fallback to Solid\n   * if not supported.\n   */\n  @Input()\n  iconTheme: FontAwesomeIconTheme = 'light';\n\n  /**\n   * The subject text of the Modal.\n   */\n  @Input()\n  titleText = '';\n\n  /**\n   * The fallback button text to accept the Modal. This is bypassed if you content-project the accept button.\n   */\n  @Input()\n  acceptText = 'OK';\n\n  /**\n   * Allows the Modal to be dismissed. This is different from `accept` and `deny`. If a Modal is meant to be re-opened frequently it should be dismissable.\n   */\n  @Input()\n  dismissable?: boolean;\n\n  /**\n   * Determines if the Modal should fade in/out.\n   */\n  @Input()\n  fade = true;\n\n  /**\n   * Shows a backdrop behind the Modal.\n   */\n  @Input()\n  backdrop = true;\n\n  /**\n   * Contains the dismiss button content projection.\n   */\n  @ViewChild('dismissEl', { static: false })\n  dismissEl!: ElementRef;\n\n  /**\n   * Contains the deny button content projection.\n   */\n  @ViewChild('denyEl', { static: false })\n  denyEl!: ElementRef;\n\n  /**\n   * Contains the accept button content projection.\n   */\n  @ViewChild('acceptEl', { static: false })\n  acceptEl!: ElementRef;\n\n  /**\n   * Event emitted when the Modal content is dismissed.\n   */\n  @Output()\n  dismiss = new EventEmitter<MouseEvent | Event>();\n\n  /**\n   * Event emitted when the Modal content is denied.\n   */\n  @Output()\n  deny = new EventEmitter<MouseEvent | Event>();\n\n  /**\n   * Event emitted when the Modal content is accepted.\n   */\n  @Output()\n  accept = new EventEmitter<MouseEvent | Event>();\n\n  constructor(private _interactionService: InteractionService) {\n    this._interactableId = this._interactionService.add(this);\n  }\n\n  get interactableId(): number {\n    return this._interactableId;\n  }\n\n  get valid(): boolean {\n    return this.interactableId > 0;\n  }\n\n  get shown(): boolean {\n    return this.valid && this._show;\n  }\n\n  get showWorkingIcon(): boolean {\n    return !!this.iconName && this.theme === ModalTheme.Working;\n  }\n\n  get showDestructiveIcon(): boolean {\n    return !!this.iconName && this.theme === ModalTheme.Destructive;\n  }\n\n  get acceptFallbackTheme(): ButtonTheme {\n    return this.theme === ModalTheme.Destructive ? 'danger' : 'primary';\n  }\n\n  get hideHeader(): boolean {\n    return !this.titleText && !this.dismissable;\n  }\n\n  ngOnInit(): void {\n    if (this.dismissable === undefined && this.theme === ModalTheme.Working) {\n      this.dismissable = true;\n    }\n\n    if (this.iconName === undefined && this.theme === ModalTheme.Destructive) {\n      this.iconName = 'exclamation-triangle';\n    }\n  }\n\n  ngOnDestroy(): void {\n    this._interactionService.remove(this.interactableId);\n    this._interactableId = 0;\n  }\n\n  protected handleDismiss(event: MouseEvent | Event): void {\n    if (\n      !this.dismissable ||\n      !!this.dismissEl?.nativeElement?.firstChild?.disabled\n    ) {\n      return;\n    }\n    this._show = false;\n    this.dismiss.emit(event);\n  }\n\n  protected handleDeny(event: MouseEvent | Event): void {\n    if (!!this.denyEl?.nativeElement?.firstChild?.disabled) {\n      return;\n    }\n    this._show = false;\n    this.deny.emit(event);\n  }\n\n  protected handleAccept(event: MouseEvent | Event): void {\n    console.log('this.acceptEl', this.acceptEl);\n    if (!!this.acceptEl?.nativeElement?.firstChild?.disabled) {\n      return;\n    }\n    this._show = false;\n    this.accept.emit(event);\n  }\n\n  /**\n   * Opens/shows the modal programmatically.\n   */\n  open(): void {\n    this._show = true;\n  }\n\n  /**\n   * Closes/hides the modal programmatically.\n   */\n  close(): void {\n    this._show = false;\n  }\n}\n","<div\n  *ngIf=\"valid\"\n  class=\"impdc modal\"\n  [class.fade]=\"fade\"\n  [class.in]=\"fade && shown\"\n  [class.show]=\"shown\"\n  [class.dismissable]=\"dismissable\"\n  tabindex=\"-1\"\n  (click)=\"handleDismiss($event); $event.stopPropagation()\"\n  (keydown.escape)=\"handleDismiss($event); $event.stopPropagation()\"\n  (keydown.enter)=\"handleAccept($event); $event.stopPropagation()\">\n  <div\n    class=\"modal-dialog modal-dialog-centered modal-dialog-scrollable modal-theme-{{\n      theme\n    }}\"\n    [class.modal-sm]=\"size === 'sm'\"\n    [class.modal-lg]=\"size === 'lg'\"\n    [class.modal-xl]=\"size === 'xl'\"\n    [class.modal-fullscreen]=\"size === 'fs'\"\n    [class.modal-fullscreen-lg-down]=\"size !== 'fs'\"\n    (click)=\"$event.stopPropagation()\">\n    <div class=\"modal-content\">\n      <div *ngIf=\"showDestructiveIcon\" class=\"modal-icon\">\n        <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n      </div>\n      <div #header class=\"modal-header\" [class.hide]=\"hideHeader\">\n        <h3 class=\"modal-title\">\n          <span *ngIf=\"showWorkingIcon\" class=\"modal-icon\">\n            <span impdc-icon [name]=\"iconName\" [theme]=\"iconTheme\"></span>\n          </span>\n          <span>{{ titleText }}</span>\n        </h3>\n        <button\n          impdcButton\n          *ngIf=\"dismissable\"\n          theme=\"close\"\n          (click)=\"handleDismiss($event); $event.stopPropagation()\"></button>\n      </div>\n      <div #bodyEl class=\"modal-body\">\n        <ng-content></ng-content>\n      </div>\n      <div #footer class=\"modal-footer\">\n        <span\n          #dismissEl\n          *ngIf=\"dismissable\"\n          [class.hide]=\"!dismissEl.children.length\"\n          (click)=\"handleDismiss($event); $event.stopPropagation()\">\n          <ng-content select=\"[dismiss]\"></ng-content>\n        </span>\n        <span\n          #denyEl\n          [class.hide]=\"!denyEl.children.length\"\n          (click)=\"handleDeny($event); $event.stopPropagation()\">\n          <ng-content select=\"[deny]\"></ng-content>\n        </span>\n        <span\n          #acceptEl\n          [class.hide]=\"!acceptEl.children.length\"\n          (click)=\"handleAccept($event); $event.stopPropagation()\">\n          <ng-content select=\"[accept]\"></ng-content>\n        </span>\n        <span\n          #acceptFallbackEl\n          *ngIf=\"!acceptEl.children.length\"\n          (click)=\"handleAccept($event); $event.stopPropagation()\">\n          <button\n            impdcButton\n            [text]=\"acceptText\"\n            [theme]=\"acceptFallbackTheme\"></button>\n        </span>\n      </div>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -131,4 +131,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
131
131
|
}], goToPage: [{
|
|
132
132
|
type: Output
|
|
133
133
|
}] } });
|
|
134
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.component.js","sourceRoot":"","sources":["../../../../../../projects/design-components/src/lib/pagination/pagination.component.ts","../../../../../../projects/design-components/src/lib/pagination/pagination.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { sanitize } from '../../utilities';\n\n/**\n * The `PaginationComponent` (`<impdc-pagination` or `<div impdc-pagination`) is a control for paged content such as tables or lists.\n * To use, import `PaginationModule` or another module that imports and exports that module from `@impartner/design-components`.\n * `ModalModule` imports and exports ImpdcFormsModule and [ButtonModule](./?path=/docs/design-components-button).\n */\n@Component({\n  selector: 'impdc-pagination, div[impdc-pagination]',\n  templateUrl: './pagination.component.html',\n  styleUrls: ['./pagination.component.scss']\n})\nexport class PaginationComponent {\n  private _summary: string =\n    'Showing {firstPageResult} to {lastPageResult} of {total} results';\n\n  /**\n   * The current page.\n   */\n  @Input()\n  public page: number = 0;\n\n  /**\n   * The number of items shown per page. Items are commonly displayed as rows in a table.\n   */\n  @Input()\n  public perPage: number = 10;\n\n  /**\n   * The total number of items.\n   */\n  @Input()\n  public total: number = 0;\n\n  /**\n   * A formattable text summary of pagination data. `{firstPageResult}` The first item within the current page.\n   * `{lastPageResult}` The last item within the current page. `{total}` The total number of items.\n   */\n  @Input()\n  public set summary(summary: string) {\n    this._summary = summary;\n  }\n\n  public get summary(): string {\n    return sanitize(this._summary)\n      .replace('{firstPageResult}', `<b>${this.firstPageResult}</b>`)\n      .replace('{lastPageResult}', `<b>${this.lastPageResult}</b>`)\n      .replace('{total}', `<b>${this.total}</b>`);\n  }\n\n  /**\n   * Emitted when the page number is changed, specifying the target page number.\n   */\n  @Output() public readonly goToPage = new EventEmitter<number>();\n\n  public get totalPages(): number {\n    if (this.total < 1 || this.perPage < 1) {\n      return 0;\n    }\n\n    if (this.total < this.perPage) {\n      return 1;\n    }\n\n    return Math.ceil(this.total / this.perPage);\n  }\n\n  public get firstPageResult(): number {\n    if (this.total < 1 || this.page < 1 || this.perPage < 1) {\n      return 0;\n    }\n\n    if (this.page === 1) {\n      return 1;\n    }\n\n    if (this.page > Math.ceil(this.total / this.perPage)) {\n      return this.total;\n    }\n\n    return (this.page - 1) * this.perPage + 1;\n  }\n\n  public get lastPageResult(): number {\n    if (this.total <= 0 || this.page < 1 || this.perPage < 1) {\n      return 0;\n    }\n\n    if (this.page === 1) {\n      if (this.total < this.perPage) {\n        return this.total;\n      } else {\n        return this.perPage;\n      }\n    }\n\n    if (this.total < this.perPage * this.page) {\n      return this.total;\n    } else {\n      return this.perPage * this.page;\n    }\n  }\n\n  public goToPreviousPage(): void {\n    if (this.page <= 1) {\n      return;\n    }\n\n    const previousPage = this.page - 1;\n    this.page = previousPage;\n\n    this.goToPage.emit(previousPage);\n  }\n\n  public goToNextPage(): void {\n    if (this.page >= this.totalPages) {\n      return;\n    }\n\n    const nextPage = this.page + 1;\n    this.page = nextPage;\n\n    this.goToPage.emit(nextPage);\n  }\n\n  public goToExactPageFromEvent({ target }: FocusEvent): void {\n    this.goToExactPage((target as HTMLInputElement)?.value);\n  }\n\n  public goToExactPage(page: number | string): void {\n    if (!page) {\n      return;\n    }\n\n    let allowedPageNumber = Number(page);\n\n    if (allowedPageNumber < 0) {\n      allowedPageNumber = 1;\n    }\n\n    if (allowedPageNumber > this.totalPages) {\n      allowedPageNumber = this.totalPages;\n    }\n\n    this.page = allowedPageNumber;\n\n    this.goToPage.emit(allowedPageNumber);\n  }\n}\n","<div class=\"pagination-container\">\n  <div class=\"mobile-pagination\">\n    <button\n      impdcButton\n      theme=\"secondary\"\n      [disabled]=\"page <= 1\"\n      (click)=\"goToPreviousPage()\">\n      <i class=\"fas fa-chevron-left\"></i>\n    </button>\n    <button\n      impdcButton\n      theme=\"secondary\"\n      [disabled]=\"page === totalPages\"\n      (click)=\"goToNextPage()\">\n      <i class=\"fas fa-chevron-right\"></i>\n    </button>\n  </div>\n\n  <div class=\"desktop-pagination\">\n    <div>\n      <span class=\"summary\" [innerHTML]=\"summary\"></span>\n    </div>\n    <div class=\"pagination-actions\">\n      <button\n        impdcButton\n        theme=\"secondary\"\n        aria-label=\"First Page\"\n        class=\"pagination-first-last-btns\"\n        (click)=\"goToExactPage(1)\"\n        [disabled]=\"page <= 1\">\n        <i class=\"fas fa-chevron-left first\"></i>\n        <i class=\"fas fa-chevron-left\"></i>\n        <!-- 1 -->\n      </button>\n      <button\n        impdcButton\n        theme=\"secondary\"\n        class=\"previous\"\n        aria-label=\"Previous\"\n        (click)=\"goToPreviousPage()\"\n        [disabled]=\"page <= 1\">\n        <i class=\"fas fa-chevron-left\"></i>\n      </button>\n      <div class=\"page-change\">\n        <input\n          impdcInput\n          maxlength=\"4\"\n          [value]=\"page\"\n          (blur)=\"goToExactPageFromEvent($event)\" />\n        <span class=\"total-pages\"> / {{ totalPages }} </span>\n      </div>\n      <button\n        impdcButton\n        theme=\"secondary\"\n        aria-label=\"Next\"\n        (click)=\"goToNextPage()\"\n        [disabled]=\"page === totalPages\">\n        <i class=\"fas fa-chevron-right\"></i>\n      </button>\n      <button\n        impdcButton\n        theme=\"secondary\"\n        class=\"pagination-first-last-btns\"\n        aria-label=\"Last Page\"\n        (click)=\"goToExactPage(totalPages)\"\n        [disabled]=\"page === totalPages\">\n        <i class=\"fas fa-chevron-right\"></i>\n        <i class=\"fas fa-chevron-right last\"></i>\n        <!-- {{ totalPages }} -->\n      </button>\n    </div>\n  </div>\n</div>\n"]}
|
|
134
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pagination.component.js","sourceRoot":"","sources":["../../../../../../projects/design-components/src/lib/pagination/pagination.component.ts","../../../../../../projects/design-components/src/lib/pagination/pagination.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, EventEmitter, Input, Output } from '@angular/core';\nimport { sanitize } from '../../utilities';\n\n/**\n * The `PaginationComponent` (`<impdc-pagination` or `<div impdc-pagination`) is a control for paged content such as tables or lists.\n * To use, import `PaginationModule` or another module that imports and exports that module from `@impartner/design-components`.\n * `ModalModule` imports and exports ImpdcFormsModule and [ButtonModule](./?path=/docs/design-components-button).\n */\n@Component({\n  selector: 'impdc-pagination, div[impdc-pagination]',\n  templateUrl: './pagination.component.html',\n  styleUrls: ['./pagination.component.scss']\n})\nexport class PaginationComponent {\n  private _summary =\n    'Showing {firstPageResult} to {lastPageResult} of {total} results';\n\n  /**\n   * The current page.\n   */\n  @Input()\n  public page = 0;\n\n  /**\n   * The number of items shown per page. Items are commonly displayed as rows in a table.\n   */\n  @Input()\n  public perPage = 10;\n\n  /**\n   * The total number of items.\n   */\n  @Input()\n  public total = 0;\n\n  /**\n   * A formattable text summary of pagination data. `{firstPageResult}` The first item within the current page.\n   * `{lastPageResult}` The last item within the current page. `{total}` The total number of items.\n   */\n  @Input()\n  public set summary(summary: string) {\n    this._summary = summary;\n  }\n\n  public get summary(): string {\n    return sanitize(this._summary)\n      .replace('{firstPageResult}', `<b>${this.firstPageResult}</b>`)\n      .replace('{lastPageResult}', `<b>${this.lastPageResult}</b>`)\n      .replace('{total}', `<b>${this.total}</b>`);\n  }\n\n  /**\n   * Emitted when the page number is changed, specifying the target page number.\n   */\n  @Output() public readonly goToPage = new EventEmitter<number>();\n\n  public get totalPages(): number {\n    if (this.total < 1 || this.perPage < 1) {\n      return 0;\n    }\n\n    if (this.total < this.perPage) {\n      return 1;\n    }\n\n    return Math.ceil(this.total / this.perPage);\n  }\n\n  public get firstPageResult(): number {\n    if (this.total < 1 || this.page < 1 || this.perPage < 1) {\n      return 0;\n    }\n\n    if (this.page === 1) {\n      return 1;\n    }\n\n    if (this.page > Math.ceil(this.total / this.perPage)) {\n      return this.total;\n    }\n\n    return (this.page - 1) * this.perPage + 1;\n  }\n\n  public get lastPageResult(): number {\n    if (this.total <= 0 || this.page < 1 || this.perPage < 1) {\n      return 0;\n    }\n\n    if (this.page === 1) {\n      if (this.total < this.perPage) {\n        return this.total;\n      } else {\n        return this.perPage;\n      }\n    }\n\n    if (this.total < this.perPage * this.page) {\n      return this.total;\n    } else {\n      return this.perPage * this.page;\n    }\n  }\n\n  public goToPreviousPage(): void {\n    if (this.page <= 1) {\n      return;\n    }\n\n    const previousPage = this.page - 1;\n    this.page = previousPage;\n\n    this.goToPage.emit(previousPage);\n  }\n\n  public goToNextPage(): void {\n    if (this.page >= this.totalPages) {\n      return;\n    }\n\n    const nextPage = this.page + 1;\n    this.page = nextPage;\n\n    this.goToPage.emit(nextPage);\n  }\n\n  public goToExactPageFromEvent({ target }: FocusEvent): void {\n    this.goToExactPage((target as HTMLInputElement)?.value);\n  }\n\n  public goToExactPage(page: number | string): void {\n    if (!page) {\n      return;\n    }\n\n    let allowedPageNumber = Number(page);\n\n    if (allowedPageNumber < 0) {\n      allowedPageNumber = 1;\n    }\n\n    if (allowedPageNumber > this.totalPages) {\n      allowedPageNumber = this.totalPages;\n    }\n\n    this.page = allowedPageNumber;\n\n    this.goToPage.emit(allowedPageNumber);\n  }\n}\n","<div class=\"pagination-container\">\n  <div class=\"mobile-pagination\">\n    <button\n      impdcButton\n      theme=\"secondary\"\n      [disabled]=\"page <= 1\"\n      (click)=\"goToPreviousPage()\">\n      <i class=\"fas fa-chevron-left\"></i>\n    </button>\n    <button\n      impdcButton\n      theme=\"secondary\"\n      [disabled]=\"page === totalPages\"\n      (click)=\"goToNextPage()\">\n      <i class=\"fas fa-chevron-right\"></i>\n    </button>\n  </div>\n\n  <div class=\"desktop-pagination\">\n    <div>\n      <span class=\"summary\" [innerHTML]=\"summary\"></span>\n    </div>\n    <div class=\"pagination-actions\">\n      <button\n        impdcButton\n        theme=\"secondary\"\n        aria-label=\"First Page\"\n        class=\"pagination-first-last-btns\"\n        (click)=\"goToExactPage(1)\"\n        [disabled]=\"page <= 1\">\n        <i class=\"fas fa-chevron-left first\"></i>\n        <i class=\"fas fa-chevron-left\"></i>\n        <!-- 1 -->\n      </button>\n      <button\n        impdcButton\n        theme=\"secondary\"\n        class=\"previous\"\n        aria-label=\"Previous\"\n        (click)=\"goToPreviousPage()\"\n        [disabled]=\"page <= 1\">\n        <i class=\"fas fa-chevron-left\"></i>\n      </button>\n      <div class=\"page-change\">\n        <input\n          impdcInput\n          maxlength=\"4\"\n          [value]=\"page\"\n          (blur)=\"goToExactPageFromEvent($event)\" />\n        <span class=\"total-pages\"> / {{ totalPages }} </span>\n      </div>\n      <button\n        impdcButton\n        theme=\"secondary\"\n        aria-label=\"Next\"\n        (click)=\"goToNextPage()\"\n        [disabled]=\"page === totalPages\">\n        <i class=\"fas fa-chevron-right\"></i>\n      </button>\n      <button\n        impdcButton\n        theme=\"secondary\"\n        class=\"pagination-first-last-btns\"\n        aria-label=\"Last Page\"\n        (click)=\"goToExactPage(totalPages)\"\n        [disabled]=\"page === totalPages\">\n        <i class=\"fas fa-chevron-right\"></i>\n        <i class=\"fas fa-chevron-right last\"></i>\n        <!-- {{ totalPages }} -->\n      </button>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -32,4 +32,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
32
32
|
}], progressPercentage: [{
|
|
33
33
|
type: Input
|
|
34
34
|
}] } });
|
|
35
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHJvZ3Jlc3MtYmFyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2Rlc2lnbi1jb21wb25lbnRzL3NyYy9saWIvcHJvZ3Jlc3MtYmFyL3Byb2dyZXNzLWJhci5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9kZXNpZ24tY29tcG9uZW50cy9zcmMvbGliL3Byb2dyZXNzLWJhci9wcm9ncmVzcy1iYXIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBRWpEOzs7R0FHRztBQU1ILE1BQU0sT0FBTyxvQkFBb0I7SUFtQi9CO1FBbEJBOztXQUVHO1FBRUksY0FBUyxHQUFHLEVBQUUsQ0FBQztRQUV0Qjs7V0FFRztRQUVJLGVBQVUsR0FBRyxFQUFFLENBQUM7UUFFdkI7O1dBRUc7UUFFSSx1QkFBa0IsR0FBRyxDQUFDLENBQUM7SUFFZixDQUFDOztrSEFuQkwsb0JBQW9CO3NHQUFwQixvQkFBb0Isa0tDWGpDLDhkQW1CQTs0RkRSYSxvQkFBb0I7a0JBTGhDLFNBQVM7K0JBQ0Usb0JBQW9COzBFQVN2QixTQUFTO3NCQURmLEtBQUs7Z0JBT0MsVUFBVTtzQkFEaEIsS0FBSztnQkFPQyxrQkFBa0I7c0JBRHhCLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbi8qKlxuICogVGhlIGBQcm9ncmVzc0JhckNvbXBvbmVudGAgKGA8aW1wZGMtcHJvZ3Jlc3MtYmFyYCkgaXMgYSBkaXNwbGF5LW9ubHkgc2xpZGVyIHRvIHNob3cgcHJvZ3Jlc3MuXG4gKiBUbyB1c2UsIGltcG9ydCBgUHJvZ3Jlc3NCYXJNb2R1bGVgIG9yIGFub3RoZXIgbW9kdWxlIHRoYXQgaW1wb3J0cyBhbmQgZXhwb3J0cyB0aGF0IG1vZHVsZSBmcm9tIGBAaW1wYXJ0bmVyL2Rlc2lnbi1jb21wb25lbnRzYC5cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnaW1wZGMtcHJvZ3Jlc3MtYmFyJyxcbiAgdGVtcGxhdGVVcmw6ICcuL3Byb2dyZXNzLWJhci5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL3Byb2dyZXNzLWJhci5jb21wb25lbnQuc2NzcyddXG59KVxuZXhwb3J0IGNsYXNzIFByb2dyZXNzQmFyQ29tcG9uZW50IHtcbiAgLyoqXG4gICAqIFRoZSBsZWZ0L3N1YmplY3QgdGV4dCBvZiB0aGUgUHJvZ3Jlc3MgQmFyLlxuICAgKi9cbiAgQElucHV0KClcbiAgcHVibGljIGxhYmVsVGV4dCA9ICcnO1xuXG4gIC8qKlxuICAgKiBUaGUgcmlnaHQvc3RhdHVzIHRleHQgb2YgdGhlIFByb2dyZXNzIEJhci5cbiAgICovXG4gIEBJbnB1dCgpXG4gIHB1YmxpYyBzdGF0dXNUZXh0ID0gJyc7XG5cbiAgLyoqXG4gICAqIFRoZSBjdXJyZW50IHZhbHVlIG9mIHRoZSBQcm9ncmVzcyBCYXIgcmVwcmVzZW50ZWQgYXMgYSBudW1iZXIgYmV0d2VlbiAwIGFuZCAxMDAuXG4gICAqL1xuICBASW5wdXQoKVxuICBwdWJsaWMgcHJvZ3Jlc3NQZXJjZW50YWdlID0gMDtcblxuICBjb25zdHJ1Y3RvcigpIHt9XG59XG4iLCI8ZGl2IGNsYXNzPVwicHJvZ3Jlc3MtY29udGFpbmVyXCI+XG4gIDxkaXYgY2xhc3M9XCJwcm9ncmVzcy10ZXh0XCI+XG4gICAgPGRpdiBjbGFzcz1cInRleHQtbGVmdFwiPlxuICAgICAge3sgbGFiZWxUZXh0IH19XG4gICAgPC9kaXY+XG4gICAgPGRpdiBjbGFzcz1cInRleHQtcmlnaHRcIj5cbiAgICAgIHt7IHN0YXR1c1RleHQgfX1cbiAgICA8L2Rpdj5cbiAgPC9kaXY+XG4gIDxkaXYgY2xhc3M9XCJiYXItYmdcIj5cbiAgICA8ZGl2XG4gICAgICBjbGFzcz1cImJhclwiXG4gICAgICByb2xlPVwicHJvZ3Jlc3NiYXJcIlxuICAgICAgW3N0eWxlLndpZHRoLiVdPVwicHJvZ3Jlc3NQZXJjZW50YWdlXCJcbiAgICAgIFthdHRyLmFyaWEtdmFsdWVub3ddPVwicHJvZ3Jlc3NQZXJjZW50YWdlXCJcbiAgICAgIGFyaWEtdmFsdWVtaW49XCIwXCJcbiAgICAgIGFyaWEtdmFsdWVtYXg9XCIxMDBcIj48L2Rpdj5cbiAgPC9kaXY+XG48L2Rpdj5cbiJdfQ==
|
|
@@ -9,8 +9,10 @@ import * as i2 from "../size-detection/directive/size-detector.directive";
|
|
|
9
9
|
* `ScrollableModule` also imports and exports [BrandingModule](./?path=/docs/design-components-directives-impdcbranded).
|
|
10
10
|
*/
|
|
11
11
|
export class ScrollableComponent {
|
|
12
|
-
constructor(_cd) {
|
|
12
|
+
constructor(_cd, _hostElementRef) {
|
|
13
13
|
this._cd = _cd;
|
|
14
|
+
this._hostElementRef = _hostElementRef;
|
|
15
|
+
this.GET_ANCESTOR_BG_SEARCH_DEPTH = 64;
|
|
14
16
|
/**
|
|
15
17
|
* The base amount to scroll.
|
|
16
18
|
*/
|
|
@@ -54,12 +56,11 @@ export class ScrollableComponent {
|
|
|
54
56
|
this._cd.detectChanges();
|
|
55
57
|
}
|
|
56
58
|
}
|
|
57
|
-
this._getBackgroundColorForButtons();
|
|
58
59
|
}
|
|
59
60
|
ngAfterViewInit() {
|
|
60
61
|
this._width = this.contentRef?.nativeElement.width;
|
|
61
|
-
this._getBackgroundColorForButtons();
|
|
62
62
|
this._cd.detectChanges();
|
|
63
|
+
Promise.resolve().then(() => this._getBackgroundColorForButtons());
|
|
63
64
|
}
|
|
64
65
|
async scroll(direction) {
|
|
65
66
|
if (!this._contentElement) {
|
|
@@ -118,29 +119,21 @@ export class ScrollableComponent {
|
|
|
118
119
|
return promise;
|
|
119
120
|
}
|
|
120
121
|
_getBackgroundColorForButtons() {
|
|
121
|
-
|
|
122
|
-
return;
|
|
123
|
-
}
|
|
124
|
-
const projectedElementCollection = this.contentRef.nativeElement.children;
|
|
125
|
-
if (!projectedElementCollection || !projectedElementCollection.length) {
|
|
126
|
-
return;
|
|
127
|
-
}
|
|
128
|
-
const firstElement = projectedElementCollection[0];
|
|
129
|
-
let backgroundColor = this._getBackGroundColor(firstElement);
|
|
122
|
+
let backgroundColor = this._getBackGroundColor(this._hostElementRef.nativeElement);
|
|
130
123
|
if (!backgroundColor) {
|
|
131
124
|
backgroundColor = this._getAncestorBackgroundColor();
|
|
132
125
|
}
|
|
133
|
-
this._scrollableBgColor
|
|
126
|
+
if (this._scrollableBgColor !== backgroundColor) {
|
|
127
|
+
this._scrollableBgColor = backgroundColor;
|
|
128
|
+
this._cd.detectChanges();
|
|
129
|
+
}
|
|
134
130
|
}
|
|
135
131
|
_getAncestorBackgroundColor() {
|
|
136
|
-
if (!this.contentRef) {
|
|
137
|
-
return null;
|
|
138
|
-
}
|
|
139
132
|
let result = {
|
|
140
|
-
parent: this.
|
|
133
|
+
parent: this._hostElementRef.nativeElement.parentElement,
|
|
141
134
|
backgroundColor: null
|
|
142
135
|
};
|
|
143
|
-
let count =
|
|
136
|
+
let count = this.GET_ANCESTOR_BG_SEARCH_DEPTH;
|
|
144
137
|
function getParentBackgroundColor(currentContext, getBackgroundColor) {
|
|
145
138
|
const result = { parent: null, backgroundColor: null };
|
|
146
139
|
if (currentContext && currentContext.parent) {
|
|
@@ -171,12 +164,12 @@ export class ScrollableComponent {
|
|
|
171
164
|
return element.style.backgroundColor;
|
|
172
165
|
}
|
|
173
166
|
}
|
|
174
|
-
ScrollableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ScrollableComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
167
|
+
ScrollableComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ScrollableComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
175
168
|
ScrollableComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.2.12", type: ScrollableComponent, selector: "impdc-scrollable, div[impdc-scrollable]", inputs: { scrollUnit: "scrollUnit", clickScrollMultiplier: "clickScrollMultiplier" }, host: { properties: { "style.--impdc-scrollable-bg": "this._scrollableBgColor" } }, viewQueries: [{ propertyName: "containerRef", first: true, predicate: ["container"], descendants: true }, { propertyName: "contentRef", first: true, predicate: ["content"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"impdc-scrollable\"\n #container\n impdcSizeDetector\n (resized)=\"widthChanged($event)\">\n <a class=\"impdc-scrollable-left\" (click)=\"scroll(-1)\" *ngIf=\"canScrollLeft\">\n <i class=\"fas fa-chevron-left default\"></i>\n </a>\n <div #content class=\"content-container\">\n <ng-content></ng-content>\n </div>\n <a class=\"impdc-scrollable-right\" (click)=\"scroll(1)\" *ngIf=\"canScrollRight\">\n <i class=\"fas fa-chevron-right default\"></i>\n </a>\n</div>\n", styles: [".impdc-scrollable{position:relative;overflow-y:hidden}.impdc-scrollable .impdc-scrollable-base,.impdc-scrollable .impdc-scrollable-right,.impdc-scrollable .impdc-scrollable-left{position:absolute;top:0px;bottom:0px;display:flex;align-items:center;text-align:center;width:6.9rem;z-index:1;color:var(--impd-color-motion-blue-600, #01606d)}.impdc-scrollable .impdc-scrollable-left{justify-content:left;padding-left:1.5rem;background-image:linear-gradient(to right,var(--impdc-scrollable-bg, #fff) 50%,rgba(255,255,255,.0001) 86.46%)}.impdc-scrollable .impdc-scrollable-right{right:0;padding-right:1.5rem;justify-content:right;background-image:linear-gradient(to left,var(--impdc-scrollable-bg, #fff) 50%,rgba(255,255,255,.0001) 86.46%)}.impdc-scrollable .content-container{overflow-x:hidden;-webkit-overflow-scrolling:touch;white-space:nowrap}:host(.branded) .impdc-scrollable .impdc-scrollable-left,:host(.branded) .impdc-scrollable .impdc-scrollable-right{color:var(--px-primary-color, var(--impd-color-motion-blue-600, #01606d))}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.SizeDetectorDirective, selector: "[impdcSizeDetector]", outputs: ["resized"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
176
169
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: ScrollableComponent, decorators: [{
|
|
177
170
|
type: Component,
|
|
178
171
|
args: [{ selector: 'impdc-scrollable, div[impdc-scrollable]', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div\n class=\"impdc-scrollable\"\n #container\n impdcSizeDetector\n (resized)=\"widthChanged($event)\">\n <a class=\"impdc-scrollable-left\" (click)=\"scroll(-1)\" *ngIf=\"canScrollLeft\">\n <i class=\"fas fa-chevron-left default\"></i>\n </a>\n <div #content class=\"content-container\">\n <ng-content></ng-content>\n </div>\n <a class=\"impdc-scrollable-right\" (click)=\"scroll(1)\" *ngIf=\"canScrollRight\">\n <i class=\"fas fa-chevron-right default\"></i>\n </a>\n</div>\n", styles: [".impdc-scrollable{position:relative;overflow-y:hidden}.impdc-scrollable .impdc-scrollable-base,.impdc-scrollable .impdc-scrollable-right,.impdc-scrollable .impdc-scrollable-left{position:absolute;top:0px;bottom:0px;display:flex;align-items:center;text-align:center;width:6.9rem;z-index:1;color:var(--impd-color-motion-blue-600, #01606d)}.impdc-scrollable .impdc-scrollable-left{justify-content:left;padding-left:1.5rem;background-image:linear-gradient(to right,var(--impdc-scrollable-bg, #fff) 50%,rgba(255,255,255,.0001) 86.46%)}.impdc-scrollable .impdc-scrollable-right{right:0;padding-right:1.5rem;justify-content:right;background-image:linear-gradient(to left,var(--impdc-scrollable-bg, #fff) 50%,rgba(255,255,255,.0001) 86.46%)}.impdc-scrollable .content-container{overflow-x:hidden;-webkit-overflow-scrolling:touch;white-space:nowrap}:host(.branded) .impdc-scrollable .impdc-scrollable-left,:host(.branded) .impdc-scrollable .impdc-scrollable-right{color:var(--px-primary-color, var(--impd-color-motion-blue-600, #01606d))}\n"] }]
|
|
179
|
-
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { scrollUnit: [{
|
|
172
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i0.ElementRef }]; }, propDecorators: { scrollUnit: [{
|
|
180
173
|
type: Input
|
|
181
174
|
}], clickScrollMultiplier: [{
|
|
182
175
|
type: Input
|
|
@@ -190,4 +183,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImpo
|
|
|
190
183
|
type: ViewChild,
|
|
191
184
|
args: ['content', { static: false }]
|
|
192
185
|
}] } });
|
|
193
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.component.js","sourceRoot":"","sources":["../../../../../../projects/design-components/src/lib/scrollable/scrollable.component.ts","../../../../../../projects/design-components/src/lib/scrollable/scrollable.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  ViewChild\n} from '@angular/core';\nimport { IElementSize } from '../size-detection';\n\n/**\n * The `ScrollableComponent` (`<impdc-scrollable` or `<div impdc-scrollable`) is a container that hides items that wouldn't fit\n * with arrows on the left and right to slide between them.\n * To use, import `ScrollableModule` or another module that imports and exports that module from `@impartner/design-components`.\n * `ScrollableModule` also imports and exports [BrandingModule](./?path=/docs/design-components-directives-impdcbranded).\n */\n@Component({\n  selector: 'impdc-scrollable, div[impdc-scrollable]',\n  templateUrl: './scrollable.component.html',\n  styleUrls: ['./scrollable.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScrollableComponent implements OnChanges, AfterViewInit {\n  /**\n   * The base amount to scroll.\n   */\n  @Input()\n  public scrollUnit: number = 20;\n\n  /**\n   * Used in combination with `scrollUnit` to determine how far the Scrollable scrolls when directional arrows are clicked.\n   */\n  @Input()\n  public clickScrollMultiplier = 5;\n\n  @HostBinding('style.--impdc-scrollable-bg')\n  private _scrollableBgColor: string | null = null;\n\n  public get canScrollLeft(): boolean {\n    if (!this._contentElement) {\n      return false;\n    }\n\n    return Math.floor(this._contentElement.scrollLeft) > 0;\n  }\n\n  public get canScrollRight(): boolean {\n    if (!this._contentElement) {\n      return false;\n    }\n\n    return (\n      Math.ceil(this._contentElement.scrollLeft + this._width) <\n      this._contentElement.scrollWidth\n    );\n  }\n\n  private _width: number = 0;\n\n  @ViewChild('container', { static: false })\n  public containerRef: ElementRef | undefined;\n\n  private get _containerElement(): HTMLDivElement | null {\n    if (!this.containerRef) {\n      return null;\n    }\n\n    return this.containerRef.nativeElement;\n  }\n\n  @ViewChild('content', { static: false })\n  public contentRef: ElementRef | undefined;\n\n  private get _contentElement(): HTMLDivElement | null {\n    if (!this.contentRef) {\n      return null;\n    }\n\n    return this.contentRef.nativeElement;\n  }\n\n  constructor(private _cd: ChangeDetectorRef) {}\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    if (changes['scrollUnit'] || changes['clickScrollMultiplier']) {\n      if (this._containerElement) {\n        this._width = this._containerElement?.clientWidth;\n        this._cd.detectChanges();\n      }\n    }\n    this._getBackgroundColorForButtons();\n  }\n\n  public ngAfterViewInit(): void {\n    this._width = this.contentRef?.nativeElement.width;\n    this._getBackgroundColorForButtons();\n    this._cd.detectChanges();\n  }\n\n  public async scroll(direction: number): Promise<void> {\n    if (!this._contentElement) {\n      return;\n    }\n\n    let pixels = direction * this.scrollUnit * this.clickScrollMultiplier;\n    let position = 0;\n\n    if (!pixels) {\n      if (direction === 1) {\n        position = this._contentElement.scrollWidth;\n      }\n    } else {\n      position = this._contentElement.scrollLeft + pixels;\n    }\n\n    await this._scrollToPosition(this._contentElement, position);\n\n    this._cd.detectChanges();\n  }\n\n  public widthChanged(elementSize: IElementSize): void {\n    if (!elementSize || !elementSize.width) {\n      return;\n    }\n    this._width = elementSize.width;\n    this._cd.detectChanges();\n  }\n\n  private async _scrollToPosition(container: HTMLDivElement, position: number) {\n    position = Math.round(position);\n\n    if (container.scrollLeft === position) {\n      return;\n    }\n\n    let resolveFn: (value: unknown) => void;\n    let scrollListener: () => void;\n    let timeoutId: any;\n\n    const promise = new Promise(resolve => {\n      resolveFn = resolve;\n    });\n\n    const finished = () => {\n      container.removeEventListener('scroll', scrollListener);\n      resolveFn(null);\n    };\n\n    scrollListener = () => {\n      clearTimeout(timeoutId);\n\n      // scroll is finished when either the position has been reached, or 100ms have elapsed since the last scroll event\n      if (container.scrollLeft === position) {\n        finished();\n      } else {\n        timeoutId = setTimeout(finished, 100);\n      }\n    };\n\n    container.addEventListener('scroll', scrollListener);\n\n    container.scrollTo({\n      left: position,\n      behavior: 'smooth'\n    });\n\n    return promise;\n  }\n\n  private _getBackgroundColorForButtons(): void {\n    if (!this.contentRef) {\n      return;\n    }\n\n    const projectedElementCollection: HTMLCollection =\n      this.contentRef.nativeElement.children;\n\n    if (!projectedElementCollection || !projectedElementCollection.length) {\n      return;\n    }\n\n    const firstElement = projectedElementCollection[0] as HTMLElement;\n    let backgroundColor: string | null = this._getBackGroundColor(firstElement);\n\n    if (!backgroundColor) {\n      backgroundColor = this._getAncestorBackgroundColor();\n    }\n\n    this._scrollableBgColor = backgroundColor;\n  }\n\n  private _getAncestorBackgroundColor(): string | null {\n    if (!this.contentRef) {\n      return null;\n    }\n\n    type parentBgResult = {\n      parent: HTMLElement | null;\n      backgroundColor: string | null;\n    };\n\n    let result: parentBgResult = {\n      parent: (this.contentRef.nativeElement as HTMLElement).parentElement,\n      backgroundColor: null\n    };\n\n    let count = 6;\n\n    function getParentBackgroundColor(\n      currentContext: parentBgResult,\n      getBackgroundColor: (element: HTMLElement) => string | null\n    ): parentBgResult {\n      const result: parentBgResult = { parent: null, backgroundColor: null };\n      if (currentContext && currentContext.parent) {\n        result.backgroundColor = getBackgroundColor(currentContext.parent);\n\n        if (!result.backgroundColor) {\n          result.parent = currentContext.parent.parentElement;\n        }\n      }\n\n      return result;\n    }\n\n    while (count > 0) {\n      if (result.backgroundColor) {\n        break;\n      }\n      result = getParentBackgroundColor(result, this._getBackGroundColor);\n\n      if (!result.parent) {\n        break;\n      }\n      count = count - 1;\n    }\n\n    return result.backgroundColor;\n  }\n\n  private _getBackGroundColor(element: HTMLElement): string | null {\n    const computedBackgroundColor =\n      getComputedStyle(element)['backgroundColor'];\n\n    if (computedBackgroundColor !== 'rgba(0, 0, 0, 0)') {\n      return computedBackgroundColor;\n    }\n\n    return element.style.backgroundColor;\n  }\n}\n","<div\n  class=\"impdc-scrollable\"\n  #container\n  impdcSizeDetector\n  (resized)=\"widthChanged($event)\">\n  <a class=\"impdc-scrollable-left\" (click)=\"scroll(-1)\" *ngIf=\"canScrollLeft\">\n    <i class=\"fas fa-chevron-left default\"></i>\n  </a>\n  <div #content class=\"content-container\">\n    <ng-content></ng-content>\n  </div>\n  <a class=\"impdc-scrollable-right\" (click)=\"scroll(1)\" *ngIf=\"canScrollRight\">\n    <i class=\"fas fa-chevron-right default\"></i>\n  </a>\n</div>\n"]}
|
|
186
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable.component.js","sourceRoot":"","sources":["../../../../../../projects/design-components/src/lib/scrollable/scrollable.component.ts","../../../../../../projects/design-components/src/lib/scrollable/scrollable.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  HostBinding,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  ViewChild\n} from '@angular/core';\nimport { IElementSize } from '../size-detection';\n\n/**\n * The `ScrollableComponent` (`<impdc-scrollable` or `<div impdc-scrollable`) is a container that hides items that wouldn't fit\n * with arrows on the left and right to slide between them.\n * To use, import `ScrollableModule` or another module that imports and exports that module from `@impartner/design-components`.\n * `ScrollableModule` also imports and exports [BrandingModule](./?path=/docs/design-components-directives-impdcbranded).\n */\n@Component({\n  selector: 'impdc-scrollable, div[impdc-scrollable]',\n  templateUrl: './scrollable.component.html',\n  styleUrls: ['./scrollable.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ScrollableComponent implements OnChanges, AfterViewInit {\n  private readonly GET_ANCESTOR_BG_SEARCH_DEPTH = 64;\n  /**\n   * The base amount to scroll.\n   */\n  @Input()\n  public scrollUnit = 20;\n\n  /**\n   * Used in combination with `scrollUnit` to determine how far the Scrollable scrolls when directional arrows are clicked.\n   */\n  @Input()\n  public clickScrollMultiplier = 5;\n\n  @HostBinding('style.--impdc-scrollable-bg')\n  private _scrollableBgColor: string | null = null;\n\n  public get canScrollLeft(): boolean {\n    if (!this._contentElement) {\n      return false;\n    }\n\n    return Math.floor(this._contentElement.scrollLeft) > 0;\n  }\n\n  public get canScrollRight(): boolean {\n    if (!this._contentElement) {\n      return false;\n    }\n\n    return (\n      Math.ceil(this._contentElement.scrollLeft + this._width) <\n      this._contentElement.scrollWidth\n    );\n  }\n\n  private _width = 0;\n\n  @ViewChild('container', { static: false })\n  public containerRef: ElementRef | undefined;\n\n  private get _containerElement(): HTMLDivElement | null {\n    if (!this.containerRef) {\n      return null;\n    }\n\n    return this.containerRef.nativeElement;\n  }\n\n  @ViewChild('content', { static: false })\n  public contentRef: ElementRef | undefined;\n\n  private get _contentElement(): HTMLDivElement | null {\n    if (!this.contentRef) {\n      return null;\n    }\n\n    return this.contentRef.nativeElement;\n  }\n\n  constructor(\n    private _cd: ChangeDetectorRef,\n    private readonly _hostElementRef: ElementRef\n  ) {}\n\n  public ngOnChanges(changes: SimpleChanges): void {\n    if (changes['scrollUnit'] || changes['clickScrollMultiplier']) {\n      if (this._containerElement) {\n        this._width = this._containerElement?.clientWidth;\n        this._cd.detectChanges();\n      }\n    }\n  }\n\n  public ngAfterViewInit(): void {\n    this._width = this.contentRef?.nativeElement.width;\n    this._cd.detectChanges();\n\n    Promise.resolve().then(() => this._getBackgroundColorForButtons());\n  }\n\n  public async scroll(direction: number): Promise<void> {\n    if (!this._contentElement) {\n      return;\n    }\n\n    let pixels = direction * this.scrollUnit * this.clickScrollMultiplier;\n    let position = 0;\n\n    if (!pixels) {\n      if (direction === 1) {\n        position = this._contentElement.scrollWidth;\n      }\n    } else {\n      position = this._contentElement.scrollLeft + pixels;\n    }\n\n    await this._scrollToPosition(this._contentElement, position);\n\n    this._cd.detectChanges();\n  }\n\n  public widthChanged(elementSize: IElementSize): void {\n    if (!elementSize || !elementSize.width) {\n      return;\n    }\n    this._width = elementSize.width;\n    this._cd.detectChanges();\n  }\n\n  private async _scrollToPosition(container: HTMLDivElement, position: number) {\n    position = Math.round(position);\n\n    if (container.scrollLeft === position) {\n      return;\n    }\n\n    let resolveFn: (value: unknown) => void;\n    let scrollListener: () => void;\n    let timeoutId: any;\n\n    const promise = new Promise(resolve => {\n      resolveFn = resolve;\n    });\n\n    const finished = () => {\n      container.removeEventListener('scroll', scrollListener);\n      resolveFn(null);\n    };\n\n    scrollListener = () => {\n      clearTimeout(timeoutId);\n\n      // scroll is finished when either the position has been reached, or 100ms have elapsed since the last scroll event\n      if (container.scrollLeft === position) {\n        finished();\n      } else {\n        timeoutId = setTimeout(finished, 100);\n      }\n    };\n\n    container.addEventListener('scroll', scrollListener);\n\n    container.scrollTo({\n      left: position,\n      behavior: 'smooth'\n    });\n\n    return promise;\n  }\n\n  private _getBackgroundColorForButtons(): void {\n    let backgroundColor = this._getBackGroundColor(\n      this._hostElementRef.nativeElement\n    );\n\n    if (!backgroundColor) {\n      backgroundColor = this._getAncestorBackgroundColor();\n    }\n\n    if (this._scrollableBgColor !== backgroundColor) {\n      this._scrollableBgColor = backgroundColor;\n      this._cd.detectChanges();\n    }\n  }\n\n  private _getAncestorBackgroundColor(): string | null {\n    type parentBgResult = {\n      parent: HTMLElement | null;\n      backgroundColor: string | null;\n    };\n\n    let result: parentBgResult = {\n      parent: this._hostElementRef.nativeElement.parentElement,\n      backgroundColor: null\n    };\n\n    let count = this.GET_ANCESTOR_BG_SEARCH_DEPTH;\n\n    function getParentBackgroundColor(\n      currentContext: parentBgResult,\n      getBackgroundColor: (element: HTMLElement) => string | null\n    ): parentBgResult {\n      const result: parentBgResult = { parent: null, backgroundColor: null };\n      if (currentContext && currentContext.parent) {\n        result.backgroundColor = getBackgroundColor(currentContext.parent);\n\n        if (!result.backgroundColor) {\n          result.parent = currentContext.parent.parentElement;\n        }\n      }\n\n      return result;\n    }\n\n    while (count > 0) {\n      if (result.backgroundColor) {\n        break;\n      }\n      result = getParentBackgroundColor(result, this._getBackGroundColor);\n\n      if (!result.parent) {\n        break;\n      }\n      count = count - 1;\n    }\n\n    return result.backgroundColor;\n  }\n\n  private _getBackGroundColor(element: HTMLElement): string | null {\n    const computedBackgroundColor =\n      getComputedStyle(element)['backgroundColor'];\n\n    if (computedBackgroundColor !== 'rgba(0, 0, 0, 0)') {\n      return computedBackgroundColor;\n    }\n\n    return element.style.backgroundColor;\n  }\n}\n","<div\n  class=\"impdc-scrollable\"\n  #container\n  impdcSizeDetector\n  (resized)=\"widthChanged($event)\">\n  <a class=\"impdc-scrollable-left\" (click)=\"scroll(-1)\" *ngIf=\"canScrollLeft\">\n    <i class=\"fas fa-chevron-left default\"></i>\n  </a>\n  <div #content class=\"content-container\">\n    <ng-content></ng-content>\n  </div>\n  <a class=\"impdc-scrollable-right\" (click)=\"scroll(1)\" *ngIf=\"canScrollRight\">\n    <i class=\"fas fa-chevron-right default\"></i>\n  </a>\n</div>\n"]}
|