@villedemontreal/angular-ui 2.2.1 → 3.1.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/esm2020/lib/alert/alert.component.mjs +24 -26
- package/esm2020/lib/alert/module.mjs +5 -5
- 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/badge/badge.component.mjs +3 -3
- package/esm2020/lib/badge/module.mjs +5 -5
- package/esm2020/lib/bao.module.mjs +24 -14
- package/esm2020/lib/breadcrumb/breadcrumb.component.mjs +8 -8
- package/esm2020/lib/breadcrumb/module.mjs +5 -5
- package/esm2020/lib/button/button.component.mjs +7 -7
- package/esm2020/lib/button/module.mjs +5 -5
- package/esm2020/lib/card/card.component.mjs +15 -15
- package/esm2020/lib/card/module.mjs +5 -5
- package/esm2020/lib/checkbox/checkbox-group.component.mjs +8 -9
- package/esm2020/lib/checkbox/checkbox.component.mjs +45 -46
- package/esm2020/lib/checkbox/module.mjs +5 -5
- package/esm2020/lib/common-components/error-text/errorText.component.mjs +3 -3
- package/esm2020/lib/common-components/guiding-text/guidingText.component.mjs +3 -3
- package/esm2020/lib/common-components/label-text/labelText.component.mjs +3 -3
- package/esm2020/lib/common-components/module.mjs +5 -5
- package/esm2020/lib/common-components/title-text/titleText.component.mjs +3 -3
- package/esm2020/lib/header-info/header-info.component.mjs +18 -18
- package/esm2020/lib/header-info/module.mjs +5 -5
- package/esm2020/lib/icon/bao-icon-registry.mjs +3 -3
- package/esm2020/lib/icon/icon.component.mjs +30 -34
- package/esm2020/lib/icon/icons-dictionary.mjs +3 -2
- package/esm2020/lib/icon/module.mjs +5 -5
- package/esm2020/lib/list/list.component.mjs +15 -15
- package/esm2020/lib/list/module.mjs +5 -5
- 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 +293 -0
- package/esm2020/lib/modal/module.mjs +44 -0
- package/esm2020/lib/radio/module.mjs +10 -12
- package/esm2020/lib/radio/radio-group.component.mjs +43 -44
- package/esm2020/lib/radio/radio.component.mjs +45 -46
- package/esm2020/lib/summary/list-summary.component.mjs +9 -16
- package/esm2020/lib/summary/module.mjs +5 -5
- package/esm2020/lib/summary/summary.component.mjs +10 -11
- 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/module.mjs +5 -5
- package/esm2020/lib/tag/tag.component.mjs +5 -5
- package/esm2020/public-api.mjs +4 -1
- package/fesm2015/villedemontreal-angular-ui.mjs +1778 -384
- package/fesm2015/villedemontreal-angular-ui.mjs.map +1 -1
- package/fesm2020/villedemontreal-angular-ui.mjs +1770 -384
- package/fesm2020/villedemontreal-angular-ui.mjs.map +1 -1
- package/{villedemontreal-angular-ui.d.ts → index.d.ts} +0 -0
- package/lib/alert/alert.component.d.ts +6 -7
- 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/badge/badge.component.d.ts +1 -1
- package/lib/bao.module.d.ts +4 -1
- package/lib/breadcrumb/breadcrumb.component.d.ts +2 -2
- package/lib/button/button.component.d.ts +2 -2
- package/lib/card/card.component.d.ts +5 -5
- package/lib/checkbox/checkbox-group.component.d.ts +3 -3
- package/lib/checkbox/checkbox.component.d.ts +28 -28
- package/lib/common-components/error-text/errorText.component.d.ts +1 -1
- package/lib/common-components/guiding-text/guidingText.component.d.ts +1 -1
- package/lib/common-components/label-text/labelText.component.d.ts +1 -1
- package/lib/common-components/title-text/titleText.component.d.ts +1 -1
- package/lib/header-info/header-info.component.d.ts +6 -6
- package/lib/icon/icon.component.d.ts +11 -11
- package/lib/list/list.component.d.ts +5 -5
- 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 +28 -28
- package/lib/summary/list-summary.component.d.ts +2 -2
- package/lib/summary/summary.component.d.ts +7 -7
- 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 +2 -2
- package/package.json +7 -7
- package/public-api.d.ts +3 -0
|
@@ -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: "14.2.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: "14.0.0", version: "14.2.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: "14.2.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==
|
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
import { ESCAPE, hasModifierKey } from '@angular/cdk/keycodes';
|
|
2
|
+
import { filter, Subject, take } from 'rxjs';
|
|
3
|
+
// Counter for unique modal ids.
|
|
4
|
+
let uniqueId = 0;
|
|
5
|
+
/**
|
|
6
|
+
* Reference to a modal opened via the BaoModalService.
|
|
7
|
+
*/
|
|
8
|
+
export class BaoModalRef {
|
|
9
|
+
constructor(_overlayRef, _containerInstance,
|
|
10
|
+
/** Id of the modal. */
|
|
11
|
+
id = `bao-modal-${uniqueId++}`) {
|
|
12
|
+
this._overlayRef = _overlayRef;
|
|
13
|
+
this._containerInstance = _containerInstance;
|
|
14
|
+
this.id = id;
|
|
15
|
+
/** Whether the user is allowed to close the modal. */
|
|
16
|
+
this.disableClose = this._containerInstance._config.disableClose;
|
|
17
|
+
/** Subject for notifying the user that the modal has finished opening. */
|
|
18
|
+
this._afterOpened = new Subject();
|
|
19
|
+
/** Subject for notifying the user that the modal has finished closing. */
|
|
20
|
+
this._afterClosed = new Subject();
|
|
21
|
+
/** Subject for notifying the user that the modal has started closing. */
|
|
22
|
+
this._beforeClosed = new Subject();
|
|
23
|
+
/** Current state of the modal. */
|
|
24
|
+
this._state = 0 /* OPEN */;
|
|
25
|
+
// Pass the id along to the container.
|
|
26
|
+
_containerInstance._id = id;
|
|
27
|
+
// Emit when opening animation completes
|
|
28
|
+
_containerInstance._animationStateChanged
|
|
29
|
+
.pipe(filter(event => event.state === 'opened'), take(1))
|
|
30
|
+
.subscribe(() => {
|
|
31
|
+
this._afterOpened.next();
|
|
32
|
+
this._afterOpened.complete();
|
|
33
|
+
});
|
|
34
|
+
// Dispose overlay when closing animation is complete
|
|
35
|
+
_containerInstance._animationStateChanged
|
|
36
|
+
.pipe(filter(event => event.state === 'closed'), take(1))
|
|
37
|
+
.subscribe(() => {
|
|
38
|
+
clearTimeout(this._closeFallbackTimeout);
|
|
39
|
+
this._finishModalClose();
|
|
40
|
+
});
|
|
41
|
+
_overlayRef.detachments().subscribe(() => {
|
|
42
|
+
this._beforeClosed.next(this._result);
|
|
43
|
+
this._beforeClosed.complete();
|
|
44
|
+
this._afterClosed.next(this._result);
|
|
45
|
+
this._afterClosed.complete();
|
|
46
|
+
this.componentInstance = null;
|
|
47
|
+
this._overlayRef.dispose();
|
|
48
|
+
});
|
|
49
|
+
_overlayRef
|
|
50
|
+
.keydownEvents()
|
|
51
|
+
.pipe(filter(event => {
|
|
52
|
+
return (event.keyCode === ESCAPE &&
|
|
53
|
+
!this.disableClose &&
|
|
54
|
+
!hasModifierKey(event));
|
|
55
|
+
}))
|
|
56
|
+
.subscribe(event => {
|
|
57
|
+
event.preventDefault();
|
|
58
|
+
_closeModalVia(this, 'keyboard');
|
|
59
|
+
});
|
|
60
|
+
_overlayRef.backdropClick().subscribe(async () => {
|
|
61
|
+
if (this.disableClose) {
|
|
62
|
+
await this._containerInstance._recaptureFocus();
|
|
63
|
+
}
|
|
64
|
+
else {
|
|
65
|
+
_closeModalVia(this, 'mouse');
|
|
66
|
+
}
|
|
67
|
+
});
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Close the modal.
|
|
71
|
+
* @param modalResult Optional result to return to the modal opener.
|
|
72
|
+
*/
|
|
73
|
+
close(modalResult) {
|
|
74
|
+
this._result = modalResult;
|
|
75
|
+
// Transition the backdrop in parallel to the modal.
|
|
76
|
+
this._containerInstance._animationStateChanged
|
|
77
|
+
.pipe(filter(event => event.state === 'closing'), take(1))
|
|
78
|
+
.subscribe(event => {
|
|
79
|
+
this._beforeClosed.next(modalResult);
|
|
80
|
+
this._beforeClosed.complete();
|
|
81
|
+
this._overlayRef.detachBackdrop();
|
|
82
|
+
// The logic that disposes of the overlay depends on the exit animation completing, however
|
|
83
|
+
// it isn't guaranteed if the parent view is destroyed while it's running. Add a fallback
|
|
84
|
+
// timeout which will clean everything up if the animation hasn't fired within the specified
|
|
85
|
+
// amount of time plus 100ms. We don't need to run this outside the NgZone, because for the
|
|
86
|
+
// vast majority of cases the timeout will have been cleared before it has the chance to fire.
|
|
87
|
+
this._closeFallbackTimeout = setTimeout(() => this._finishModalClose(), event.totalTime + 100);
|
|
88
|
+
});
|
|
89
|
+
this._state = 1 /* CLOSING */;
|
|
90
|
+
this._containerInstance._startExitAnimation();
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Gets an observable that is notified when the modal is finished opening.
|
|
94
|
+
*/
|
|
95
|
+
afterOpened() {
|
|
96
|
+
return this._afterOpened;
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* Gets an observable that is notified when the modal is finished closing.
|
|
100
|
+
*/
|
|
101
|
+
afterClosed() {
|
|
102
|
+
return this._afterClosed;
|
|
103
|
+
}
|
|
104
|
+
/**
|
|
105
|
+
* Gets an observable that is notified when the modal has started closing.
|
|
106
|
+
*/
|
|
107
|
+
beforeClosed() {
|
|
108
|
+
return this._beforeClosed;
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Gets an observable that emits when the overlay's backdrop has been clicked.
|
|
112
|
+
*/
|
|
113
|
+
backdropClick() {
|
|
114
|
+
return this._overlayRef.backdropClick();
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Gets an observable that emits when keydown events are targeted on the overlay.
|
|
118
|
+
*/
|
|
119
|
+
keydownEvents() {
|
|
120
|
+
return this._overlayRef.keydownEvents();
|
|
121
|
+
}
|
|
122
|
+
/**
|
|
123
|
+
* Updates the dialog's position.
|
|
124
|
+
*/
|
|
125
|
+
updatePosition(position) {
|
|
126
|
+
const strategy = this._getPositionStrategy();
|
|
127
|
+
if (position && (position.left || position.right)) {
|
|
128
|
+
position.left
|
|
129
|
+
? strategy.left(position.left)
|
|
130
|
+
: strategy.right(position.right);
|
|
131
|
+
}
|
|
132
|
+
else {
|
|
133
|
+
strategy.centerHorizontally();
|
|
134
|
+
}
|
|
135
|
+
if (position && (position.top || position.bottom)) {
|
|
136
|
+
position.top
|
|
137
|
+
? strategy.top(position.top)
|
|
138
|
+
: strategy.bottom(position.bottom);
|
|
139
|
+
}
|
|
140
|
+
else {
|
|
141
|
+
strategy.centerVertically();
|
|
142
|
+
}
|
|
143
|
+
this._overlayRef.updatePosition();
|
|
144
|
+
return this;
|
|
145
|
+
}
|
|
146
|
+
/**
|
|
147
|
+
* Updates the modal's width and height.
|
|
148
|
+
*/
|
|
149
|
+
updateSize(width = '', height = '') {
|
|
150
|
+
this._overlayRef.updateSize({ width, height });
|
|
151
|
+
this._overlayRef.updatePosition();
|
|
152
|
+
return this;
|
|
153
|
+
}
|
|
154
|
+
/** Add a CSS class or an array of classes to the overlay pane. */
|
|
155
|
+
addPanelClass(classes) {
|
|
156
|
+
this._overlayRef.addPanelClass(classes);
|
|
157
|
+
return this;
|
|
158
|
+
}
|
|
159
|
+
/** Remove a CSS class or an array of classes from the overlay pane. */
|
|
160
|
+
removePanelClass(classes) {
|
|
161
|
+
this._overlayRef.removePanelClass(classes);
|
|
162
|
+
return this;
|
|
163
|
+
}
|
|
164
|
+
/** Gets the current state of the modal's lifecycle. */
|
|
165
|
+
getState() {
|
|
166
|
+
return this._state;
|
|
167
|
+
}
|
|
168
|
+
/**
|
|
169
|
+
* Finishes the modal close by updating the state of the modal
|
|
170
|
+
* and disposing the overlay.
|
|
171
|
+
*/
|
|
172
|
+
_finishModalClose() {
|
|
173
|
+
this._state = 2 /* CLOSED */;
|
|
174
|
+
this._overlayRef.dispose();
|
|
175
|
+
}
|
|
176
|
+
/** Fetches the position strategy object from the overlay ref. */
|
|
177
|
+
_getPositionStrategy() {
|
|
178
|
+
return this._overlayRef.getConfig()
|
|
179
|
+
.positionStrategy;
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
/**
|
|
183
|
+
* Closes the modal with the specified interaction type. This is currently not part of
|
|
184
|
+
* `BaoModalRef` as that would conflict with custom modal ref mocks provided in tests.
|
|
185
|
+
* More details. See: https://github.com/angular/components/pull/9257#issuecomment-651342226.
|
|
186
|
+
*/
|
|
187
|
+
export function _closeModalVia(ref, interactionType, result) {
|
|
188
|
+
// Some mock modal ref instances in tests do not have the `_containerInstance` property.
|
|
189
|
+
// For those, we keep the behavior as is and do not deal with the interaction type.
|
|
190
|
+
if (ref._containerInstance !== undefined) {
|
|
191
|
+
ref._containerInstance._closeInteractionType = interactionType;
|
|
192
|
+
}
|
|
193
|
+
return ref.close(result);
|
|
194
|
+
}
|
|
195
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-ref.js","sourceRoot":"","sources":["../../../../../projects/angular-ui/src/lib/modal/modal-ref.ts"],"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 { FocusOrigin } from '@angular/cdk/a11y';\nimport { ESCAPE, hasModifierKey } from '@angular/cdk/keycodes';\nimport { GlobalPositionStrategy, OverlayRef } from '@angular/cdk/overlay';\nimport { filter, Observable, Subject, take } from 'rxjs';\nimport { ModalPosition } from './modal-config';\nimport { _BaoModalContainerBase } from './modal-container';\n\n// Counter for unique modal ids.\nlet uniqueId = 0;\n\n/** Possible states of the lifecycle of a modal. */\nexport const enum BaoModalState {\n  OPEN,\n  CLOSING,\n  CLOSED\n}\n\n/**\n * Reference to a modal opened via the BaoModalService.\n */\nexport class BaoModalRef<T, R = unknown> {\n  /** The instance of component opened into the modal. */\n  public componentInstance: T;\n\n  /** Whether the user is allowed to close the modal. */\n  public disableClose: boolean | undefined =\n    this._containerInstance._config.disableClose;\n\n  /** Subject for notifying the user that the modal has finished opening. */\n  private readonly _afterOpened = new Subject<void>();\n\n  /** Subject for notifying the user that the modal has finished closing. */\n  private readonly _afterClosed = new Subject<R | undefined>();\n\n  /** Subject for notifying the user that the modal has started closing. */\n  private readonly _beforeClosed = new Subject<R | undefined>();\n\n  /** Result to be passed to afterClosed. */\n  private _result: R | undefined;\n\n  /** Handle to the timeout that's running as a fallback in case the exit animation doesn't fire. */\n  private _closeFallbackTimeout: any;\n\n  /** Current state of the modal. */\n  private _state = BaoModalState.OPEN;\n\n  constructor(\n    private _overlayRef: OverlayRef,\n    public _containerInstance: _BaoModalContainerBase,\n    /** Id of the modal. */\n    readonly id: string = `bao-modal-${uniqueId++}`\n  ) {\n    // Pass the id along to the container.\n    _containerInstance._id = id;\n\n    // Emit when opening animation completes\n    _containerInstance._animationStateChanged\n      .pipe(\n        filter(event => event.state === 'opened'),\n        take(1)\n      )\n      .subscribe(() => {\n        this._afterOpened.next();\n        this._afterOpened.complete();\n      });\n\n    // Dispose overlay when closing animation is complete\n    _containerInstance._animationStateChanged\n      .pipe(\n        filter(event => event.state === 'closed'),\n        take(1)\n      )\n      .subscribe(() => {\n        clearTimeout(this._closeFallbackTimeout);\n        this._finishModalClose();\n      });\n\n    _overlayRef.detachments().subscribe(() => {\n      this._beforeClosed.next(this._result);\n      this._beforeClosed.complete();\n      this._afterClosed.next(this._result);\n      this._afterClosed.complete();\n      this.componentInstance = null;\n      this._overlayRef.dispose();\n    });\n\n    _overlayRef\n      .keydownEvents()\n      .pipe(\n        filter(event => {\n          return (\n            event.keyCode === ESCAPE &&\n            !this.disableClose &&\n            !hasModifierKey(event)\n          );\n        })\n      )\n      .subscribe(event => {\n        event.preventDefault();\n        _closeModalVia(this, 'keyboard');\n      });\n\n    _overlayRef.backdropClick().subscribe(async () => {\n      if (this.disableClose) {\n        await this._containerInstance._recaptureFocus();\n      } else {\n        _closeModalVia(this, 'mouse');\n      }\n    });\n  }\n\n  /**\n   * Close the modal.\n   * @param modalResult Optional result to return to the modal opener.\n   */\n  public close(modalResult?: R): void {\n    this._result = modalResult;\n\n    // Transition the backdrop in parallel to the modal.\n    this._containerInstance._animationStateChanged\n      .pipe(\n        filter(event => event.state === 'closing'),\n        take(1)\n      )\n      .subscribe(event => {\n        this._beforeClosed.next(modalResult);\n        this._beforeClosed.complete();\n        this._overlayRef.detachBackdrop();\n        // The logic that disposes of the overlay depends on the exit animation completing, however\n        // it isn't guaranteed if the parent view is destroyed while it's running. Add a fallback\n        // timeout which will clean everything up if the animation hasn't fired within the specified\n        // amount of time plus 100ms. We don't need to run this outside the NgZone, because for the\n        // vast majority of cases the timeout will have been cleared before it has the chance to fire.\n        this._closeFallbackTimeout = setTimeout(\n          () => this._finishModalClose(),\n          (event.totalTime as number) + 100\n        );\n      });\n\n    this._state = BaoModalState.CLOSING;\n    this._containerInstance._startExitAnimation();\n  }\n\n  /**\n   * Gets an observable that is notified when the modal is finished opening.\n   */\n  public afterOpened(): Observable<void> {\n    return this._afterOpened;\n  }\n\n  /**\n   * Gets an observable that is notified when the modal is finished closing.\n   */\n  public afterClosed(): Observable<R | undefined> {\n    return this._afterClosed;\n  }\n\n  /**\n   * Gets an observable that is notified when the modal has started closing.\n   */\n  public beforeClosed(): Observable<R | undefined> {\n    return this._beforeClosed;\n  }\n\n  /**\n   * Gets an observable that emits when the overlay's backdrop has been clicked.\n   */\n  public backdropClick(): Observable<MouseEvent> {\n    return this._overlayRef.backdropClick();\n  }\n\n  /**\n   * Gets an observable that emits when keydown events are targeted on the overlay.\n   */\n  public keydownEvents(): Observable<KeyboardEvent> {\n    return this._overlayRef.keydownEvents();\n  }\n\n  /**\n   * Updates the dialog's position.\n   */\n  public updatePosition(position?: ModalPosition): this {\n    const strategy = this._getPositionStrategy();\n\n    if (position && (position.left || position.right)) {\n      position.left\n        ? strategy.left(position.left)\n        : strategy.right(position.right);\n    } else {\n      strategy.centerHorizontally();\n    }\n\n    if (position && (position.top || position.bottom)) {\n      position.top\n        ? strategy.top(position.top)\n        : strategy.bottom(position.bottom);\n    } else {\n      strategy.centerVertically();\n    }\n\n    this._overlayRef.updatePosition();\n\n    return this;\n  }\n\n  /**\n   * Updates the modal's width and height.\n   */\n  public updateSize(width = '', height = ''): this {\n    this._overlayRef.updateSize({ width, height });\n    this._overlayRef.updatePosition();\n    return this;\n  }\n\n  /** Add a CSS class or an array of classes to the overlay pane. */\n  public addPanelClass(classes: string | string[]): this {\n    this._overlayRef.addPanelClass(classes);\n    return this;\n  }\n\n  /** Remove a CSS class or an array of classes from the overlay pane. */\n  public removePanelClass(classes: string | string[]): this {\n    this._overlayRef.removePanelClass(classes);\n    return this;\n  }\n\n  /** Gets the current state of the modal's lifecycle. */\n  public getState(): BaoModalState {\n    return this._state;\n  }\n\n  /**\n   * Finishes the modal close by updating the state of the modal\n   * and disposing the overlay.\n   */\n  private _finishModalClose() {\n    this._state = BaoModalState.CLOSED;\n    this._overlayRef.dispose();\n  }\n\n  /** Fetches the position strategy object from the overlay ref. */\n  private _getPositionStrategy(): GlobalPositionStrategy {\n    return this._overlayRef.getConfig()\n      .positionStrategy as GlobalPositionStrategy;\n  }\n}\n\n/**\n * Closes the modal with the specified interaction type. This is currently not part of\n * `BaoModalRef` as that would conflict with custom modal ref mocks provided in tests.\n * More details. See: https://github.com/angular/components/pull/9257#issuecomment-651342226.\n */\nexport function _closeModalVia<R>(\n  ref: BaoModalRef<R>,\n  interactionType: FocusOrigin,\n  result?: R\n) {\n  // Some mock modal ref instances in tests do not have the `_containerInstance` property.\n  // For those, we keep the behavior as is and do not deal with the interaction type.\n  if (ref._containerInstance !== undefined) {\n    ref._containerInstance._closeInteractionType = interactionType;\n  }\n  return ref.close(result);\n}\n"]}
|