@villedemontreal/angular-ui 2.2.0 → 3.1.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/alert/alert.component.mjs +7 -9
- package/esm2020/lib/avatar/avatar.component.mjs +102 -0
- package/esm2020/lib/avatar/index.mjs +8 -0
- package/esm2020/lib/avatar/module.mjs +24 -0
- package/esm2020/lib/bao.module.mjs +16 -4
- package/esm2020/lib/breadcrumb/breadcrumb.component.mjs +5 -5
- package/esm2020/lib/button/button.component.mjs +3 -3
- package/esm2020/lib/checkbox/checkbox-group.component.mjs +5 -6
- package/esm2020/lib/checkbox/checkbox.component.mjs +38 -39
- package/esm2020/lib/core/colors.mjs +10 -27
- package/esm2020/lib/icon/icon.component.mjs +29 -33
- package/esm2020/lib/icon/icons-dictionary.mjs +3 -2
- package/esm2020/lib/modal/index.mjs +12 -0
- package/esm2020/lib/modal/modal-animations.mjs +29 -0
- package/esm2020/lib/modal/modal-config.mjs +65 -0
- package/esm2020/lib/modal/modal-container.mjs +254 -0
- package/esm2020/lib/modal/modal-directives.mjs +84 -0
- package/esm2020/lib/modal/modal-ref.mjs +195 -0
- package/esm2020/lib/modal/modal.mjs +291 -0
- package/esm2020/lib/modal/module.mjs +46 -0
- package/esm2020/lib/radio/radio-group.component.mjs +40 -41
- package/esm2020/lib/radio/radio.component.mjs +38 -39
- package/esm2020/lib/shared/enum/display-mode.mjs +2 -11
- package/esm2020/lib/summary/summary.component.mjs +5 -7
- package/esm2020/lib/tabs/index.mjs +8 -0
- package/esm2020/lib/tabs/module.mjs +35 -0
- package/esm2020/lib/tabs/tabs.component.mjs +295 -0
- package/esm2020/lib/tag/tag.component.mjs +3 -3
- package/esm2020/public-api.mjs +5 -1
- package/fesm2015/villedemontreal-angular-ui.mjs +1607 -230
- package/fesm2015/villedemontreal-angular-ui.mjs.map +1 -1
- package/fesm2020/villedemontreal-angular-ui.mjs +1597 -228
- package/fesm2020/villedemontreal-angular-ui.mjs.map +1 -1
- package/lib/alert/alert.component.d.ts +2 -3
- package/lib/avatar/avatar.component.d.ts +26 -0
- package/lib/avatar/index.d.ts +2 -0
- package/lib/avatar/module.d.ts +8 -0
- package/lib/bao.module.d.ts +4 -1
- package/lib/breadcrumb/breadcrumb.component.d.ts +1 -1
- package/lib/button/button.component.d.ts +1 -1
- package/lib/checkbox/checkbox-group.component.d.ts +2 -2
- package/lib/checkbox/checkbox.component.d.ts +27 -27
- package/lib/core/colors.d.ts +1 -1
- package/lib/icon/icon.component.d.ts +11 -11
- package/lib/modal/index.d.ts +6 -0
- package/lib/modal/modal-animations.d.ts +8 -0
- package/lib/modal/modal-config.d.ts +105 -0
- package/lib/modal/modal-container.d.ts +106 -0
- package/lib/modal/modal-directives.d.ts +25 -0
- package/lib/modal/modal-ref.d.ts +91 -0
- package/lib/modal/modal.d.ts +91 -0
- package/lib/modal/module.d.ts +12 -0
- package/lib/radio/radio-group.component.d.ts +19 -19
- package/lib/radio/radio.component.d.ts +27 -27
- package/lib/shared/enum/display-mode.d.ts +1 -1
- package/lib/summary/summary.component.d.ts +5 -5
- package/lib/tabs/index.d.ts +2 -0
- package/lib/tabs/module.d.ts +8 -0
- package/lib/tabs/tabs.component.d.ts +95 -0
- package/lib/tag/tag.component.d.ts +1 -1
- package/package.json +1 -1
- package/public-api.d.ts +4 -0
|
@@ -0,0 +1,254 @@
|
|
|
1
|
+
import { _getFocusedElementPierceShadowDom } from '@angular/cdk/platform';
|
|
2
|
+
import { BasePortalOutlet, CdkPortalOutlet } from '@angular/cdk/portal';
|
|
3
|
+
import { DOCUMENT } from '@angular/common';
|
|
4
|
+
import { ChangeDetectionStrategy, Component, Directive, EventEmitter, Inject, Optional, ViewChild, ViewEncapsulation } from '@angular/core';
|
|
5
|
+
import { baoModalAnimations } from './modal-animations';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/cdk/a11y";
|
|
8
|
+
import * as i2 from "./modal-config";
|
|
9
|
+
import * as i3 from "@angular/cdk/portal";
|
|
10
|
+
/**
|
|
11
|
+
* Throws an exception for the case when a ComponentPortal is
|
|
12
|
+
* attached to a DomPortalOutlet without an origin.
|
|
13
|
+
* @docs-private
|
|
14
|
+
*/
|
|
15
|
+
export function throwBaoModalContentAlreadyAttachedError() {
|
|
16
|
+
throw Error('Attempting to attach modal content after content is already attached');
|
|
17
|
+
}
|
|
18
|
+
/**
|
|
19
|
+
* Base class for the `BaoModalContainer`. The base class does not implement
|
|
20
|
+
* animations as these are left to implementers of the modal container.
|
|
21
|
+
*/
|
|
22
|
+
export class _BaoModalContainerBase extends BasePortalOutlet {
|
|
23
|
+
constructor(_elementRef, _focusTrapFactory, _changeDetectorRef, _document,
|
|
24
|
+
/** The modal configuration. */
|
|
25
|
+
_config, _interactivityChecker, _ngZone, _focusMonitor) {
|
|
26
|
+
super();
|
|
27
|
+
this._elementRef = _elementRef;
|
|
28
|
+
this._focusTrapFactory = _focusTrapFactory;
|
|
29
|
+
this._changeDetectorRef = _changeDetectorRef;
|
|
30
|
+
this._config = _config;
|
|
31
|
+
this._interactivityChecker = _interactivityChecker;
|
|
32
|
+
this._ngZone = _ngZone;
|
|
33
|
+
this._focusMonitor = _focusMonitor;
|
|
34
|
+
/** Emits when an animation state changes. */
|
|
35
|
+
this._animationStateChanged = new EventEmitter();
|
|
36
|
+
/**
|
|
37
|
+
* Type of interaction that led to the modal being closed. This is used to determine
|
|
38
|
+
* whether the focus style will be applied when returning focus to its original location
|
|
39
|
+
* after the modal is closed.
|
|
40
|
+
*/
|
|
41
|
+
this._closeInteractionType = null;
|
|
42
|
+
/** Element that was focused before the modal was opened. Save this to restore upon close. */
|
|
43
|
+
this._elementFocusedBeforeDialogWasOpened = null;
|
|
44
|
+
this._ariaLabelledBy = _config.ariaLabelledBy || null;
|
|
45
|
+
this._document = _document;
|
|
46
|
+
}
|
|
47
|
+
/** Initializes the modal container with the attached content. */
|
|
48
|
+
_initializeWithAttachedContent() {
|
|
49
|
+
this._setupFocusTrap();
|
|
50
|
+
// Save the previously focused element. This element will be re-focused
|
|
51
|
+
// when the modal closes.
|
|
52
|
+
this._capturePreviouslyFocusedElement();
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Attach a ComponentPortal as content to this modal container.
|
|
56
|
+
*/
|
|
57
|
+
attachComponentPortal(portal) {
|
|
58
|
+
return this._portalOutlet.attachComponentPortal(portal);
|
|
59
|
+
}
|
|
60
|
+
/**
|
|
61
|
+
* Attach a TemplatePortal as content to this modal container.
|
|
62
|
+
*/
|
|
63
|
+
attachTemplatePortal(portal) {
|
|
64
|
+
return this._portalOutlet.attachTemplatePortal(portal);
|
|
65
|
+
}
|
|
66
|
+
/** Moves focus back into the modal if it was moved out. */
|
|
67
|
+
async _recaptureFocus() {
|
|
68
|
+
if (!this._containsFocus()) {
|
|
69
|
+
await this._trapFocus();
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Moves the focus inside the focus trap. When autoFocus is not set to 'modal', if focus
|
|
74
|
+
* cannot be moved then focus will go to the modal container.
|
|
75
|
+
*/
|
|
76
|
+
async _trapFocus() {
|
|
77
|
+
const element = this._elementRef.nativeElement;
|
|
78
|
+
// If were to attempt to focus immediately, then the content of the modal would not yet be
|
|
79
|
+
// ready in instances where change detection has to run first. To deal with this, we simply
|
|
80
|
+
// wait for the microtask queue to be empty when setting focus when autoFocus isn't set to
|
|
81
|
+
// modal. If the element inside the modal can't be focused, then the container is focused
|
|
82
|
+
// so the user can't tab into other elements behind it.
|
|
83
|
+
switch (this._config.autoFocus) {
|
|
84
|
+
case 'modal':
|
|
85
|
+
// Ensure that focus is on the modal container. It's possible that a different
|
|
86
|
+
// component tried to move focus while the open animation was running. See:
|
|
87
|
+
// https://github.com/angular/components/issues/16215. Note that we only want to do this
|
|
88
|
+
// if the focus isn't inside the modal already, because it's possible that the consumer
|
|
89
|
+
// turned off `autoFocus` in order to move focus themselves.
|
|
90
|
+
if (!this._containsFocus()) {
|
|
91
|
+
element.focus();
|
|
92
|
+
}
|
|
93
|
+
break;
|
|
94
|
+
case 'first-tabbable':
|
|
95
|
+
// If we weren't able to find a focusable element in the modal, then focus the modal
|
|
96
|
+
// container instead.
|
|
97
|
+
const focusedSuccessfully = await this._focusTrap.focusInitialElementWhenReady();
|
|
98
|
+
if (!focusedSuccessfully) {
|
|
99
|
+
this._focusDialogContainer();
|
|
100
|
+
}
|
|
101
|
+
break;
|
|
102
|
+
case 'first-heading':
|
|
103
|
+
this._focusByCssSelector('h1, h2, h3, h4, h5, h6, [role="heading"]');
|
|
104
|
+
break;
|
|
105
|
+
default:
|
|
106
|
+
this._focusByCssSelector(this._config.autoFocus);
|
|
107
|
+
break;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
/** Restores focus to the element that was focused before the modal opened. */
|
|
111
|
+
_restoreFocus() {
|
|
112
|
+
const previousElement = this._elementFocusedBeforeDialogWasOpened;
|
|
113
|
+
// We need the extra check, because IE can set the `activeElement` to null in some cases.
|
|
114
|
+
if (this._config.restoreFocus &&
|
|
115
|
+
previousElement &&
|
|
116
|
+
typeof previousElement.focus === 'function') {
|
|
117
|
+
const activeElement = _getFocusedElementPierceShadowDom();
|
|
118
|
+
const element = this._elementRef.nativeElement;
|
|
119
|
+
// Make sure that focus is still inside the modal or is on the body (usually because a
|
|
120
|
+
// non-focusable element like the backdrop was clicked) before moving it. It's possible that
|
|
121
|
+
// the consumer moved it themselves before the animation was done, in which case we shouldn't
|
|
122
|
+
// do anything.
|
|
123
|
+
if (!activeElement ||
|
|
124
|
+
activeElement === this._document.body ||
|
|
125
|
+
activeElement === element ||
|
|
126
|
+
element.contains(activeElement)) {
|
|
127
|
+
if (this._focusMonitor) {
|
|
128
|
+
this._focusMonitor.focusVia(previousElement, this._closeInteractionType);
|
|
129
|
+
this._closeInteractionType = null;
|
|
130
|
+
}
|
|
131
|
+
else {
|
|
132
|
+
previousElement.focus();
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
if (this._focusTrap) {
|
|
137
|
+
this._focusTrap.destroy();
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
/**
|
|
141
|
+
* Focuses the provided element. If the element is not focusable, it will add a tabIndex
|
|
142
|
+
* attribute to forcefully focus it. The attribute is removed after focus is moved.
|
|
143
|
+
*/
|
|
144
|
+
_forceFocus(element, options) {
|
|
145
|
+
if (!this._interactivityChecker.isFocusable(element)) {
|
|
146
|
+
element.tabIndex = -1;
|
|
147
|
+
// The tabindex attribute should be removed to avoid navigating to that element again
|
|
148
|
+
this._ngZone.runOutsideAngular(() => {
|
|
149
|
+
element.addEventListener('blur', () => element.removeAttribute('tabindex'));
|
|
150
|
+
element.addEventListener('mousedown', () => element.removeAttribute('tabindex'));
|
|
151
|
+
});
|
|
152
|
+
}
|
|
153
|
+
element.focus(options);
|
|
154
|
+
}
|
|
155
|
+
/**
|
|
156
|
+
* Focuses the first element that matches the given selector within the focus trap.
|
|
157
|
+
*/
|
|
158
|
+
_focusByCssSelector(selector, options) {
|
|
159
|
+
const elementToFocus = this._elementRef.nativeElement.querySelector(selector);
|
|
160
|
+
if (elementToFocus) {
|
|
161
|
+
this._forceFocus(elementToFocus, options);
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
/** Sets up the focus trap. */
|
|
165
|
+
_setupFocusTrap() {
|
|
166
|
+
this._focusTrap = this._focusTrapFactory.create(this._elementRef.nativeElement);
|
|
167
|
+
}
|
|
168
|
+
/** Captures the element that was focused before the modal was opened. */
|
|
169
|
+
_capturePreviouslyFocusedElement() {
|
|
170
|
+
if (this._document) {
|
|
171
|
+
this._elementFocusedBeforeDialogWasOpened =
|
|
172
|
+
_getFocusedElementPierceShadowDom();
|
|
173
|
+
}
|
|
174
|
+
}
|
|
175
|
+
/** Focuses the modal container. */
|
|
176
|
+
_focusDialogContainer() {
|
|
177
|
+
// Note that there is no focus method when rendering on the server.
|
|
178
|
+
if (this._elementRef.nativeElement.focus) {
|
|
179
|
+
this._elementRef.nativeElement.focus();
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
/** Returns whether focus is inside the modal. */
|
|
183
|
+
_containsFocus() {
|
|
184
|
+
const element = this._elementRef.nativeElement;
|
|
185
|
+
const activeElement = _getFocusedElementPierceShadowDom();
|
|
186
|
+
return element === activeElement || element.contains(activeElement);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
_BaoModalContainerBase.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: _BaoModalContainerBase, deps: [{ token: i0.ElementRef }, { token: i1.ConfigurableFocusTrapFactory }, { token: i0.ChangeDetectorRef }, { token: DOCUMENT, optional: true }, { token: i2.BaoModalInitialConfig }, { token: i1.InteractivityChecker }, { token: i0.NgZone }, { token: i1.FocusMonitor }], target: i0.ɵɵFactoryTarget.Directive });
|
|
190
|
+
_BaoModalContainerBase.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.1.1", type: _BaoModalContainerBase, viewQueries: [{ propertyName: "_portalOutlet", first: true, predicate: CdkPortalOutlet, descendants: true, static: true }], usesInheritance: true, ngImport: i0 });
|
|
191
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: _BaoModalContainerBase, decorators: [{
|
|
192
|
+
type: Directive
|
|
193
|
+
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.ConfigurableFocusTrapFactory }, { type: i0.ChangeDetectorRef }, { type: undefined, decorators: [{
|
|
194
|
+
type: Optional
|
|
195
|
+
}, {
|
|
196
|
+
type: Inject,
|
|
197
|
+
args: [DOCUMENT]
|
|
198
|
+
}] }, { type: i2.BaoModalInitialConfig }, { type: i1.InteractivityChecker }, { type: i0.NgZone }, { type: i1.FocusMonitor }]; }, propDecorators: { _portalOutlet: [{
|
|
199
|
+
type: ViewChild,
|
|
200
|
+
args: [CdkPortalOutlet, { static: true }]
|
|
201
|
+
}] } });
|
|
202
|
+
export class BaoModalContainer extends _BaoModalContainerBase {
|
|
203
|
+
constructor() {
|
|
204
|
+
super(...arguments);
|
|
205
|
+
/** State of the modal animation. */
|
|
206
|
+
this._state = 'enter';
|
|
207
|
+
}
|
|
208
|
+
/** Callback, invoked whenever an animation on the host completes. */
|
|
209
|
+
async _onAnimationDone({ toState, totalTime }) {
|
|
210
|
+
if (toState === 'enter') {
|
|
211
|
+
await this._trapFocus();
|
|
212
|
+
this._animationStateChanged.next({ state: 'opened', totalTime });
|
|
213
|
+
}
|
|
214
|
+
else if (toState === 'exit') {
|
|
215
|
+
this._restoreFocus();
|
|
216
|
+
this._animationStateChanged.next({ state: 'closed', totalTime });
|
|
217
|
+
}
|
|
218
|
+
}
|
|
219
|
+
/** Callback, invoked when an animation on the host starts. */
|
|
220
|
+
_onAnimationStart({ toState, totalTime }) {
|
|
221
|
+
if (toState === 'enter') {
|
|
222
|
+
this._animationStateChanged.next({ state: 'opening', totalTime });
|
|
223
|
+
}
|
|
224
|
+
else if (toState === 'exit' || toState === 'void') {
|
|
225
|
+
this._animationStateChanged.next({ state: 'closing', totalTime });
|
|
226
|
+
}
|
|
227
|
+
}
|
|
228
|
+
/** Starts the modal exit animation. */
|
|
229
|
+
_startExitAnimation() {
|
|
230
|
+
this._state = 'exit';
|
|
231
|
+
// Mark the container for check so it can react if the
|
|
232
|
+
// view container is using OnPush change detection.
|
|
233
|
+
this._changeDetectorRef.markForCheck();
|
|
234
|
+
}
|
|
235
|
+
}
|
|
236
|
+
BaoModalContainer.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: BaoModalContainer, deps: null, target: i0.ɵɵFactoryTarget.Component });
|
|
237
|
+
BaoModalContainer.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.1.1", type: BaoModalContainer, selector: "bao-modal-container", host: { attributes: { "tabindex": "-1", "aria-modal": "true" }, listeners: { "@modalContainer.start": "_onAnimationStart($event)", "@modalContainer.done": "_onAnimationDone($event)" }, properties: { "id": "_id", "attr.role": "_config.role", "attr.aria-labelledby": "_config.ariaLabel ? null : _ariaLabelledBy", "attr.aria-label": "_config.ariaLabel", "attr.aria-describedby": "_config.ariaDescribedBy || null", "@modalContainer": "_state" }, classAttribute: "bao-modal-container" }, usesInheritance: true, ngImport: i0, template: "<ng-template cdkPortalOutlet></ng-template>\n", styles: [".bao-container{padding-right:16px;padding-left:16px;margin-right:auto;margin-left:auto;width:100%}@media (min-width: 576px){.bao-container{max-width:576px}}@media (min-width: 768px){.bao-container{max-width:768px}}@media (min-width: 992px){.bao-container{max-width:992px}}@media (min-width: 1200px){.bao-container{max-width:1200px}}.bao-row{display:flex;flex-wrap:wrap;margin-right:-16px;margin-left:-16px}.bao-col-12,.bao-col-lg-7{position:relative;width:100%;padding-right:1rem;padding-left:1rem}@media (min-width: 992px){.bao-col-lg-7{flex:0 0 58.33333%;max-width:58.33333%}}.cdk-overlay-container,.cdk-global-overlay-wrapper{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container{position:fixed;z-index:1000}.cdk-overlay-container:empty{display:none}.cdk-global-overlay-wrapper{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex}.cdk-overlay-pane.bao-modal-mobil-full{width:100%;height:100%}.cdk-overlay-pane.bao-modal-mobil-compact{width:300px;height:100%}@media (min-width: 768px){.cdk-overlay-pane{width:500px;height:auto}.cdk-overlay-pane.bao-modal-lg{width:calc(100% - 4rem);height:calc(100% - 4rem)}.cdk-overlay-pane.bao-modal-md,.cdk-overlay-pane.bao-modal-sm{width:500px;height:auto}}@media (min-width: 992px){.cdk-overlay-pane{width:500px;height:auto}.cdk-overlay-pane.bao-modal-lg{width:calc(100% - 4rem);height:calc(100% - 4rem)}.cdk-overlay-pane.bao-modal-md{width:800px;height:auto}.cdk-overlay-pane.bao-modal-sm{width:500px;height:auto}}.cdk-overlay-backdrop{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity .4s cubic-bezier(.25,.8,.25,1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:.5}.cdk-high-contrast-active .cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:.6}.cdk-overlay-dark-backdrop{background:black}.cdk-overlay-transparent-backdrop,.cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing{opacity:.5}.cdk-overlay-connected-position-bounding-box{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock{position:fixed;width:100%;overflow-y:scroll}.bao-modal-container{display:block;overflow:auto;width:100%;height:100%;min-height:inherit;max-height:inherit;background-color:#fff;background-clip:padding-box;border:0 solid rgba(0,0,0,.2);border-radius:.5rem;outline:0}.bao-modal-content{display:flex;flex-direction:column;height:100%}.bao-modal-header{flex:0 0 auto;display:flex;align-items:flex-start;justify-content:space-between;border-bottom:1px solid #ced4da;border-top-left-radius:.5rem;border-top-right-radius:.5rem}.bao-modal-header .bao-modal-title{font-size:1rem;line-height:1.5rem;margin:1rem 0 1rem 1rem}.bao-modal-header button{margin:.5rem}.bao-modal-body{display:block;padding:1rem;overflow:auto;flex-grow:1}@media (min-width: 768px){.bao-modal-body{padding:2rem}}.bao-modal-footer{display:flex;flex-wrap:wrap;padding:1rem;border-top:1px solid #ced4da;border-bottom-right-radius:.5rem;border-bottom-left-radius:.5rem}.bao-modal-footer.bao-modal-footer-order{justify-content:unset;flex-direction:column-reverse}.bao-modal-footer button{text-align:center}.bao-modal-footer .bao-button-primary,.bao-modal-footer .bao-button-secondary{width:100%;display:block}.bao-modal-footer .bao-button-secondary{margin-bottom:.5rem}.bao-modal-footer .bao-button-tertiary{display:none}@media (min-width: 768px){.bao-modal-footer{justify-content:flex-end}.bao-modal-footer.bao-modal-footer-order{flex-direction:row-reverse}.bao-modal-footer .bao-button-primary{margin-left:1rem}.bao-modal-footer .bao-button-secondary{margin-bottom:0}.bao-modal-footer .bao-button-primary,.bao-modal-footer .bao-button-secondary{width:auto}.bao-modal-footer .bao-button-tertiary{display:block}}\n"], directives: [{ type: i3.CdkPortalOutlet, selector: "[cdkPortalOutlet]", inputs: ["cdkPortalOutlet"], outputs: ["attached"], exportAs: ["cdkPortalOutlet"] }], animations: [baoModalAnimations.modalContainer], changeDetection: i0.ChangeDetectionStrategy.Default, encapsulation: i0.ViewEncapsulation.None });
|
|
238
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: BaoModalContainer, decorators: [{
|
|
239
|
+
type: Component,
|
|
240
|
+
args: [{ selector: 'bao-modal-container', encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.Default, animations: [baoModalAnimations.modalContainer], host: {
|
|
241
|
+
class: 'bao-modal-container',
|
|
242
|
+
tabindex: '-1',
|
|
243
|
+
'aria-modal': 'true',
|
|
244
|
+
'[id]': '_id',
|
|
245
|
+
'[attr.role]': '_config.role',
|
|
246
|
+
'[attr.aria-labelledby]': '_config.ariaLabel ? null : _ariaLabelledBy',
|
|
247
|
+
'[attr.aria-label]': '_config.ariaLabel',
|
|
248
|
+
'[attr.aria-describedby]': '_config.ariaDescribedBy || null',
|
|
249
|
+
'[@modalContainer]': '_state',
|
|
250
|
+
'(@modalContainer.start)': '_onAnimationStart($event)',
|
|
251
|
+
'(@modalContainer.done)': '_onAnimationDone($event)'
|
|
252
|
+
}, template: "<ng-template cdkPortalOutlet></ng-template>\n", styles: [".bao-container{padding-right:16px;padding-left:16px;margin-right:auto;margin-left:auto;width:100%}@media (min-width: 576px){.bao-container{max-width:576px}}@media (min-width: 768px){.bao-container{max-width:768px}}@media (min-width: 992px){.bao-container{max-width:992px}}@media (min-width: 1200px){.bao-container{max-width:1200px}}.bao-row{display:flex;flex-wrap:wrap;margin-right:-16px;margin-left:-16px}.bao-col-12,.bao-col-lg-7{position:relative;width:100%;padding-right:1rem;padding-left:1rem}@media (min-width: 992px){.bao-col-lg-7{flex:0 0 58.33333%;max-width:58.33333%}}.cdk-overlay-container,.cdk-global-overlay-wrapper{pointer-events:none;top:0;left:0;height:100%;width:100%}.cdk-overlay-container{position:fixed;z-index:1000}.cdk-overlay-container:empty{display:none}.cdk-global-overlay-wrapper{display:flex;position:absolute;z-index:1000}.cdk-overlay-pane{position:absolute;pointer-events:auto;box-sizing:border-box;z-index:1000;display:flex}.cdk-overlay-pane.bao-modal-mobil-full{width:100%;height:100%}.cdk-overlay-pane.bao-modal-mobil-compact{width:300px;height:100%}@media (min-width: 768px){.cdk-overlay-pane{width:500px;height:auto}.cdk-overlay-pane.bao-modal-lg{width:calc(100% - 4rem);height:calc(100% - 4rem)}.cdk-overlay-pane.bao-modal-md,.cdk-overlay-pane.bao-modal-sm{width:500px;height:auto}}@media (min-width: 992px){.cdk-overlay-pane{width:500px;height:auto}.cdk-overlay-pane.bao-modal-lg{width:calc(100% - 4rem);height:calc(100% - 4rem)}.cdk-overlay-pane.bao-modal-md{width:800px;height:auto}.cdk-overlay-pane.bao-modal-sm{width:500px;height:auto}}.cdk-overlay-backdrop{position:absolute;top:0;bottom:0;left:0;right:0;z-index:1000;pointer-events:auto;-webkit-tap-highlight-color:transparent;transition:opacity .4s cubic-bezier(.25,.8,.25,1);opacity:0}.cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:.5}.cdk-high-contrast-active .cdk-overlay-backdrop.cdk-overlay-backdrop-showing{opacity:.6}.cdk-overlay-dark-backdrop{background:black}.cdk-overlay-transparent-backdrop,.cdk-overlay-transparent-backdrop.cdk-overlay-backdrop-showing{opacity:.5}.cdk-overlay-connected-position-bounding-box{position:absolute;z-index:1000;display:flex;flex-direction:column;min-width:1px;min-height:1px}.cdk-global-scrollblock{position:fixed;width:100%;overflow-y:scroll}.bao-modal-container{display:block;overflow:auto;width:100%;height:100%;min-height:inherit;max-height:inherit;background-color:#fff;background-clip:padding-box;border:0 solid rgba(0,0,0,.2);border-radius:.5rem;outline:0}.bao-modal-content{display:flex;flex-direction:column;height:100%}.bao-modal-header{flex:0 0 auto;display:flex;align-items:flex-start;justify-content:space-between;border-bottom:1px solid #ced4da;border-top-left-radius:.5rem;border-top-right-radius:.5rem}.bao-modal-header .bao-modal-title{font-size:1rem;line-height:1.5rem;margin:1rem 0 1rem 1rem}.bao-modal-header button{margin:.5rem}.bao-modal-body{display:block;padding:1rem;overflow:auto;flex-grow:1}@media (min-width: 768px){.bao-modal-body{padding:2rem}}.bao-modal-footer{display:flex;flex-wrap:wrap;padding:1rem;border-top:1px solid #ced4da;border-bottom-right-radius:.5rem;border-bottom-left-radius:.5rem}.bao-modal-footer.bao-modal-footer-order{justify-content:unset;flex-direction:column-reverse}.bao-modal-footer button{text-align:center}.bao-modal-footer .bao-button-primary,.bao-modal-footer .bao-button-secondary{width:100%;display:block}.bao-modal-footer .bao-button-secondary{margin-bottom:.5rem}.bao-modal-footer .bao-button-tertiary{display:none}@media (min-width: 768px){.bao-modal-footer{justify-content:flex-end}.bao-modal-footer.bao-modal-footer-order{flex-direction:row-reverse}.bao-modal-footer .bao-button-primary{margin-left:1rem}.bao-modal-footer .bao-button-secondary{margin-bottom:0}.bao-modal-footer .bao-button-primary,.bao-modal-footer .bao-button-secondary{width:auto}.bao-modal-footer .bao-button-tertiary{display:block}}\n"] }]
|
|
253
|
+
}] });
|
|
254
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-container.js","sourceRoot":"","sources":["../../../../../projects/angular-ui/src/lib/modal/modal-container.ts","../../../../../projects/angular-ui/src/lib/modal/modal-container.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2022 Ville de Montreal. All rights reserved.\n * Licensed under the MIT license.\n * See LICENSE file in the project root for full license information.\n */\nimport { AnimationEvent } from '@angular/animations';\nimport {\n  FocusMonitor,\n  FocusOrigin,\n  ConfigurableFocusTrap,\n  ConfigurableFocusTrapFactory,\n  InteractivityChecker\n} from '@angular/cdk/a11y';\nimport { _getFocusedElementPierceShadowDom } from '@angular/cdk/platform';\nimport {\n  BasePortalOutlet,\n  CdkPortalOutlet,\n  ComponentPortal,\n  TemplatePortal\n} from '@angular/cdk/portal';\nimport { DOCUMENT } from '@angular/common';\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ComponentRef,\n  Directive,\n  ElementRef,\n  EmbeddedViewRef,\n  EventEmitter,\n  Inject,\n  NgZone,\n  Optional,\n  ViewChild,\n  ViewEncapsulation\n} from '@angular/core';\nimport { baoModalAnimations } from './modal-animations';\nimport { BaoModalInitialConfig } from './modal-config';\n\n/** Event that captures the state of modal container animations. */\ninterface DialogAnimationEvent {\n  state: 'opened' | 'opening' | 'closing' | 'closed';\n  totalTime: number;\n}\n\n/**\n * Throws an exception for the case when a ComponentPortal is\n * attached to a DomPortalOutlet without an origin.\n * @docs-private\n */\nexport function throwBaoModalContentAlreadyAttachedError() {\n  throw Error(\n    'Attempting to attach modal content after content is already attached'\n  );\n}\n\n/**\n * Base class for the `BaoModalContainer`. The base class does not implement\n * animations as these are left to implementers of the modal container.\n */\n@Directive()\nexport abstract class _BaoModalContainerBase extends BasePortalOutlet {\n  /** The portal outlet inside of this container into which the modal content will be loaded. */\n  @ViewChild(CdkPortalOutlet, { static: true }) _portalOutlet: CdkPortalOutlet;\n\n  /** Emits when an animation state changes. */\n  public _animationStateChanged = new EventEmitter<DialogAnimationEvent>();\n\n  /**\n   * Type of interaction that led to the modal being closed. This is used to determine\n   * whether the focus style will be applied when returning focus to its original location\n   * after the modal is closed.\n   */\n  public _closeInteractionType: FocusOrigin | null = null;\n\n  /** ID of the element that should be considered as the modal's label. */\n  public _ariaLabelledBy: string | null;\n\n  /** ID for the container DOM element. */\n  public _id: string;\n\n  protected _document: Document;\n\n  /** The class that traps and manages focus within the modal. */\n  private _focusTrap: ConfigurableFocusTrap;\n\n  /** Element that was focused before the modal was opened. Save this to restore upon close. */\n  private _elementFocusedBeforeDialogWasOpened: HTMLElement | null = null;\n\n  constructor(\n    protected _elementRef: ElementRef,\n    protected _focusTrapFactory: ConfigurableFocusTrapFactory,\n    protected _changeDetectorRef: ChangeDetectorRef,\n    @Optional() @Inject(DOCUMENT) _document: any,\n    /** The modal configuration. */\n    public _config: BaoModalInitialConfig,\n    private readonly _interactivityChecker: InteractivityChecker,\n    private readonly _ngZone: NgZone,\n    private _focusMonitor?: FocusMonitor\n  ) {\n    super();\n    this._ariaLabelledBy = _config.ariaLabelledBy || null;\n    this._document = _document;\n  }\n\n  /** Initializes the modal container with the attached content. */\n  public _initializeWithAttachedContent() {\n    this._setupFocusTrap();\n    // Save the previously focused element. This element will be re-focused\n    // when the modal closes.\n    this._capturePreviouslyFocusedElement();\n  }\n\n  /**\n   * Attach a ComponentPortal as content to this modal container.\n   */\n  public attachComponentPortal<T>(portal: ComponentPortal<T>): ComponentRef<T> {\n    return this._portalOutlet.attachComponentPortal(portal);\n  }\n\n  /**\n   * Attach a TemplatePortal as content to this modal container.\n   */\n  public attachTemplatePortal<C>(\n    portal: TemplatePortal<C>\n  ): EmbeddedViewRef<C> {\n    return this._portalOutlet.attachTemplatePortal(portal);\n  }\n\n  /** Moves focus back into the modal if it was moved out. */\n  public async _recaptureFocus() {\n    if (!this._containsFocus()) {\n      await this._trapFocus();\n    }\n  }\n\n  /**\n   * Moves the focus inside the focus trap. When autoFocus is not set to 'modal', if focus\n   * cannot be moved then focus will go to the modal container.\n   */\n  protected async _trapFocus() {\n    const element = this._elementRef.nativeElement;\n    // If were to attempt to focus immediately, then the content of the modal would not yet be\n    // ready in instances where change detection has to run first. To deal with this, we simply\n    // wait for the microtask queue to be empty when setting focus when autoFocus isn't set to\n    // modal. If the element inside the modal can't be focused, then the container is focused\n    // so the user can't tab into other elements behind it.\n    switch (this._config.autoFocus) {\n      case 'modal':\n        // Ensure that focus is on the modal container. It's possible that a different\n        // component tried to move focus while the open animation was running. See:\n        // https://github.com/angular/components/issues/16215. Note that we only want to do this\n        // if the focus isn't inside the modal already, because it's possible that the consumer\n        // turned off `autoFocus` in order to move focus themselves.\n        if (!this._containsFocus()) {\n          element.focus();\n        }\n        break;\n      case 'first-tabbable':\n        // If we weren't able to find a focusable element in the modal, then focus the modal\n        // container instead.\n        const focusedSuccessfully =\n          await this._focusTrap.focusInitialElementWhenReady();\n        if (!focusedSuccessfully) {\n          this._focusDialogContainer();\n        }\n        break;\n      case 'first-heading':\n        this._focusByCssSelector('h1, h2, h3, h4, h5, h6, [role=\"heading\"]');\n        break;\n      default:\n        this._focusByCssSelector(this._config.autoFocus);\n        break;\n    }\n  }\n\n  /** Restores focus to the element that was focused before the modal opened. */\n  protected _restoreFocus() {\n    const previousElement = this._elementFocusedBeforeDialogWasOpened;\n\n    // We need the extra check, because IE can set the `activeElement` to null in some cases.\n    if (\n      this._config.restoreFocus &&\n      previousElement &&\n      typeof previousElement.focus === 'function'\n    ) {\n      const activeElement = _getFocusedElementPierceShadowDom();\n      const element = this._elementRef.nativeElement;\n\n      // Make sure that focus is still inside the modal or is on the body (usually because a\n      // non-focusable element like the backdrop was clicked) before moving it. It's possible that\n      // the consumer moved it themselves before the animation was done, in which case we shouldn't\n      // do anything.\n      if (\n        !activeElement ||\n        activeElement === this._document.body ||\n        activeElement === element ||\n        element.contains(activeElement)\n      ) {\n        if (this._focusMonitor) {\n          this._focusMonitor.focusVia(\n            previousElement,\n            this._closeInteractionType\n          );\n          this._closeInteractionType = null;\n        } else {\n          previousElement.focus();\n        }\n      }\n    }\n\n    if (this._focusTrap) {\n      this._focusTrap.destroy();\n    }\n  }\n\n  /**\n   * Focuses the provided element. If the element is not focusable, it will add a tabIndex\n   * attribute to forcefully focus it. The attribute is removed after focus is moved.\n   */\n  private _forceFocus(element: HTMLElement, options?: FocusOptions) {\n    if (!this._interactivityChecker.isFocusable(element)) {\n      element.tabIndex = -1;\n      // The tabindex attribute should be removed to avoid navigating to that element again\n      this._ngZone.runOutsideAngular(() => {\n        element.addEventListener('blur', () =>\n          element.removeAttribute('tabindex')\n        );\n        element.addEventListener('mousedown', () =>\n          element.removeAttribute('tabindex')\n        );\n      });\n    }\n    element.focus(options);\n  }\n\n  /**\n   * Focuses the first element that matches the given selector within the focus trap.\n   */\n  private _focusByCssSelector(selector: string, options?: FocusOptions) {\n    const elementToFocus = this._elementRef.nativeElement.querySelector(\n      selector\n    ) as HTMLElement | null;\n    if (elementToFocus) {\n      this._forceFocus(elementToFocus, options);\n    }\n  }\n\n  /** Sets up the focus trap. */\n  private _setupFocusTrap() {\n    this._focusTrap = this._focusTrapFactory.create(\n      this._elementRef.nativeElement\n    );\n  }\n\n  /** Captures the element that was focused before the modal was opened. */\n  private _capturePreviouslyFocusedElement() {\n    if (this._document) {\n      this._elementFocusedBeforeDialogWasOpened =\n        _getFocusedElementPierceShadowDom();\n    }\n  }\n\n  /** Focuses the modal container. */\n  private _focusDialogContainer() {\n    // Note that there is no focus method when rendering on the server.\n    if (this._elementRef.nativeElement.focus) {\n      this._elementRef.nativeElement.focus();\n    }\n  }\n\n  /** Returns whether focus is inside the modal. */\n  private _containsFocus() {\n    const element = this._elementRef.nativeElement as HTMLElement;\n    const activeElement = _getFocusedElementPierceShadowDom();\n    return element === activeElement || element.contains(activeElement);\n  }\n\n  /** Starts the modal exit animation. */\n  abstract _startExitAnimation(): void;\n}\n\n@Component({\n  selector: 'bao-modal-container',\n  templateUrl: 'modal-container.component.html',\n  styleUrls: ['modal-container.component.scss'],\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.Default,\n  animations: [baoModalAnimations.modalContainer],\n  host: {\n    class: 'bao-modal-container',\n    tabindex: '-1',\n    'aria-modal': 'true',\n    '[id]': '_id',\n    '[attr.role]': '_config.role',\n    '[attr.aria-labelledby]': '_config.ariaLabel ? null : _ariaLabelledBy',\n    '[attr.aria-label]': '_config.ariaLabel',\n    '[attr.aria-describedby]': '_config.ariaDescribedBy || null',\n    '[@modalContainer]': '_state',\n    '(@modalContainer.start)': '_onAnimationStart($event)',\n    '(@modalContainer.done)': '_onAnimationDone($event)'\n  }\n})\nexport class BaoModalContainer extends _BaoModalContainerBase {\n  /** State of the modal animation. */\n  _state: 'void' | 'enter' | 'exit' = 'enter';\n\n  /** Callback, invoked whenever an animation on the host completes. */\n  public async _onAnimationDone({ toState, totalTime }: AnimationEvent) {\n    if (toState === 'enter') {\n      await this._trapFocus();\n      this._animationStateChanged.next({ state: 'opened', totalTime });\n    } else if (toState === 'exit') {\n      this._restoreFocus();\n      this._animationStateChanged.next({ state: 'closed', totalTime });\n    }\n  }\n\n  /** Callback, invoked when an animation on the host starts. */\n  public _onAnimationStart({ toState, totalTime }: AnimationEvent) {\n    if (toState === 'enter') {\n      this._animationStateChanged.next({ state: 'opening', totalTime });\n    } else if (toState === 'exit' || toState === 'void') {\n      this._animationStateChanged.next({ state: 'closing', totalTime });\n    }\n  }\n\n  /** Starts the modal exit animation. */\n  public _startExitAnimation(): void {\n    this._state = 'exit';\n\n    // Mark the container for check so it can react if the\n    // view container is using OnPush change detection.\n    this._changeDetectorRef.markForCheck();\n  }\n}\n","<ng-template cdkPortalOutlet></ng-template>\n"]}
|
|
@@ -0,0 +1,84 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2022 Ville de Montreal. All rights reserved.
|
|
3
|
+
* Licensed under the MIT license.
|
|
4
|
+
* See LICENSE file in the project root for full license information.
|
|
5
|
+
*/
|
|
6
|
+
import { Directive, Input, Optional } from '@angular/core';
|
|
7
|
+
import { _closeModalVia } from './modal-ref';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "./modal-ref";
|
|
10
|
+
import * as i2 from "./modal";
|
|
11
|
+
/**
|
|
12
|
+
* Button that will close the current dialog.
|
|
13
|
+
*/
|
|
14
|
+
export class BaoModalClose {
|
|
15
|
+
constructor(modalRef, _elementRef, _dialog) {
|
|
16
|
+
this.modalRef = modalRef;
|
|
17
|
+
this._elementRef = _elementRef;
|
|
18
|
+
this._dialog = _dialog;
|
|
19
|
+
/** Default to "button" to prevents accidental form submits. */
|
|
20
|
+
this.type = 'button';
|
|
21
|
+
}
|
|
22
|
+
ngOnInit() {
|
|
23
|
+
if (!this.modalRef) {
|
|
24
|
+
// When this directive is included in a dialog via TemplateRef (rather than being
|
|
25
|
+
// in a Component), the modalRef isn't available via injection because embedded
|
|
26
|
+
// views cannot be given a custom injector. Instead, we look up the modalRef by
|
|
27
|
+
// ID. This must occur in `onInit`, as the ID binding for the dialog container won't
|
|
28
|
+
// be resolved at constructor time.
|
|
29
|
+
this.modalRef =
|
|
30
|
+
getClosestDialog(this._elementRef, this._dialog.openModals) || null;
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
ngOnChanges(changes) {
|
|
34
|
+
const proxiedChange = changes['_baoModalClose'] || changes['_baoModalCloseResult'];
|
|
35
|
+
if (proxiedChange) {
|
|
36
|
+
this.dialogResult = proxiedChange.currentValue;
|
|
37
|
+
}
|
|
38
|
+
}
|
|
39
|
+
_onButtonClick(event) {
|
|
40
|
+
// Determinate the focus origin using the click event, because using the FocusMonitor will
|
|
41
|
+
// result in incorrect origins. Most of the time, close buttons will be auto focused in the
|
|
42
|
+
// dialog, and therefore clicking the button won't result in a focus change. This means that
|
|
43
|
+
// the FocusMonitor won't detect any origin change, and will always output `program`.
|
|
44
|
+
_closeModalVia(this.modalRef, event.screenX === 0 && event.screenY === 0 ? 'keyboard' : 'mouse', this.dialogResult);
|
|
45
|
+
}
|
|
46
|
+
}
|
|
47
|
+
BaoModalClose.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: BaoModalClose, deps: [{ token: i1.BaoModalRef, optional: true }, { token: i0.ElementRef }, { token: i2.BaoModal }], target: i0.ɵɵFactoryTarget.Directive });
|
|
48
|
+
BaoModalClose.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "12.0.0", version: "13.1.1", type: BaoModalClose, selector: "[bao-modal-close], [baoModalClose]", inputs: { ariaLabel: ["aria-label", "ariaLabel"], type: "type", dialogResult: ["bao-modal-close", "dialogResult"], _baoModalClose: ["baoModalClose", "_baoModalClose"] }, host: { listeners: { "click": "_onButtonClick($event)" }, properties: { "attr.aria-label": "ariaLabel || null", "attr.type": "type" } }, exportAs: ["BaoModalClose"], usesOnChanges: true, ngImport: i0 });
|
|
49
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.1.1", ngImport: i0, type: BaoModalClose, decorators: [{
|
|
50
|
+
type: Directive,
|
|
51
|
+
args: [{
|
|
52
|
+
selector: '[bao-modal-close], [baoModalClose]',
|
|
53
|
+
exportAs: 'BaoModalClose',
|
|
54
|
+
host: {
|
|
55
|
+
'(click)': '_onButtonClick($event)',
|
|
56
|
+
'[attr.aria-label]': 'ariaLabel || null',
|
|
57
|
+
'[attr.type]': 'type'
|
|
58
|
+
}
|
|
59
|
+
}]
|
|
60
|
+
}], ctorParameters: function () { return [{ type: i1.BaoModalRef, decorators: [{
|
|
61
|
+
type: Optional
|
|
62
|
+
}] }, { type: i0.ElementRef }, { type: i2.BaoModal }]; }, propDecorators: { ariaLabel: [{
|
|
63
|
+
type: Input,
|
|
64
|
+
args: ['aria-label']
|
|
65
|
+
}], type: [{
|
|
66
|
+
type: Input
|
|
67
|
+
}], dialogResult: [{
|
|
68
|
+
type: Input,
|
|
69
|
+
args: ['bao-modal-close']
|
|
70
|
+
}], _baoModalClose: [{
|
|
71
|
+
type: Input,
|
|
72
|
+
args: ['baoModalClose']
|
|
73
|
+
}] } });
|
|
74
|
+
/**
|
|
75
|
+
* Finds the closest BaoModalRef to an element by looking at the DOM.
|
|
76
|
+
*/
|
|
77
|
+
function getClosestDialog(element, openDialogs) {
|
|
78
|
+
let parent = element.nativeElement.parentElement;
|
|
79
|
+
while (parent && !parent.classList.contains('bao-modal-container')) {
|
|
80
|
+
parent = parent.parentElement;
|
|
81
|
+
}
|
|
82
|
+
return parent ? openDialogs.find(dialog => dialog.id === parent.id) : null;
|
|
83
|
+
}
|
|
84
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kYWwtZGlyZWN0aXZlcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2FuZ3VsYXItdWkvc3JjL2xpYi9tb2RhbC9tb2RhbC1kaXJlY3RpdmVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7O0dBSUc7QUFFSCxPQUFPLEVBQ0wsU0FBUyxFQUNULEtBQUssRUFHTCxRQUFRLEVBR1QsTUFBTSxlQUFlLENBQUM7QUFFdkIsT0FBTyxFQUFFLGNBQWMsRUFBZSxNQUFNLGFBQWEsQ0FBQzs7OztBQUUxRDs7R0FFRztBQVVILE1BQU0sT0FBTyxhQUFhO0lBWXhCLFlBQ3FCLFFBQXFDLEVBQ2hELFdBQW9DLEVBQ3BDLE9BQWlCO1FBRk4sYUFBUSxHQUFSLFFBQVEsQ0FBNkI7UUFDaEQsZ0JBQVcsR0FBWCxXQUFXLENBQXlCO1FBQ3BDLFlBQU8sR0FBUCxPQUFPLENBQVU7UUFYM0IsK0RBQStEO1FBQ3RELFNBQUksR0FBa0MsUUFBUSxDQUFDO0lBV3JELENBQUM7SUFFSixRQUFRO1FBQ04sSUFBSSxDQUFDLElBQUksQ0FBQyxRQUFRLEVBQUU7WUFDbEIsaUZBQWlGO1lBQ2pGLCtFQUErRTtZQUMvRSwrRUFBK0U7WUFDL0Usb0ZBQW9GO1lBQ3BGLG1DQUFtQztZQUNuQyxJQUFJLENBQUMsUUFBUTtnQkFDWCxnQkFBZ0IsQ0FBQyxJQUFJLENBQUMsV0FBVyxFQUFFLElBQUksQ0FBQyxPQUFPLENBQUMsVUFBVSxDQUFDLElBQUksSUFBSSxDQUFDO1NBQ3ZFO0lBQ0gsQ0FBQztJQUVELFdBQVcsQ0FBQyxPQUFzQjtRQUNoQyxNQUFNLGFBQWEsR0FDakIsT0FBTyxDQUFDLGdCQUFnQixDQUFDLElBQUksT0FBTyxDQUFDLHNCQUFzQixDQUFDLENBQUM7UUFFL0QsSUFBSSxhQUFhLEVBQUU7WUFDakIsSUFBSSxDQUFDLFlBQVksR0FBRyxhQUFhLENBQUMsWUFBWSxDQUFDO1NBQ2hEO0lBQ0gsQ0FBQztJQUVELGNBQWMsQ0FBQyxLQUFpQjtRQUM5QiwwRkFBMEY7UUFDMUYsMkZBQTJGO1FBQzNGLDRGQUE0RjtRQUM1RixxRkFBcUY7UUFDckYsY0FBYyxDQUNaLElBQUksQ0FBQyxRQUFRLEVBQ2IsS0FBSyxDQUFDLE9BQU8sS0FBSyxDQUFDLElBQUksS0FBSyxDQUFDLE9BQU8sS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQUMsT0FBTyxFQUNqRSxJQUFJLENBQUMsWUFBWSxDQUNsQixDQUFDO0lBQ0osQ0FBQzs7MEdBakRVLGFBQWE7OEZBQWIsYUFBYTsyRkFBYixhQUFhO2tCQVR6QixTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSxvQ0FBb0M7b0JBQzlDLFFBQVEsRUFBRSxlQUFlO29CQUN6QixJQUFJLEVBQUU7d0JBQ0osU0FBUyxFQUFFLHdCQUF3Qjt3QkFDbkMsbUJBQW1CLEVBQUUsbUJBQW1CO3dCQUN4QyxhQUFhLEVBQUUsTUFBTTtxQkFDdEI7aUJBQ0Y7OzBCQWNJLFFBQVE7NEZBWFUsU0FBUztzQkFBN0IsS0FBSzt1QkFBQyxZQUFZO2dCQUdWLElBQUk7c0JBQVosS0FBSztnQkFHb0IsWUFBWTtzQkFBckMsS0FBSzt1QkFBQyxpQkFBaUI7Z0JBRUEsY0FBYztzQkFBckMsS0FBSzt1QkFBQyxlQUFlOztBQTBDeEI7O0dBRUc7QUFDSCxTQUFTLGdCQUFnQixDQUN2QixPQUFnQyxFQUNoQyxXQUFtQztJQUVuQyxJQUFJLE1BQU0sR0FBdUIsT0FBTyxDQUFDLGFBQWEsQ0FBQyxhQUFhLENBQUM7SUFFckUsT0FBTyxNQUFNLElBQUksQ0FBQyxNQUFNLENBQUMsU0FBUyxDQUFDLFFBQVEsQ0FBQyxxQkFBcUIsQ0FBQyxFQUFFO1FBQ2xFLE1BQU0sR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDO0tBQy9CO0lBRUQsT0FBTyxNQUFNLENBQUMsQ0FBQyxDQUFDLFdBQVcsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxNQUFNLENBQUMsRUFBRSxLQUFLLE1BQU0sQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDO0FBQzdFLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDIyIFZpbGxlIGRlIE1vbnRyZWFsLiBBbGwgcmlnaHRzIHJlc2VydmVkLlxuICogTGljZW5zZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLlxuICogU2VlIExJQ0VOU0UgZmlsZSBpbiB0aGUgcHJvamVjdCByb290IGZvciBmdWxsIGxpY2Vuc2UgaW5mb3JtYXRpb24uXG4gKi9cblxuaW1wb3J0IHtcbiAgRGlyZWN0aXZlLFxuICBJbnB1dCxcbiAgT25DaGFuZ2VzLFxuICBPbkluaXQsXG4gIE9wdGlvbmFsLFxuICBTaW1wbGVDaGFuZ2VzLFxuICBFbGVtZW50UmVmXG59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQmFvTW9kYWwgfSBmcm9tICcuL21vZGFsJztcbmltcG9ydCB7IF9jbG9zZU1vZGFsVmlhLCBCYW9Nb2RhbFJlZiB9IGZyb20gJy4vbW9kYWwtcmVmJztcblxuLyoqXG4gKiBCdXR0b24gdGhhdCB3aWxsIGNsb3NlIHRoZSBjdXJyZW50IGRpYWxvZy5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gIHNlbGVjdG9yOiAnW2Jhby1tb2RhbC1jbG9zZV0sIFtiYW9Nb2RhbENsb3NlXScsXG4gIGV4cG9ydEFzOiAnQmFvTW9kYWxDbG9zZScsXG4gIGhvc3Q6IHtcbiAgICAnKGNsaWNrKSc6ICdfb25CdXR0b25DbGljaygkZXZlbnQpJyxcbiAgICAnW2F0dHIuYXJpYS1sYWJlbF0nOiAnYXJpYUxhYmVsIHx8IG51bGwnLFxuICAgICdbYXR0ci50eXBlXSc6ICd0eXBlJ1xuICB9XG59KVxuZXhwb3J0IGNsYXNzIEJhb01vZGFsQ2xvc2UgaW1wbGVtZW50cyBPbkluaXQsIE9uQ2hhbmdlcyB7XG4gIC8qKiBTY3JlZW5yZWFkZXIgbGFiZWwgZm9yIHRoZSBidXR0b24uICovXG4gIEBJbnB1dCgnYXJpYS1sYWJlbCcpIGFyaWFMYWJlbDogc3RyaW5nO1xuXG4gIC8qKiBEZWZhdWx0IHRvIFwiYnV0dG9uXCIgdG8gcHJldmVudHMgYWNjaWRlbnRhbCBmb3JtIHN1Ym1pdHMuICovXG4gIEBJbnB1dCgpIHR5cGU6ICdzdWJtaXQnIHwgJ2J1dHRvbicgfCAncmVzZXQnID0gJ2J1dHRvbic7XG5cbiAgLyoqIERpYWxvZyBjbG9zZSBpbnB1dC4gKi9cbiAgQElucHV0KCdiYW8tbW9kYWwtY2xvc2UnKSBkaWFsb2dSZXN1bHQ6IHVua25vd247XG5cbiAgQElucHV0KCdiYW9Nb2RhbENsb3NlJykgX2Jhb01vZGFsQ2xvc2U6IHVua25vd247XG5cbiAgY29uc3RydWN0b3IoXG4gICAgQE9wdGlvbmFsKCkgcHVibGljIG1vZGFsUmVmOiBCYW9Nb2RhbFJlZjx1bmtub3duPiB8IG51bGwsXG4gICAgcHJpdmF0ZSBfZWxlbWVudFJlZjogRWxlbWVudFJlZjxIVE1MRWxlbWVudD4sXG4gICAgcHJpdmF0ZSBfZGlhbG9nOiBCYW9Nb2RhbFxuICApIHt9XG5cbiAgbmdPbkluaXQoKSB7XG4gICAgaWYgKCF0aGlzLm1vZGFsUmVmKSB7XG4gICAgICAvLyBXaGVuIHRoaXMgZGlyZWN0aXZlIGlzIGluY2x1ZGVkIGluIGEgZGlhbG9nIHZpYSBUZW1wbGF0ZVJlZiAocmF0aGVyIHRoYW4gYmVpbmdcbiAgICAgIC8vIGluIGEgQ29tcG9uZW50KSwgdGhlIG1vZGFsUmVmIGlzbid0IGF2YWlsYWJsZSB2aWEgaW5qZWN0aW9uIGJlY2F1c2UgZW1iZWRkZWRcbiAgICAgIC8vIHZpZXdzIGNhbm5vdCBiZSBnaXZlbiBhIGN1c3RvbSBpbmplY3Rvci4gSW5zdGVhZCwgd2UgbG9vayB1cCB0aGUgbW9kYWxSZWYgYnlcbiAgICAgIC8vIElELiBUaGlzIG11c3Qgb2NjdXIgaW4gYG9uSW5pdGAsIGFzIHRoZSBJRCBiaW5kaW5nIGZvciB0aGUgZGlhbG9nIGNvbnRhaW5lciB3b24ndFxuICAgICAgLy8gYmUgcmVzb2x2ZWQgYXQgY29uc3RydWN0b3IgdGltZS5cbiAgICAgIHRoaXMubW9kYWxSZWYgPVxuICAgICAgICBnZXRDbG9zZXN0RGlhbG9nKHRoaXMuX2VsZW1lbnRSZWYsIHRoaXMuX2RpYWxvZy5vcGVuTW9kYWxzKSB8fCBudWxsO1xuICAgIH1cbiAgfVxuXG4gIG5nT25DaGFuZ2VzKGNoYW5nZXM6IFNpbXBsZUNoYW5nZXMpIHtcbiAgICBjb25zdCBwcm94aWVkQ2hhbmdlID1cbiAgICAgIGNoYW5nZXNbJ19iYW9Nb2RhbENsb3NlJ10gfHwgY2hhbmdlc1snX2Jhb01vZGFsQ2xvc2VSZXN1bHQnXTtcblxuICAgIGlmIChwcm94aWVkQ2hhbmdlKSB7XG4gICAgICB0aGlzLmRpYWxvZ1Jlc3VsdCA9IHByb3hpZWRDaGFuZ2UuY3VycmVudFZhbHVlO1xuICAgIH1cbiAgfVxuXG4gIF9vbkJ1dHRvbkNsaWNrKGV2ZW50OiBNb3VzZUV2ZW50KSB7XG4gICAgLy8gRGV0ZXJtaW5hdGUgdGhlIGZvY3VzIG9yaWdpbiB1c2luZyB0aGUgY2xpY2sgZXZlbnQsIGJlY2F1c2UgdXNpbmcgdGhlIEZvY3VzTW9uaXRvciB3aWxsXG4gICAgLy8gcmVzdWx0IGluIGluY29ycmVjdCBvcmlnaW5zLiBNb3N0IG9mIHRoZSB0aW1lLCBjbG9zZSBidXR0b25zIHdpbGwgYmUgYXV0byBmb2N1c2VkIGluIHRoZVxuICAgIC8vIGRpYWxvZywgYW5kIHRoZXJlZm9yZSBjbGlja2luZyB0aGUgYnV0dG9uIHdvbid0IHJlc3VsdCBpbiBhIGZvY3VzIGNoYW5nZS4gVGhpcyBtZWFucyB0aGF0XG4gICAgLy8gdGhlIEZvY3VzTW9uaXRvciB3b24ndCBkZXRlY3QgYW55IG9yaWdpbiBjaGFuZ2UsIGFuZCB3aWxsIGFsd2F5cyBvdXRwdXQgYHByb2dyYW1gLlxuICAgIF9jbG9zZU1vZGFsVmlhKFxuICAgICAgdGhpcy5tb2RhbFJlZixcbiAgICAgIGV2ZW50LnNjcmVlblggPT09IDAgJiYgZXZlbnQuc2NyZWVuWSA9PT0gMCA/ICdrZXlib2FyZCcgOiAnbW91c2UnLFxuICAgICAgdGhpcy5kaWFsb2dSZXN1bHRcbiAgICApO1xuICB9XG59XG5cbi8qKlxuICogRmluZHMgdGhlIGNsb3Nlc3QgQmFvTW9kYWxSZWYgdG8gYW4gZWxlbWVudCBieSBsb29raW5nIGF0IHRoZSBET00uXG4gKi9cbmZ1bmN0aW9uIGdldENsb3Nlc3REaWFsb2coXG4gIGVsZW1lbnQ6IEVsZW1lbnRSZWY8SFRNTEVsZW1lbnQ+LFxuICBvcGVuRGlhbG9nczogQmFvTW9kYWxSZWY8dW5rbm93bj5bXVxuKTogQmFvTW9kYWxSZWY8dW5rbm93bj4gfCBudWxsIHtcbiAgbGV0IHBhcmVudDogSFRNTEVsZW1lbnQgfCBudWxsID0gZWxlbWVudC5uYXRpdmVFbGVtZW50LnBhcmVudEVsZW1lbnQ7XG5cbiAgd2hpbGUgKHBhcmVudCAmJiAhcGFyZW50LmNsYXNzTGlzdC5jb250YWlucygnYmFvLW1vZGFsLWNvbnRhaW5lcicpKSB7XG4gICAgcGFyZW50ID0gcGFyZW50LnBhcmVudEVsZW1lbnQ7XG4gIH1cblxuICByZXR1cm4gcGFyZW50ID8gb3BlbkRpYWxvZ3MuZmluZChkaWFsb2cgPT4gZGlhbG9nLmlkID09PSBwYXJlbnQuaWQpIDogbnVsbDtcbn1cbiJdfQ==
|