-
Notifications
You must be signed in to change notification settings - Fork 28
/
Copy pathDisplayContainer.js
211 lines (194 loc) · 8.13 KB
/
DisplayContainer.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
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
/** @module delite/DisplayContainer */
define(["dcl/dcl", "./Container"], function (dcl, Container) {
/**
* Dispatched before child is shown.
* @example
* document.addEventListener("delite-before-show", function (evt) {
* console.log("about to show child", evt.child);
* });
* @event module:delite/DisplayContainer#delite-before-show
* @property {Element} child - reference to child element
*/
/**
* Dispatched after child is shown.
* @example
* document.addEventListener("delite-after-show", function (evt) {
* console.log("just displayed child", evt.child);
* });
* @event module:delite/DisplayContainer#delite-after-show
* @property {Element} child - reference to child element
*/
/**
* Dispatched to let an application level listener create/load the child node.
* @example
* document.addEventListener("delite-display-load", function (evt) {
* evt.setChild(new Promise(function (resolve, reject) {
* // fetch the data for the specified id, then create a node with that data
* fetchData(evt.dest).then(function(data) {
* var child = document.createElement("div");
* child.innerHTML = data;
* resolve({child: child});
* });
* );
* });
* @event module:delite/DisplayContainer#delite-display-load
* @property {Function} setChild - method to set child element, or Promise for child element
*/
/**
* Dispatched before child is hidden.
* @example
* document.addEventListener("delite-before-hide", function (evt) {
* console.log("about to hide child", evt.child);
* });
* @event module:delite/DisplayContainer#delite-before-hide
* @property {Element} child - reference to child element
*/
/**
* Dispatched after child is hidden.
* @example
* document.addEventListener("delite-after-hide", function (evt) {
* console.log("just hid child", evt.child);
* });
* @event module:delite/DisplayContainer#delite-after-hide
* @property {Element} child - reference to child element
*/
function mix (a, b) {
for (var n in b) {
a[n] = b[n];
}
}
/**
* Mixin for widget containers that need to show on or off a child.
*
* When the show method is called a container extending this mixin is able to be notified that one of
* its children must be displayed. Before displaying it, it will fire the `delite-display-load` event
* giving a chance to a listener to load and create the child if not yet available before proceeding with
* the display. After the display has been performed a `delite-display-complete` event will be fired.
* @mixin module:delite/DisplayContainer
* @augments module:delite/Container
*/
return dcl(Container, /** @lends module:delite/DisplayContainer# */ {
declaredClass: "delite/DisplayContainer",
/**
* This method must be called to display a particular destination child on this container.
* @param {Element|string} dest - Element or Element id that points to the child this container must
* display.
* @param {Object} [params] - Optional params that might be taken into account when displaying the child.
* This can be the type of visual transitions involved. This might vary from one DisplayContainer to another.
* @returns {Promise} A promise that will be resolved when the display & transition effect will have been
* performed.
* @fires module:delite/DisplayContainer#delite-before-show
* @fires module:delite/DisplayContainer#delite-after-show
* @fires module:delite/DisplayContainer#delite-display-load
*/
show: function (dest, params) {
var self = this;
return this.loadChild(dest, params).then(function (value) {
// if view is not already a child this means we loaded a new view (div), add it
if (self.getIndexOfChild(value.child) === -1) {
self.addChild(value.child, value.index);
}
// the child is here, actually perform the display
// notify everyone we are going to proceed
var event = {
dest: dest,
cancelable: false
};
mix(event, params);
mix(event, value);
self.emit("delite-before-show", event);
return Promise.resolve(self.changeDisplay(value.child, event)).then(function () {
self.emit("delite-after-show", event);
return value;
});
});
},
/**
* This method must be called to hide a particular destination child on this container.
* @param {Element|string} dest - Element or Element id that points to the child this container must
* hide.
* @param {Object} [params] - Optional params that might be taken into account when hiding the child.
* This can be the type of visual transitions involved. This might vary from one DisplayContainer to another.
* @returns {Promise} A promise that will be resolved when the display & transition effect will have been
* performed.
* @fires module:delite/DisplayContainer#delite-display-load
* @fires module:delite/DisplayContainer#delite-before-hide
* @fires module:delite/DisplayContainer#delite-after-hide
*/
hide: function (dest, params) {
var args = {hide: true};
mix(args, params);
var self = this;
return this.loadChild(dest, args).then(function (value) {
// the child is here, actually perform the display
// notify everyone we are going to proceed
var event = {
dest: dest,
bubbles: true,
cancelable: false,
hide: true
};
mix(event, params);
mix(event, value);
self.emit("delite-before-hide", event);
return Promise.resolve(self.changeDisplay(value.child, event)).then(function () {
// one might listen to that event and actuall remove the child if needed (view unload feature)
self.emit("delite-after-hide", event);
return value;
});
});
},
/**
* This method must perform the display and possible transition effect. It is meant to be specialized by
* subclasses.
* @param {Element|string} widget - Element or Element id that points to the child this container must
* show or hide.
* @param {Object} [params] - Optional params that might be taken into account when displaying the child. This
* can be the type of visual transitions involved. This might vary from one DisplayContainer to another.
* By default on the "hide" param is supporting meaning that the transition should hide the widget
* not display it.
* @returns {Promise} Optionally a promise that will be resolved when the display & transition effect will have
* been performed.
*/
changeDisplay: function (widget, params) {
if (params.hide === true) {
widget.style.visibility = "hidden";
widget.style.display = "none";
} else {
widget.style.visibility = "visible";
widget.style.display = "";
}
},
/**
* This method must be called to load a particular child on this container.
* A `delite-display-load` event is fired giving the chance to a controller to load/create the child.
* This method can be redefined to actually load a child of the container. If a controller is not present,
* it just looks up elements by id.
* @param {Element|string} dest - Element or Element id that points to the child this container must
* load.
* @param {Object} [params] - Optional params that might be taken into account when removing the child.
* @returns {Promise} A promise that will be resolved when the child will have been
* loaded with an object of the following form: `{ child: childElement }` or with an optional index
* `{ child: childElement, index: index }`. Other properties might be added to the object if needed.
* @fires module:delite/DisplayContainer#delite-display-load
*/
loadChild: function (dest, params) {
// we need to warn potential app controller we are going to load a view
var child, event = {
dest: dest,
setChild: function (val) {
child = val;
}
};
mix(event, params);
// we now need to warn potential app controller we need to load a new child.
// if the controller specifies the child then use it,
// otherwise call the container load method
this.emit("delite-display-load", event);
if (!child) {
child = { child: typeof dest === "string" ? this.ownerDocument.getElementById(dest) : dest };
}
return Promise.resolve(child);
}
});
});