forked from ibm-js/delite
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathDialogUnderlay.js
144 lines (126 loc) · 4.21 KB
/
DialogUnderlay.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
/** @module delite/DialogUnderlay */
define([
"./register",
"./Widget",
"./BackgroundIframe",
"./Viewport",
"./theme!./DialogUnderlay/themes/{{theme}}/DialogUnderlay.css"
], function (register, Widget, BackgroundIframe, Viewport) {
// TODO: having show() methods on the instance and also on the module is confusing,
// at least when looking at the API doc page. Should one be renamed?
/**
* A component used to block input behind a Dialog widget.
*
* Normally this widget should not be instantiated directly, but rather shown and hidden via
* `DialogUnderlay.show()` and `DialogUnderlay.hide()`. And usually the module is not accessed directly
* at all, since the underlay is shown and hidden by the Dialog.
*
* @class module:delite/DialogUnderlay
*/
var DialogUnderlay = register("d-dialog-underlay", [HTMLElement, Widget],
/** @lends module:delite/DialogUnderlay# */ {
// This will get overwritten as soon as show() is call, but leave an empty array in case hide() or destroy()
// is called first. The array is shared between instances but that's OK because we never write into it.
_modalConnects: [],
buildRendering: function () {
// Outer div is used for fade-in/fade-out, and also to hold background iframe.
// Inner div has opacity specified in CSS file.
this.className = "d-dialog-underlay";
},
postCreate: function () {
// Append the underlay to the body
this.ownerDocument.body.appendChild(this);
this.on("keydown", this._onKeyDown.bind(this));
},
/**
* Sets the background to the size of the viewport (rather than the size
* of the document) since we need to cover the whole browser window, even
* if the document is only a few lines long.
* @private
*/
layout: function () {
var s = this.style;
// hide the background temporarily, so that the background itself isn't
// causing scrollbars to appear (might happen when user shrinks browser
// window and then we are called to resize)
s.display = "none";
// then resize and show
// could alternately use $(window).scrollTop() and $(window).height(), etc.
var html = this.ownerDocument.documentElement;
s.width = html.clientWidth + "px";
s.height = html.clientHeight + "px";
s.display = "";
},
/**
* Show the dialog underlay (instance method).
*/
show: function () {
if (!this._open) {
this.style.display = "block";
this._open = true;
this.layout();
this.bgIframe = new BackgroundIframe(this);
this._modalConnects = [
Viewport.on("resize", function () { this.layout(); }.bind(this))
];
}
},
/**
* Hide the dialog underlay (instance method).
*/
hide: function () {
if (this._open) {
this.bgIframe.destroy();
delete this.bgIframe;
this.style.display = "none";
while (this._modalConnects.length) {
(this._modalConnects.pop()).remove();
}
this._open = false;
}
},
destroy: register.before(function () {
while (this._modalConnects.length) {
(this._modalConnects.pop()).remove();
}
}),
/**
* Extension point so Dialog can monitor keyboard events on the underlay.
* @protected
*/
_onKeyDown: function () {
}
});
/**
* Static method to display the underlay with the given attributes set. If the underlay is already displayed,
* then adjust it's attributes as specified.
* @memberof module:delite/DialogUnderlay
* @param {Object} attrs - The parameters to create DialogUnderlay with.
* @param {number} zIndex - z-index of the underlay.
*/
DialogUnderlay.show = function (attrs, zIndex) {
var underlay = DialogUnderlay._singleton;
if (!underlay || underlay._destroyed) {
underlay = DialogUnderlay._singleton = new DialogUnderlay(attrs);
} else {
if (attrs) {
underlay.mix(attrs);
}
}
underlay.style.zIndex = zIndex;
underlay.show();
};
/**
* Static method to hide the underlay.
* @memberof module:delite/DialogUnderlay
*/
DialogUnderlay.hide = function () {
// Guard code in case the underlay widget has already been destroyed
// because we are being called during page unload (when all widgets are destroyed)
var underlay = DialogUnderlay._singleton;
if (underlay && !underlay._destroyed) {
underlay.hide();
}
};
return DialogUnderlay;
});