@skyux/modals 8.7.2 → 9.0.0-alpha.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/documentation.json +56 -56
- package/esm2022/lib/modules/confirm/confirm.component.mjs +112 -0
- package/{esm2020 → esm2022}/lib/modules/confirm/confirm.module.mjs +13 -13
- package/esm2022/lib/modules/confirm/confirm.service.mjs +54 -0
- package/esm2022/lib/modules/modal/modal-adapter.service.mjs +99 -0
- package/esm2022/lib/modules/modal/modal-component-adapter.service.mjs +101 -0
- package/{esm2020 → esm2022}/lib/modules/modal/modal-configuration.mjs +4 -4
- package/{esm2020 → esm2022}/lib/modules/modal/modal-content.component.mjs +4 -4
- package/esm2022/lib/modules/modal/modal-errors.service.mjs +21 -0
- package/{esm2020 → esm2022}/lib/modules/modal/modal-footer.component.mjs +4 -4
- package/{esm2020 → esm2022}/lib/modules/modal/modal-header.component.mjs +4 -4
- package/{esm2020 → esm2022}/lib/modules/modal/modal-host-context.mjs +4 -4
- package/esm2022/lib/modules/modal/modal-host.component.mjs +163 -0
- package/esm2022/lib/modules/modal/modal-host.service.mjs +62 -0
- package/esm2022/lib/modules/modal/modal-instance.mjs +123 -0
- package/esm2022/lib/modules/modal/modal-is-dirty.directive.mjs +94 -0
- package/esm2022/lib/modules/modal/modal-scroll-shadow.directive.mjs +129 -0
- package/esm2022/lib/modules/modal/modal.component.mjs +229 -0
- package/{esm2020 → esm2022}/lib/modules/modal/modal.module.mjs +29 -29
- package/esm2022/lib/modules/modal/modal.service.mjs +97 -0
- package/esm2022/lib/modules/shared/sky-modals-resources.module.mjs +61 -0
- package/esm2022/testing/confirm/confirm-button-harness.mjs +46 -0
- package/esm2022/testing/confirm/confirm-harness.mjs +90 -0
- package/esm2022/testing/modal/modal-harness.mjs +75 -0
- package/esm2022/testing/modal-fixture.mjs +143 -0
- package/{fesm2020 → fesm2022}/skyux-modals-testing.mjs +60 -73
- package/{fesm2020 → fesm2022}/skyux-modals-testing.mjs.map +1 -1
- package/fesm2022/skyux-modals.mjs +1502 -0
- package/{fesm2015 → fesm2022}/skyux-modals.mjs.map +1 -1
- package/lib/modules/modal/modal-is-dirty.directive.d.ts +1 -1
- package/lib/modules/modal/modal.component.d.ts +1 -1
- package/package.json +16 -24
- package/esm2020/lib/modules/confirm/confirm.component.mjs +0 -114
- package/esm2020/lib/modules/confirm/confirm.service.mjs +0 -57
- package/esm2020/lib/modules/modal/modal-adapter.service.mjs +0 -101
- package/esm2020/lib/modules/modal/modal-component-adapter.service.mjs +0 -104
- package/esm2020/lib/modules/modal/modal-errors.service.mjs +0 -24
- package/esm2020/lib/modules/modal/modal-host.component.mjs +0 -164
- package/esm2020/lib/modules/modal/modal-host.service.mjs +0 -65
- package/esm2020/lib/modules/modal/modal-instance.mjs +0 -127
- package/esm2020/lib/modules/modal/modal-is-dirty.directive.mjs +0 -93
- package/esm2020/lib/modules/modal/modal-scroll-shadow.directive.mjs +0 -127
- package/esm2020/lib/modules/modal/modal.component.mjs +0 -228
- package/esm2020/lib/modules/modal/modal.service.mjs +0 -99
- package/esm2020/lib/modules/shared/sky-modals-resources.module.mjs +0 -61
- package/esm2020/testing/confirm/confirm-button-harness.mjs +0 -46
- package/esm2020/testing/confirm/confirm-harness.mjs +0 -97
- package/esm2020/testing/modal/modal-harness.mjs +0 -81
- package/esm2020/testing/modal-fixture.mjs +0 -145
- package/fesm2015/skyux-modals-testing.mjs +0 -409
- package/fesm2015/skyux-modals-testing.mjs.map +0 -1
- package/fesm2015/skyux-modals.mjs +0 -1522
- package/fesm2020/skyux-modals.mjs +0 -1510
- package/fesm2020/skyux-modals.mjs.map +0 -1
- /package/{esm2020 → esm2022}/index.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-button-action.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-button-config.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-button-style-type.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-button.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-closed-event-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-config-token.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-config.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-instance.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/confirm/confirm-type.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/modal/modal-before-close-handler.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/modal/modal-close-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/modal/modal-error.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/modal/modal-host-context-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/modal/modal-scroll-shadow-event-args.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/modules/modal/modal.interface.mjs +0 -0
- /package/{esm2020 → esm2022}/skyux-modals.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/confirm/confirm-button-harness-filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/modal/modal-harness-filters.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/public-api.mjs +0 -0
- /package/{esm2020 → esm2022}/testing/skyux-modals-testing.mjs +0 -0
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { Subject } from 'rxjs';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
const BASE_Z_INDEX = 1040;
|
|
5
|
+
const modalHosts = [];
|
|
6
|
+
/**
|
|
7
|
+
* @internal
|
|
8
|
+
* @dynamic
|
|
9
|
+
*/
|
|
10
|
+
export class SkyModalHostService {
|
|
11
|
+
static get openModalCount() {
|
|
12
|
+
return modalHosts.length;
|
|
13
|
+
}
|
|
14
|
+
static get fullPageModalCount() {
|
|
15
|
+
const fullPageModals = modalHosts.filter((modal) => modal.fullPage);
|
|
16
|
+
return fullPageModals.length;
|
|
17
|
+
}
|
|
18
|
+
static get backdropZIndex() {
|
|
19
|
+
return BASE_Z_INDEX + modalHosts.length * 10;
|
|
20
|
+
}
|
|
21
|
+
static get topModal() {
|
|
22
|
+
return modalHosts[modalHosts.length - 1];
|
|
23
|
+
}
|
|
24
|
+
constructor() {
|
|
25
|
+
this.close = new Subject();
|
|
26
|
+
this.fullPage = false;
|
|
27
|
+
this.openHelp = new Subject();
|
|
28
|
+
this.zIndex = this.#calculateZIndex();
|
|
29
|
+
modalHosts.push(this);
|
|
30
|
+
}
|
|
31
|
+
getModalZIndex() {
|
|
32
|
+
return this.zIndex;
|
|
33
|
+
}
|
|
34
|
+
onClose() {
|
|
35
|
+
this.close.next();
|
|
36
|
+
}
|
|
37
|
+
onOpenHelp(helpKey) {
|
|
38
|
+
this.openHelp.next(helpKey);
|
|
39
|
+
}
|
|
40
|
+
destroy() {
|
|
41
|
+
modalHosts.splice(modalHosts.indexOf(this), 1);
|
|
42
|
+
}
|
|
43
|
+
#calculateZIndex() {
|
|
44
|
+
const zIndexArray = modalHosts.map((hostService) => hostService.zIndex);
|
|
45
|
+
if (zIndexArray.length === 0) {
|
|
46
|
+
return BASE_Z_INDEX + 11;
|
|
47
|
+
}
|
|
48
|
+
else {
|
|
49
|
+
const currentMaxZIndex = Math.max(...zIndexArray);
|
|
50
|
+
return currentMaxZIndex + 10;
|
|
51
|
+
}
|
|
52
|
+
}
|
|
53
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyModalHostService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
54
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyModalHostService, providedIn: 'root' }); }
|
|
55
|
+
}
|
|
56
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyModalHostService, decorators: [{
|
|
57
|
+
type: Injectable,
|
|
58
|
+
args: [{
|
|
59
|
+
providedIn: 'root',
|
|
60
|
+
}]
|
|
61
|
+
}], ctorParameters: function () { return []; } });
|
|
62
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kYWwtaG9zdC5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9jb21wb25lbnRzL21vZGFscy9zcmMvbGliL21vZHVsZXMvbW9kYWwvbW9kYWwtaG9zdC5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFM0MsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLE1BQU0sQ0FBQzs7QUFFL0IsTUFBTSxZQUFZLEdBQUcsSUFBSSxDQUFDO0FBQzFCLE1BQU0sVUFBVSxHQUEwQixFQUFFLENBQUM7QUFFN0M7OztHQUdHO0FBSUgsTUFBTSxPQUFPLG1CQUFtQjtJQUN2QixNQUFNLEtBQUssY0FBYztRQUM5QixPQUFPLFVBQVUsQ0FBQyxNQUFNLENBQUM7SUFDM0IsQ0FBQztJQUVNLE1BQU0sS0FBSyxrQkFBa0I7UUFDbEMsTUFBTSxjQUFjLEdBQUcsVUFBVSxDQUFDLE1BQU0sQ0FBQyxDQUFDLEtBQUssRUFBRSxFQUFFLENBQUMsS0FBSyxDQUFDLFFBQVEsQ0FBQyxDQUFDO1FBQ3BFLE9BQU8sY0FBYyxDQUFDLE1BQU0sQ0FBQztJQUMvQixDQUFDO0lBRU0sTUFBTSxLQUFLLGNBQWM7UUFDOUIsT0FBTyxZQUFZLEdBQUcsVUFBVSxDQUFDLE1BQU0sR0FBRyxFQUFFLENBQUM7SUFDL0MsQ0FBQztJQUVNLE1BQU0sS0FBSyxRQUFRO1FBQ3hCLE9BQU8sVUFBVSxDQUFDLFVBQVUsQ0FBQyxNQUFNLEdBQUcsQ0FBQyxDQUFDLENBQUM7SUFDM0MsQ0FBQztJQU9EO1FBTE8sVUFBSyxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7UUFDNUIsYUFBUSxHQUFHLEtBQUssQ0FBQztRQUNqQixhQUFRLEdBQUcsSUFBSSxPQUFPLEVBQVUsQ0FBQztRQUl0QyxJQUFJLENBQUMsTUFBTSxHQUFHLElBQUksQ0FBQyxnQkFBZ0IsRUFBRSxDQUFDO1FBQ3RDLFVBQVUsQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDeEIsQ0FBQztJQUVNLGNBQWM7UUFDbkIsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDO0lBQ3JCLENBQUM7SUFFTSxPQUFPO1FBQ1osSUFBSSxDQUFDLEtBQUssQ0FBQyxJQUFJLEVBQUUsQ0FBQztJQUNwQixDQUFDO0lBRU0sVUFBVSxDQUFDLE9BQWU7UUFDL0IsSUFBSSxDQUFDLFFBQVEsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7SUFDOUIsQ0FBQztJQUVNLE9BQU87UUFDWixVQUFVLENBQUMsTUFBTSxDQUFDLFVBQVUsQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDLENBQUM7SUFDakQsQ0FBQztJQUVELGdCQUFnQjtRQUNkLE1BQU0sV0FBVyxHQUFHLFVBQVUsQ0FBQyxHQUFHLENBQUMsQ0FBQyxXQUFXLEVBQUUsRUFBRSxDQUFDLFdBQVcsQ0FBQyxNQUFNLENBQUMsQ0FBQztRQUN4RSxJQUFJLFdBQVcsQ0FBQyxNQUFNLEtBQUssQ0FBQyxFQUFFO1lBQzVCLE9BQU8sWUFBWSxHQUFHLEVBQUUsQ0FBQztTQUMxQjthQUFNO1lBQ0wsTUFBTSxnQkFBZ0IsR0FBRyxJQUFJLENBQUMsR0FBRyxDQUFDLEdBQUcsV0FBVyxDQUFDLENBQUM7WUFDbEQsT0FBTyxnQkFBZ0IsR0FBRyxFQUFFLENBQUM7U0FDOUI7SUFDSCxDQUFDOzhHQXBEVSxtQkFBbUI7a0hBQW5CLG1CQUFtQixjQUZsQixNQUFNOzsyRkFFUCxtQkFBbUI7a0JBSC9CLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5pbXBvcnQgeyBTdWJqZWN0IH0gZnJvbSAncnhqcyc7XG5cbmNvbnN0IEJBU0VfWl9JTkRFWCA9IDEwNDA7XG5jb25zdCBtb2RhbEhvc3RzOiBTa3lNb2RhbEhvc3RTZXJ2aWNlW10gPSBbXTtcblxuLyoqXG4gKiBAaW50ZXJuYWxcbiAqIEBkeW5hbWljXG4gKi9cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnLFxufSlcbmV4cG9ydCBjbGFzcyBTa3lNb2RhbEhvc3RTZXJ2aWNlIHtcbiAgcHVibGljIHN0YXRpYyBnZXQgb3Blbk1vZGFsQ291bnQoKTogbnVtYmVyIHtcbiAgICByZXR1cm4gbW9kYWxIb3N0cy5sZW5ndGg7XG4gIH1cblxuICBwdWJsaWMgc3RhdGljIGdldCBmdWxsUGFnZU1vZGFsQ291bnQoKTogbnVtYmVyIHtcbiAgICBjb25zdCBmdWxsUGFnZU1vZGFscyA9IG1vZGFsSG9zdHMuZmlsdGVyKChtb2RhbCkgPT4gbW9kYWwuZnVsbFBhZ2UpO1xuICAgIHJldHVybiBmdWxsUGFnZU1vZGFscy5sZW5ndGg7XG4gIH1cblxuICBwdWJsaWMgc3RhdGljIGdldCBiYWNrZHJvcFpJbmRleCgpOiBudW1iZXIge1xuICAgIHJldHVybiBCQVNFX1pfSU5ERVggKyBtb2RhbEhvc3RzLmxlbmd0aCAqIDEwO1xuICB9XG5cbiAgcHVibGljIHN0YXRpYyBnZXQgdG9wTW9kYWwoKTogU2t5TW9kYWxIb3N0U2VydmljZSB7XG4gICAgcmV0dXJuIG1vZGFsSG9zdHNbbW9kYWxIb3N0cy5sZW5ndGggLSAxXTtcbiAgfVxuXG4gIHB1YmxpYyBjbG9zZSA9IG5ldyBTdWJqZWN0PHZvaWQ+KCk7XG4gIHB1YmxpYyBmdWxsUGFnZSA9IGZhbHNlO1xuICBwdWJsaWMgb3BlbkhlbHAgPSBuZXcgU3ViamVjdDxzdHJpbmc+KCk7XG4gIHB1YmxpYyB6SW5kZXg6IG51bWJlcjtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLnpJbmRleCA9IHRoaXMuI2NhbGN1bGF0ZVpJbmRleCgpO1xuICAgIG1vZGFsSG9zdHMucHVzaCh0aGlzKTtcbiAgfVxuXG4gIHB1YmxpYyBnZXRNb2RhbFpJbmRleCgpOiBudW1iZXIge1xuICAgIHJldHVybiB0aGlzLnpJbmRleDtcbiAgfVxuXG4gIHB1YmxpYyBvbkNsb3NlKCk6IHZvaWQge1xuICAgIHRoaXMuY2xvc2UubmV4dCgpO1xuICB9XG5cbiAgcHVibGljIG9uT3BlbkhlbHAoaGVscEtleTogc3RyaW5nKSB7XG4gICAgdGhpcy5vcGVuSGVscC5uZXh0KGhlbHBLZXkpO1xuICB9XG5cbiAgcHVibGljIGRlc3Ryb3koKTogdm9pZCB7XG4gICAgbW9kYWxIb3N0cy5zcGxpY2UobW9kYWxIb3N0cy5pbmRleE9mKHRoaXMpLCAxKTtcbiAgfVxuXG4gICNjYWxjdWxhdGVaSW5kZXgoKTogbnVtYmVyIHtcbiAgICBjb25zdCB6SW5kZXhBcnJheSA9IG1vZGFsSG9zdHMubWFwKChob3N0U2VydmljZSkgPT4gaG9zdFNlcnZpY2UuekluZGV4KTtcbiAgICBpZiAoekluZGV4QXJyYXkubGVuZ3RoID09PSAwKSB7XG4gICAgICByZXR1cm4gQkFTRV9aX0lOREVYICsgMTE7XG4gICAgfSBlbHNlIHtcbiAgICAgIGNvbnN0IGN1cnJlbnRNYXhaSW5kZXggPSBNYXRoLm1heCguLi56SW5kZXhBcnJheSk7XG4gICAgICByZXR1cm4gY3VycmVudE1heFpJbmRleCArIDEwO1xuICAgIH1cbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,123 @@
|
|
|
1
|
+
import { Subject } from 'rxjs';
|
|
2
|
+
import { SkyModalBeforeCloseHandler } from './modal-before-close-handler';
|
|
3
|
+
import { SkyModalCloseArgs } from './modal-close-args';
|
|
4
|
+
export class SkyModalInstance {
|
|
5
|
+
/**
|
|
6
|
+
* An event that the modal instance emits when it is about to close.
|
|
7
|
+
* It emits a `SkyModalBeforeCloseHandler` object with a `closeModal` method
|
|
8
|
+
* that closes the modal. If a subscription exists for this event,
|
|
9
|
+
* the modal does not close until the subscriber calls the `closeModal` method.
|
|
10
|
+
*/
|
|
11
|
+
get beforeClose() {
|
|
12
|
+
return this.#_beforeClose;
|
|
13
|
+
}
|
|
14
|
+
/**
|
|
15
|
+
* An event that the modal instance emits when it closes.
|
|
16
|
+
* It emits a `SkyModalCloseArgs` object with a `data` property that includes
|
|
17
|
+
* data passed from users on close or save and a `reason` property that indicates
|
|
18
|
+
* whether the modal was saved or closed without saving.
|
|
19
|
+
* The `reason` property accepts any string value.
|
|
20
|
+
* Common examples include `"cancel"`, `"close"`, and `"save"`.
|
|
21
|
+
*/
|
|
22
|
+
get closed() {
|
|
23
|
+
return this.#_closed;
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* An event that the modal instance emits when users click
|
|
27
|
+
* the <i class="fa fa-question-circle" aria-hidden="true"></i> button.
|
|
28
|
+
* If a `helpKey` parameter was specified, the `helpOpened` event broadcasts the `helpKey`.
|
|
29
|
+
*/
|
|
30
|
+
get helpOpened() {
|
|
31
|
+
return this.#_helpOpened;
|
|
32
|
+
}
|
|
33
|
+
/**
|
|
34
|
+
* Sets the component adapter for the instance. This is used internally for actions such as scrolling the content.
|
|
35
|
+
* @internal
|
|
36
|
+
*/
|
|
37
|
+
set adapter(value) {
|
|
38
|
+
this.#adapter = value;
|
|
39
|
+
}
|
|
40
|
+
/**
|
|
41
|
+
* Sets the component ref for the instance. This is used to extract the component instance for the public API and the element ref for internal use.
|
|
42
|
+
* @internal
|
|
43
|
+
*/
|
|
44
|
+
set componentRef(value) {
|
|
45
|
+
this.componentInstance = value.instance;
|
|
46
|
+
this.#elementRef = value.location;
|
|
47
|
+
}
|
|
48
|
+
#_beforeClose = new Subject();
|
|
49
|
+
#_closed = new Subject();
|
|
50
|
+
#_helpOpened = new Subject();
|
|
51
|
+
#adapter;
|
|
52
|
+
#elementRef;
|
|
53
|
+
/**
|
|
54
|
+
* Closes the modal instance.
|
|
55
|
+
* @param result Specifies an object to emit to subscribers of the `closed` event of the
|
|
56
|
+
* modal instance. The `SkyModalInstance` provider can be injected into a component's constructor
|
|
57
|
+
* so that this `close` function can be called from a button in the `sky-modal-footer`.
|
|
58
|
+
* @param reason Specifies the reason for the modal closing, with the default reason of `"close"`.
|
|
59
|
+
* @param ignoreBeforeClose Indicates whether to ignore the modal instance's `beforeClose` event.
|
|
60
|
+
*/
|
|
61
|
+
close(result, reason, ignoreBeforeClose) {
|
|
62
|
+
if (reason === undefined) {
|
|
63
|
+
reason = 'close';
|
|
64
|
+
}
|
|
65
|
+
this.#closeModal(reason, result, ignoreBeforeClose);
|
|
66
|
+
}
|
|
67
|
+
/**
|
|
68
|
+
* Closes the modal instance with `reason="cancel"`.
|
|
69
|
+
* @param result Specifies an object to emit to subscribers of the `closed` event of the modal
|
|
70
|
+
* instance. The `SkyModalInstance` provider can be injected into a component's constructor so
|
|
71
|
+
* that this cancel function can be called from a button in the `sky-modal-footer`.
|
|
72
|
+
*/
|
|
73
|
+
cancel(result) {
|
|
74
|
+
this.#closeModal('cancel', result);
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Closes the modal instance with `reason="save"`.
|
|
78
|
+
* @param result Specifies an object to emit to subscribers of the `closed` event of the modal
|
|
79
|
+
* instance. The `SkyModalInstance` provider can be injected into a component's constructor so
|
|
80
|
+
* that this `save` function can be called from a button in `the sky-modal-footer`.
|
|
81
|
+
*/
|
|
82
|
+
save(result) {
|
|
83
|
+
this.#closeModal('save', result);
|
|
84
|
+
}
|
|
85
|
+
/**
|
|
86
|
+
* Scrolls the modal content area to the top of its scrollable area.
|
|
87
|
+
*/
|
|
88
|
+
scrollContentToTop() {
|
|
89
|
+
if (this.#adapter && this.#elementRef) {
|
|
90
|
+
this.#adapter.scrollContentToTop(this.#elementRef);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
/**
|
|
94
|
+
* Triggers the `helpOpened` event that broadcasts a `helpKey` parameter to open
|
|
95
|
+
* when users click the <i class="fa fa-question-circle" aria-hidden="true"></i> button.
|
|
96
|
+
* @param helpKey Specifies a string to emit to subscribers of
|
|
97
|
+
* the modal instance's `helpOpened` event. Consumers can inject the `SkyModalInstance` provider
|
|
98
|
+
* into a component's constructor to call the `openHelp` function in the modal template.
|
|
99
|
+
*/
|
|
100
|
+
openHelp(helpKey) {
|
|
101
|
+
this.#_helpOpened.next(helpKey);
|
|
102
|
+
}
|
|
103
|
+
#closeModal(type, result, ignoreBeforeClose = false) {
|
|
104
|
+
const args = new SkyModalCloseArgs();
|
|
105
|
+
args.reason = type;
|
|
106
|
+
args.data = result;
|
|
107
|
+
if (this.#_beforeClose.observers.length === 0 || ignoreBeforeClose) {
|
|
108
|
+
this.#notifyClosed(args);
|
|
109
|
+
}
|
|
110
|
+
else {
|
|
111
|
+
this.#_beforeClose.next(new SkyModalBeforeCloseHandler(() => {
|
|
112
|
+
this.#notifyClosed(args);
|
|
113
|
+
}, args));
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
#notifyClosed(args) {
|
|
117
|
+
this.#_closed.next(args);
|
|
118
|
+
this.#_closed.complete();
|
|
119
|
+
this.#_beforeClose.complete();
|
|
120
|
+
this.#_helpOpened.complete();
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-instance.js","sourceRoot":"","sources":["../../../../../../../../libs/components/modals/src/lib/modules/modal/modal-instance.ts"],"names":[],"mappings":"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","sourcesContent":["import { ComponentRef, ElementRef } from '@angular/core';\n\nimport { Observable, Subject } from 'rxjs';\n\nimport { SkyModalAdapterService } from './modal-adapter.service';\nimport { SkyModalBeforeCloseHandler } from './modal-before-close-handler';\nimport { SkyModalCloseArgs } from './modal-close-args';\n\nexport class SkyModalInstance {\n  /**\n   * An event that the modal instance emits when it is about to close.\n   * It emits a `SkyModalBeforeCloseHandler` object with a `closeModal` method\n   * that closes the modal. If a subscription exists for this event,\n   * the modal does not close until the subscriber calls the `closeModal` method.\n   */\n  public get beforeClose(): Observable<SkyModalBeforeCloseHandler> {\n    return this.#_beforeClose;\n  }\n\n  /**\n   * An event that the modal instance emits when it closes.\n   * It emits a `SkyModalCloseArgs` object with a `data` property that includes\n   * data passed from users on close or save and a `reason` property that indicates\n   * whether the modal was saved or closed without saving.\n   * The `reason` property accepts any string value.\n   * Common examples include `\"cancel\"`, `\"close\"`, and `\"save\"`.\n   */\n  public get closed(): Observable<SkyModalCloseArgs> {\n    return this.#_closed;\n  }\n\n  /**\n   * An event that the modal instance emits when users click\n   * the <i class=\"fa fa-question-circle\" aria-hidden=\"true\"></i> button.\n   * If a `helpKey` parameter was specified, the `helpOpened` event broadcasts the `helpKey`.\n   */\n  public get helpOpened(): Observable<string> {\n    return this.#_helpOpened;\n  }\n\n  /**\n   * A direct reference to the provided component's class.\n   */\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  public componentInstance: any;\n\n  /**\n   * Sets the component adapter for the instance. This is used internally for actions such as scrolling the content.\n   * @internal\n   */\n  public set adapter(value: SkyModalAdapterService) {\n    this.#adapter = value;\n  }\n\n  /**\n   * Sets the component ref for the instance. This is used to extract the component instance for the public API and the element ref for internal use.\n   * @internal\n   */\n  public set componentRef(value: ComponentRef<any>) {\n    this.componentInstance = value.instance;\n    this.#elementRef = value.location;\n  }\n\n  #_beforeClose = new Subject<SkyModalBeforeCloseHandler>();\n\n  #_closed = new Subject<SkyModalCloseArgs>();\n\n  #_helpOpened = new Subject<string>();\n\n  #adapter: SkyModalAdapterService | undefined;\n  #elementRef: ElementRef | undefined;\n\n  /**\n   * Closes the modal instance.\n   * @param result Specifies an object to emit to subscribers of the `closed` event of the\n   * modal instance. The `SkyModalInstance` provider can be injected into a component's constructor\n   * so that this `close` function can be called from a button in the `sky-modal-footer`.\n   * @param reason Specifies the reason for the modal closing, with the default reason of `\"close\"`.\n   * @param ignoreBeforeClose Indicates whether to ignore the modal instance's `beforeClose` event.\n   */\n  public close(\n    result?: any,\n    reason?: string,\n    ignoreBeforeClose?: boolean\n  ): void {\n    if (reason === undefined) {\n      reason = 'close';\n    }\n\n    this.#closeModal(reason, result, ignoreBeforeClose);\n  }\n\n  /**\n   * Closes the modal instance with `reason=\"cancel\"`.\n   * @param result Specifies an object to emit to subscribers of the `closed` event of the modal\n   * instance. The `SkyModalInstance` provider can be injected into a component's constructor so\n   * that this cancel function can be called from a button in the `sky-modal-footer`.\n   */\n  public cancel(result?: any): void {\n    this.#closeModal('cancel', result);\n  }\n\n  /**\n   * Closes the modal instance with `reason=\"save\"`.\n   * @param result Specifies an object to emit to subscribers of the `closed` event of the modal\n   * instance. The `SkyModalInstance` provider can be injected into a component's constructor so\n   * that this `save` function can be called from a button in `the sky-modal-footer`.\n   */\n  public save(result?: any): void {\n    this.#closeModal('save', result);\n  }\n\n  /**\n   * Scrolls the modal content area to the top of its scrollable area.\n   */\n  public scrollContentToTop(): void {\n    if (this.#adapter && this.#elementRef) {\n      this.#adapter.scrollContentToTop(this.#elementRef);\n    }\n  }\n\n  /**\n   * Triggers the `helpOpened` event that broadcasts a `helpKey` parameter to open\n   * when users click the <i class=\"fa fa-question-circle\" aria-hidden=\"true\"></i> button.\n   * @param helpKey Specifies a string to emit to subscribers of\n   * the modal instance's `helpOpened` event. Consumers can inject the `SkyModalInstance` provider\n   * into a component's constructor to call the `openHelp` function in the modal template.\n   */\n  public openHelp(helpKey: string): void {\n    this.#_helpOpened.next(helpKey);\n  }\n\n  #closeModal(type: string, result?: any, ignoreBeforeClose = false): void {\n    const args = new SkyModalCloseArgs();\n\n    args.reason = type;\n    args.data = result;\n\n    if (this.#_beforeClose.observers.length === 0 || ignoreBeforeClose) {\n      this.#notifyClosed(args);\n    } else {\n      this.#_beforeClose.next(\n        new SkyModalBeforeCloseHandler(() => {\n          this.#notifyClosed(args);\n        }, args)\n      );\n    }\n  }\n\n  #notifyClosed(args: SkyModalCloseArgs): void {\n    this.#_closed.next(args);\n    this.#_closed.complete();\n    this.#_beforeClose.complete();\n    this.#_helpOpened.complete();\n  }\n}\n"]}
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
import { Directive, HostBinding, Input, inject, } from '@angular/core';
|
|
2
|
+
import { SkyLibResourcesService } from '@skyux/i18n';
|
|
3
|
+
import { Subject } from 'rxjs';
|
|
4
|
+
import { takeUntil } from 'rxjs/operators';
|
|
5
|
+
import { SkyConfirmType } from '../confirm/confirm-type';
|
|
6
|
+
import { SkyConfirmService } from '../confirm/confirm.service';
|
|
7
|
+
import { SkyModalInstance } from './modal-instance';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
/**
|
|
10
|
+
* Provides a way to mark a modal as "dirty" and displays a confirmation
|
|
11
|
+
* message when a user closes the modal without saving.
|
|
12
|
+
*/
|
|
13
|
+
export class SkyModalIsDirtyDirective {
|
|
14
|
+
constructor() {
|
|
15
|
+
/**
|
|
16
|
+
* Whether the user edited an input on the modal.
|
|
17
|
+
* @required
|
|
18
|
+
*/
|
|
19
|
+
this.isDirty = false;
|
|
20
|
+
this.#ngUnsubscribe = new Subject();
|
|
21
|
+
this.#modalInstance = inject(SkyModalInstance);
|
|
22
|
+
this.#confirmSvc = inject(SkyConfirmService);
|
|
23
|
+
this.#resourcesSvc = inject(SkyLibResourcesService);
|
|
24
|
+
}
|
|
25
|
+
#ngUnsubscribe;
|
|
26
|
+
#modalInstance;
|
|
27
|
+
#confirmSvc;
|
|
28
|
+
#resourcesSvc;
|
|
29
|
+
ngOnInit() {
|
|
30
|
+
this.#modalInstance.beforeClose
|
|
31
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
32
|
+
.subscribe((handler) => this.#promptIfDirty(handler));
|
|
33
|
+
}
|
|
34
|
+
ngOnDestroy() {
|
|
35
|
+
this.#ngUnsubscribe.next();
|
|
36
|
+
this.#ngUnsubscribe.complete();
|
|
37
|
+
}
|
|
38
|
+
#promptIfDirty(handler) {
|
|
39
|
+
if (this.isDirty && handler.closeArgs.reason === 'close') {
|
|
40
|
+
this.#resourcesSvc
|
|
41
|
+
.getStrings({
|
|
42
|
+
message: 'skyux_modal_dirty_default_message',
|
|
43
|
+
discardActionText: 'skyux_modal_dirty_default_discard_changes_text',
|
|
44
|
+
keepActionText: 'skyux_modal_dirty_default_keep_working_text',
|
|
45
|
+
})
|
|
46
|
+
.subscribe((textValues) => {
|
|
47
|
+
const discardAction = 'discard';
|
|
48
|
+
const keepAction = 'keep';
|
|
49
|
+
this.#confirmSvc
|
|
50
|
+
.open({
|
|
51
|
+
message: textValues.message,
|
|
52
|
+
buttons: [
|
|
53
|
+
{
|
|
54
|
+
action: discardAction,
|
|
55
|
+
text: textValues.discardActionText,
|
|
56
|
+
styleType: 'primary',
|
|
57
|
+
},
|
|
58
|
+
{
|
|
59
|
+
action: keepAction,
|
|
60
|
+
text: textValues.keepActionText,
|
|
61
|
+
styleType: 'link',
|
|
62
|
+
},
|
|
63
|
+
],
|
|
64
|
+
type: SkyConfirmType.Custom,
|
|
65
|
+
})
|
|
66
|
+
.closed.subscribe((args) => {
|
|
67
|
+
if (args.action === discardAction) {
|
|
68
|
+
handler.closeModal();
|
|
69
|
+
}
|
|
70
|
+
});
|
|
71
|
+
});
|
|
72
|
+
}
|
|
73
|
+
else {
|
|
74
|
+
handler.closeModal();
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyModalIsDirtyDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
78
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.1.7", type: SkyModalIsDirtyDirective, selector: "sky-modal[isDirty]", inputs: { isDirty: "isDirty" }, host: { properties: { "attr.data-sky-modal-is-dirty": "this.isDirty" } }, ngImport: i0 }); }
|
|
79
|
+
}
|
|
80
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyModalIsDirtyDirective, decorators: [{
|
|
81
|
+
type: Directive,
|
|
82
|
+
args: [{
|
|
83
|
+
// Since this is limited to sky-modal, it should be safe to
|
|
84
|
+
// leave off the sky prefix.
|
|
85
|
+
// eslint-disable-next-line @angular-eslint/directive-selector
|
|
86
|
+
selector: 'sky-modal[isDirty]',
|
|
87
|
+
}]
|
|
88
|
+
}], propDecorators: { isDirty: [{
|
|
89
|
+
type: Input
|
|
90
|
+
}, {
|
|
91
|
+
type: HostBinding,
|
|
92
|
+
args: ['attr.data-sky-modal-is-dirty']
|
|
93
|
+
}] } });
|
|
94
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kYWwtaXMtZGlydHkuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9jb21wb25lbnRzL21vZGFscy9zcmMvbGliL21vZHVsZXMvbW9kYWwvbW9kYWwtaXMtZGlydHkuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFDTCxTQUFTLEVBQ1QsV0FBVyxFQUNYLEtBQUssRUFHTCxNQUFNLEdBQ1AsTUFBTSxlQUFlLENBQUM7QUFDdkIsT0FBTyxFQUFFLHNCQUFzQixFQUFFLE1BQU0sYUFBYSxDQUFDO0FBRXJELE9BQU8sRUFBRSxPQUFPLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDL0IsT0FBTyxFQUFFLFNBQVMsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBRTNDLE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUN6RCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUcvRCxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQzs7QUFFcEQ7OztHQUdHO0FBT0gsTUFBTSxPQUFPLHdCQUF3QjtJQU5yQztRQU9FOzs7V0FHRztRQUtJLFlBQU8sR0FBRyxLQUFLLENBQUM7UUFFZCxtQkFBYyxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7UUFFckMsbUJBQWMsR0FBRyxNQUFNLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztRQUMxQyxnQkFBVyxHQUFHLE1BQU0sQ0FBQyxpQkFBaUIsQ0FBQyxDQUFDO1FBQ3hDLGtCQUFhLEdBQUcsTUFBTSxDQUFDLHNCQUFzQixDQUFDLENBQUM7S0FvRHpEO0lBeERVLGNBQWMsQ0FBdUI7SUFFckMsY0FBYyxDQUE0QjtJQUMxQyxXQUFXLENBQTZCO0lBQ3hDLGFBQWEsQ0FBa0M7SUFFakQsUUFBUTtRQUNiLElBQUksQ0FBQyxjQUFjLENBQUMsV0FBVzthQUM1QixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksQ0FBQyxjQUFjLENBQUMsQ0FBQzthQUNwQyxTQUFTLENBQUMsQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDLElBQUksQ0FBQyxjQUFjLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQztJQUMxRCxDQUFDO0lBRU0sV0FBVztRQUNoQixJQUFJLENBQUMsY0FBYyxDQUFDLElBQUksRUFBRSxDQUFDO1FBQzNCLElBQUksQ0FBQyxjQUFjLENBQUMsUUFBUSxFQUFFLENBQUM7SUFDakMsQ0FBQztJQUVELGNBQWMsQ0FBQyxPQUFtQztRQUNoRCxJQUFJLElBQUksQ0FBQyxPQUFPLElBQUksT0FBTyxDQUFDLFNBQVMsQ0FBQyxNQUFNLEtBQUssT0FBTyxFQUFFO1lBQ3hELElBQUksQ0FBQyxhQUFhO2lCQUNmLFVBQVUsQ0FBQztnQkFDVixPQUFPLEVBQUUsbUNBQW1DO2dCQUM1QyxpQkFBaUIsRUFBRSxnREFBZ0Q7Z0JBQ25FLGNBQWMsRUFBRSw2Q0FBNkM7YUFDOUQsQ0FBQztpQkFDRCxTQUFTLENBQUMsQ0FBQyxVQUFVLEVBQUUsRUFBRTtnQkFDeEIsTUFBTSxhQUFhLEdBQUcsU0FBUyxDQUFDO2dCQUNoQyxNQUFNLFVBQVUsR0FBRyxNQUFNLENBQUM7Z0JBRTFCLElBQUksQ0FBQyxXQUFXO3FCQUNiLElBQUksQ0FBQztvQkFDSixPQUFPLEVBQUUsVUFBVSxDQUFDLE9BQU87b0JBQzNCLE9BQU8sRUFBRTt3QkFDUDs0QkFDRSxNQUFNLEVBQUUsYUFBYTs0QkFDckIsSUFBSSxFQUFFLFVBQVUsQ0FBQyxpQkFBaUI7NEJBQ2xDLFNBQVMsRUFBRSxTQUFTO3lCQUNyQjt3QkFDRDs0QkFDRSxNQUFNLEVBQUUsVUFBVTs0QkFDbEIsSUFBSSxFQUFFLFVBQVUsQ0FBQyxjQUFjOzRCQUMvQixTQUFTLEVBQUUsTUFBTTt5QkFDbEI7cUJBQ0Y7b0JBQ0QsSUFBSSxFQUFFLGNBQWMsQ0FBQyxNQUFNO2lCQUM1QixDQUFDO3FCQUNELE1BQU0sQ0FBQyxTQUFTLENBQUMsQ0FBQyxJQUFJLEVBQUUsRUFBRTtvQkFDekIsSUFBSSxJQUFJLENBQUMsTUFBTSxLQUFLLGFBQWEsRUFBRTt3QkFDakMsT0FBTyxDQUFDLFVBQVUsRUFBRSxDQUFDO3FCQUN0QjtnQkFDSCxDQUFDLENBQUMsQ0FBQztZQUNQLENBQUMsQ0FBQyxDQUFDO1NBQ047YUFBTTtZQUNMLE9BQU8sQ0FBQyxVQUFVLEVBQUUsQ0FBQztTQUN0QjtJQUNILENBQUM7OEdBbEVVLHdCQUF3QjtrR0FBeEIsd0JBQXdCOzsyRkFBeEIsd0JBQXdCO2tCQU5wQyxTQUFTO21CQUFDO29CQUNULDJEQUEyRDtvQkFDM0QsNEJBQTRCO29CQUM1Qiw4REFBOEQ7b0JBQzlELFFBQVEsRUFBRSxvQkFBb0I7aUJBQy9COzhCQVVRLE9BQU87c0JBSmIsS0FBSzs7c0JBR0wsV0FBVzt1QkFBQyw4QkFBOEIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBEaXJlY3RpdmUsXG4gIEhvc3RCaW5kaW5nLFxuICBJbnB1dCxcbiAgT25EZXN0cm95LFxuICBPbkluaXQsXG4gIGluamVjdCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBTa3lMaWJSZXNvdXJjZXNTZXJ2aWNlIH0gZnJvbSAnQHNreXV4L2kxOG4nO1xuXG5pbXBvcnQgeyBTdWJqZWN0IH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyB0YWtlVW50aWwgfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5cbmltcG9ydCB7IFNreUNvbmZpcm1UeXBlIH0gZnJvbSAnLi4vY29uZmlybS9jb25maXJtLXR5cGUnO1xuaW1wb3J0IHsgU2t5Q29uZmlybVNlcnZpY2UgfSBmcm9tICcuLi9jb25maXJtL2NvbmZpcm0uc2VydmljZSc7XG5cbmltcG9ydCB7IFNreU1vZGFsQmVmb3JlQ2xvc2VIYW5kbGVyIH0gZnJvbSAnLi9tb2RhbC1iZWZvcmUtY2xvc2UtaGFuZGxlcic7XG5pbXBvcnQgeyBTa3lNb2RhbEluc3RhbmNlIH0gZnJvbSAnLi9tb2RhbC1pbnN0YW5jZSc7XG5cbi8qKlxuICogUHJvdmlkZXMgYSB3YXkgdG8gbWFyayBhIG1vZGFsIGFzIFwiZGlydHlcIiBhbmQgZGlzcGxheXMgYSBjb25maXJtYXRpb25cbiAqIG1lc3NhZ2Ugd2hlbiBhIHVzZXIgY2xvc2VzIHRoZSBtb2RhbCB3aXRob3V0IHNhdmluZy5cbiAqL1xuQERpcmVjdGl2ZSh7XG4gIC8vIFNpbmNlIHRoaXMgaXMgbGltaXRlZCB0byBza3ktbW9kYWwsIGl0IHNob3VsZCBiZSBzYWZlIHRvXG4gIC8vIGxlYXZlIG9mZiB0aGUgc2t5IHByZWZpeC5cbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEBhbmd1bGFyLWVzbGludC9kaXJlY3RpdmUtc2VsZWN0b3JcbiAgc2VsZWN0b3I6ICdza3ktbW9kYWxbaXNEaXJ0eV0nLFxufSlcbmV4cG9ydCBjbGFzcyBTa3lNb2RhbElzRGlydHlEaXJlY3RpdmUgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSB7XG4gIC8qKlxuICAgKiBXaGV0aGVyIHRoZSB1c2VyIGVkaXRlZCBhbiBpbnB1dCBvbiB0aGUgbW9kYWwuXG4gICAqIEByZXF1aXJlZFxuICAgKi9cbiAgQElucHV0KClcbiAgLy8gVGhpcyBhdHRyaWJ1dGUgaXMgYmVpbmcgYXBwbGllZCB0byB0aGUgaG9zdCB0byBzdXBwb3J0XG4gIC8vIHVuaXQgdGVzdGluZyB0aGlzIGZlYXR1cmUuXG4gIEBIb3N0QmluZGluZygnYXR0ci5kYXRhLXNreS1tb2RhbC1pcy1kaXJ0eScpXG4gIHB1YmxpYyBpc0RpcnR5ID0gZmFsc2U7XG5cbiAgcmVhZG9ubHkgI25nVW5zdWJzY3JpYmUgPSBuZXcgU3ViamVjdDx2b2lkPigpO1xuXG4gIHJlYWRvbmx5ICNtb2RhbEluc3RhbmNlID0gaW5qZWN0KFNreU1vZGFsSW5zdGFuY2UpO1xuICByZWFkb25seSAjY29uZmlybVN2YyA9IGluamVjdChTa3lDb25maXJtU2VydmljZSk7XG4gIHJlYWRvbmx5ICNyZXNvdXJjZXNTdmMgPSBpbmplY3QoU2t5TGliUmVzb3VyY2VzU2VydmljZSk7XG5cbiAgcHVibGljIG5nT25Jbml0KCk6IHZvaWQge1xuICAgIHRoaXMuI21vZGFsSW5zdGFuY2UuYmVmb3JlQ2xvc2VcbiAgICAgIC5waXBlKHRha2VVbnRpbCh0aGlzLiNuZ1Vuc3Vic2NyaWJlKSlcbiAgICAgIC5zdWJzY3JpYmUoKGhhbmRsZXIpID0+IHRoaXMuI3Byb21wdElmRGlydHkoaGFuZGxlcikpO1xuICB9XG5cbiAgcHVibGljIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMuI25nVW5zdWJzY3JpYmUubmV4dCgpO1xuICAgIHRoaXMuI25nVW5zdWJzY3JpYmUuY29tcGxldGUoKTtcbiAgfVxuXG4gICNwcm9tcHRJZkRpcnR5KGhhbmRsZXI6IFNreU1vZGFsQmVmb3JlQ2xvc2VIYW5kbGVyKTogdm9pZCB7XG4gICAgaWYgKHRoaXMuaXNEaXJ0eSAmJiBoYW5kbGVyLmNsb3NlQXJncy5yZWFzb24gPT09ICdjbG9zZScpIHtcbiAgICAgIHRoaXMuI3Jlc291cmNlc1N2Y1xuICAgICAgICAuZ2V0U3RyaW5ncyh7XG4gICAgICAgICAgbWVzc2FnZTogJ3NreXV4X21vZGFsX2RpcnR5X2RlZmF1bHRfbWVzc2FnZScsXG4gICAgICAgICAgZGlzY2FyZEFjdGlvblRleHQ6ICdza3l1eF9tb2RhbF9kaXJ0eV9kZWZhdWx0X2Rpc2NhcmRfY2hhbmdlc190ZXh0JyxcbiAgICAgICAgICBrZWVwQWN0aW9uVGV4dDogJ3NreXV4X21vZGFsX2RpcnR5X2RlZmF1bHRfa2VlcF93b3JraW5nX3RleHQnLFxuICAgICAgICB9KVxuICAgICAgICAuc3Vic2NyaWJlKCh0ZXh0VmFsdWVzKSA9PiB7XG4gICAgICAgICAgY29uc3QgZGlzY2FyZEFjdGlvbiA9ICdkaXNjYXJkJztcbiAgICAgICAgICBjb25zdCBrZWVwQWN0aW9uID0gJ2tlZXAnO1xuXG4gICAgICAgICAgdGhpcy4jY29uZmlybVN2Y1xuICAgICAgICAgICAgLm9wZW4oe1xuICAgICAgICAgICAgICBtZXNzYWdlOiB0ZXh0VmFsdWVzLm1lc3NhZ2UsXG4gICAgICAgICAgICAgIGJ1dHRvbnM6IFtcbiAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICBhY3Rpb246IGRpc2NhcmRBY3Rpb24sXG4gICAgICAgICAgICAgICAgICB0ZXh0OiB0ZXh0VmFsdWVzLmRpc2NhcmRBY3Rpb25UZXh0LFxuICAgICAgICAgICAgICAgICAgc3R5bGVUeXBlOiAncHJpbWFyeScsXG4gICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgICB7XG4gICAgICAgICAgICAgICAgICBhY3Rpb246IGtlZXBBY3Rpb24sXG4gICAgICAgICAgICAgICAgICB0ZXh0OiB0ZXh0VmFsdWVzLmtlZXBBY3Rpb25UZXh0LFxuICAgICAgICAgICAgICAgICAgc3R5bGVUeXBlOiAnbGluaycsXG4gICAgICAgICAgICAgICAgfSxcbiAgICAgICAgICAgICAgXSxcbiAgICAgICAgICAgICAgdHlwZTogU2t5Q29uZmlybVR5cGUuQ3VzdG9tLFxuICAgICAgICAgICAgfSlcbiAgICAgICAgICAgIC5jbG9zZWQuc3Vic2NyaWJlKChhcmdzKSA9PiB7XG4gICAgICAgICAgICAgIGlmIChhcmdzLmFjdGlvbiA9PT0gZGlzY2FyZEFjdGlvbikge1xuICAgICAgICAgICAgICAgIGhhbmRsZXIuY2xvc2VNb2RhbCgpO1xuICAgICAgICAgICAgICB9XG4gICAgICAgICAgICB9KTtcbiAgICAgICAgfSk7XG4gICAgfSBlbHNlIHtcbiAgICAgIGhhbmRsZXIuY2xvc2VNb2RhbCgpO1xuICAgIH1cbiAgfVxufVxuIl19
|
|
@@ -0,0 +1,129 @@
|
|
|
1
|
+
import { Directive, EventEmitter, HostListener, Optional, Output, } from '@angular/core';
|
|
2
|
+
import { SkyTheme } from '@skyux/theme';
|
|
3
|
+
import { Subject } from 'rxjs';
|
|
4
|
+
import { takeUntil } from 'rxjs/operators';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "@skyux/core";
|
|
7
|
+
import * as i2 from "@skyux/theme";
|
|
8
|
+
/**
|
|
9
|
+
* Raises an event when the box shadow for the modal header or footer should be adjusted
|
|
10
|
+
* based on the scroll position of the host element.
|
|
11
|
+
* @internal
|
|
12
|
+
*/
|
|
13
|
+
export class SkyModalScrollShadowDirective {
|
|
14
|
+
#currentShadow;
|
|
15
|
+
#currentTheme;
|
|
16
|
+
#mutationObserver;
|
|
17
|
+
#ngUnsubscribe;
|
|
18
|
+
#elRef;
|
|
19
|
+
#mutationObserverSvc;
|
|
20
|
+
#ngZone;
|
|
21
|
+
#themeSvc;
|
|
22
|
+
constructor(elRef, mutationObserverSvc, ngZone, themeSvc) {
|
|
23
|
+
this.skyModalScrollShadow = new EventEmitter();
|
|
24
|
+
this.#ngUnsubscribe = new Subject();
|
|
25
|
+
this.#elRef = elRef;
|
|
26
|
+
this.#mutationObserverSvc = mutationObserverSvc;
|
|
27
|
+
this.#ngZone = ngZone;
|
|
28
|
+
this.#themeSvc = themeSvc;
|
|
29
|
+
}
|
|
30
|
+
windowResize() {
|
|
31
|
+
this.#checkForShadow();
|
|
32
|
+
}
|
|
33
|
+
scroll() {
|
|
34
|
+
this.#checkForShadow();
|
|
35
|
+
}
|
|
36
|
+
ngOnInit() {
|
|
37
|
+
if (this.#themeSvc) {
|
|
38
|
+
this.#themeSvc.settingsChange
|
|
39
|
+
.pipe(takeUntil(this.#ngUnsubscribe))
|
|
40
|
+
.subscribe((themeSettings) => {
|
|
41
|
+
this.#currentTheme = themeSettings.currentSettings.theme;
|
|
42
|
+
if (this.#currentTheme === SkyTheme.presets.modern) {
|
|
43
|
+
this.#initMutationObserver();
|
|
44
|
+
}
|
|
45
|
+
else {
|
|
46
|
+
this.#emitShadow({
|
|
47
|
+
bottomShadow: 'none',
|
|
48
|
+
topShadow: 'none',
|
|
49
|
+
});
|
|
50
|
+
this.#destroyMutationObserver();
|
|
51
|
+
}
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
ngOnDestroy() {
|
|
56
|
+
this.#ngUnsubscribe.next();
|
|
57
|
+
this.#ngUnsubscribe.complete();
|
|
58
|
+
this.#destroyMutationObserver();
|
|
59
|
+
}
|
|
60
|
+
#initMutationObserver() {
|
|
61
|
+
if (!this.#mutationObserver) {
|
|
62
|
+
const el = this.#elRef.nativeElement;
|
|
63
|
+
// MutationObserver is patched by Zone.js and therefore becomes part of the
|
|
64
|
+
// Angular change detection cycle, but this can lead to infinite loops in some
|
|
65
|
+
// scenarios. This will keep MutationObserver from triggering change detection.
|
|
66
|
+
this.#ngZone.runOutsideAngular(() => {
|
|
67
|
+
this.#mutationObserver = this.#mutationObserverSvc.create(() => {
|
|
68
|
+
this.#checkForShadow();
|
|
69
|
+
});
|
|
70
|
+
this.#mutationObserver.observe(el, {
|
|
71
|
+
attributes: true,
|
|
72
|
+
characterData: true,
|
|
73
|
+
childList: true,
|
|
74
|
+
subtree: true,
|
|
75
|
+
});
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
}
|
|
79
|
+
#destroyMutationObserver() {
|
|
80
|
+
if (this.#mutationObserver) {
|
|
81
|
+
this.#mutationObserver.disconnect();
|
|
82
|
+
this.#mutationObserver = undefined;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
#checkForShadow() {
|
|
86
|
+
if (this.#currentTheme === SkyTheme.presets.modern) {
|
|
87
|
+
const el = this.#elRef.nativeElement;
|
|
88
|
+
const topShadow = this.#buildShadowStyle(el.scrollTop);
|
|
89
|
+
const bottomShadow = this.#buildShadowStyle(el.scrollHeight - el.scrollTop - el.clientHeight);
|
|
90
|
+
this.#emitShadow({
|
|
91
|
+
bottomShadow,
|
|
92
|
+
topShadow,
|
|
93
|
+
});
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
#buildShadowStyle(pixelsFromEnd) {
|
|
97
|
+
// Progressively darken the shadow until the user scrolls 30 pixels from the top or bottom
|
|
98
|
+
// of the scrollable element, with a max opacity of 0.3.
|
|
99
|
+
const opacity = Math.min(pixelsFromEnd / 30, 1) * 0.3;
|
|
100
|
+
return opacity > 0 ? `0px 1px 8px 0px rgba(0, 0, 0, ${opacity})` : 'none';
|
|
101
|
+
}
|
|
102
|
+
#emitShadow(shadow) {
|
|
103
|
+
if (!this.#currentShadow ||
|
|
104
|
+
this.#currentShadow.bottomShadow !== shadow.bottomShadow ||
|
|
105
|
+
this.#currentShadow.topShadow !== shadow.topShadow) {
|
|
106
|
+
this.skyModalScrollShadow.emit(shadow);
|
|
107
|
+
this.#currentShadow = shadow;
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyModalScrollShadowDirective, deps: [{ token: i0.ElementRef }, { token: i1.SkyMutationObserverService }, { token: i0.NgZone }, { token: i2.SkyThemeService, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
111
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "16.1.7", type: SkyModalScrollShadowDirective, selector: "[skyModalScrollShadow]", outputs: { skyModalScrollShadow: "skyModalScrollShadow" }, host: { listeners: { "window:resize": "windowResize()", "scroll": "scroll()" } }, ngImport: i0 }); }
|
|
112
|
+
}
|
|
113
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.1.7", ngImport: i0, type: SkyModalScrollShadowDirective, decorators: [{
|
|
114
|
+
type: Directive,
|
|
115
|
+
args: [{
|
|
116
|
+
selector: '[skyModalScrollShadow]',
|
|
117
|
+
}]
|
|
118
|
+
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i1.SkyMutationObserverService }, { type: i0.NgZone }, { type: i2.SkyThemeService, decorators: [{
|
|
119
|
+
type: Optional
|
|
120
|
+
}] }]; }, propDecorators: { skyModalScrollShadow: [{
|
|
121
|
+
type: Output
|
|
122
|
+
}], windowResize: [{
|
|
123
|
+
type: HostListener,
|
|
124
|
+
args: ['window:resize']
|
|
125
|
+
}], scroll: [{
|
|
126
|
+
type: HostListener,
|
|
127
|
+
args: ['scroll']
|
|
128
|
+
}] } });
|
|
129
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"modal-scroll-shadow.directive.js","sourceRoot":"","sources":["../../../../../../../../libs/components/modals/src/lib/modules/modal/modal-scroll-shadow.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  ElementRef,\n  EventEmitter,\n  HostListener,\n  NgZone,\n  OnDestroy,\n  OnInit,\n  Optional,\n  Output,\n} from '@angular/core';\nimport { SkyMutationObserverService } from '@skyux/core';\nimport { SkyTheme, SkyThemeService } from '@skyux/theme';\n\nimport { Subject } from 'rxjs';\nimport { takeUntil } from 'rxjs/operators';\n\nimport { SkyModalScrollShadowEventArgs } from './modal-scroll-shadow-event-args';\n\n/**\n * Raises an event when the box shadow for the modal header or footer should be adjusted\n * based on the scroll position of the host element.\n * @internal\n */\n@Directive({\n  selector: '[skyModalScrollShadow]',\n})\nexport class SkyModalScrollShadowDirective implements OnInit, OnDestroy {\n  @Output()\n  public skyModalScrollShadow =\n    new EventEmitter<SkyModalScrollShadowEventArgs>();\n\n  #currentShadow: SkyModalScrollShadowEventArgs | undefined;\n\n  #currentTheme: SkyTheme | undefined;\n\n  #mutationObserver: MutationObserver | undefined;\n\n  #ngUnsubscribe = new Subject<void>();\n\n  #elRef: ElementRef;\n  #mutationObserverSvc: SkyMutationObserverService;\n  #ngZone: NgZone;\n  #themeSvc: SkyThemeService | undefined;\n\n  constructor(\n    elRef: ElementRef,\n    mutationObserverSvc: SkyMutationObserverService,\n    ngZone: NgZone,\n    @Optional() themeSvc?: SkyThemeService\n  ) {\n    this.#elRef = elRef;\n    this.#mutationObserverSvc = mutationObserverSvc;\n    this.#ngZone = ngZone;\n    this.#themeSvc = themeSvc;\n  }\n\n  @HostListener('window:resize')\n  public windowResize(): void {\n    this.#checkForShadow();\n  }\n\n  @HostListener('scroll')\n  public scroll(): void {\n    this.#checkForShadow();\n  }\n\n  public ngOnInit(): void {\n    if (this.#themeSvc) {\n      this.#themeSvc.settingsChange\n        .pipe(takeUntil(this.#ngUnsubscribe))\n        .subscribe((themeSettings) => {\n          this.#currentTheme = themeSettings.currentSettings.theme;\n\n          if (this.#currentTheme === SkyTheme.presets.modern) {\n            this.#initMutationObserver();\n          } else {\n            this.#emitShadow({\n              bottomShadow: 'none',\n              topShadow: 'none',\n            });\n\n            this.#destroyMutationObserver();\n          }\n        });\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this.#ngUnsubscribe.next();\n    this.#ngUnsubscribe.complete();\n\n    this.#destroyMutationObserver();\n  }\n\n  #initMutationObserver(): void {\n    if (!this.#mutationObserver) {\n      const el = this.#elRef.nativeElement;\n\n      // MutationObserver is patched by Zone.js and therefore becomes part of the\n      // Angular change detection cycle, but this can lead to infinite loops in some\n      // scenarios. This will keep MutationObserver from triggering change detection.\n      this.#ngZone.runOutsideAngular(() => {\n        this.#mutationObserver = this.#mutationObserverSvc.create(() => {\n          this.#checkForShadow();\n        });\n\n        this.#mutationObserver.observe(el, {\n          attributes: true,\n          characterData: true,\n          childList: true,\n          subtree: true,\n        });\n      });\n    }\n  }\n\n  #destroyMutationObserver(): void {\n    if (this.#mutationObserver) {\n      this.#mutationObserver.disconnect();\n      this.#mutationObserver = undefined;\n    }\n  }\n\n  #checkForShadow(): void {\n    if (this.#currentTheme === SkyTheme.presets.modern) {\n      const el = this.#elRef.nativeElement;\n\n      const topShadow = this.#buildShadowStyle(el.scrollTop);\n\n      const bottomShadow = this.#buildShadowStyle(\n        el.scrollHeight - el.scrollTop - el.clientHeight\n      );\n\n      this.#emitShadow({\n        bottomShadow,\n        topShadow,\n      });\n    }\n  }\n\n  #buildShadowStyle(pixelsFromEnd: number): string {\n    // Progressively darken the shadow until the user scrolls 30 pixels from the top or bottom\n    // of the scrollable element, with a max opacity of 0.3.\n    const opacity = Math.min(pixelsFromEnd / 30, 1) * 0.3;\n\n    return opacity > 0 ? `0px 1px 8px 0px rgba(0, 0, 0, ${opacity})` : 'none';\n  }\n\n  #emitShadow(shadow: SkyModalScrollShadowEventArgs): void {\n    if (\n      !this.#currentShadow ||\n      this.#currentShadow.bottomShadow !== shadow.bottomShadow ||\n      this.#currentShadow.topShadow !== shadow.topShadow\n    ) {\n      this.skyModalScrollShadow.emit(shadow);\n      this.#currentShadow = shadow;\n    }\n  }\n}\n"]}
|