Project

General

Profile

Statistics
| Revision:

root / trunk / web / dojo / dijit / dijit.js.uncompressed.js @ 10

History | View | Annotate | Download (194 KB)

1 9 andrej.cim
/*
2
        Copyright (c) 2004-2010, The Dojo Foundation All Rights Reserved.
3
        Available via Academic Free License >= 2.1 OR the modified BSD license.
4
        see: http://dojotoolkit.org/license for details
5
*/
6
7
/*
8
        This is an optimized version of Dojo, built for deployment and not for
9
        development. To get sources and documentation, please visit:
10

11
                http://dojotoolkit.org
12
*/
13
14
if(!dojo._hasResource["dojo.window"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
15
dojo._hasResource["dojo.window"] = true;
16
dojo.provide("dojo.window");
17
18
dojo.window.getBox = function(){
19
        // summary:
20
        //                Returns the dimensions and scroll position of the viewable area of a browser window
21
22
        var scrollRoot = (dojo.doc.compatMode == 'BackCompat') ? dojo.body() : dojo.doc.documentElement;
23
24
        // get scroll position
25
        var scroll = dojo._docScroll(); // scrollRoot.scrollTop/Left should work
26
        return { w: scrollRoot.clientWidth, h: scrollRoot.clientHeight, l: scroll.x, t: scroll.y };
27
};
28
29
dojo.window.get = function(doc){
30
        // summary:
31
        //                 Get window object associated with document doc
32
33
        // In some IE versions (at least 6.0), document.parentWindow does not return a
34
        // reference to the real window object (maybe a copy), so we must fix it as well
35
        // We use IE specific execScript to attach the real window reference to
36
        // document._parentWindow for later use
37
        if(dojo.isIE && window !== document.parentWindow){
38
                /*
39
                In IE 6, only the variable "window" can be used to connect events (others
40
                may be only copies).
41
                */
42
                doc.parentWindow.execScript("document._parentWindow = window;", "Javascript");
43
                //to prevent memory leak, unset it after use
44
                //another possibility is to add an onUnload handler which seems overkill to me (liucougar)
45
                var win = doc._parentWindow;
46
                doc._parentWindow = null;
47
                return win;        //        Window
48
        }
49
50
        return doc.parentWindow || doc.defaultView;        //        Window
51
};
52
53
dojo.window.scrollIntoView = function(/*DomNode*/ node, /*Object?*/ pos){
54
        // summary:
55
        //                Scroll the passed node into view, if it is not already.
56
57
        // don't rely on node.scrollIntoView working just because the function is there
58
59
        try{ // catch unexpected/unrecreatable errors (#7808) since we can recover using a semi-acceptable native method
60
                node = dojo.byId(node);
61
                var doc = node.ownerDocument || dojo.doc,
62
                        body = doc.body || dojo.body(),
63
                        html = doc.documentElement || body.parentNode,
64
                        isIE = dojo.isIE, isWK = dojo.isWebKit;
65
                // if an untested browser, then use the native method
66
                if((!(dojo.isMoz || isIE || isWK || dojo.isOpera) || node == body || node == html) && (typeof node.scrollIntoView != "undefined")){
67
                        node.scrollIntoView(false); // short-circuit to native if possible
68
                        return;
69
                }
70
                var backCompat = doc.compatMode == 'BackCompat',
71
                        clientAreaRoot = backCompat? body : html,
72
                        scrollRoot = isWK ? body : clientAreaRoot,
73
                        rootWidth = clientAreaRoot.clientWidth,
74
                        rootHeight = clientAreaRoot.clientHeight,
75
                        rtl = !dojo._isBodyLtr(),
76
                        nodePos = pos || dojo.position(node),
77
                        el = node.parentNode,
78
                        isFixed = function(el){
79
                                return ((isIE <= 6 || (isIE && backCompat))? false : (dojo.style(el, 'position').toLowerCase() == "fixed"));
80
                        };
81
                if(isFixed(node)){ return; } // nothing to do
82
83
                while(el){
84
                        if(el == body){ el = scrollRoot; }
85
                        var elPos = dojo.position(el),
86
                                fixedPos = isFixed(el);
87
88
                        if(el == scrollRoot){
89
                                elPos.w = rootWidth; elPos.h = rootHeight;
90
                                if(scrollRoot == html && isIE && rtl){ elPos.x += scrollRoot.offsetWidth-elPos.w; } // IE workaround where scrollbar causes negative x
91
                                if(elPos.x < 0 || !isIE){ elPos.x = 0; } // IE can have values > 0
92
                                if(elPos.y < 0 || !isIE){ elPos.y = 0; }
93
                        }else{
94
                                var pb = dojo._getPadBorderExtents(el);
95
                                elPos.w -= pb.w; elPos.h -= pb.h; elPos.x += pb.l; elPos.y += pb.t;
96
                        }
97
98
                        if(el != scrollRoot){ // body, html sizes already have the scrollbar removed
99
                                var clientSize = el.clientWidth,
100
                                        scrollBarSize = elPos.w - clientSize;
101
                                if(clientSize > 0 && scrollBarSize > 0){
102
                                        elPos.w = clientSize;
103
                                        if(isIE && rtl){ elPos.x += scrollBarSize; }
104
                                }
105
                                clientSize = el.clientHeight;
106
                                scrollBarSize = elPos.h - clientSize;
107
                                if(clientSize > 0 && scrollBarSize > 0){
108
                                        elPos.h = clientSize;
109
                                }
110
                        }
111
                        if(fixedPos){ // bounded by viewport, not parents
112
                                if(elPos.y < 0){
113
                                        elPos.h += elPos.y; elPos.y = 0;
114
                                }
115
                                if(elPos.x < 0){
116
                                        elPos.w += elPos.x; elPos.x = 0;
117
                                }
118
                                if(elPos.y + elPos.h > rootHeight){
119
                                        elPos.h = rootHeight - elPos.y;
120
                                }
121
                                if(elPos.x + elPos.w > rootWidth){
122
                                        elPos.w = rootWidth - elPos.x;
123
                                }
124
                        }
125
                        // calculate overflow in all 4 directions
126
                        var l = nodePos.x - elPos.x, // beyond left: < 0
127
                                t = nodePos.y - Math.max(elPos.y, 0), // beyond top: < 0
128
                                r = l + nodePos.w - elPos.w, // beyond right: > 0
129
                                bot = t + nodePos.h - elPos.h; // beyond bottom: > 0
130
                        if(r * l > 0){
131
                                var s = Math[l < 0? "max" : "min"](l, r);
132
                                nodePos.x += el.scrollLeft;
133
                                el.scrollLeft += (isIE >= 8 && !backCompat && rtl)? -s : s;
134
                                nodePos.x -= el.scrollLeft;
135
                        }
136
                        if(bot * t > 0){
137
                                nodePos.y += el.scrollTop;
138
                                el.scrollTop += Math[t < 0? "max" : "min"](t, bot);
139
                                nodePos.y -= el.scrollTop;
140
                        }
141
                        el = (el != scrollRoot) && !fixedPos && el.parentNode;
142
                }
143
        }catch(error){
144
                console.error('scrollIntoView: ' + error);
145
                node.scrollIntoView(false);
146
        }
147
};
148
149
}
150
151
if(!dojo._hasResource["dijit._base.manager"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
152
dojo._hasResource["dijit._base.manager"] = true;
153
dojo.provide("dijit._base.manager");
154
155
dojo.declare("dijit.WidgetSet", null, {
156
        // summary:
157
        //                A set of widgets indexed by id. A default instance of this class is
158
        //                available as `dijit.registry`
159
        //
160
        // example:
161
        //                Create a small list of widgets:
162
        //                |        var ws = new dijit.WidgetSet();
163
        //                |        ws.add(dijit.byId("one"));
164
        //                |         ws.add(dijit.byId("two"));
165
        //                |        // destroy both:
166
        //                |        ws.forEach(function(w){ w.destroy(); });
167
        //
168
        // example:
169
        //                Using dijit.registry:
170
        //                |        dijit.registry.forEach(function(w){ /* do something */ });
171
172
        constructor: function(){
173
                this._hash = {};
174
                this.length = 0;
175
        },
176
177
        add: function(/*dijit._Widget*/ widget){
178
                // summary:
179
                //                Add a widget to this list. If a duplicate ID is detected, a error is thrown.
180
                //
181
                // widget: dijit._Widget
182
                //                Any dijit._Widget subclass.
183
                if(this._hash[widget.id]){
184
                        throw new Error("Tried to register widget with id==" + widget.id + " but that id is already registered");
185
                }
186
                this._hash[widget.id] = widget;
187
                this.length++;
188
        },
189
190
        remove: function(/*String*/ id){
191
                // summary:
192
                //                Remove a widget from this WidgetSet. Does not destroy the widget; simply
193
                //                removes the reference.
194
                if(this._hash[id]){
195
                        delete this._hash[id];
196
                        this.length--;
197
                }
198
        },
199
200
        forEach: function(/*Function*/ func, /* Object? */thisObj){
201
                // summary:
202
                //                Call specified function for each widget in this set.
203
                //
204
                // func:
205
                //                A callback function to run for each item. Is passed the widget, the index
206
                //                in the iteration, and the full hash, similar to `dojo.forEach`.
207
                //
208
                // thisObj:
209
                //                An optional scope parameter
210
                //
211
                // example:
212
                //                Using the default `dijit.registry` instance:
213
                //                |        dijit.registry.forEach(function(widget){
214
                //                |                console.log(widget.declaredClass);
215
                //                |        });
216
                //
217
                // returns:
218
                //                Returns self, in order to allow for further chaining.
219
220
                thisObj = thisObj || dojo.global;
221
                var i = 0, id;
222
                for(id in this._hash){
223
                        func.call(thisObj, this._hash[id], i++, this._hash);
224
                }
225
                return this;        // dijit.WidgetSet
226
        },
227
228
        filter: function(/*Function*/ filter, /* Object? */thisObj){
229
                // summary:
230
                //                Filter down this WidgetSet to a smaller new WidgetSet
231
                //                Works the same as `dojo.filter` and `dojo.NodeList.filter`
232
                //
233
                // filter:
234
                //                Callback function to test truthiness. Is passed the widget
235
                //                reference and the pseudo-index in the object.
236
                //
237
                // thisObj: Object?
238
                //                Option scope to use for the filter function.
239
                //
240
                // example:
241
                //                Arbitrary: select the odd widgets in this list
242
                //                |        dijit.registry.filter(function(w, i){
243
                //                |                return i % 2 == 0;
244
                //                |        }).forEach(function(w){ /* odd ones */ });
245
246
                thisObj = thisObj || dojo.global;
247
                var res = new dijit.WidgetSet(), i = 0, id;
248
                for(id in this._hash){
249
                        var w = this._hash[id];
250
                        if(filter.call(thisObj, w, i++, this._hash)){
251
                                res.add(w);
252
                        }
253
                }
254
                return res; // dijit.WidgetSet
255
        },
256
257
        byId: function(/*String*/ id){
258
                // summary:
259
                //                Find a widget in this list by it's id.
260
                // example:
261
                //                Test if an id is in a particular WidgetSet
262
                //                | var ws = new dijit.WidgetSet();
263
                //                | ws.add(dijit.byId("bar"));
264
                //                | var t = ws.byId("bar") // returns a widget
265
                //                | var x = ws.byId("foo"); // returns undefined
266
267
                return this._hash[id];        // dijit._Widget
268
        },
269
270
        byClass: function(/*String*/ cls){
271
                // summary:
272
                //                Reduce this widgetset to a new WidgetSet of a particular `declaredClass`
273
                //
274
                // cls: String
275
                //                The Class to scan for. Full dot-notated string.
276
                //
277
                // example:
278
                //                Find all `dijit.TitlePane`s in a page:
279
                //                |        dijit.registry.byClass("dijit.TitlePane").forEach(function(tp){ tp.close(); });
280
281
                var res = new dijit.WidgetSet(), id, widget;
282
                for(id in this._hash){
283
                        widget = this._hash[id];
284
                        if(widget.declaredClass == cls){
285
                                res.add(widget);
286
                        }
287
                 }
288
                 return res; // dijit.WidgetSet
289
},
290
291
        toArray: function(){
292
                // summary:
293
                //                Convert this WidgetSet into a true Array
294
                //
295
                // example:
296
                //                Work with the widget .domNodes in a real Array
297
                //                |        dojo.map(dijit.registry.toArray(), function(w){ return w.domNode; });
298
299
                var ar = [];
300
                for(var id in this._hash){
301
                        ar.push(this._hash[id]);
302
                }
303
                return ar;        // dijit._Widget[]
304
},
305
306
        map: function(/* Function */func, /* Object? */thisObj){
307
                // summary:
308
                //                Create a new Array from this WidgetSet, following the same rules as `dojo.map`
309
                // example:
310
                //                |        var nodes = dijit.registry.map(function(w){ return w.domNode; });
311
                //
312
                // returns:
313
                //                A new array of the returned values.
314
                return dojo.map(this.toArray(), func, thisObj); // Array
315
        },
316
317
        every: function(func, thisObj){
318
                // summary:
319
                //                 A synthetic clone of `dojo.every` acting explicitly on this WidgetSet
320
                //
321
                // func: Function
322
                //                A callback function run for every widget in this list. Exits loop
323
                //                when the first false return is encountered.
324
                //
325
                // thisObj: Object?
326
                //                Optional scope parameter to use for the callback
327
328
                thisObj = thisObj || dojo.global;
329
                var x = 0, i;
330
                for(i in this._hash){
331
                        if(!func.call(thisObj, this._hash[i], x++, this._hash)){
332
                                return false; // Boolean
333
                        }
334
                }
335
                return true; // Boolean
336
        },
337
338
        some: function(func, thisObj){
339
                // summary:
340
                //                 A synthetic clone of `dojo.some` acting explictly on this WidgetSet
341
                //
342
                // func: Function
343
                //                A callback function run for every widget in this list. Exits loop
344
                //                when the first true return is encountered.
345
                //
346
                // thisObj: Object?
347
                //                Optional scope parameter to use for the callback
348
349
                thisObj = thisObj || dojo.global;
350
                var x = 0, i;
351
                for(i in this._hash){
352
                        if(func.call(thisObj, this._hash[i], x++, this._hash)){
353
                                return true; // Boolean
354
                        }
355
                }
356
                return false; // Boolean
357
        }
358
359
});
360
361
(function(){
362
363
        /*=====
364
        dijit.registry = {
365
                // summary:
366
                //                A list of widgets on a page.
367
                // description:
368
                //                Is an instance of `dijit.WidgetSet`
369
        };
370
        =====*/
371
        dijit.registry = new dijit.WidgetSet();
372
373
        var hash = dijit.registry._hash,
374
                attr = dojo.attr,
375
                hasAttr = dojo.hasAttr,
376
                style = dojo.style;
377
378
        dijit.byId = function(/*String|dijit._Widget*/ id){
379
                // summary:
380
                //                Returns a widget by it's id, or if passed a widget, no-op (like dojo.byId())
381
                return typeof id == "string" ? hash[id] : id; // dijit._Widget
382
        };
383
384
        var _widgetTypeCtr = {};
385
        dijit.getUniqueId = function(/*String*/widgetType){
386
                // summary:
387
                //                Generates a unique id for a given widgetType
388
389
                var id;
390
                do{
391
                        id = widgetType + "_" +
392
                                (widgetType in _widgetTypeCtr ?
393
                                        ++_widgetTypeCtr[widgetType] : _widgetTypeCtr[widgetType] = 0);
394
                }while(hash[id]);
395
                return dijit._scopeName == "dijit" ? id : dijit._scopeName + "_" + id; // String
396
        };
397
398
        dijit.findWidgets = function(/*DomNode*/ root){
399
                // summary:
400
                //                Search subtree under root returning widgets found.
401
                //                Doesn't search for nested widgets (ie, widgets inside other widgets).
402
403
                var outAry = [];
404
405
                function getChildrenHelper(root){
406
                        for(var node = root.firstChild; node; node = node.nextSibling){
407
                                if(node.nodeType == 1){
408
                                        var widgetId = node.getAttribute("widgetId");
409
                                        if(widgetId){
410
                                                outAry.push(hash[widgetId]);
411
                                        }else{
412
                                                getChildrenHelper(node);
413
                                        }
414
                                }
415
                        }
416
                }
417
418
                getChildrenHelper(root);
419
                return outAry;
420
        };
421
422
        dijit._destroyAll = function(){
423
                // summary:
424
                //                Code to destroy all widgets and do other cleanup on page unload
425
426
                // Clean up focus manager lingering references to widgets and nodes
427
                dijit._curFocus = null;
428
                dijit._prevFocus = null;
429
                dijit._activeStack = [];
430
431
                // Destroy all the widgets, top down
432
                dojo.forEach(dijit.findWidgets(dojo.body()), function(widget){
433
                        // Avoid double destroy of widgets like Menu that are attached to <body>
434
                        // even though they are logically children of other widgets.
435
                        if(!widget._destroyed){
436
                                if(widget.destroyRecursive){
437
                                        widget.destroyRecursive();
438
                                }else if(widget.destroy){
439
                                        widget.destroy();
440
                                }
441
                        }
442
                });
443
        };
444
445
        if(dojo.isIE){
446
                // Only run _destroyAll() for IE because we think it's only necessary in that case,
447
                // and because it causes problems on FF.  See bug #3531 for details.
448
                dojo.addOnWindowUnload(function(){
449
                        dijit._destroyAll();
450
                });
451
        }
452
453
        dijit.byNode = function(/*DOMNode*/ node){
454
                // summary:
455
                //                Returns the widget corresponding to the given DOMNode
456
                return hash[node.getAttribute("widgetId")]; // dijit._Widget
457
        };
458
459
        dijit.getEnclosingWidget = function(/*DOMNode*/ node){
460
                // summary:
461
                //                Returns the widget whose DOM tree contains the specified DOMNode, or null if
462
                //                the node is not contained within the DOM tree of any widget
463
                while(node){
464
                        var id = node.getAttribute && node.getAttribute("widgetId");
465
                        if(id){
466
                                return hash[id];
467
                        }
468
                        node = node.parentNode;
469
                }
470
                return null;
471
        };
472
473
        var shown = (dijit._isElementShown = function(/*Element*/ elem){
474
                var s = style(elem);
475
                return (s.visibility != "hidden")
476
                        && (s.visibility != "collapsed")
477
                        && (s.display != "none")
478
                        && (attr(elem, "type") != "hidden");
479
        });
480
481
        dijit.hasDefaultTabStop = function(/*Element*/ elem){
482
                // summary:
483
                //                Tests if element is tab-navigable even without an explicit tabIndex setting
484
485
                // No explicit tabIndex setting, need to investigate node type
486
                switch(elem.nodeName.toLowerCase()){
487
                        case "a":
488
                                // An <a> w/out a tabindex is only navigable if it has an href
489
                                return hasAttr(elem, "href");
490
                        case "area":
491
                        case "button":
492
                        case "input":
493
                        case "object":
494
                        case "select":
495
                        case "textarea":
496
                                // These are navigable by default
497
                                return true;
498
                        case "iframe":
499
                                // If it's an editor <iframe> then it's tab navigable.
500
                                //TODO: feature detect "designMode" in elem.contentDocument?
501
                                if(dojo.isMoz){
502
                                        try{
503
                                                return elem.contentDocument.designMode == "on";
504
                                        }catch(err){
505
                                                return false;
506
                                        }
507
                                }else if(dojo.isWebKit){
508
                                        var doc = elem.contentDocument,
509
                                                body = doc && doc.body;
510
                                        return body && body.contentEditable == 'true';
511
                                }else{
512
                                        // contentWindow.document isn't accessible within IE7/8
513
                                        // if the iframe.src points to a foreign url and this
514
                                        // page contains an element, that could get focus
515
                                        try{
516
                                                doc = elem.contentWindow.document;
517
                                                body = doc && doc.body;
518
                                                return body && body.firstChild && body.firstChild.contentEditable == 'true';
519
                                        }catch(e){
520
                                                return false;
521
                                        }
522
                                }
523
                        default:
524
                                return elem.contentEditable == 'true';
525
                }
526
        };
527
528
        var isTabNavigable = (dijit.isTabNavigable = function(/*Element*/ elem){
529
                // summary:
530
                //                Tests if an element is tab-navigable
531
532
                // TODO: convert (and rename method) to return effective tabIndex; will save time in _getTabNavigable()
533
                if(attr(elem, "disabled")){
534
                        return false;
535
                }else if(hasAttr(elem, "tabIndex")){
536
                        // Explicit tab index setting
537
                        return attr(elem, "tabIndex") >= 0; // boolean
538
                }else{
539
                        // No explicit tabIndex setting, so depends on node type
540
                        return dijit.hasDefaultTabStop(elem);
541
                }
542
        });
543
544
        dijit._getTabNavigable = function(/*DOMNode*/ root){
545
                // summary:
546
                //                Finds descendants of the specified root node.
547
                //
548
                // description:
549
                //                Finds the following descendants of the specified root node:
550
                //                * the first tab-navigable element in document order
551
                //                  without a tabIndex or with tabIndex="0"
552
                //                * the last tab-navigable element in document order
553
                //                  without a tabIndex or with tabIndex="0"
554
                //                * the first element in document order with the lowest
555
                //                  positive tabIndex value
556
                //                * the last element in document order with the highest
557
                //                  positive tabIndex value
558
                var first, last, lowest, lowestTabindex, highest, highestTabindex;
559
                var walkTree = function(/*DOMNode*/parent){
560
                        dojo.query("> *", parent).forEach(function(child){
561
                                // Skip hidden elements, and also non-HTML elements (those in custom namespaces) in IE,
562
                                // since show() invokes getAttribute("type"), which crash on VML nodes in IE.
563
                                if((dojo.isIE && child.scopeName!=="HTML") || !shown(child)){
564
                                        return;
565
                                }
566
567
                                if(isTabNavigable(child)){
568
                                        var tabindex = attr(child, "tabIndex");
569
                                        if(!hasAttr(child, "tabIndex") || tabindex == 0){
570
                                                if(!first){ first = child; }
571
                                                last = child;
572
                                        }else if(tabindex > 0){
573
                                                if(!lowest || tabindex < lowestTabindex){
574
                                                        lowestTabindex = tabindex;
575
                                                        lowest = child;
576
                                                }
577
                                                if(!highest || tabindex >= highestTabindex){
578
                                                        highestTabindex = tabindex;
579
                                                        highest = child;
580
                                                }
581
                                        }
582
                                }
583
                                if(child.nodeName.toUpperCase() != 'SELECT'){
584
                                        walkTree(child);
585
                                }
586
                        });
587
                };
588
                if(shown(root)){ walkTree(root) }
589
                return { first: first, last: last, lowest: lowest, highest: highest };
590
        }
591
        dijit.getFirstInTabbingOrder = function(/*String|DOMNode*/ root){
592
                // summary:
593
                //                Finds the descendant of the specified root node
594
                //                that is first in the tabbing order
595
                var elems = dijit._getTabNavigable(dojo.byId(root));
596
                return elems.lowest ? elems.lowest : elems.first; // DomNode
597
        };
598
599
        dijit.getLastInTabbingOrder = function(/*String|DOMNode*/ root){
600
                // summary:
601
                //                Finds the descendant of the specified root node
602
                //                that is last in the tabbing order
603
                var elems = dijit._getTabNavigable(dojo.byId(root));
604
                return elems.last ? elems.last : elems.highest; // DomNode
605
        };
606
607
        /*=====
608
        dojo.mixin(dijit, {
609
                // defaultDuration: Integer
610
                //                The default animation speed (in ms) to use for all Dijit
611
                //                transitional animations, unless otherwise specified
612
                //                on a per-instance basis. Defaults to 200, overrided by
613
                //                `djConfig.defaultDuration`
614
                defaultDuration: 200
615
        });
616
        =====*/
617
618
        dijit.defaultDuration = dojo.config["defaultDuration"] || 200;
619
620
})();
621
622
}
623
624
if(!dojo._hasResource["dijit._base.focus"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
625
dojo._hasResource["dijit._base.focus"] = true;
626
dojo.provide("dijit._base.focus");
627
628
629
        // for dijit.isTabNavigable()
630
631
// summary:
632
//                These functions are used to query or set the focus and selection.
633
//
634
//                Also, they trace when widgets become activated/deactivated,
635
//                so that the widget can fire _onFocus/_onBlur events.
636
//                "Active" here means something similar to "focused", but
637
//                "focus" isn't quite the right word because we keep track of
638
//                a whole stack of "active" widgets.  Example: ComboButton --> Menu -->
639
//                MenuItem.  The onBlur event for ComboButton doesn't fire due to focusing
640
//                on the Menu or a MenuItem, since they are considered part of the
641
//                ComboButton widget.  It only happens when focus is shifted
642
//                somewhere completely different.
643
644
dojo.mixin(dijit, {
645
        // _curFocus: DomNode
646
        //                Currently focused item on screen
647
        _curFocus: null,
648
649
        // _prevFocus: DomNode
650
        //                Previously focused item on screen
651
        _prevFocus: null,
652
653
        isCollapsed: function(){
654
                // summary:
655
                //                Returns true if there is no text selected
656
                return dijit.getBookmark().isCollapsed;
657
        },
658
659
        getBookmark: function(){
660
                // summary:
661
                //                Retrieves a bookmark that can be used with moveToBookmark to return to the same range
662
                var bm, rg, tg, sel = dojo.doc.selection, cf = dijit._curFocus;
663
664
                if(dojo.global.getSelection){
665
                        //W3C Range API for selections.
666
                        sel = dojo.global.getSelection();
667
                        if(sel){
668
                                if(sel.isCollapsed){
669
                                        tg = cf? cf.tagName : "";
670
                                        if(tg){
671
                                                //Create a fake rangelike item to restore selections.
672
                                                tg = tg.toLowerCase();
673
                                                if(tg == "textarea" ||
674
                                                                (tg == "input" && (!cf.type || cf.type.toLowerCase() == "text"))){
675
                                                        sel = {
676
                                                                start: cf.selectionStart,
677
                                                                end: cf.selectionEnd,
678
                                                                node: cf,
679
                                                                pRange: true
680
                                                        };
681
                                                        return {isCollapsed: (sel.end <= sel.start), mark: sel}; //Object.
682
                                                }
683
                                        }
684
                                        bm = {isCollapsed:true};
685
                                }else{
686
                                        rg = sel.getRangeAt(0);
687
                                        bm = {isCollapsed: false, mark: rg.cloneRange()};
688
                                }
689
                        }
690
                }else if(sel){
691
                        // If the current focus was a input of some sort and no selection, don't bother saving
692
                        // a native bookmark.  This is because it causes issues with dialog/page selection restore.
693
                        // So, we need to create psuedo bookmarks to work with.
694
                        tg = cf ? cf.tagName : "";
695
                        tg = tg.toLowerCase();
696
                        if(cf && tg && (tg == "button" || tg == "textarea" || tg == "input")){
697
                                if(sel.type && sel.type.toLowerCase() == "none"){
698
                                        return {
699
                                                isCollapsed: true,
700
                                                mark: null
701
                                        }
702
                                }else{
703
                                        rg = sel.createRange();
704
                                        return {
705
                                                isCollapsed: rg.text && rg.text.length?false:true,
706
                                                mark: {
707
                                                        range: rg,
708
                                                        pRange: true
709
                                                }
710
                                        };
711
                                }
712
                        }
713
                        bm = {};
714
715
                        //'IE' way for selections.
716
                        try{
717
                                // createRange() throws exception when dojo in iframe
718
                                //and nothing selected, see #9632
719
                                rg = sel.createRange();
720
                                bm.isCollapsed = !(sel.type == 'Text' ? rg.htmlText.length : rg.length);
721
                        }catch(e){
722
                                bm.isCollapsed = true;
723
                                return bm;
724
                        }
725
                        if(sel.type.toUpperCase() == 'CONTROL'){
726
                                if(rg.length){
727
                                        bm.mark=[];
728
                                        var i=0,len=rg.length;
729
                                        while(i<len){
730
                                                bm.mark.push(rg.item(i++));
731
                                        }
732
                                }else{
733
                                        bm.isCollapsed = true;
734
                                        bm.mark = null;
735
                                }
736
                        }else{
737
                                bm.mark = rg.getBookmark();
738
                        }
739
                }else{
740
                        console.warn("No idea how to store the current selection for this browser!");
741
                }
742
                return bm; // Object
743
        },
744
745
        moveToBookmark: function(/*Object*/bookmark){
746
                // summary:
747
                //                Moves current selection to a bookmark
748
                // bookmark:
749
                //                This should be a returned object from dijit.getBookmark()
750
751
                var _doc = dojo.doc,
752
                        mark = bookmark.mark;
753
                if(mark){
754
                        if(dojo.global.getSelection){
755
                                //W3C Rangi API (FF, WebKit, Opera, etc)
756
                                var sel = dojo.global.getSelection();
757
                                if(sel && sel.removeAllRanges){
758
                                        if(mark.pRange){
759
                                                var r = mark;
760
                                                var n = r.node;
761
                                                n.selectionStart = r.start;
762
                                                n.selectionEnd = r.end;
763
                                        }else{
764
                                                sel.removeAllRanges();
765
                                                sel.addRange(mark);
766
                                        }
767
                                }else{
768
                                        console.warn("No idea how to restore selection for this browser!");
769
                                }
770
                        }else if(_doc.selection && mark){
771
                                //'IE' way.
772
                                var rg;
773
                                if(mark.pRange){
774
                                        rg = mark.range;
775
                                }else if(dojo.isArray(mark)){
776
                                        rg = _doc.body.createControlRange();
777
                                        //rg.addElement does not have call/apply method, so can not call it directly
778
                                        //rg is not available in "range.addElement(item)", so can't use that either
779
                                        dojo.forEach(mark, function(n){
780
                                                rg.addElement(n);
781
                                        });
782
                                }else{
783
                                        rg = _doc.body.createTextRange();
784
                                        rg.moveToBookmark(mark);
785
                                }
786
                                rg.select();
787
                        }
788
                }
789
        },
790
791
        getFocus: function(/*Widget?*/ menu, /*Window?*/ openedForWindow){
792
                // summary:
793
                //                Called as getFocus(), this returns an Object showing the current focus
794
                //                and selected text.
795
                //
796
                //                Called as getFocus(widget), where widget is a (widget representing) a button
797
                //                that was just pressed, it returns where focus was before that button
798
                //                was pressed.   (Pressing the button may have either shifted focus to the button,
799
                //                or removed focus altogether.)   In this case the selected text is not returned,
800
                //                since it can't be accurately determined.
801
                //
802
                // menu: dijit._Widget or {domNode: DomNode} structure
803
                //                The button that was just pressed.  If focus has disappeared or moved
804
                //                to this button, returns the previous focus.  In this case the bookmark
805
                //                information is already lost, and null is returned.
806
                //
807
                // openedForWindow:
808
                //                iframe in which menu was opened
809
                //
810
                // returns:
811
                //                A handle to restore focus/selection, to be passed to `dijit.focus`
812
                var node = !dijit._curFocus || (menu && dojo.isDescendant(dijit._curFocus, menu.domNode)) ? dijit._prevFocus : dijit._curFocus;
813
                return {
814
                        node: node,
815
                        bookmark: (node == dijit._curFocus) && dojo.withGlobal(openedForWindow || dojo.global, dijit.getBookmark),
816
                        openedForWindow: openedForWindow
817
                }; // Object
818
        },
819
820
        focus: function(/*Object || DomNode */ handle){
821
                // summary:
822
                //                Sets the focused node and the selection according to argument.
823
                //                To set focus to an iframe's content, pass in the iframe itself.
824
                // handle:
825
                //                object returned by get(), or a DomNode
826
827
                if(!handle){ return; }
828
829
                var node = "node" in handle ? handle.node : handle,                // because handle is either DomNode or a composite object
830
                        bookmark = handle.bookmark,
831
                        openedForWindow = handle.openedForWindow,
832
                        collapsed = bookmark ? bookmark.isCollapsed : false;
833
834
                // Set the focus
835
                // Note that for iframe's we need to use the <iframe> to follow the parentNode chain,
836
                // but we need to set focus to iframe.contentWindow
837
                if(node){
838
                        var focusNode = (node.tagName.toLowerCase() == "iframe") ? node.contentWindow : node;
839
                        if(focusNode && focusNode.focus){
840
                                try{
841
                                        // Gecko throws sometimes if setting focus is impossible,
842
                                        // node not displayed or something like that
843
                                        focusNode.focus();
844
                                }catch(e){/*quiet*/}
845
                        }
846
                        dijit._onFocusNode(node);
847
                }
848
849
                // set the selection
850
                // do not need to restore if current selection is not empty
851
                // (use keyboard to select a menu item) or if previous selection was collapsed
852
                // as it may cause focus shift (Esp in IE).
853
                if(bookmark && dojo.withGlobal(openedForWindow || dojo.global, dijit.isCollapsed) && !collapsed){
854
                        if(openedForWindow){
855
                                openedForWindow.focus();
856
                        }
857
                        try{
858
                                dojo.withGlobal(openedForWindow || dojo.global, dijit.moveToBookmark, null, [bookmark]);
859
                        }catch(e2){
860
                                /*squelch IE internal error, see http://trac.dojotoolkit.org/ticket/1984 */
861
                        }
862
                }
863
        },
864
865
        // _activeStack: dijit._Widget[]
866
        //                List of currently active widgets (focused widget and it's ancestors)
867
        _activeStack: [],
868
869
        registerIframe: function(/*DomNode*/ iframe){
870
                // summary:
871
                //                Registers listeners on the specified iframe so that any click
872
                //                or focus event on that iframe (or anything in it) is reported
873
                //                as a focus/click event on the <iframe> itself.
874
                // description:
875
                //                Currently only used by editor.
876
                // returns:
877
                //                Handle to pass to unregisterIframe()
878
                return dijit.registerWin(iframe.contentWindow, iframe);
879
        },
880
881
        unregisterIframe: function(/*Object*/ handle){
882
                // summary:
883
                //                Unregisters listeners on the specified iframe created by registerIframe.
884
                //                After calling be sure to delete or null out the handle itself.
885
                // handle:
886
                //                Handle returned by registerIframe()
887
888
                dijit.unregisterWin(handle);
889
        },
890
891
        registerWin: function(/*Window?*/targetWindow, /*DomNode?*/ effectiveNode){
892
                // summary:
893
                //                Registers listeners on the specified window (either the main
894
                //                window or an iframe's window) to detect when the user has clicked somewhere
895
                //                or focused somewhere.
896
                // description:
897
                //                Users should call registerIframe() instead of this method.
898
                // targetWindow:
899
                //                If specified this is the window associated with the iframe,
900
                //                i.e. iframe.contentWindow.
901
                // effectiveNode:
902
                //                If specified, report any focus events inside targetWindow as
903
                //                an event on effectiveNode, rather than on evt.target.
904
                // returns:
905
                //                Handle to pass to unregisterWin()
906
907
                // TODO: make this function private in 2.0; Editor/users should call registerIframe(),
908
909
                var mousedownListener = function(evt){
910
                        dijit._justMouseDowned = true;
911
                        setTimeout(function(){ dijit._justMouseDowned = false; }, 0);
912
913
                        // workaround weird IE bug where the click is on an orphaned node
914
                        // (first time clicking a Select/DropDownButton inside a TooltipDialog)
915
                        if(dojo.isIE && evt && evt.srcElement && evt.srcElement.parentNode == null){
916
                                return;
917
                        }
918
919
                        dijit._onTouchNode(effectiveNode || evt.target || evt.srcElement, "mouse");
920
                };
921
                //dojo.connect(targetWindow, "onscroll", ???);
922
923
                // Listen for blur and focus events on targetWindow's document.
924
                // IIRC, I'm using attachEvent() rather than dojo.connect() because focus/blur events don't bubble
925
                // through dojo.connect(), and also maybe to catch the focus events early, before onfocus handlers
926
                // fire.
927
                // Connect to <html> (rather than document) on IE to avoid memory leaks, but document on other browsers because
928
                // (at least for FF) the focus event doesn't fire on <html> or <body>.
929
                var doc = dojo.isIE ? targetWindow.document.documentElement : targetWindow.document;
930
                if(doc){
931
                        if(dojo.isIE){
932
                                doc.attachEvent('onmousedown', mousedownListener);
933
                                var activateListener = function(evt){
934
                                        // IE reports that nodes like <body> have gotten focus, even though they have tabIndex=-1,
935
                                        // Should consider those more like a mouse-click than a focus....
936
                                        if(evt.srcElement.tagName.toLowerCase() != "#document" &&
937
                                                dijit.isTabNavigable(evt.srcElement)){
938
                                                dijit._onFocusNode(effectiveNode || evt.srcElement);
939
                                        }else{
940
                                                dijit._onTouchNode(effectiveNode || evt.srcElement);
941
                                        }
942
                                };
943
                                doc.attachEvent('onactivate', activateListener);
944
                                var deactivateListener =  function(evt){
945
                                        dijit._onBlurNode(effectiveNode || evt.srcElement);
946
                                };
947
                                doc.attachEvent('ondeactivate', deactivateListener);
948
949
                                return function(){
950
                                        doc.detachEvent('onmousedown', mousedownListener);
951
                                        doc.detachEvent('onactivate', activateListener);
952
                                        doc.detachEvent('ondeactivate', deactivateListener);
953
                                        doc = null;        // prevent memory leak (apparent circular reference via closure)
954
                                };
955
                        }else{
956
                                doc.addEventListener('mousedown', mousedownListener, true);
957
                                var focusListener = function(evt){
958
                                        dijit._onFocusNode(effectiveNode || evt.target);
959
                                };
960
                                doc.addEventListener('focus', focusListener, true);
961
                                var blurListener = function(evt){
962
                                        dijit._onBlurNode(effectiveNode || evt.target);
963
                                };
964
                                doc.addEventListener('blur', blurListener, true);
965
966
                                return function(){
967
                                        doc.removeEventListener('mousedown', mousedownListener, true);
968
                                        doc.removeEventListener('focus', focusListener, true);
969
                                        doc.removeEventListener('blur', blurListener, true);
970
                                        doc = null;        // prevent memory leak (apparent circular reference via closure)
971
                                };
972
                        }
973
                }
974
        },
975
976
        unregisterWin: function(/*Handle*/ handle){
977
                // summary:
978
                //                Unregisters listeners on the specified window (either the main
979
                //                window or an iframe's window) according to handle returned from registerWin().
980
                //                After calling be sure to delete or null out the handle itself.
981
982
                // Currently our handle is actually a function
983
                handle && handle();
984
        },
985
986
        _onBlurNode: function(/*DomNode*/ node){
987
                // summary:
988
                //                 Called when focus leaves a node.
989
                //                Usually ignored, _unless_ it *isn't* follwed by touching another node,
990
                //                which indicates that we tabbed off the last field on the page,
991
                //                in which case every widget is marked inactive
992
                dijit._prevFocus = dijit._curFocus;
993
                dijit._curFocus = null;
994
995
                if(dijit._justMouseDowned){
996
                        // the mouse down caused a new widget to be marked as active; this blur event
997
                        // is coming late, so ignore it.
998
                        return;
999
                }
1000
1001
                // if the blur event isn't followed by a focus event then mark all widgets as inactive.
1002
                if(dijit._clearActiveWidgetsTimer){
1003
                        clearTimeout(dijit._clearActiveWidgetsTimer);
1004
                }
1005
                dijit._clearActiveWidgetsTimer = setTimeout(function(){
1006
                        delete dijit._clearActiveWidgetsTimer;
1007
                        dijit._setStack([]);
1008
                        dijit._prevFocus = null;
1009
                }, 100);
1010
        },
1011
1012
        _onTouchNode: function(/*DomNode*/ node, /*String*/ by){
1013
                // summary:
1014
                //                Callback when node is focused or mouse-downed
1015
                // node:
1016
                //                The node that was touched.
1017
                // by:
1018
                //                "mouse" if the focus/touch was caused by a mouse down event
1019
1020
                // ignore the recent blurNode event
1021
                if(dijit._clearActiveWidgetsTimer){
1022
                        clearTimeout(dijit._clearActiveWidgetsTimer);
1023
                        delete dijit._clearActiveWidgetsTimer;
1024
                }
1025
1026
                // compute stack of active widgets (ex: ComboButton --> Menu --> MenuItem)
1027
                var newStack=[];
1028
                try{
1029
                        while(node){
1030
                                var popupParent = dojo.attr(node, "dijitPopupParent");
1031
                                if(popupParent){
1032
                                        node=dijit.byId(popupParent).domNode;
1033
                                }else if(node.tagName && node.tagName.toLowerCase() == "body"){
1034
                                        // is this the root of the document or just the root of an iframe?
1035
                                        if(node === dojo.body()){
1036
                                                // node is the root of the main document
1037
                                                break;
1038
                                        }
1039
                                        // otherwise, find the iframe this node refers to (can't access it via parentNode,
1040
                                        // need to do this trick instead). window.frameElement is supported in IE/FF/Webkit
1041
                                        node=dojo.window.get(node.ownerDocument).frameElement;
1042
                                }else{
1043
                                        // if this node is the root node of a widget, then add widget id to stack,
1044
                                        // except ignore clicks on disabled widgets (actually focusing a disabled widget still works,
1045
                                        // to support MenuItem)
1046
                                        var id = node.getAttribute && node.getAttribute("widgetId"),
1047
                                                widget = id && dijit.byId(id);
1048
                                        if(widget && !(by == "mouse" && widget.get("disabled"))){
1049
                                                newStack.unshift(id);
1050
                                        }
1051
                                        node=node.parentNode;
1052
                                }
1053
                        }
1054
                }catch(e){ /* squelch */ }
1055
1056
                dijit._setStack(newStack, by);
1057
        },
1058
1059
        _onFocusNode: function(/*DomNode*/ node){
1060
                // summary:
1061
                //                Callback when node is focused
1062
1063
                if(!node){
1064
                        return;
1065
                }
1066
1067
                if(node.nodeType == 9){
1068
                        // Ignore focus events on the document itself.  This is here so that
1069
                        // (for example) clicking the up/down arrows of a spinner
1070
                        // (which don't get focus) won't cause that widget to blur. (FF issue)
1071
                        return;
1072
                }
1073
1074
                dijit._onTouchNode(node);
1075
1076
                if(node == dijit._curFocus){ return; }
1077
                if(dijit._curFocus){
1078
                        dijit._prevFocus = dijit._curFocus;
1079
                }
1080
                dijit._curFocus = node;
1081
                dojo.publish("focusNode", [node]);
1082
        },
1083
1084
        _setStack: function(/*String[]*/ newStack, /*String*/ by){
1085
                // summary:
1086
                //                The stack of active widgets has changed.  Send out appropriate events and records new stack.
1087
                // newStack:
1088
                //                array of widget id's, starting from the top (outermost) widget
1089
                // by:
1090
                //                "mouse" if the focus/touch was caused by a mouse down event
1091
1092
                var oldStack = dijit._activeStack;
1093
                dijit._activeStack = newStack;
1094
1095
                // compare old stack to new stack to see how many elements they have in common
1096
                for(var nCommon=0; nCommon<Math.min(oldStack.length, newStack.length); nCommon++){
1097
                        if(oldStack[nCommon] != newStack[nCommon]){
1098
                                break;
1099
                        }
1100
                }
1101
1102
                var widget;
1103
                // for all elements that have gone out of focus, send blur event
1104
                for(var i=oldStack.length-1; i>=nCommon; i--){
1105
                        widget = dijit.byId(oldStack[i]);
1106
                        if(widget){
1107
                                widget._focused = false;
1108
                                widget._hasBeenBlurred = true;
1109
                                if(widget._onBlur){
1110
                                        widget._onBlur(by);
1111
                                }
1112
                                dojo.publish("widgetBlur", [widget, by]);
1113
                        }
1114
                }
1115
1116
                // for all element that have come into focus, send focus event
1117
                for(i=nCommon; i<newStack.length; i++){
1118
                        widget = dijit.byId(newStack[i]);
1119
                        if(widget){
1120
                                widget._focused = true;
1121
                                if(widget._onFocus){
1122
                                        widget._onFocus(by);
1123
                                }
1124
                                dojo.publish("widgetFocus", [widget, by]);
1125
                        }
1126
                }
1127
        }
1128
});
1129
1130
// register top window and all the iframes it contains
1131
dojo.addOnLoad(function(){
1132
        var handle = dijit.registerWin(window);
1133
        if(dojo.isIE){
1134
                dojo.addOnWindowUnload(function(){
1135
                        dijit.unregisterWin(handle);
1136
                        handle = null;
1137
                })
1138
        }
1139
});
1140
1141
}
1142
1143
if(!dojo._hasResource["dojo.AdapterRegistry"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
1144
dojo._hasResource["dojo.AdapterRegistry"] = true;
1145
dojo.provide("dojo.AdapterRegistry");
1146
1147
dojo.AdapterRegistry = function(/*Boolean?*/ returnWrappers){
1148
        //        summary:
1149
        //                A registry to make contextual calling/searching easier.
1150
        //        description:
1151
        //                Objects of this class keep list of arrays in the form [name, check,
1152
        //                wrap, directReturn] that are used to determine what the contextual
1153
        //                result of a set of checked arguments is. All check/wrap functions
1154
        //                in this registry should be of the same arity.
1155
        //        example:
1156
        //        |        // create a new registry
1157
        //        |        var reg = new dojo.AdapterRegistry();
1158
        //        |        reg.register("handleString",
1159
        //        |                dojo.isString,
1160
        //        |                function(str){
1161
        //        |                        // do something with the string here
1162
        //        |                }
1163
        //        |        );
1164
        //        |        reg.register("handleArr",
1165
        //        |                dojo.isArray,
1166
        //        |                function(arr){
1167
        //        |                        // do something with the array here
1168
        //        |                }
1169
        //        |        );
1170
        //        |
1171
        //        |        // now we can pass reg.match() *either* an array or a string and
1172
        //        |        // the value we pass will get handled by the right function
1173
        //        |        reg.match("someValue"); // will call the first function
1174
        //        |        reg.match(["someValue"]); // will call the second
1175
1176
        this.pairs = [];
1177
        this.returnWrappers = returnWrappers || false; // Boolean
1178
}
1179
1180
dojo.extend(dojo.AdapterRegistry, {
1181
        register: function(/*String*/ name, /*Function*/ check, /*Function*/ wrap, /*Boolean?*/ directReturn, /*Boolean?*/ override){
1182
                //        summary:
1183
                //                register a check function to determine if the wrap function or
1184
                //                object gets selected
1185
                //        name:
1186
                //                a way to identify this matcher.
1187
                //        check:
1188
                //                a function that arguments are passed to from the adapter's
1189
                //                match() function.  The check function should return true if the
1190
                //                given arguments are appropriate for the wrap function.
1191
                //        directReturn:
1192
                //                If directReturn is true, the value passed in for wrap will be
1193
                //                returned instead of being called. Alternately, the
1194
                //                AdapterRegistry can be set globally to "return not call" using
1195
                //                the returnWrappers property. Either way, this behavior allows
1196
                //                the registry to act as a "search" function instead of a
1197
                //                function interception library.
1198
                //        override:
1199
                //                If override is given and true, the check function will be given
1200
                //                highest priority. Otherwise, it will be the lowest priority
1201
                //                adapter.
1202
                this.pairs[((override) ? "unshift" : "push")]([name, check, wrap, directReturn]);
1203
        },
1204
1205
        match: function(/* ... */){
1206
                // summary:
1207
                //                Find an adapter for the given arguments. If no suitable adapter
1208
                //                is found, throws an exception. match() accepts any number of
1209
                //                arguments, all of which are passed to all matching functions
1210
                //                from the registered pairs.
1211
                for(var i = 0; i < this.pairs.length; i++){
1212
                        var pair = this.pairs[i];
1213
                        if(pair[1].apply(this, arguments)){
1214
                                if((pair[3])||(this.returnWrappers)){
1215
                                        return pair[2];
1216
                                }else{
1217
                                        return pair[2].apply(this, arguments);
1218
                                }
1219
                        }
1220
                }
1221
                throw new Error("No match found");
1222
        },
1223
1224
        unregister: function(name){
1225
                // summary: Remove a named adapter from the registry
1226
1227
                // FIXME: this is kind of a dumb way to handle this. On a large
1228
                // registry this will be slow-ish and we can use the name as a lookup
1229
                // should we choose to trade memory for speed.
1230
                for(var i = 0; i < this.pairs.length; i++){
1231
                        var pair = this.pairs[i];
1232
                        if(pair[0] == name){
1233
                                this.pairs.splice(i, 1);
1234
                                return true;
1235
                        }
1236
                }
1237
                return false;
1238
        }
1239
});
1240
1241
}
1242
1243
if(!dojo._hasResource["dijit._base.place"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
1244
dojo._hasResource["dijit._base.place"] = true;
1245
dojo.provide("dijit._base.place");
1246
1247
1248
1249
1250
1251
dijit.getViewport = function(){
1252
        // summary:
1253
        //                Returns the dimensions and scroll position of the viewable area of a browser window
1254
1255
        return dojo.window.getBox();
1256
};
1257
1258
/*=====
1259
dijit.__Position = function(){
1260
        // x: Integer
1261
        //                horizontal coordinate in pixels, relative to document body
1262
        // y: Integer
1263
        //                vertical coordinate in pixels, relative to document body
1264

1265
        thix.x = x;
1266
        this.y = y;
1267
}
1268
=====*/
1269
1270
1271
dijit.placeOnScreen = function(
1272
        /* DomNode */                        node,
1273
        /* dijit.__Position */        pos,
1274
        /* String[] */                        corners,
1275
        /* dijit.__Position? */        padding){
1276
        // summary:
1277
        //                Positions one of the node's corners at specified position
1278
        //                such that node is fully visible in viewport.
1279
        // description:
1280
        //                NOTE: node is assumed to be absolutely or relatively positioned.
1281
        //        pos:
1282
        //                Object like {x: 10, y: 20}
1283
        //        corners:
1284
        //                Array of Strings representing order to try corners in, like ["TR", "BL"].
1285
        //                Possible values are:
1286
        //                        * "BL" - bottom left
1287
        //                        * "BR" - bottom right
1288
        //                        * "TL" - top left
1289
        //                        * "TR" - top right
1290
        //        padding:
1291
        //                set padding to put some buffer around the element you want to position.
1292
        // example:
1293
        //                Try to place node's top right corner at (10,20).
1294
        //                If that makes node go (partially) off screen, then try placing
1295
        //                bottom left corner at (10,20).
1296
        //        |        placeOnScreen(node, {x: 10, y: 20}, ["TR", "BL"])
1297
1298
        var choices = dojo.map(corners, function(corner){
1299
                var c = { corner: corner, pos: {x:pos.x,y:pos.y} };
1300
                if(padding){
1301
                        c.pos.x += corner.charAt(1) == 'L' ? padding.x : -padding.x;
1302
                        c.pos.y += corner.charAt(0) == 'T' ? padding.y : -padding.y;
1303
                }
1304
                return c;
1305
        });
1306
1307
        return dijit._place(node, choices);
1308
}
1309
1310
dijit._place = function(/*DomNode*/ node, /* Array */ choices, /* Function */ layoutNode){
1311
        // summary:
1312
        //                Given a list of spots to put node, put it at the first spot where it fits,
1313
        //                of if it doesn't fit anywhere then the place with the least overflow
1314
        // choices: Array
1315
        //                Array of elements like: {corner: 'TL', pos: {x: 10, y: 20} }
1316
        //                Above example says to put the top-left corner of the node at (10,20)
1317
        // layoutNode: Function(node, aroundNodeCorner, nodeCorner)
1318
        //                for things like tooltip, they are displayed differently (and have different dimensions)
1319
        //                based on their orientation relative to the parent.   This adjusts the popup based on orientation.
1320
1321
        // get {x: 10, y: 10, w: 100, h:100} type obj representing position of
1322
        // viewport over document
1323
        var view = dojo.window.getBox();
1324
1325
        // This won't work if the node is inside a <div style="position: relative">,
1326
        // so reattach it to dojo.doc.body.   (Otherwise, the positioning will be wrong
1327
        // and also it might get cutoff)
1328
        if(!node.parentNode || String(node.parentNode.tagName).toLowerCase() != "body"){
1329
                dojo.body().appendChild(node);
1330
        }
1331
1332
        var best = null;
1333
        dojo.some(choices, function(choice){
1334
                var corner = choice.corner;
1335
                var pos = choice.pos;
1336
1337
                // configure node to be displayed in given position relative to button
1338
                // (need to do this in order to get an accurate size for the node, because
1339
                // a tooltips size changes based on position, due to triangle)
1340
                if(layoutNode){
1341
                        layoutNode(node, choice.aroundCorner, corner);
1342
                }
1343
1344
                // get node's size
1345
                var style = node.style;
1346
                var oldDisplay = style.display;
1347
                var oldVis = style.visibility;
1348
                style.visibility = "hidden";
1349
                style.display = "";
1350
                var mb = dojo.marginBox(node);
1351
                style.display = oldDisplay;
1352
                style.visibility = oldVis;
1353
1354
                // coordinates and size of node with specified corner placed at pos,
1355
                // and clipped by viewport
1356
                var startX = Math.max(view.l, corner.charAt(1) == 'L' ? pos.x : (pos.x - mb.w)),
1357
                        startY = Math.max(view.t, corner.charAt(0) == 'T' ? pos.y : (pos.y - mb.h)),
1358
                        endX = Math.min(view.l + view.w, corner.charAt(1) == 'L' ? (startX + mb.w) : pos.x),
1359
                        endY = Math.min(view.t + view.h, corner.charAt(0) == 'T' ? (startY + mb.h) : pos.y),
1360
                        width = endX - startX,
1361
                        height = endY - startY,
1362
                        overflow = (mb.w - width) + (mb.h - height);
1363
1364
                if(best == null || overflow < best.overflow){
1365
                        best = {
1366
                                corner: corner,
1367
                                aroundCorner: choice.aroundCorner,
1368
                                x: startX,
1369
                                y: startY,
1370
                                w: width,
1371
                                h: height,
1372
                                overflow: overflow
1373
                        };
1374
                }
1375
                return !overflow;
1376
        });
1377
1378
        node.style.left = best.x + "px";
1379
        node.style.top = best.y + "px";
1380
        if(best.overflow && layoutNode){
1381
                layoutNode(node, best.aroundCorner, best.corner);
1382
        }
1383
        return best;
1384
}
1385
1386
dijit.placeOnScreenAroundNode = function(
1387
        /* DomNode */                node,
1388
        /* DomNode */                aroundNode,
1389
        /* Object */                aroundCorners,
1390
        /* Function? */                layoutNode){
1391
1392
        // summary:
1393
        //                Position node adjacent or kitty-corner to aroundNode
1394
        //                such that it's fully visible in viewport.
1395
        //
1396
        // description:
1397
        //                Place node such that corner of node touches a corner of
1398
        //                aroundNode, and that node is fully visible.
1399
        //
1400
        // aroundCorners:
1401
        //                Ordered list of pairs of corners to try matching up.
1402
        //                Each pair of corners is represented as a key/value in the hash,
1403
        //                where the key corresponds to the aroundNode's corner, and
1404
        //                the value corresponds to the node's corner:
1405
        //
1406
        //        |        { aroundNodeCorner1: nodeCorner1, aroundNodeCorner2: nodeCorner2, ...}
1407
        //
1408
        //                The following strings are used to represent the four corners:
1409
        //                        * "BL" - bottom left
1410
        //                        * "BR" - bottom right
1411
        //                        * "TL" - top left
1412
        //                        * "TR" - top right
1413
        //
1414
        // layoutNode: Function(node, aroundNodeCorner, nodeCorner)
1415
        //                For things like tooltip, they are displayed differently (and have different dimensions)
1416
        //                based on their orientation relative to the parent.   This adjusts the popup based on orientation.
1417
        //
1418
        // example:
1419
        //        |        dijit.placeOnScreenAroundNode(node, aroundNode, {'BL':'TL', 'TR':'BR'});
1420
        //                This will try to position node such that node's top-left corner is at the same position
1421
        //                as the bottom left corner of the aroundNode (ie, put node below
1422
        //                aroundNode, with left edges aligned).  If that fails it will try to put
1423
        //                 the bottom-right corner of node where the top right corner of aroundNode is
1424
        //                (ie, put node above aroundNode, with right edges aligned)
1425
        //
1426
1427
        // get coordinates of aroundNode
1428
        aroundNode = dojo.byId(aroundNode);
1429
        var oldDisplay = aroundNode.style.display;
1430
        aroundNode.style.display="";
1431
        // #3172: use the slightly tighter border box instead of marginBox
1432
        var aroundNodePos = dojo.position(aroundNode, true);
1433
        aroundNode.style.display=oldDisplay;
1434
1435
        // place the node around the calculated rectangle
1436
        return dijit._placeOnScreenAroundRect(node,
1437
                aroundNodePos.x, aroundNodePos.y, aroundNodePos.w, aroundNodePos.h,        // rectangle
1438
                aroundCorners, layoutNode);
1439
};
1440
1441
/*=====
1442
dijit.__Rectangle = function(){
1443
        // x: Integer
1444
        //                horizontal offset in pixels, relative to document body
1445
        // y: Integer
1446
        //                vertical offset in pixels, relative to document body
1447
        // width: Integer
1448
        //                width in pixels
1449
        // height: Integer
1450
        //                height in pixels
1451

1452
        this.x = x;
1453
        this.y = y;
1454
        this.width = width;
1455
        this.height = height;
1456
}
1457
=====*/
1458
1459
1460
dijit.placeOnScreenAroundRectangle = function(
1461
        /* DomNode */                        node,
1462
        /* dijit.__Rectangle */        aroundRect,
1463
        /* Object */                        aroundCorners,
1464
        /* Function */                        layoutNode){
1465
1466
        // summary:
1467
        //                Like dijit.placeOnScreenAroundNode(), except that the "around"
1468
        //                parameter is an arbitrary rectangle on the screen (x, y, width, height)
1469
        //                instead of a dom node.
1470
1471
        return dijit._placeOnScreenAroundRect(node,
1472
                aroundRect.x, aroundRect.y, aroundRect.width, aroundRect.height,        // rectangle
1473
                aroundCorners, layoutNode);
1474
};
1475
1476
dijit._placeOnScreenAroundRect = function(
1477
        /* DomNode */                node,
1478
        /* Number */                x,
1479
        /* Number */                y,
1480
        /* Number */                width,
1481
        /* Number */                height,
1482
        /* Object */                aroundCorners,
1483
        /* Function */                layoutNode){
1484
1485
        // summary:
1486
        //                Like dijit.placeOnScreenAroundNode(), except it accepts coordinates
1487
        //                of a rectangle to place node adjacent to.
1488
1489
        // TODO: combine with placeOnScreenAroundRectangle()
1490
1491
        // Generate list of possible positions for node
1492
        var choices = [];
1493
        for(var nodeCorner in aroundCorners){
1494
                choices.push( {
1495
                        aroundCorner: nodeCorner,
1496
                        corner: aroundCorners[nodeCorner],
1497
                        pos: {
1498
                                x: x + (nodeCorner.charAt(1) == 'L' ? 0 : width),
1499
                                y: y + (nodeCorner.charAt(0) == 'T' ? 0 : height)
1500
                        }
1501
                });
1502
        }
1503
1504
        return dijit._place(node, choices, layoutNode);
1505
};
1506
1507
dijit.placementRegistry= new dojo.AdapterRegistry();
1508
dijit.placementRegistry.register("node",
1509
        function(n, x){
1510
                return typeof x == "object" &&
1511
                        typeof x.offsetWidth != "undefined" && typeof x.offsetHeight != "undefined";
1512
        },
1513
        dijit.placeOnScreenAroundNode);
1514
dijit.placementRegistry.register("rect",
1515
        function(n, x){
1516
                return typeof x == "object" &&
1517
                        "x" in x && "y" in x && "width" in x && "height" in x;
1518
        },
1519
        dijit.placeOnScreenAroundRectangle);
1520
1521
dijit.placeOnScreenAroundElement = function(
1522
        /* DomNode */                node,
1523
        /* Object */                aroundElement,
1524
        /* Object */                aroundCorners,
1525
        /* Function */                layoutNode){
1526
1527
        // summary:
1528
        //                Like dijit.placeOnScreenAroundNode(), except it accepts an arbitrary object
1529
        //                for the "around" argument and finds a proper processor to place a node.
1530
1531
        return dijit.placementRegistry.match.apply(dijit.placementRegistry, arguments);
1532
};
1533
1534
dijit.getPopupAroundAlignment = function(/*Array*/ position, /*Boolean*/ leftToRight){
1535
        // summary:
1536
        //                Transforms the passed array of preferred positions into a format suitable for passing as the aroundCorners argument to dijit.placeOnScreenAroundElement.
1537
        //
1538
        // position: String[]
1539
        //                This variable controls the position of the drop down.
1540
        //                It's an array of strings with the following values:
1541
        //
1542
        //                        * before: places drop down to the left of the target node/widget, or to the right in
1543
        //                          the case of RTL scripts like Hebrew and Arabic
1544
        //                        * after: places drop down to the right of the target node/widget, or to the left in
1545
        //                          the case of RTL scripts like Hebrew and Arabic
1546
        //                        * above: drop down goes above target node
1547
        //                        * below: drop down goes below target node
1548
        //
1549
        //                The list is positions is tried, in order, until a position is found where the drop down fits
1550
        //                within the viewport.
1551
        //
1552
        // leftToRight: Boolean
1553
        //                Whether the popup will be displaying in leftToRight mode.
1554
        //
1555
        var align = {};
1556
        dojo.forEach(position, function(pos){
1557
                switch(pos){
1558
                        case "after":
1559
                                align[leftToRight ? "BR" : "BL"] = leftToRight ? "BL" : "BR";
1560
                                break;
1561
                        case "before":
1562
                                align[leftToRight ? "BL" : "BR"] = leftToRight ? "BR" : "BL";
1563
                                break;
1564
                        case "below":
1565
                                // first try to align left borders, next try to align right borders (or reverse for RTL mode)
1566
                                align[leftToRight ? "BL" : "BR"] = leftToRight ? "TL" : "TR";
1567
                                align[leftToRight ? "BR" : "BL"] = leftToRight ? "TR" : "TL";
1568
                                break;
1569
                        case "above":
1570
                        default:
1571
                                // first try to align left borders, next try to align right borders (or reverse for RTL mode)
1572
                                align[leftToRight ? "TL" : "TR"] = leftToRight ? "BL" : "BR";
1573
                                align[leftToRight ? "TR" : "TL"] = leftToRight ? "BR" : "BL";
1574
                                break;
1575
                }
1576
        });
1577
        return align;
1578
};
1579
1580
}
1581
1582
if(!dojo._hasResource["dijit._base.window"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
1583
dojo._hasResource["dijit._base.window"] = true;
1584
dojo.provide("dijit._base.window");
1585
1586
1587
1588
dijit.getDocumentWindow = function(doc){
1589
        return dojo.window.get(doc);
1590
};
1591
1592
}
1593
1594
if(!dojo._hasResource["dijit._base.popup"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
1595
dojo._hasResource["dijit._base.popup"] = true;
1596
dojo.provide("dijit._base.popup");
1597
1598
1599
1600
1601
1602
/*=====
1603
dijit.popup.__OpenArgs = function(){
1604
        // popup: Widget
1605
        //                widget to display
1606
        // parent: Widget
1607
        //                the button etc. that is displaying this popup
1608
        // around: DomNode
1609
        //                DOM node (typically a button); place popup relative to this node.  (Specify this *or* "x" and "y" parameters.)
1610
        // x: Integer
1611
        //                Absolute horizontal position (in pixels) to place node at.  (Specify this *or* "around" parameter.)
1612
        // y: Integer
1613
        //                Absolute vertical position (in pixels) to place node at.  (Specify this *or* "around" parameter.)
1614
        // orient: Object|String
1615
        //                When the around parameter is specified, orient should be an
1616
        //                ordered list of tuples of the form (around-node-corner, popup-node-corner).
1617
        //                dijit.popup.open() tries to position the popup according to each tuple in the list, in order,
1618
        //                until the popup appears fully within the viewport.
1619
        //
1620
        //                The default value is {BL:'TL', TL:'BL'}, which represents a list of two tuples:
1621
        //                        1. (BL, TL)
1622
        //                        2. (TL, BL)
1623
        //                where BL means "bottom left" and "TL" means "top left".
1624
        //                So by default, it first tries putting the popup below the around node, left-aligning them,
1625
        //                and then tries to put it above the around node, still left-aligning them.   Note that the
1626
        //                default is horizontally reversed when in RTL mode.
1627
        //
1628
        //                When an (x,y) position is specified rather than an around node, orient is either
1629
        //                "R" or "L".  R (for right) means that it tries to put the popup to the right of the mouse,
1630
        //                specifically positioning the popup's top-right corner at the mouse position, and if that doesn't
1631
        //                fit in the viewport, then it tries, in order, the bottom-right corner, the top left corner,
1632
        //                and the top-right corner.
1633
        // onCancel: Function
1634
        //                callback when user has canceled the popup by
1635
        //                        1. hitting ESC or
1636
        //                        2. by using the popup widget's proprietary cancel mechanism (like a cancel button in a dialog);
1637
        //                           i.e. whenever popupWidget.onCancel() is called, args.onCancel is called
1638
        // onClose: Function
1639
        //                callback whenever this popup is closed
1640
        // onExecute: Function
1641
        //                callback when user "executed" on the popup/sub-popup by selecting a menu choice, etc. (top menu only)
1642
        // padding: dijit.__Position
1643
        //                adding a buffer around the opening position. This is only useful when around is not set.
1644
        this.popup = popup;
1645
        this.parent = parent;
1646
        this.around = around;
1647
        this.x = x;
1648
        this.y = y;
1649
        this.orient = orient;
1650
        this.onCancel = onCancel;
1651
        this.onClose = onClose;
1652
        this.onExecute = onExecute;
1653
        this.padding = padding;
1654
}
1655
=====*/
1656
1657
dijit.popup = {
1658
        // summary:
1659
        //                This singleton is used to show/hide widgets as popups.
1660
1661
        // _stack: dijit._Widget[]
1662
        //                Stack of currently popped up widgets.
1663
        //                (someone opened _stack[0], and then it opened _stack[1], etc.)
1664
        _stack: [],
1665
1666
        // _beginZIndex: Number
1667
        //                Z-index of the first popup.   (If first popup opens other
1668
        //                popups they get a higher z-index.)
1669
        _beginZIndex: 1000,
1670
1671
        _idGen: 1,
1672
1673
        moveOffScreen: function(/*DomNode*/ node){
1674
                // summary:
1675
                //                Initialization for nodes that will be used as popups
1676
                //
1677
                // description:
1678
                //                Puts node inside a wrapper <div>, and
1679
                //                positions wrapper div off screen, but not display:none, so that
1680
                //                the widget doesn't appear in the page flow and/or cause a blank
1681
                //                area at the bottom of the viewport (making scrollbar longer), but
1682
                //                initialization of contained widgets works correctly
1683
1684
                var wrapper = node.parentNode;
1685
1686
                // Create a wrapper widget for when this node (in the future) will be used as a popup.
1687
                // This is done early because of IE bugs where creating/moving DOM nodes causes focus
1688
                // to go wonky, see tests/robot/Toolbar.html to reproduce
1689
                if(!wrapper || !dojo.hasClass(wrapper, "dijitPopup")){
1690
                        wrapper = dojo.create("div",{
1691
                                "class":"dijitPopup",
1692
                                style:{
1693
                                        visibility:"hidden",
1694
                                        top: "-9999px"
1695
                                }
1696
                        }, dojo.body());
1697
                        dijit.setWaiRole(wrapper, "presentation");
1698
                        wrapper.appendChild(node);
1699
                }
1700
1701
1702
                var s = node.style;
1703
                s.display = "";
1704
                s.visibility = "";
1705
                s.position = "";
1706
                s.top = "0px";
1707
1708
                dojo.style(wrapper, {
1709
                        visibility: "hidden",
1710
                        // prevent transient scrollbar causing misalign (#5776), and initial flash in upper left (#10111)
1711
                        top: "-9999px"
1712
                });
1713
        },
1714
1715
        getTopPopup: function(){
1716
                // summary:
1717
                //                Compute the closest ancestor popup that's *not* a child of another popup.
1718
                //                Ex: For a TooltipDialog with a button that spawns a tree of menus, find the popup of the button.
1719
                var stack = this._stack;
1720
                for(var pi=stack.length-1; pi > 0 && stack[pi].parent === stack[pi-1].widget; pi--){
1721
                        /* do nothing, just trying to get right value for pi */
1722
                }
1723
                return stack[pi];
1724
        },
1725
1726
        open: function(/*dijit.popup.__OpenArgs*/ args){
1727
                // summary:
1728
                //                Popup the widget at the specified position
1729
                //
1730
                // example:
1731
                //                opening at the mouse position
1732
                //                |                dijit.popup.open({popup: menuWidget, x: evt.pageX, y: evt.pageY});
1733
                //
1734
                // example:
1735
                //                opening the widget as a dropdown
1736
                //                |                dijit.popup.open({parent: this, popup: menuWidget, around: this.domNode, onClose: function(){...}});
1737
                //
1738
                //                Note that whatever widget called dijit.popup.open() should also listen to its own _onBlur callback
1739
                //                (fired from _base/focus.js) to know that focus has moved somewhere else and thus the popup should be closed.
1740
1741
                var stack = this._stack,
1742
                        widget = args.popup,
1743
                        orient = args.orient || (
1744
                                (args.parent ? args.parent.isLeftToRight() : dojo._isBodyLtr()) ?
1745
                                {'BL':'TL', 'BR':'TR', 'TL':'BL', 'TR':'BR'} :
1746
                                {'BR':'TR', 'BL':'TL', 'TR':'BR', 'TL':'BL'}
1747
                        ),
1748
                        around = args.around,
1749
                        id = (args.around && args.around.id) ? (args.around.id+"_dropdown") : ("popup_"+this._idGen++);
1750
1751
1752
                // The wrapper may have already been created, but in case it wasn't, create here
1753
                var wrapper = widget.domNode.parentNode;
1754
                if(!wrapper || !dojo.hasClass(wrapper, "dijitPopup")){
1755
                        this.moveOffScreen(widget.domNode);
1756
                        wrapper = widget.domNode.parentNode;
1757
                }
1758
1759
                dojo.attr(wrapper, {
1760
                        id: id,
1761
                        style: {
1762
                                zIndex: this._beginZIndex + stack.length
1763
                        },
1764
                        "class": "dijitPopup " + (widget.baseClass || widget["class"] || "").split(" ")[0] +"Popup",
1765
                        dijitPopupParent: args.parent ? args.parent.id : ""
1766
                });
1767
1768
                if(dojo.isIE || dojo.isMoz){
1769
                        var iframe = wrapper.childNodes[1];
1770
                        if(!iframe){
1771
                                iframe = new dijit.BackgroundIframe(wrapper);
1772
                        }
1773
                }
1774
1775
                // position the wrapper node and make it visible
1776
                var best = around ?
1777
                        dijit.placeOnScreenAroundElement(wrapper, around, orient, widget.orient ? dojo.hitch(widget, "orient") : null) :
1778
                        dijit.placeOnScreen(wrapper, args, orient == 'R' ? ['TR','BR','TL','BL'] : ['TL','BL','TR','BR'], args.padding);
1779
1780
                wrapper.style.visibility = "visible";
1781
                widget.domNode.style.visibility = "visible";        // counteract effects from _HasDropDown
1782
1783
                var handlers = [];
1784
1785
                // provide default escape and tab key handling
1786
                // (this will work for any widget, not just menu)
1787
                handlers.push(dojo.connect(wrapper, "onkeypress", this, function(evt){
1788
                        if(evt.charOrCode == dojo.keys.ESCAPE && args.onCancel){
1789
                                dojo.stopEvent(evt);
1790
                                args.onCancel();
1791
                        }else if(evt.charOrCode === dojo.keys.TAB){
1792
                                dojo.stopEvent(evt);
1793
                                var topPopup = this.getTopPopup();
1794
                                if(topPopup && topPopup.onCancel){
1795
                                        topPopup.onCancel();
1796
                                }
1797
                        }
1798
                }));
1799
1800
                // watch for cancel/execute events on the popup and notify the caller
1801
                // (for a menu, "execute" means clicking an item)
1802
                if(widget.onCancel){
1803
                        handlers.push(dojo.connect(widget, "onCancel", args.onCancel));
1804
                }
1805
1806
                handlers.push(dojo.connect(widget, widget.onExecute ? "onExecute" : "onChange", this, function(){
1807
                        var topPopup = this.getTopPopup();
1808
                        if(topPopup && topPopup.onExecute){
1809
                                topPopup.onExecute();
1810
                        }
1811
                }));
1812
1813
                stack.push({
1814
                        wrapper: wrapper,
1815
                        iframe: iframe,
1816
                        widget: widget,
1817
                        parent: args.parent,
1818
                        onExecute: args.onExecute,
1819
                        onCancel: args.onCancel,
1820
                         onClose: args.onClose,
1821
                        handlers: handlers
1822
                });
1823
1824
                if(widget.onOpen){
1825
                        // TODO: in 2.0 standardize onShow() (used by StackContainer) and onOpen() (used here)
1826
                        widget.onOpen(best);
1827
                }
1828
1829
                return best;
1830
        },
1831
1832
        close: function(/*dijit._Widget*/ popup){
1833
                // summary:
1834
                //                Close specified popup and any popups that it parented
1835
1836
                var stack = this._stack;
1837
1838
                // Basically work backwards from the top of the stack closing popups
1839
                // until we hit the specified popup, but IIRC there was some issue where closing
1840
                // a popup would cause others to close too.  Thus if we are trying to close B in [A,B,C]
1841
                // closing C might close B indirectly and then the while() condition will run where stack==[A]...
1842
                // so the while condition is constructed defensively.
1843
                while(dojo.some(stack, function(elem){return elem.widget == popup;})){
1844
                        var top = stack.pop(),
1845
                                wrapper = top.wrapper,
1846
                                iframe = top.iframe,
1847
                                widget = top.widget,
1848
                                onClose = top.onClose;
1849
1850
                        if(widget.onClose){
1851
                                // TODO: in 2.0 standardize onHide() (used by StackContainer) and onClose() (used here)
1852
                                widget.onClose();
1853
                        }
1854
                        dojo.forEach(top.handlers, dojo.disconnect);
1855
1856
                        // Move the widget plus it's wrapper off screen, unless it has already been destroyed in above onClose() etc.
1857
                        if(widget && widget.domNode){
1858
                                this.moveOffScreen(widget.domNode);
1859
                        }else{
1860
                                dojo.destroy(wrapper);
1861
                        }
1862
1863
                        if(onClose){
1864
                                onClose();
1865
                        }
1866
                }
1867
        }
1868
};
1869
1870
dijit._frames = new function(){
1871
        // summary:
1872
        //                cache of iframes
1873
        var queue = [];
1874
1875
        this.pop = function(){
1876
                var iframe;
1877
                if(queue.length){
1878
                        iframe = queue.pop();
1879
                        iframe.style.display="";
1880
                }else{
1881
                        if(dojo.isIE){
1882
                                var burl = dojo.config["dojoBlankHtmlUrl"] || (dojo.moduleUrl("dojo", "resources/blank.html")+"") || "javascript:\"\"";
1883
                                var html="<iframe src='" + burl + "'"
1884
                                        + " style='position: absolute; left: 0px; top: 0px;"
1885
                                        + "z-index: -1; filter:Alpha(Opacity=\"0\");'>";
1886
                                iframe = dojo.doc.createElement(html);
1887
                        }else{
1888
                                 iframe = dojo.create("iframe");
1889
                                iframe.src = 'javascript:""';
1890
                                iframe.className = "dijitBackgroundIframe";
1891
                                dojo.style(iframe, "opacity", 0.1);
1892
                        }
1893
                        iframe.tabIndex = -1; // Magic to prevent iframe from getting focus on tab keypress - as style didnt work.
1894
                        dijit.setWaiRole(iframe,"presentation");
1895
                }
1896
                return iframe;
1897
        };
1898
1899
        this.push = function(iframe){
1900
                iframe.style.display="none";
1901
                queue.push(iframe);
1902
        }
1903
}();
1904
1905
1906
dijit.BackgroundIframe = function(/* DomNode */node){
1907
        // summary:
1908
        //                For IE/FF z-index schenanigans. id attribute is required.
1909
        //
1910
        // description:
1911
        //                new dijit.BackgroundIframe(node)
1912
        //                        Makes a background iframe as a child of node, that fills
1913
        //                        area (and position) of node
1914
1915
        if(!node.id){ throw new Error("no id"); }
1916
        if(dojo.isIE || dojo.isMoz){
1917
                var iframe = dijit._frames.pop();
1918
                node.appendChild(iframe);
1919
                if(dojo.isIE<7){
1920
                        this.resize(node);
1921
                        this._conn = dojo.connect(node, 'onresize', this, function(){
1922
                                this.resize(node);
1923
                        });
1924
                }else{
1925
                        dojo.style(iframe, {
1926
                                width: '100%',
1927
                                height: '100%'
1928
                        });
1929
                }
1930
                this.iframe = iframe;
1931
        }
1932
};
1933
1934
dojo.extend(dijit.BackgroundIframe, {
1935
        resize: function(node){
1936
                // summary:
1937
                //                 resize the iframe so its the same size as node
1938
                // description:
1939
                //                this function is a no-op in all browsers except
1940
                //                IE6, which does not support 100% width/height
1941
                //                of absolute positioned iframes
1942
                if(this.iframe && dojo.isIE<7){
1943
                        dojo.style(this.iframe, {
1944
                                width: node.offsetWidth + 'px',
1945
                                height: node.offsetHeight + 'px'
1946
                        });
1947
                }
1948
        },
1949
        destroy: function(){
1950
                // summary:
1951
                //                destroy the iframe
1952
                if(this._conn){
1953
                        dojo.disconnect(this._conn);
1954
                        this._conn = null;
1955
                }
1956
                if(this.iframe){
1957
                        dijit._frames.push(this.iframe);
1958
                        delete this.iframe;
1959
                }
1960
        }
1961
});
1962
1963
}
1964
1965
if(!dojo._hasResource["dijit._base.scroll"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
1966
dojo._hasResource["dijit._base.scroll"] = true;
1967
dojo.provide("dijit._base.scroll");
1968
1969
1970
1971
dijit.scrollIntoView = function(/*DomNode*/ node, /*Object?*/ pos){
1972
        // summary:
1973
        //                Scroll the passed node into view, if it is not already.
1974
        //                Deprecated, use `dojo.window.scrollIntoView` instead.
1975
1976
        dojo.window.scrollIntoView(node, pos);
1977
};
1978
1979
}
1980
1981
if(!dojo._hasResource["dojo.uacss"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
1982
dojo._hasResource["dojo.uacss"] = true;
1983
dojo.provide("dojo.uacss");
1984
1985
(function(){
1986
        // summary:
1987
        //                Applies pre-set CSS classes to the top-level HTML node, based on:
1988
        //                         - browser (ex: dj_ie)
1989
        //                        - browser version (ex: dj_ie6)
1990
        //                        - box model (ex: dj_contentBox)
1991
        //                        - text direction (ex: dijitRtl)
1992
        //
1993
        //                In addition, browser, browser version, and box model are
1994
        //                combined with an RTL flag when browser text is RTL.  ex: dj_ie-rtl.
1995
1996
        var d = dojo,
1997
                html = d.doc.documentElement,
1998
                ie = d.isIE,
1999
                opera = d.isOpera,
2000
                maj = Math.floor,
2001
                ff = d.isFF,
2002
                boxModel = d.boxModel.replace(/-/,''),
2003
2004
                classes = {
2005
                        dj_ie: ie,
2006
                        dj_ie6: maj(ie) == 6,
2007
                        dj_ie7: maj(ie) == 7,
2008
                        dj_ie8: maj(ie) == 8,
2009
                        dj_quirks: d.isQuirks,
2010
                        dj_iequirks: ie && d.isQuirks,
2011
2012
                        // NOTE: Opera not supported by dijit
2013
                        dj_opera: opera,
2014
2015
                        dj_khtml: d.isKhtml,
2016
2017
                        dj_webkit: d.isWebKit,
2018
                        dj_safari: d.isSafari,
2019
                        dj_chrome: d.isChrome,
2020
2021
                        dj_gecko: d.isMozilla,
2022
                        dj_ff3: maj(ff) == 3
2023
                }; // no dojo unsupported browsers
2024
2025
        classes["dj_" + boxModel] = true;
2026
2027
        // apply browser, browser version, and box model class names
2028
        var classStr = "";
2029
        for(var clz in classes){
2030
                if(classes[clz]){
2031
                        classStr += clz + " ";
2032
                }
2033
        }
2034
        html.className = d.trim(html.className + " " + classStr);
2035
2036
        // If RTL mode, then add dj_rtl flag plus repeat existing classes with -rtl extension.
2037
        // We can't run the code below until the <body> tag has loaded (so we can check for dir=rtl).
2038
        // Unshift() is to run sniff code before the parser.
2039
        dojo._loaders.unshift(function(){
2040
                if(!dojo._isBodyLtr()){
2041
                        var rtlClassStr = "dj_rtl dijitRtl " + classStr.replace(/ /g, "-rtl ")
2042
                        html.className = d.trim(html.className + " " + rtlClassStr);
2043
                }
2044
        });
2045
})();
2046
2047
}
2048
2049
if(!dojo._hasResource["dijit._base.sniff"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
2050
dojo._hasResource["dijit._base.sniff"] = true;
2051
// summary:
2052
//                Applies pre-set CSS classes to the top-level HTML node, see
2053
//                `dojo.uacss` for details.
2054
//
2055
//                Simply doing a require on this module will
2056
//                establish this CSS.  Modified version of Morris' CSS hack.
2057
2058
dojo.provide("dijit._base.sniff");
2059
2060
2061
2062
}
2063
2064
if(!dojo._hasResource["dijit._base.typematic"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
2065
dojo._hasResource["dijit._base.typematic"] = true;
2066
dojo.provide("dijit._base.typematic");
2067
2068
dijit.typematic = {
2069
        // summary:
2070
        //                These functions are used to repetitively call a user specified callback
2071
        //                method when a specific key or mouse click over a specific DOM node is
2072
        //                held down for a specific amount of time.
2073
        //                Only 1 such event is allowed to occur on the browser page at 1 time.
2074
2075
        _fireEventAndReload: function(){
2076
                this._timer = null;
2077
                this._callback(++this._count, this._node, this._evt);
2078
2079
                // Schedule next event, timer is at most minDelay (default 10ms) to avoid
2080
                // browser overload (particularly avoiding starving DOH robot so it never gets to send a mouseup)
2081
                this._currentTimeout = Math.max(
2082
                        this._currentTimeout < 0 ? this._initialDelay :
2083
                                (this._subsequentDelay > 1 ? this._subsequentDelay : Math.round(this._currentTimeout * this._subsequentDelay)),
2084
                        this._minDelay);
2085
                this._timer = setTimeout(dojo.hitch(this, "_fireEventAndReload"), this._currentTimeout);
2086
        },
2087
2088
        trigger: function(/*Event*/ evt, /*Object*/ _this, /*DOMNode*/ node, /*Function*/ callback, /*Object*/ obj, /*Number*/ subsequentDelay, /*Number*/ initialDelay, /*Number?*/ minDelay){
2089
                // summary:
2090
                //                Start a timed, repeating callback sequence.
2091
                //                If already started, the function call is ignored.
2092
                //                This method is not normally called by the user but can be
2093
                //                when the normal listener code is insufficient.
2094
                // evt:
2095
                //                key or mouse event object to pass to the user callback
2096
                // _this:
2097
                //                pointer to the user's widget space.
2098
                // node:
2099
                //                the DOM node object to pass the the callback function
2100
                // callback:
2101
                //                function to call until the sequence is stopped called with 3 parameters:
2102
                // count:
2103
                //                integer representing number of repeated calls (0..n) with -1 indicating the iteration has stopped
2104
                // node:
2105
                //                the DOM node object passed in
2106
                // evt:
2107
                //                key or mouse event object
2108
                // obj:
2109
                //                user space object used to uniquely identify each typematic sequence
2110
                // subsequentDelay (optional):
2111
                //                if > 1, the number of milliseconds until the 3->n events occur
2112
                //                or else the fractional time multiplier for the next event's delay, default=0.9
2113
                // initialDelay (optional):
2114
                //                the number of milliseconds until the 2nd event occurs, default=500ms
2115
                // minDelay (optional):
2116
                //                the maximum delay in milliseconds for event to fire, default=10ms
2117
                if(obj != this._obj){
2118
                        this.stop();
2119
                        this._initialDelay = initialDelay || 500;
2120
                        this._subsequentDelay = subsequentDelay || 0.90;
2121
                        this._minDelay = minDelay || 10;
2122
                        this._obj = obj;
2123
                        this._evt = evt;
2124
                        this._node = node;
2125
                        this._currentTimeout = -1;
2126
                        this._count = -1;
2127
                        this._callback = dojo.hitch(_this, callback);
2128
                        this._fireEventAndReload();
2129
                        this._evt = dojo.mixin({faux: true}, evt);
2130
                }
2131
        },
2132
2133
        stop: function(){
2134
                // summary:
2135
                //                Stop an ongoing timed, repeating callback sequence.
2136
                if(this._timer){
2137
                        clearTimeout(this._timer);
2138
                        this._timer = null;
2139
                }
2140
                if(this._obj){
2141
                        this._callback(-1, this._node, this._evt);
2142
                        this._obj = null;
2143
                }
2144
        },
2145
2146
        addKeyListener: function(/*DOMNode*/ node, /*Object*/ keyObject, /*Object*/ _this, /*Function*/ callback, /*Number*/ subsequentDelay, /*Number*/ initialDelay, /*Number?*/ minDelay){
2147
                // summary:
2148
                //                Start listening for a specific typematic key.
2149
                //                See also the trigger method for other parameters.
2150
                // keyObject:
2151
                //                an object defining the key to listen for:
2152
                //                 charOrCode:
2153
                //                        the printable character (string) or keyCode (number) to listen for.
2154
                //                 keyCode:
2155
                //                        (deprecated - use charOrCode) the keyCode (number) to listen for (implies charCode = 0).
2156
                //                 charCode:
2157
                //                        (deprecated - use charOrCode) the charCode (number) to listen for.
2158
                //                 ctrlKey:
2159
                //                        desired ctrl key state to initiate the callback sequence:
2160
                //                        - pressed (true)
2161
                //                        - released (false)
2162
                //                        - either (unspecified)
2163
                //                 altKey:
2164
                //                        same as ctrlKey but for the alt key
2165
                //                 shiftKey:
2166
                //                        same as ctrlKey but for the shift key
2167
                // returns:
2168
                //                an array of dojo.connect handles
2169
                if(keyObject.keyCode){
2170
                        keyObject.charOrCode = keyObject.keyCode;
2171
                        dojo.deprecated("keyCode attribute parameter for dijit.typematic.addKeyListener is deprecated. Use charOrCode instead.", "", "2.0");
2172
                }else if(keyObject.charCode){
2173
                        keyObject.charOrCode = String.fromCharCode(keyObject.charCode);
2174
                        dojo.deprecated("charCode attribute parameter for dijit.typematic.addKeyListener is deprecated. Use charOrCode instead.", "", "2.0");
2175
                }
2176
                return [
2177
                        dojo.connect(node, "onkeypress", this, function(evt){
2178
                                if(evt.charOrCode == keyObject.charOrCode &&
2179
                                (keyObject.ctrlKey === undefined || keyObject.ctrlKey == evt.ctrlKey) &&
2180
                                (keyObject.altKey === undefined || keyObject.altKey == evt.altKey) &&
2181
                                (keyObject.metaKey === undefined || keyObject.metaKey == (evt.metaKey || false)) && // IE doesn't even set metaKey
2182
                                (keyObject.shiftKey === undefined || keyObject.shiftKey == evt.shiftKey)){
2183
                                        dojo.stopEvent(evt);
2184
                                        dijit.typematic.trigger(evt, _this, node, callback, keyObject, subsequentDelay, initialDelay, minDelay);
2185
                                }else if(dijit.typematic._obj == keyObject){
2186
                                        dijit.typematic.stop();
2187
                                }
2188
                        }),
2189
                        dojo.connect(node, "onkeyup", this, function(evt){
2190
                                if(dijit.typematic._obj == keyObject){
2191
                                        dijit.typematic.stop();
2192
                                }
2193
                        })
2194
                ];
2195
        },
2196
2197
        addMouseListener: function(/*DOMNode*/ node, /*Object*/ _this, /*Function*/ callback, /*Number*/ subsequentDelay, /*Number*/ initialDelay, /*Number?*/ minDelay){
2198
                // summary:
2199
                //                Start listening for a typematic mouse click.
2200
                //                See the trigger method for other parameters.
2201
                // returns:
2202
                //                an array of dojo.connect handles
2203
                var dc = dojo.connect;
2204
                return [
2205
                        dc(node, "mousedown", this, function(evt){
2206
                                dojo.stopEvent(evt);
2207
                                dijit.typematic.trigger(evt, _this, node, callback, node, subsequentDelay, initialDelay, minDelay);
2208
                        }),
2209
                        dc(node, "mouseup", this, function(evt){
2210
                                dojo.stopEvent(evt);
2211
                                dijit.typematic.stop();
2212
                        }),
2213
                        dc(node, "mouseout", this, function(evt){
2214
                                dojo.stopEvent(evt);
2215
                                dijit.typematic.stop();
2216
                        }),
2217
                        dc(node, "mousemove", this, function(evt){
2218
                                evt.preventDefault();
2219
                        }),
2220
                        dc(node, "dblclick", this, function(evt){
2221
                                dojo.stopEvent(evt);
2222
                                if(dojo.isIE){
2223
                                        dijit.typematic.trigger(evt, _this, node, callback, node, subsequentDelay, initialDelay, minDelay);
2224
                                        setTimeout(dojo.hitch(this, dijit.typematic.stop), 50);
2225
                                }
2226
                        })
2227
                ];
2228
        },
2229
2230
        addListener: function(/*Node*/ mouseNode, /*Node*/ keyNode, /*Object*/ keyObject, /*Object*/ _this, /*Function*/ callback, /*Number*/ subsequentDelay, /*Number*/ initialDelay, /*Number?*/ minDelay){
2231
                // summary:
2232
                //                Start listening for a specific typematic key and mouseclick.
2233
                //                This is a thin wrapper to addKeyListener and addMouseListener.
2234
                //                See the addMouseListener and addKeyListener methods for other parameters.
2235
                // mouseNode:
2236
                //                the DOM node object to listen on for mouse events.
2237
                // keyNode:
2238
                //                the DOM node object to listen on for key events.
2239
                // returns:
2240
                //                an array of dojo.connect handles
2241
                return this.addKeyListener(keyNode, keyObject, _this, callback, subsequentDelay, initialDelay, minDelay).concat(
2242
                        this.addMouseListener(mouseNode, _this, callback, subsequentDelay, initialDelay, minDelay));
2243
        }
2244
};
2245
2246
}
2247
2248
if(!dojo._hasResource["dijit._base.wai"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
2249
dojo._hasResource["dijit._base.wai"] = true;
2250
dojo.provide("dijit._base.wai");
2251
2252
dijit.wai = {
2253
        onload: function(){
2254
                // summary:
2255
                //                Detects if we are in high-contrast mode or not
2256
2257
                // This must be a named function and not an anonymous
2258
                // function, so that the widget parsing code can make sure it
2259
                // registers its onload function after this function.
2260
                // DO NOT USE "this" within this function.
2261
2262
                // create div for testing if high contrast mode is on or images are turned off
2263
                var div = dojo.create("div",{
2264
                        id: "a11yTestNode",
2265
                        style:{
2266
                                cssText:'border: 1px solid;'
2267
                                        + 'border-color:red green;'
2268
                                        + 'position: absolute;'
2269
                                        + 'height: 5px;'
2270
                                        + 'top: -999px;'
2271
                                        + 'background-image: url("' + (dojo.config.blankGif || dojo.moduleUrl("dojo", "resources/blank.gif")) + '");'
2272
                        }
2273
                }, dojo.body());
2274
2275
                // test it
2276
                var cs = dojo.getComputedStyle(div);
2277
                if(cs){
2278
                        var bkImg = cs.backgroundImage;
2279
                        var needsA11y = (cs.borderTopColor == cs.borderRightColor) || (bkImg != null && (bkImg == "none" || bkImg == "url(invalid-url:)" ));
2280
                        dojo[needsA11y ? "addClass" : "removeClass"](dojo.body(), "dijit_a11y");
2281
                        if(dojo.isIE){
2282
                                div.outerHTML = "";                // prevent mixed-content warning, see http://support.microsoft.com/kb/925014
2283
                        }else{
2284
                                dojo.body().removeChild(div);
2285
                        }
2286
                }
2287
        }
2288
};
2289
2290
// Test if computer is in high contrast mode.
2291
// Make sure the a11y test runs first, before widgets are instantiated.
2292
if(dojo.isIE || dojo.isMoz){        // NOTE: checking in Safari messes things up
2293
        dojo._loaders.unshift(dijit.wai.onload);
2294
}
2295
2296
dojo.mixin(dijit, {
2297
        _XhtmlRoles: /banner|contentinfo|definition|main|navigation|search|note|secondary|seealso/,
2298
2299
        hasWaiRole: function(/*Element*/ elem, /*String*/ role){
2300
                // summary:
2301
                //                Determines if an element has a particular non-XHTML role.
2302
                // returns:
2303
                //                True if elem has the specific non-XHTML role attribute and false if not.
2304
                //                 For backwards compatibility if role parameter not provided,
2305
                //                 returns true if has non XHTML role
2306
                var waiRole = this.getWaiRole(elem);
2307
                return role ? (waiRole.indexOf(role) > -1) : (waiRole.length > 0);
2308
        },
2309
2310
        getWaiRole: function(/*Element*/ elem){
2311
                // summary:
2312
                //                Gets the non-XHTML role for an element (which should be a wai role).
2313
                // returns:
2314
                //                The non-XHTML role of elem or an empty string if elem
2315
                //                does not have a role.
2316
                 return dojo.trim((dojo.attr(elem, "role") || "").replace(this._XhtmlRoles,"").replace("wairole:",""));
2317
        },
2318
2319
        setWaiRole: function(/*Element*/ elem, /*String*/ role){
2320
                // summary:
2321
                //                Sets the role on an element.
2322
                // description:
2323
                //                Replace existing role attribute with new role.
2324
                //                If elem already has an XHTML role, append this role to XHTML role
2325
                //                and remove other ARIA roles.
2326
2327
                var curRole = dojo.attr(elem, "role") || "";
2328
                if(!this._XhtmlRoles.test(curRole)){
2329
                        dojo.attr(elem, "role", role);
2330
                }else{
2331
                        if((" "+ curRole +" ").indexOf(" " + role + " ") < 0){
2332
                                var clearXhtml = dojo.trim(curRole.replace(this._XhtmlRoles, ""));
2333
                                var cleanRole = dojo.trim(curRole.replace(clearXhtml, ""));
2334
                                dojo.attr(elem, "role", cleanRole + (cleanRole ? ' ' : '') + role);
2335
                        }
2336
                }
2337
        },
2338
2339
        removeWaiRole: function(/*Element*/ elem, /*String*/ role){
2340
                // summary:
2341
                //                Removes the specified non-XHTML role from an element.
2342
                //                 Removes role attribute if no specific role provided (for backwards compat.)
2343
2344
                var roleValue = dojo.attr(elem, "role");
2345
                if(!roleValue){ return; }
2346
                if(role){
2347
                        var t = dojo.trim((" " + roleValue + " ").replace(" " + role + " ", " "));
2348
                        dojo.attr(elem, "role", t);
2349
                }else{
2350
                        elem.removeAttribute("role");
2351
                }
2352
        },
2353
2354
        hasWaiState: function(/*Element*/ elem, /*String*/ state){
2355
                // summary:
2356
                //                Determines if an element has a given state.
2357
                // description:
2358
                //                Checks for an attribute called "aria-"+state.
2359
                // returns:
2360
                //                true if elem has a value for the given state and
2361
                //                false if it does not.
2362
2363
                return elem.hasAttribute ? elem.hasAttribute("aria-"+state) : !!elem.getAttribute("aria-"+state);
2364
        },
2365
2366
        getWaiState: function(/*Element*/ elem, /*String*/ state){
2367
                // summary:
2368
                //                Gets the value of a state on an element.
2369
                // description:
2370
                //                Checks for an attribute called "aria-"+state.
2371
                // returns:
2372
                //                The value of the requested state on elem
2373
                //                or an empty string if elem has no value for state.
2374
2375
                return elem.getAttribute("aria-"+state) || "";
2376
        },
2377
2378
        setWaiState: function(/*Element*/ elem, /*String*/ state, /*String*/ value){
2379
                // summary:
2380
                //                Sets a state on an element.
2381
                // description:
2382
                //                Sets an attribute called "aria-"+state.
2383
2384
                elem.setAttribute("aria-"+state, value);
2385
        },
2386
2387
        removeWaiState: function(/*Element*/ elem, /*String*/ state){
2388
                // summary:
2389
                //                Removes a state from an element.
2390
                // description:
2391
                //                Sets an attribute called "aria-"+state.
2392
2393
                elem.removeAttribute("aria-"+state);
2394
        }
2395
});
2396
2397
}
2398
2399
if(!dojo._hasResource["dijit._base"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
2400
dojo._hasResource["dijit._base"] = true;
2401
dojo.provide("dijit._base");
2402
2403
2404
2405
2406
2407
2408
2409
2410
2411
2412
2413
}
2414
2415
if(!dojo._hasResource["dojo.date.stamp"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
2416
dojo._hasResource["dojo.date.stamp"] = true;
2417
dojo.provide("dojo.date.stamp");
2418
2419
// Methods to convert dates to or from a wire (string) format using well-known conventions
2420
2421
dojo.date.stamp.fromISOString = function(/*String*/formattedString, /*Number?*/defaultTime){
2422
        //        summary:
2423
        //                Returns a Date object given a string formatted according to a subset of the ISO-8601 standard.
2424
        //
2425
        //        description:
2426
        //                Accepts a string formatted according to a profile of ISO8601 as defined by
2427
        //                [RFC3339](http://www.ietf.org/rfc/rfc3339.txt), except that partial input is allowed.
2428
        //                Can also process dates as specified [by the W3C](http://www.w3.org/TR/NOTE-datetime)
2429
        //                The following combinations are valid:
2430
        //
2431
        //                        * dates only
2432
        //                        |        * yyyy
2433
        //                        |        * yyyy-MM
2434
        //                        |        * yyyy-MM-dd
2435
        //                         * times only, with an optional time zone appended
2436
        //                        |        * THH:mm
2437
        //                        |        * THH:mm:ss
2438
        //                        |        * THH:mm:ss.SSS
2439
        //                         * and "datetimes" which could be any combination of the above
2440
        //
2441
        //                timezones may be specified as Z (for UTC) or +/- followed by a time expression HH:mm
2442
        //                Assumes the local time zone if not specified.  Does not validate.  Improperly formatted
2443
        //                input may return null.  Arguments which are out of bounds will be handled
2444
        //                 by the Date constructor (e.g. January 32nd typically gets resolved to February 1st)
2445
        //                Only years between 100 and 9999 are supported.
2446
        //
2447
          //        formattedString:
2448
        //                A string such as 2005-06-30T08:05:00-07:00 or 2005-06-30 or T08:05:00
2449
        //
2450
        //        defaultTime:
2451
        //                Used for defaults for fields omitted in the formattedString.
2452
        //                Uses 1970-01-01T00:00:00.0Z by default.
2453
2454
        if(!dojo.date.stamp._isoRegExp){
2455
                dojo.date.stamp._isoRegExp =
2456
//TODO: could be more restrictive and check for 00-59, etc.
2457
                        /^(?:(\d{4})(?:-(\d{2})(?:-(\d{2}))?)?)?(?:T(\d{2}):(\d{2})(?::(\d{2})(.\d+)?)?((?:[+-](\d{2}):(\d{2}))|Z)?)?$/;
2458
        }
2459
2460
        var match = dojo.date.stamp._isoRegExp.exec(formattedString),
2461
                result = null;
2462
2463
        if(match){
2464
                match.shift();
2465
                if(match[1]){match[1]--;} // Javascript Date months are 0-based
2466
                if(match[6]){match[6] *= 1000;} // Javascript Date expects fractional seconds as milliseconds
2467
2468
                if(defaultTime){
2469
                        // mix in defaultTime.  Relatively expensive, so use || operators for the fast path of defaultTime === 0
2470
                        defaultTime = new Date(defaultTime);
2471
                        dojo.forEach(dojo.map(["FullYear", "Month", "Date", "Hours", "Minutes", "Seconds", "Milliseconds"], function(prop){
2472
                                return defaultTime["get" + prop]();
2473
                        }), function(value, index){
2474
                                match[index] = match[index] || value;
2475
                        });
2476
                }
2477
                result = new Date(match[0]||1970, match[1]||0, match[2]||1, match[3]||0, match[4]||0, match[5]||0, match[6]||0); //TODO: UTC defaults
2478
                if(match[0] < 100){
2479
                        result.setFullYear(match[0] || 1970);
2480
                }
2481
2482
                var offset = 0,
2483
                        zoneSign = match[7] && match[7].charAt(0);
2484
                if(zoneSign != 'Z'){
2485
                        offset = ((match[8] || 0) * 60) + (Number(match[9]) || 0);
2486
                        if(zoneSign != '-'){ offset *= -1; }
2487
                }
2488
                if(zoneSign){
2489
                        offset -= result.getTimezoneOffset();
2490
                }
2491
                if(offset){
2492
                        result.setTime(result.getTime() + offset * 60000);
2493
                }
2494
        }
2495
2496
        return result; // Date or null
2497
}
2498
2499
/*=====
2500
        dojo.date.stamp.__Options = function(){
2501
                //        selector: String
2502
                //                "date" or "time" for partial formatting of the Date object.
2503
                //                Both date and time will be formatted by default.
2504
                //        zulu: Boolean
2505
                //                if true, UTC/GMT is used for a timezone
2506
                //        milliseconds: Boolean
2507
                //                if true, output milliseconds
2508
                this.selector = selector;
2509
                this.zulu = zulu;
2510
                this.milliseconds = milliseconds;
2511
        }
2512
=====*/
2513
2514
dojo.date.stamp.toISOString = function(/*Date*/dateObject, /*dojo.date.stamp.__Options?*/options){
2515
        //        summary:
2516
        //                Format a Date object as a string according a subset of the ISO-8601 standard
2517
        //
2518
        //        description:
2519
        //                When options.selector is omitted, output follows [RFC3339](http://www.ietf.org/rfc/rfc3339.txt)
2520
        //                The local time zone is included as an offset from GMT, except when selector=='time' (time without a date)
2521
        //                Does not check bounds.  Only years between 100 and 9999 are supported.
2522
        //
2523
        //        dateObject:
2524
        //                A Date object
2525
2526
        var _ = function(n){ return (n < 10) ? "0" + n : n; };
2527
        options = options || {};
2528
        var formattedDate = [],
2529
                getter = options.zulu ? "getUTC" : "get",
2530
                date = "";
2531
        if(options.selector != "time"){
2532
                var year = dateObject[getter+"FullYear"]();
2533
                date = ["0000".substr((year+"").length)+year, _(dateObject[getter+"Month"]()+1), _(dateObject[getter+"Date"]())].join('-');
2534
        }
2535
        formattedDate.push(date);
2536
        if(options.selector != "date"){
2537
                var time = [_(dateObject[getter+"Hours"]()), _(dateObject[getter+"Minutes"]()), _(dateObject[getter+"Seconds"]())].join(':');
2538
                var millis = dateObject[getter+"Milliseconds"]();
2539
                if(options.milliseconds){
2540
                        time += "."+ (millis < 100 ? "0" : "") + _(millis);
2541
                }
2542
                if(options.zulu){
2543
                        time += "Z";
2544
                }else if(options.selector != "time"){
2545
                        var timezoneOffset = dateObject.getTimezoneOffset();
2546
                        var absOffset = Math.abs(timezoneOffset);
2547
                        time += (timezoneOffset > 0 ? "-" : "+") +
2548
                                _(Math.floor(absOffset/60)) + ":" + _(absOffset%60);
2549
                }
2550
                formattedDate.push(time);
2551
        }
2552
        return formattedDate.join('T'); // String
2553
}
2554
2555
}
2556
2557
if(!dojo._hasResource["dojo.parser"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
2558
dojo._hasResource["dojo.parser"] = true;
2559
dojo.provide("dojo.parser");
2560
2561
2562
new Date("X"); // workaround for #11279, new Date("") == NaN
2563
2564
dojo.parser = new function(){
2565
        // summary: The Dom/Widget parsing package
2566
2567
        var d = dojo;
2568
        this._attrName = d._scopeName + "Type";
2569
        this._query = "[" + this._attrName + "]";
2570
2571
        function val2type(/*Object*/ value){
2572
                // summary:
2573
                //                Returns name of type of given value.
2574
2575
                if(d.isString(value)){ return "string"; }
2576
                if(typeof value == "number"){ return "number"; }
2577
                if(typeof value == "boolean"){ return "boolean"; }
2578
                if(d.isFunction(value)){ return "function"; }
2579
                if(d.isArray(value)){ return "array"; } // typeof [] == "object"
2580
                if(value instanceof Date) { return "date"; } // assume timestamp
2581
                if(value instanceof d._Url){ return "url"; }
2582
                return "object";
2583
        }
2584
2585
        function str2obj(/*String*/ value, /*String*/ type){
2586
                // summary:
2587
                //                Convert given string value to given type
2588
                switch(type){
2589
                        case "string":
2590
                                return value;
2591
                        case "number":
2592
                                return value.length ? Number(value) : NaN;
2593
                        case "boolean":
2594
                                // for checked/disabled value might be "" or "checked".  interpret as true.
2595
                                return typeof value == "boolean" ? value : !(value.toLowerCase()=="false");
2596
                        case "function":
2597
                                if(d.isFunction(value)){
2598
                                        // IE gives us a function, even when we say something like onClick="foo"
2599
                                        // (in which case it gives us an invalid function "function(){ foo }").
2600
                                        //  Therefore, convert to string
2601
                                        value=value.toString();
2602
                                        value=d.trim(value.substring(value.indexOf('{')+1, value.length-1));
2603
                                }
2604
                                try{
2605
                                        if(value === "" || value.search(/[^\w\.]+/i) != -1){
2606
                                                // The user has specified some text for a function like "return x+5"
2607
                                                return new Function(value);
2608
                                        }else{
2609
                                                // The user has specified the name of a function like "myOnClick"
2610
                                                // or a single word function "return"
2611
                                                return d.getObject(value, false) || new Function(value);
2612
                                        }
2613
                                }catch(e){ return new Function(); }
2614
                        case "array":
2615
                                return value ? value.split(/\s*,\s*/) : [];
2616
                        case "date":
2617
                                switch(value){
2618
                                        case "": return new Date("");        // the NaN of dates
2619
                                        case "now": return new Date();        // current date
2620
                                        default: return d.date.stamp.fromISOString(value);
2621
                                }
2622
                        case "url":
2623
                                return d.baseUrl + value;
2624
                        default:
2625
                                return d.fromJson(value);
2626
                }
2627
        }
2628
2629
        var instanceClasses = {
2630
                // map from fully qualified name (like "dijit.Button") to structure like
2631
                // { cls: dijit.Button, params: {label: "string", disabled: "boolean"} }
2632
        };
2633
2634
        // Widgets like BorderContainer add properties to _Widget via dojo.extend().
2635
        // If BorderContainer is loaded after _Widget's parameter list has been cached,
2636
        // we need to refresh that parameter list (for _Widget and all widgets that extend _Widget).
2637
        dojo.connect(dojo, "extend", function(){
2638
                instanceClasses = {};
2639
        });
2640
2641
        function getClassInfo(/*String*/ className){
2642
                // className:
2643
                //                fully qualified name (like "dijit.form.Button")
2644
                // returns:
2645
                //                structure like
2646
                //                        {
2647
                //                                cls: dijit.Button,
2648
                //                                params: { label: "string", disabled: "boolean"}
2649
                //                        }
2650
2651
                if(!instanceClasses[className]){
2652
                        // get pointer to widget class
2653
                        var cls = d.getObject(className);
2654
                        if(!cls){ return null; }                // class not defined [yet]
2655
2656
                        var proto = cls.prototype;
2657
2658
                        // get table of parameter names & types
2659
                        var params = {}, dummyClass = {};
2660
                        for(var name in proto){
2661
                                if(name.charAt(0)=="_"){ continue; }         // skip internal properties
2662
                                if(name in dummyClass){ continue; }                // skip "constructor" and "toString"
2663
                                var defVal = proto[name];
2664
                                params[name]=val2type(defVal);
2665
                        }
2666
2667
                        instanceClasses[className] = { cls: cls, params: params };
2668
                }
2669
                return instanceClasses[className];
2670
        }
2671
2672
        this._functionFromScript = function(script){
2673
                var preamble = "";
2674
                var suffix = "";
2675
                var argsStr = script.getAttribute("args");
2676
                if(argsStr){
2677
                        d.forEach(argsStr.split(/\s*,\s*/), function(part, idx){
2678
                                preamble += "var "+part+" = arguments["+idx+"]; ";
2679
                        });
2680
                }
2681
                var withStr = script.getAttribute("with");
2682
                if(withStr && withStr.length){
2683
                        d.forEach(withStr.split(/\s*,\s*/), function(part){
2684
                                preamble += "with("+part+"){";
2685
                                suffix += "}";
2686
                        });
2687
                }
2688
                return new Function(preamble+script.innerHTML+suffix);
2689
        }
2690
2691
        this.instantiate = function(/* Array */nodes, /* Object? */mixin, /* Object? */args){
2692
                // summary:
2693
                //                Takes array of nodes, and turns them into class instances and
2694
                //                potentially calls a startup method to allow them to connect with
2695
                //                any children.
2696
                // nodes: Array
2697
                //                Array of nodes or objects like
2698
                //        |                {
2699
                //        |                        type: "dijit.form.Button",
2700
                //        |                        node: DOMNode,
2701
                //        |                        scripts: [ ... ],        // array of <script type="dojo/..."> children of node
2702
                //        |                        inherited: { ... }        // settings inherited from ancestors like dir, theme, etc.
2703
                //        |                }
2704
                // mixin: Object?
2705
                //                An object that will be mixed in with each node in the array.
2706
                //                Values in the mixin will override values in the node, if they
2707
                //                exist.
2708
                // args: Object?
2709
                //                An object used to hold kwArgs for instantiation.
2710
                //                Supports 'noStart' and inherited.
2711
                var thelist = [], dp = dojo.parser;
2712
                mixin = mixin||{};
2713
                args = args||{};
2714
2715
                d.forEach(nodes, function(obj){
2716
                        if(!obj){ return; }
2717
2718
                        // Get pointers to DOMNode, dojoType string, and clsInfo (metadata about the dojoType), etc.s
2719
                        var node, type, clsInfo, clazz, scripts;
2720
                        if(obj.node){
2721
                                // new format of nodes[] array, object w/lots of properties pre-computed for me
2722
                                node = obj.node;
2723
                                type = obj.type;
2724
                                clsInfo = obj.clsInfo || (type && getClassInfo(type));
2725
                                clazz = clsInfo && clsInfo.cls;
2726
                                scripts = obj.scripts;
2727
                        }else{
2728
                                // old (backwards compatible) format of nodes[] array, simple array of DOMNodes
2729
                                node = obj;
2730
                                type = dp._attrName in mixin ? mixin[dp._attrName] : node.getAttribute(dp._attrName);
2731
                                clsInfo = type && getClassInfo(type);
2732
                                clazz = clsInfo && clsInfo.cls;
2733
                                scripts = (clazz && (clazz._noScript || clazz.prototype._noScript) ? [] :
2734
                                                        d.query("> script[type^='dojo/']", node));
2735
                        }
2736
                        if(!clsInfo){
2737
                                throw new Error("Could not load class '" + type);
2738
                        }
2739
2740
                        // Setup hash to hold parameter settings for this widget.   Start with the parameter
2741
                        // settings inherited from ancestors ("dir" and "lang").
2742
                        // Inherited setting may later be overridden by explicit settings on node itself.
2743
                        var params = {},
2744
                                attributes = node.attributes;
2745
                        if(args.defaults){
2746
                                // settings for the document itself (or whatever subtree is being parsed)
2747
                                dojo.mixin(params, args.defaults);
2748
                        }
2749
                        if(obj.inherited){
2750
                                // settings from dir=rtl or lang=... on a node above this node
2751
                                dojo.mixin(params, obj.inherited);
2752
                        }
2753
2754
                        // read parameters (ie, attributes) specified on DOMNode
2755
                        // clsInfo.params lists expected params like {"checked": "boolean", "n": "number"}
2756
                        for(var name in clsInfo.params){
2757
                                var item = name in mixin?{value:mixin[name],specified:true}:attributes.getNamedItem(name);
2758
                                if(!item || (!item.specified && (!dojo.isIE || name.toLowerCase()!="value"))){ continue; }
2759
                                var value = item.value;
2760
                                // Deal with IE quirks for 'class' and 'style'
2761
                                switch(name){
2762
                                case "class":
2763
                                        value = "className" in mixin?mixin.className:node.className;
2764
                                        break;
2765
                                case "style":
2766
                                        value = "style" in mixin?mixin.style:(node.style && node.style.cssText); // FIXME: Opera?
2767
                                }
2768
                                var _type = clsInfo.params[name];
2769
                                if(typeof value == "string"){
2770
                                        params[name] = str2obj(value, _type);
2771
                                }else{
2772
                                        params[name] = value;
2773
                                }
2774
                        }
2775
2776
                        // Process <script type="dojo/*"> script tags
2777
                        // <script type="dojo/method" event="foo"> tags are added to params, and passed to
2778
                        // the widget on instantiation.
2779
                        // <script type="dojo/method"> tags (with no event) are executed after instantiation
2780
                        // <script type="dojo/connect" event="foo"> tags are dojo.connected after instantiation
2781
                        // note: dojo/* script tags cannot exist in self closing widgets, like <input />
2782
                        var connects = [],        // functions to connect after instantiation
2783
                                calls = [];                // functions to call after instantiation
2784
2785
                        d.forEach(scripts, function(script){
2786
                                node.removeChild(script);
2787
                                var event = script.getAttribute("event"),
2788
                                        type = script.getAttribute("type"),
2789
                                        nf = d.parser._functionFromScript(script);
2790
                                if(event){
2791
                                        if(type == "dojo/connect"){
2792
                                                connects.push({event: event, func: nf});
2793
                                        }else{
2794
                                                params[event] = nf;
2795
                                        }
2796
                                }else{
2797
                                        calls.push(nf);
2798
                                }
2799
                        });
2800
2801
                        var markupFactory = clazz.markupFactory || clazz.prototype && clazz.prototype.markupFactory;
2802
                        // create the instance
2803
                        var instance = markupFactory ? markupFactory(params, node, clazz) : new clazz(params, node);
2804
                        thelist.push(instance);
2805
2806
                        // map it to the JS namespace if that makes sense
2807
                        var jsname = node.getAttribute("jsId");
2808
                        if(jsname){
2809
                                d.setObject(jsname, instance);
2810
                        }
2811
2812
                        // process connections and startup functions
2813
                        d.forEach(connects, function(connect){
2814
                                d.connect(instance, connect.event, null, connect.func);
2815
                        });
2816
                        d.forEach(calls, function(func){
2817
                                func.call(instance);
2818
                        });
2819
                });
2820
2821
                // Call startup on each top level instance if it makes sense (as for
2822
                // widgets).  Parent widgets will recursively call startup on their
2823
                // (non-top level) children
2824
                if(!mixin._started){
2825
                        // TODO: for 2.0, when old instantiate() API is desupported, store parent-child
2826
                        // relationships in the nodes[] array so that no getParent() call is needed.
2827
                        // Note that will  require a parse() call from ContentPane setting a param that the
2828
                        // ContentPane is the parent widget (so that the parse doesn't call startup() on the
2829
                        // ContentPane's children)
2830
                        d.forEach(thelist, function(instance){
2831
                                if(        !args.noStart && instance  &&
2832
                                        instance.startup &&
2833
                                        !instance._started &&
2834
                                        (!instance.getParent || !instance.getParent())
2835
                                ){
2836
                                        instance.startup();
2837
                                }
2838
                        });
2839
                }
2840
                return thelist;
2841
        };
2842
2843
        this.parse = function(/*DomNode?*/ rootNode, /* Object? */ args){
2844
                // summary:
2845
                //                Scan the DOM for class instances, and instantiate them.
2846
                //
2847
                // description:
2848
                //                Search specified node (or root node) recursively for class instances,
2849
                //                and instantiate them Searches for
2850
                //                dojoType="qualified.class.name"
2851
                //
2852
                // rootNode: DomNode?
2853
                //                A default starting root node from which to start the parsing. Can be
2854
                //                omitted, defaulting to the entire document. If omitted, the `args`
2855
                //                object can be passed in this place. If the `args` object has a
2856
                //                `rootNode` member, that is used.
2857
                //
2858
                // args:
2859
                //                a kwArgs object passed along to instantiate()
2860
                //
2861
                //                        * noStart: Boolean?
2862
                //                                when set will prevent the parser from calling .startup()
2863
                //                                when locating the nodes.
2864
                //                        * rootNode: DomNode?
2865
                //                                identical to the function's `rootNode` argument, though
2866
                //                                allowed to be passed in via this `args object.
2867
                //                        * inherited: Object
2868
                //                                Hash possibly containing dir and lang settings to be applied to
2869
                //                                parsed widgets, unless there's another setting on a sub-node that overrides
2870
                //
2871
                //
2872
                // example:
2873
                //                Parse all widgets on a page:
2874
                //        |                dojo.parser.parse();
2875
                //
2876
                // example:
2877
                //                Parse all classes within the node with id="foo"
2878
                //        |                dojo.parser.parse(dojo.byId(foo));
2879
                //
2880
                // example:
2881
                //                Parse all classes in a page, but do not call .startup() on any
2882
                //                child
2883
                //        |                dojo.parser.parse({ noStart: true })
2884
                //
2885
                // example:
2886
                //                Parse all classes in a node, but do not call .startup()
2887
                //        |                dojo.parser.parse(someNode, { noStart:true });
2888
                //        |                // or
2889
                //         |                dojo.parser.parse({ noStart:true, rootNode: someNode });
2890
2891
                // determine the root node based on the passed arguments.
2892
                var root;
2893
                if(!args && rootNode && rootNode.rootNode){
2894
                        args = rootNode;
2895
                        root = args.rootNode;
2896
                }else{
2897
                        root = rootNode;
2898
                }
2899
2900
                var attrName = this._attrName;
2901
                function scan(parent, list){
2902
                        // summary:
2903
                        //                Parent is an Object representing a DOMNode, with or without a dojoType specified.
2904
                        //                Scan parent's children looking for nodes with dojoType specified, storing in list[].
2905
                        //                If parent has a dojoType, also collects <script type=dojo/*> children and stores in parent.scripts[].
2906
                        // parent: Object
2907
                        //                Object representing the parent node, like
2908
                        //        |        {
2909
                        //        |                node: DomNode,                         // scan children of this node
2910
                        //        |                inherited: {dir: "rtl"},        // dir/lang setting inherited from above node
2911
                        //        |
2912
                        //        |                // attributes only set if node has dojoType specified
2913
                        //        |                scripts: [],                        // empty array, put <script type=dojo/*> in here
2914
                        //        |                clsInfo: { cls: dijit.form.Button, ...}
2915
                        //        |        }
2916
                        // list: DomNode[]
2917
                        //                Output array of objects (same format as parent) representing nodes to be turned into widgets
2918
2919
                        // Effective dir and lang settings on parent node, either set directly or inherited from grandparent
2920
                        var inherited = dojo.clone(parent.inherited);
2921
                        dojo.forEach(["dir", "lang"], function(name){
2922
                                var val = parent.node.getAttribute(name);
2923
                                if(val){
2924
                                        inherited[name] = val;
2925
                                }
2926
                        });
2927
2928
                        // if parent is a widget, then search for <script type=dojo/*> tags and put them in scripts[].
2929
                        var scripts = parent.scripts;
2930
2931
                        // unless parent is a widget with the stopParser flag set, continue search for dojoType, recursively
2932
                        var recurse = !parent.clsInfo || !parent.clsInfo.cls.prototype.stopParser;
2933
2934
                        // scan parent's children looking for dojoType and <script type=dojo/*>
2935
                        for(var child = parent.node.firstChild; child; child = child.nextSibling){
2936
                                if(child.nodeType == 1){
2937
                                        var type = recurse && child.getAttribute(attrName);
2938
                                        if(type){
2939
                                                // if dojoType specified, add to output array of nodes to instantiate
2940
                                                var params = {
2941
                                                        "type": type,
2942
                                                        clsInfo: getClassInfo(type),        // note: won't find classes declared via dojo.Declaration
2943
                                                        node: child,
2944
                                                        scripts: [], // <script> nodes that are parent's children
2945
                                                        inherited: inherited // dir & lang attributes inherited from parent
2946
                                                };
2947
                                                list.push(params);
2948
2949
                                                // Recurse, collecting <script type="dojo/..."> children, and also looking for
2950
                                                // descendant nodes with dojoType specified (unless the widget has the stopParser flag),
2951
                                                scan(params, list);
2952
                                        }else if(scripts && child.nodeName.toLowerCase() == "script"){
2953
                                                // if <script type="dojo/...">, save in scripts[]
2954
                                                type = child.getAttribute("type");
2955
                                                if (type && /^dojo\//i.test(type)) {
2956
                                                        scripts.push(child);
2957
                                                }
2958
                                        }else if(recurse){
2959
                                                // Recurse, looking for grandchild nodes with dojoType specified
2960
                                                scan({
2961
                                                        node: child,
2962
                                                        inherited: inherited
2963
                                                }, list);
2964
                                        }
2965
                                }
2966
                        }
2967
                }
2968
2969
                // Make list of all nodes on page w/dojoType specified
2970
                var list = [];
2971
                scan({
2972
                        node: root ? dojo.byId(root) : dojo.body(),
2973
                        inherited: (args && args.inherited) || {
2974
                                dir: dojo._isBodyLtr() ? "ltr" : "rtl"
2975
                        }
2976
                }, list);
2977
2978
                // go build the object instances
2979
                return this.instantiate(list, null, args); // Array
2980
        };
2981
}();
2982
2983
//Register the parser callback. It should be the first callback
2984
//after the a11y test.
2985
2986
(function(){
2987
        var parseRunner = function(){
2988
                if(dojo.config.parseOnLoad){
2989
                        dojo.parser.parse();
2990
                }
2991
        };
2992
2993
        // FIXME: need to clobber cross-dependency!!
2994
        if(dojo.exists("dijit.wai.onload") && (dijit.wai.onload === dojo._loaders[0])){
2995
                dojo._loaders.splice(1, 0, parseRunner);
2996
        }else{
2997
                dojo._loaders.unshift(parseRunner);
2998
        }
2999
})();
3000
3001
}
3002
3003
if(!dojo._hasResource["dijit._Widget"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
3004
dojo._hasResource["dijit._Widget"] = true;
3005
dojo.provide("dijit._Widget");
3006
3007
dojo.require( "dijit._base" );
3008
3009
3010
// This code is to assist deferring dojo.connect() calls in widgets (connecting to events on the widgets'
3011
// DOM nodes) until someone actually needs to monitor that event.
3012
dojo.connect(dojo, "_connect",
3013
        function(/*dijit._Widget*/ widget, /*String*/ event){
3014
                if(widget && dojo.isFunction(widget._onConnect)){
3015
                        widget._onConnect(event);
3016
                }
3017
        });
3018
3019
dijit._connectOnUseEventHandler = function(/*Event*/ event){};
3020
3021
// Keep track of where the last keydown event was, to help avoid generating
3022
// spurious ondijitclick events when:
3023
// 1. focus is on a <button> or <a>
3024
// 2. user presses then releases the ENTER key
3025
// 3. onclick handler fires and shifts focus to another node, with an ondijitclick handler
3026
// 4. onkeyup event fires, causing the ondijitclick handler to fire
3027
dijit._lastKeyDownNode = null;
3028
if(dojo.isIE){
3029
        (function(){
3030
                var keydownCallback = function(evt){
3031
                        dijit._lastKeyDownNode = evt.srcElement;
3032
                };
3033
                dojo.doc.attachEvent('onkeydown', keydownCallback);
3034
                dojo.addOnWindowUnload(function(){
3035
                        dojo.doc.detachEvent('onkeydown', keydownCallback);
3036
                });
3037
        })();
3038
}else{
3039
        dojo.doc.addEventListener('keydown', function(evt){
3040
                dijit._lastKeyDownNode = evt.target;
3041
        }, true);
3042
}
3043
3044
(function(){
3045
3046
var _attrReg = {},        // cached results from getSetterAttributes
3047
        getSetterAttributes = function(widget){
3048
                // summary:
3049
                //                Returns list of attributes with custom setters for specified widget
3050
                var dc = widget.declaredClass;
3051
                if(!_attrReg[dc]){
3052
                        var r = [],
3053
                                attrs,
3054
                                proto = widget.constructor.prototype;
3055
                        for(var fxName in proto){
3056
                                if(dojo.isFunction(proto[fxName]) && (attrs = fxName.match(/^_set([a-zA-Z]*)Attr$/)) && attrs[1]){
3057
                                        r.push(attrs[1].charAt(0).toLowerCase() + attrs[1].substr(1));
3058
                                }
3059
                        }
3060
                        _attrReg[dc] = r;
3061
                }
3062
                return _attrReg[dc] || [];        // String[]
3063
        };
3064
3065
dojo.declare("dijit._Widget", null, {
3066
        // summary:
3067
        //                Base class for all Dijit widgets.
3068
3069
        // id: [const] String
3070
        //                A unique, opaque ID string that can be assigned by users or by the
3071
        //                system. If the developer passes an ID which is known not to be
3072
        //                unique, the specified ID is ignored and the system-generated ID is
3073
        //                used instead.
3074
        id: "",
3075
3076
        // lang: [const] String
3077
        //                Rarely used.  Overrides the default Dojo locale used to render this widget,
3078
        //                as defined by the [HTML LANG](http://www.w3.org/TR/html401/struct/dirlang.html#adef-lang) attribute.
3079
        //                Value must be among the list of locales specified during by the Dojo bootstrap,
3080
        //                formatted according to [RFC 3066](http://www.ietf.org/rfc/rfc3066.txt) (like en-us).
3081
        lang: "",
3082
3083
        // dir: [const] String
3084
        //                Bi-directional support, as defined by the [HTML DIR](http://www.w3.org/TR/html401/struct/dirlang.html#adef-dir)
3085
        //                attribute. Either left-to-right "ltr" or right-to-left "rtl".  If undefined, widgets renders in page's
3086
        //                default direction.
3087
        dir: "",
3088
3089
        // class: String
3090
        //                HTML class attribute
3091
        "class": "",
3092
3093
        // style: String||Object
3094
        //                HTML style attributes as cssText string or name/value hash
3095
        style: "",
3096
3097
        // title: String
3098
        //                HTML title attribute.
3099
        //
3100
        //                For form widgets this specifies a tooltip to display when hovering over
3101
        //                the widget (just like the native HTML title attribute).
3102
        //
3103
        //                For TitlePane or for when this widget is a child of a TabContainer, AccordionContainer,
3104
        //                etc., it's used to specify the tab label, accordion pane title, etc.
3105
        title: "",
3106
3107
        // tooltip: String
3108
        //                When this widget's title attribute is used to for a tab label, accordion pane title, etc.,
3109
        //                this specifies the tooltip to appear when the mouse is hovered over that text.
3110
        tooltip: "",
3111
3112
        // baseClass: [protected] String
3113
        //                Root CSS class of the widget (ex: dijitTextBox), used to construct CSS classes to indicate
3114
        //                widget state.
3115
        baseClass: "",
3116
3117
        // srcNodeRef: [readonly] DomNode
3118
        //                pointer to original DOM node
3119
        srcNodeRef: null,
3120
3121
        // domNode: [readonly] DomNode
3122
        //                This is our visible representation of the widget! Other DOM
3123
        //                Nodes may by assigned to other properties, usually through the
3124
        //                template system's dojoAttachPoint syntax, but the domNode
3125
        //                property is the canonical "top level" node in widget UI.
3126
        domNode: null,
3127
3128
        // containerNode: [readonly] DomNode
3129
        //                Designates where children of the source DOM node will be placed.
3130
        //                "Children" in this case refers to both DOM nodes and widgets.
3131
        //                For example, for myWidget:
3132
        //
3133
        //                |        <div dojoType=myWidget>
3134
        //                |                <b> here's a plain DOM node
3135
        //                |                <span dojoType=subWidget>and a widget</span>
3136
        //                |                <i> and another plain DOM node </i>
3137
        //                |        </div>
3138
        //
3139
        //                containerNode would point to:
3140
        //
3141
        //                |                <b> here's a plain DOM node
3142
        //                |                <span dojoType=subWidget>and a widget</span>
3143
        //                |                <i> and another plain DOM node </i>
3144
        //
3145
        //                In templated widgets, "containerNode" is set via a
3146
        //                dojoAttachPoint assignment.
3147
        //
3148
        //                containerNode must be defined for any widget that accepts innerHTML
3149
        //                (like ContentPane or BorderContainer or even Button), and conversely
3150
        //                is null for widgets that don't, like TextBox.
3151
        containerNode: null,
3152
3153
/*=====
3154
        // _started: Boolean
3155
        //                startup() has completed.
3156
        _started: false,
3157
=====*/
3158
3159
        // attributeMap: [protected] Object
3160
        //                attributeMap sets up a "binding" between attributes (aka properties)
3161
        //                of the widget and the widget's DOM.
3162
        //                Changes to widget attributes listed in attributeMap will be
3163
        //                reflected into the DOM.
3164
        //
3165
        //                For example, calling attr('title', 'hello')
3166
        //                on a TitlePane will automatically cause the TitlePane's DOM to update
3167
        //                with the new title.
3168
        //
3169
        //                attributeMap is a hash where the key is an attribute of the widget,
3170
        //                and the value reflects a binding to a:
3171
        //
3172
        //                - DOM node attribute
3173
        // |                focus: {node: "focusNode", type: "attribute"}
3174
        //                 Maps this.focus to this.focusNode.focus
3175
        //
3176
        //                - DOM node innerHTML
3177
        //        |                title: { node: "titleNode", type: "innerHTML" }
3178
        //                Maps this.title to this.titleNode.innerHTML
3179
        //
3180
        //                - DOM node innerText
3181
        //        |                title: { node: "titleNode", type: "innerText" }
3182
        //                Maps this.title to this.titleNode.innerText
3183
        //
3184
        //                - DOM node CSS class
3185
        // |                myClass: { node: "domNode", type: "class" }
3186
        //                Maps this.myClass to this.domNode.className
3187
        //
3188
        //                If the value is an array, then each element in the array matches one of the
3189
        //                formats of the above list.
3190
        //
3191
        //                There are also some shorthands for backwards compatibility:
3192
        //                - string --> { node: string, type: "attribute" }, for example:
3193
        //        |        "focusNode" ---> { node: "focusNode", type: "attribute" }
3194
        //                - "" --> { node: "domNode", type: "attribute" }
3195
        attributeMap: {id:"", dir:"", lang:"", "class":"", style:"", title:""},
3196
3197
        // _deferredConnects: [protected] Object
3198
        //                attributeMap addendum for event handlers that should be connected only on first use
3199
        _deferredConnects: {
3200
                onClick: "",
3201
                onDblClick: "",
3202
                onKeyDown: "",
3203
                onKeyPress: "",
3204
                onKeyUp: "",
3205
                onMouseMove: "",
3206
                onMouseDown: "",
3207
                onMouseOut: "",
3208
                onMouseOver: "",
3209
                onMouseLeave: "",
3210
                onMouseEnter: "",
3211
                onMouseUp: ""
3212
        },
3213
3214
        onClick: dijit._connectOnUseEventHandler,
3215
        /*=====
3216
        onClick: function(event){
3217
                // summary:
3218
                //                Connect to this function to receive notifications of mouse click events.
3219
                // event:
3220
                //                mouse Event
3221
                // tags:
3222
                //                callback
3223
        },
3224
        =====*/
3225
        onDblClick: dijit._connectOnUseEventHandler,
3226
        /*=====
3227
        onDblClick: function(event){
3228
                // summary:
3229
                //                Connect to this function to receive notifications of mouse double click events.
3230
                // event:
3231
                //                mouse Event
3232
                // tags:
3233
                //                callback
3234
        },
3235
        =====*/
3236
        onKeyDown: dijit._connectOnUseEventHandler,
3237
        /*=====
3238
        onKeyDown: function(event){
3239
                // summary:
3240
                //                Connect to this function to receive notifications of keys being pressed down.
3241
                // event:
3242
                //                key Event
3243
                // tags:
3244
                //                callback
3245
        },
3246
        =====*/
3247
        onKeyPress: dijit._connectOnUseEventHandler,
3248
        /*=====
3249
        onKeyPress: function(event){
3250
                // summary:
3251
                //                Connect to this function to receive notifications of printable keys being typed.
3252
                // event:
3253
                //                key Event
3254
                // tags:
3255
                //                callback
3256
        },
3257
        =====*/
3258
        onKeyUp: dijit._connectOnUseEventHandler,
3259
        /*=====
3260
        onKeyUp: function(event){
3261
                // summary:
3262
                //                Connect to this function to receive notifications of keys being released.
3263
                // event:
3264
                //                key Event
3265
                // tags:
3266
                //                callback
3267
        },
3268
        =====*/
3269
        onMouseDown: dijit._connectOnUseEventHandler,
3270
        /*=====
3271
        onMouseDown: function(event){
3272
                // summary:
3273
                //                Connect to this function to receive notifications of when the mouse button is pressed down.
3274
                // event:
3275
                //                mouse Event
3276
                // tags:
3277
                //                callback
3278
        },
3279
        =====*/
3280
        onMouseMove: dijit._connectOnUseEventHandler,
3281
        /*=====
3282
        onMouseMove: function(event){
3283
                // summary:
3284
                //                Connect to this function to receive notifications of when the mouse moves over nodes contained within this widget.
3285
                // event:
3286
                //                mouse Event
3287
                // tags:
3288
                //                callback
3289
        },
3290
        =====*/
3291
        onMouseOut: dijit._connectOnUseEventHandler,
3292
        /*=====
3293
        onMouseOut: function(event){
3294
                // summary:
3295
                //                Connect to this function to receive notifications of when the mouse moves off of nodes contained within this widget.
3296
                // event:
3297
                //                mouse Event
3298
                // tags:
3299
                //                callback
3300
        },
3301
        =====*/
3302
        onMouseOver: dijit._connectOnUseEventHandler,
3303
        /*=====
3304
        onMouseOver: function(event){
3305
                // summary:
3306
                //                Connect to this function to receive notifications of when the mouse moves onto nodes contained within this widget.
3307
                // event:
3308
                //                mouse Event
3309
                // tags:
3310
                //                callback
3311
        },
3312
        =====*/
3313
        onMouseLeave: dijit._connectOnUseEventHandler,
3314
        /*=====
3315
        onMouseLeave: function(event){
3316
                // summary:
3317
                //                Connect to this function to receive notifications of when the mouse moves off of this widget.
3318
                // event:
3319
                //                mouse Event
3320
                // tags:
3321
                //                callback
3322
        },
3323
        =====*/
3324
        onMouseEnter: dijit._connectOnUseEventHandler,
3325
        /*=====
3326
        onMouseEnter: function(event){
3327
                // summary:
3328
                //                Connect to this function to receive notifications of when the mouse moves onto this widget.
3329
                // event:
3330
                //                mouse Event
3331
                // tags:
3332
                //                callback
3333
        },
3334
        =====*/
3335
        onMouseUp: dijit._connectOnUseEventHandler,
3336
        /*=====
3337
        onMouseUp: function(event){
3338
                // summary:
3339
                //                Connect to this function to receive notifications of when the mouse button is released.
3340
                // event:
3341
                //                mouse Event
3342
                // tags:
3343
                //                callback
3344
        },
3345
        =====*/
3346
3347
        // Constants used in templates
3348
3349
        // _blankGif: [protected] String
3350
        //                Path to a blank 1x1 image.
3351
        //                Used by <img> nodes in templates that really get their image via CSS background-image.
3352
        _blankGif: (dojo.config.blankGif || dojo.moduleUrl("dojo", "resources/blank.gif")).toString(),
3353
3354
        //////////// INITIALIZATION METHODS ///////////////////////////////////////
3355
3356
        postscript: function(/*Object?*/params, /*DomNode|String*/srcNodeRef){
3357
                // summary:
3358
                //                Kicks off widget instantiation.  See create() for details.
3359
                // tags:
3360
                //                private
3361
                this.create(params, srcNodeRef);
3362
        },
3363
3364
        create: function(/*Object?*/params, /*DomNode|String?*/srcNodeRef){
3365
                // summary:
3366
                //                Kick off the life-cycle of a widget
3367
                // params:
3368
                //                Hash of initialization parameters for widget, including
3369
                //                scalar values (like title, duration etc.) and functions,
3370
                //                typically callbacks like onClick.
3371
                // srcNodeRef:
3372
                //                If a srcNodeRef (DOM node) is specified:
3373
                //                        - use srcNodeRef.innerHTML as my contents
3374
                //                        - if this is a behavioral widget then apply behavior
3375
                //                          to that srcNodeRef
3376
                //                        - otherwise, replace srcNodeRef with my generated DOM
3377
                //                          tree
3378
                // description:
3379
                //                Create calls a number of widget methods (postMixInProperties, buildRendering, postCreate,
3380
                //                etc.), some of which of you'll want to override. See http://docs.dojocampus.org/dijit/_Widget
3381
                //                for a discussion of the widget creation lifecycle.
3382
                //
3383
                //                Of course, adventurous developers could override create entirely, but this should
3384
                //                only be done as a last resort.
3385
                // tags:
3386
                //                private
3387
3388
                // store pointer to original DOM tree
3389
                this.srcNodeRef = dojo.byId(srcNodeRef);
3390
3391
                // For garbage collection.  An array of handles returned by Widget.connect()
3392
                // Each handle returned from Widget.connect() is an array of handles from dojo.connect()
3393
                this._connects = [];
3394
3395
                // For garbage collection.  An array of handles returned by Widget.subscribe()
3396
                // The handle returned from Widget.subscribe() is the handle returned from dojo.subscribe()
3397
                this._subscribes = [];
3398
3399
                // To avoid double-connects, remove entries from _deferredConnects
3400
                // that have been setup manually by a subclass (ex, by dojoAttachEvent).
3401
                // If a subclass has redefined a callback (ex: onClick) then assume it's being
3402
                // connected to manually.
3403
                this._deferredConnects = dojo.clone(this._deferredConnects);
3404
                for(var attr in this.attributeMap){
3405
                        delete this._deferredConnects[attr]; // can't be in both attributeMap and _deferredConnects
3406
                }
3407
                for(attr in this._deferredConnects){
3408
                        if(this[attr] !== dijit._connectOnUseEventHandler){
3409
                                delete this._deferredConnects[attr];        // redefined, probably dojoAttachEvent exists
3410
                        }
3411
                }
3412
3413
                //mixin our passed parameters
3414
                if(this.srcNodeRef && (typeof this.srcNodeRef.id == "string")){ this.id = this.srcNodeRef.id; }
3415
                if(params){
3416
                        this.params = params;
3417
                        dojo.mixin(this,params);
3418
                }
3419
                this.postMixInProperties();
3420
3421
                // generate an id for the widget if one wasn't specified
3422
                // (be sure to do this before buildRendering() because that function might
3423
                // expect the id to be there.)
3424
                if(!this.id){
3425
                        this.id = dijit.getUniqueId(this.declaredClass.replace(/\./g,"_"));
3426
                }
3427
                dijit.registry.add(this);
3428
3429
                this.buildRendering();
3430
3431
                if(this.domNode){
3432
                        // Copy attributes listed in attributeMap into the [newly created] DOM for the widget.
3433
                        this._applyAttributes();
3434
3435
                        var source = this.srcNodeRef;
3436
                        if(source && source.parentNode){
3437
                                source.parentNode.replaceChild(this.domNode, source);
3438
                        }
3439
3440
                        // If the developer has specified a handler as a widget parameter
3441
                        // (ex: new Button({onClick: ...})
3442
                        // then naturally need to connect from DOM node to that handler immediately,
3443
                        for(attr in this.params){
3444
                                this._onConnect(attr);
3445
                        }
3446
                }
3447
3448
                if(this.domNode){
3449
                        this.domNode.setAttribute("widgetId", this.id);
3450
                }
3451
                this.postCreate();
3452
3453
                // If srcNodeRef has been processed and removed from the DOM (e.g. TemplatedWidget) then delete it to allow GC.
3454
                if(this.srcNodeRef && !this.srcNodeRef.parentNode){
3455
                        delete this.srcNodeRef;
3456
                }
3457
3458
                this._created = true;
3459
        },
3460
3461
        _applyAttributes: function(){
3462
                // summary:
3463
                //                Step during widget creation to copy all widget attributes to the
3464
                //                DOM as per attributeMap and _setXXXAttr functions.
3465
                // description:
3466
                //                Skips over blank/false attribute values, unless they were explicitly specified
3467
                //                as parameters to the widget, since those are the default anyway,
3468
                //                and setting tabIndex="" is different than not setting tabIndex at all.
3469
                //
3470
                //                It processes the attributes in the attribute map first, and then
3471
                //                it goes through and processes the attributes for the _setXXXAttr
3472
                //                functions that have been specified
3473
                // tags:
3474
                //                private
3475
                var condAttrApply = function(attr, scope){
3476
                        if((scope.params && attr in scope.params) || scope[attr]){
3477
                                scope.set(attr, scope[attr]);
3478
                        }
3479
                };
3480
3481
                // Do the attributes in attributeMap
3482
                for(var attr in this.attributeMap){
3483
                        condAttrApply(attr, this);
3484
                }
3485
3486
                // And also any attributes with custom setters
3487
                dojo.forEach(getSetterAttributes(this), function(a){
3488
                        if(!(a in this.attributeMap)){
3489
                                condAttrApply(a, this);
3490
                        }
3491
                }, this);
3492
        },
3493
3494
        postMixInProperties: function(){
3495
                // summary:
3496
                //                Called after the parameters to the widget have been read-in,
3497
                //                but before the widget template is instantiated. Especially
3498
                //                useful to set properties that are referenced in the widget
3499
                //                template.
3500
                // tags:
3501
                //                protected
3502
        },
3503
3504
        buildRendering: function(){
3505
                // summary:
3506
                //                Construct the UI for this widget, setting this.domNode
3507
                // description:
3508
                //                Most widgets will mixin `dijit._Templated`, which implements this
3509
                //                method.
3510
                // tags:
3511
                //                protected
3512
                this.domNode = this.srcNodeRef || dojo.create('div');
3513
        },
3514
3515
        postCreate: function(){
3516
                // summary:
3517
                //                Processing after the DOM fragment is created
3518
                // description:
3519
                //                Called after the DOM fragment has been created, but not necessarily
3520
                //                added to the document.  Do not include any operations which rely on
3521
                //                node dimensions or placement.
3522
                // tags:
3523
                //                protected
3524
3525
                // baseClass is a single class name or occasionally a space-separated list of names.
3526
                // Add those classes to the DOMNod.  If RTL mode then also add with Rtl suffix.
3527
                if(this.baseClass){
3528
                        var classes = this.baseClass.split(" ");
3529
                        if(!this.isLeftToRight()){
3530
                                classes = classes.concat( dojo.map(classes, function(name){ return name+"Rtl"; }));
3531
                        }
3532
                        dojo.addClass(this.domNode, classes);
3533
                }
3534
        },
3535
3536
        startup: function(){
3537
                // summary:
3538
                //                Processing after the DOM fragment is added to the document
3539
                // description:
3540
                //                Called after a widget and its children have been created and added to the page,
3541
                //                and all related widgets have finished their create() cycle, up through postCreate().
3542
                //                This is useful for composite widgets that need to control or layout sub-widgets.
3543
                //                Many layout widgets can use this as a wiring phase.
3544
                this._started = true;
3545
        },
3546
3547
        //////////// DESTROY FUNCTIONS ////////////////////////////////
3548
3549
        destroyRecursive: function(/*Boolean?*/ preserveDom){
3550
                // summary:
3551
                //                 Destroy this widget and its descendants
3552
                // description:
3553
                //                This is the generic "destructor" function that all widget users
3554
                //                 should call to cleanly discard with a widget. Once a widget is
3555
                //                 destroyed, it is removed from the manager object.
3556
                // preserveDom:
3557
                //                If true, this method will leave the original DOM structure
3558
                //                alone of descendant Widgets. Note: This will NOT work with
3559
                //                dijit._Templated widgets.
3560
3561
                this._beingDestroyed = true;
3562
                this.destroyDescendants(preserveDom);
3563
                this.destroy(preserveDom);
3564
        },
3565
3566
        destroy: function(/*Boolean*/ preserveDom){
3567
                // summary:
3568
                //                 Destroy this widget, but not its descendants.
3569
                //                This method will, however, destroy internal widgets such as those used within a template.
3570
                // preserveDom: Boolean
3571
                //                If true, this method will leave the original DOM structure alone.
3572
                //                Note: This will not yet work with _Templated widgets
3573
3574
                this._beingDestroyed = true;
3575
                this.uninitialize();
3576
                var d = dojo,
3577
                        dfe = d.forEach,
3578
                        dun = d.unsubscribe;
3579
                dfe(this._connects, function(array){
3580
                        dfe(array, d.disconnect);
3581
                });
3582
                dfe(this._subscribes, function(handle){
3583
                        dun(handle);
3584
                });
3585
3586
                // destroy widgets created as part of template, etc.
3587
                dfe(this._supportingWidgets || [], function(w){
3588
                        if(w.destroyRecursive){
3589
                                w.destroyRecursive();
3590
                        }else if(w.destroy){
3591
                                w.destroy();
3592
                        }
3593
                });
3594
3595
                this.destroyRendering(preserveDom);
3596
                dijit.registry.remove(this.id);
3597
                this._destroyed = true;
3598
        },
3599
3600
        destroyRendering: function(/*Boolean?*/ preserveDom){
3601
                // summary:
3602
                //                Destroys the DOM nodes associated with this widget
3603
                // preserveDom:
3604
                //                If true, this method will leave the original DOM structure alone
3605
                //                during tear-down. Note: this will not work with _Templated
3606
                //                widgets yet.
3607
                // tags:
3608
                //                protected
3609
3610
                if(this.bgIframe){
3611
                        this.bgIframe.destroy(preserveDom);
3612
                        delete this.bgIframe;
3613
                }
3614
3615
                if(this.domNode){
3616
                        if(preserveDom){
3617
                                dojo.removeAttr(this.domNode, "widgetId");
3618
                        }else{
3619
                                dojo.destroy(this.domNode);
3620
                        }
3621
                        delete this.domNode;
3622
                }
3623
3624
                if(this.srcNodeRef){
3625
                        if(!preserveDom){
3626
                                dojo.destroy(this.srcNodeRef);
3627
                        }
3628
                        delete this.srcNodeRef;
3629
                }
3630
        },
3631
3632
        destroyDescendants: function(/*Boolean?*/ preserveDom){
3633
                // summary:
3634
                //                Recursively destroy the children of this widget and their
3635
                //                descendants.
3636
                // preserveDom:
3637
                //                If true, the preserveDom attribute is passed to all descendant
3638
                //                widget's .destroy() method. Not for use with _Templated
3639
                //                widgets.
3640
3641
                // get all direct descendants and destroy them recursively
3642
                dojo.forEach(this.getChildren(), function(widget){
3643
                        if(widget.destroyRecursive){
3644
                                widget.destroyRecursive(preserveDom);
3645
                        }
3646
                });
3647
        },
3648
3649
3650
        uninitialize: function(){
3651
                // summary:
3652
                //                Stub function. Override to implement custom widget tear-down
3653
                //                behavior.
3654
                // tags:
3655
                //                protected
3656
                return false;
3657
        },
3658
3659
        ////////////////// MISCELLANEOUS METHODS ///////////////////
3660
3661
        onFocus: function(){
3662
                // summary:
3663
                //                Called when the widget becomes "active" because
3664
                //                it or a widget inside of it either has focus, or has recently
3665
                //                been clicked.
3666
                // tags:
3667
                //                callback
3668
        },
3669
3670
        onBlur: function(){
3671
                // summary:
3672
                //                Called when the widget stops being "active" because
3673
                //                focus moved to something outside of it, or the user
3674
                //                clicked somewhere outside of it, or the widget was
3675
                //                hidden.
3676
                // tags:
3677
                //                callback
3678
        },
3679
3680
        _onFocus: function(e){
3681
                // summary:
3682
                //                This is where widgets do processing for when they are active,
3683
                //                such as changing CSS classes.  See onFocus() for more details.
3684
                // tags:
3685
                //                protected
3686
                this.onFocus();
3687
        },
3688
3689
        _onBlur: function(){
3690
                // summary:
3691
                //                This is where widgets do processing for when they stop being active,
3692
                //                such as changing CSS classes.  See onBlur() for more details.
3693
                // tags:
3694
                //                protected
3695
                this.onBlur();
3696
        },
3697
3698
        _onConnect: function(/*String*/ event){
3699
                // summary:
3700
                //                Called when someone connects to one of my handlers.
3701
                //                "Turn on" that handler if it isn't active yet.
3702
                //
3703
                //                This is also called for every single initialization parameter
3704
                //                so need to do nothing for parameters like "id".
3705
                // tags:
3706
                //                private
3707
                if(event in this._deferredConnects){
3708
                        var mapNode = this[this._deferredConnects[event] || 'domNode'];
3709
                        this.connect(mapNode, event.toLowerCase(), event);
3710
                        delete this._deferredConnects[event];
3711
                }
3712
        },
3713
3714
        _setClassAttr: function(/*String*/ value){
3715
                // summary:
3716
                //                Custom setter for the CSS "class" attribute
3717
                // tags:
3718
                //                protected
3719
                var mapNode = this[this.attributeMap["class"] || 'domNode'];
3720
                dojo.removeClass(mapNode, this["class"])
3721
                this["class"] = value;
3722
                dojo.addClass(mapNode, value);
3723
        },
3724
3725
        _setStyleAttr: function(/*String||Object*/ value){
3726
                // summary:
3727
                //                Sets the style attribut of the widget according to value,
3728
                //                which is either a hash like {height: "5px", width: "3px"}
3729
                //                or a plain string
3730
                // description:
3731
                //                Determines which node to set the style on based on style setting
3732
                //                in attributeMap.
3733
                // tags:
3734
                //                protected
3735
3736
                var mapNode = this[this.attributeMap.style || 'domNode'];
3737
3738
                // Note: technically we should revert any style setting made in a previous call
3739
                // to his method, but that's difficult to keep track of.
3740
3741
                if(dojo.isObject(value)){
3742
                        dojo.style(mapNode, value);
3743
                }else{
3744
                        if(mapNode.style.cssText){
3745
                                mapNode.style.cssText += "; " + value;
3746
                        }else{
3747
                                mapNode.style.cssText = value;
3748
                        }
3749
                }
3750
3751
                this.style = value;
3752
        },
3753
3754
        setAttribute: function(/*String*/ attr, /*anything*/ value){
3755
                // summary:
3756
                //                Deprecated.  Use set() instead.
3757
                // tags:
3758
                //                deprecated
3759
                dojo.deprecated(this.declaredClass+"::setAttribute(attr, value) is deprecated. Use set() instead.", "", "2.0");
3760
                this.set(attr, value);
3761
        },
3762
3763
        _attrToDom: function(/*String*/ attr, /*String*/ value){
3764
                // summary:
3765
                //                Reflect a widget attribute (title, tabIndex, duration etc.) to
3766
                //                the widget DOM, as specified in attributeMap.
3767
                //
3768
                // description:
3769
                //                Also sets this["attr"] to the new value.
3770
                //                Note some attributes like "type"
3771
                //                cannot be processed this way as they are not mutable.
3772
                //
3773
                // tags:
3774
                //                private
3775
3776
                var commands = this.attributeMap[attr];
3777
                dojo.forEach(dojo.isArray(commands) ? commands : [commands], function(command){
3778
3779
                        // Get target node and what we are doing to that node
3780
                        var mapNode = this[command.node || command || "domNode"];        // DOM node
3781
                        var type = command.type || "attribute";        // class, innerHTML, innerText, or attribute
3782
3783
                        switch(type){
3784
                                case "attribute":
3785
                                        if(dojo.isFunction(value)){ // functions execute in the context of the widget
3786
                                                value = dojo.hitch(this, value);
3787
                                        }
3788
3789
                                        // Get the name of the DOM node attribute; usually it's the same
3790
                                        // as the name of the attribute in the widget (attr), but can be overridden.
3791
                                        // Also maps handler names to lowercase, like onSubmit --> onsubmit
3792
                                        var attrName = command.attribute ? command.attribute :
3793
                                                (/^on[A-Z][a-zA-Z]*$/.test(attr) ? attr.toLowerCase() : attr);
3794
3795
                                        dojo.attr(mapNode, attrName, value);
3796
                                        break;
3797
                                case "innerText":
3798
                                        mapNode.innerHTML = "";
3799
                                        mapNode.appendChild(dojo.doc.createTextNode(value));
3800
                                        break;
3801
                                case "innerHTML":
3802
                                        mapNode.innerHTML = value;
3803
                                        break;
3804
                                case "class":
3805
                                        dojo.removeClass(mapNode, this[attr]);
3806
                                        dojo.addClass(mapNode, value);
3807
                                        break;
3808
                        }
3809
                }, this);
3810
                this[attr] = value;
3811
        },
3812
3813
        attr: function(/*String|Object*/name, /*Object?*/value){
3814
                // summary:
3815
                //                Set or get properties on a widget instance.
3816
                //        name:
3817
                //                The property to get or set. If an object is passed here and not
3818
                //                a string, its keys are used as names of attributes to be set
3819
                //                and the value of the object as values to set in the widget.
3820
                //        value:
3821
                //                Optional. If provided, attr() operates as a setter. If omitted,
3822
                //                the current value of the named property is returned.
3823
                // description:
3824
                //                This method is deprecated, use get() or set() directly.
3825
3826
                // Print deprecation warning but only once per calling function
3827
                if(dojo.config.isDebug){
3828
                        var alreadyCalledHash = arguments.callee._ach || (arguments.callee._ach = {}),
3829
                                caller = (arguments.callee.caller || "unknown caller").toString();
3830
                        if(!alreadyCalledHash[caller]){
3831
                                dojo.deprecated(this.declaredClass + "::attr() is deprecated. Use get() or set() instead, called from " +
3832
                                caller, "", "2.0");
3833
                                alreadyCalledHash[caller] = true;
3834
                        }
3835
                }
3836
3837
                var args = arguments.length;
3838
                if(args >= 2 || typeof name === "object"){ // setter
3839
                        return this.set.apply(this, arguments);
3840
                }else{ // getter
3841
                        return this.get(name);
3842
                }
3843
        },
3844
3845
        get: function(name){
3846
                // summary:
3847
                //                Get a property from a widget.
3848
                //        name:
3849
                //                The property to get.
3850
                // description:
3851
                //                Get a named property from a widget. The property may
3852
                //                potentially be retrieved via a getter method. If no getter is defined, this
3853
                //                 just retrieves the object's property.
3854
                //                 For example, if the widget has a properties "foo"
3855
                //                and "bar" and a method named "_getFooAttr", calling:
3856
                //        |        myWidget.get("foo");
3857
                //                would be equivalent to writing:
3858
                //        |        widget._getFooAttr();
3859
                //                and:
3860
                //        |        myWidget.get("bar");
3861
                //                would be equivalent to writing:
3862
                //        |        widget.bar;
3863
                var names = this._getAttrNames(name);
3864
                return this[names.g] ? this[names.g]() : this[name];
3865
        },
3866
3867
        set: function(name, value){
3868
                // summary:
3869
                //                Set a property on a widget
3870
                //        name:
3871
                //                The property to set.
3872
                //        value:
3873
                //                The value to set in the property.
3874
                // description:
3875
                //                Sets named properties on a widget which may potentially be handled by a
3876
                //                 setter in the widget.
3877
                //                 For example, if the widget has a properties "foo"
3878
                //                and "bar" and a method named "_setFooAttr", calling:
3879
                //        |        myWidget.set("foo", "Howdy!");
3880
                //                would be equivalent to writing:
3881
                //        |        widget._setFooAttr("Howdy!");
3882
                //                and:
3883
                //        |        myWidget.set("bar", 3);
3884
                //                would be equivalent to writing:
3885
                //        |        widget.bar = 3;
3886
                //
3887
                //        set() may also be called with a hash of name/value pairs, ex:
3888
                //        |        myWidget.set({
3889
                //        |                foo: "Howdy",
3890
                //        |                bar: 3
3891
                //        |        })
3892
                //        This is equivalent to calling set(foo, "Howdy") and set(bar, 3)
3893
3894
                if(typeof name === "object"){
3895
                        for(var x in name){
3896
                                this.set(x, name[x]);
3897
                        }
3898
                        return this;
3899
                }
3900
                var names = this._getAttrNames(name);
3901
                if(this[names.s]){
3902
                        // use the explicit setter
3903
                        var result = this[names.s].apply(this, Array.prototype.slice.call(arguments, 1));
3904
                }else{
3905
                        // if param is specified as DOM node attribute, copy it
3906
                        if(name in this.attributeMap){
3907
                                this._attrToDom(name, value);
3908
                        }
3909
                        var oldValue = this[name];
3910
                        // FIXME: what about function assignments? Any way to connect() here?
3911
                        this[name] = value;
3912
                }
3913
                return result || this;
3914
        },
3915
3916
        _attrPairNames: {},                // shared between all widgets
3917
        _getAttrNames: function(name){
3918
                // summary:
3919
                //                Helper function for get() and set().
3920
                //                Caches attribute name values so we don't do the string ops every time.
3921
                // tags:
3922
                //                private
3923
3924
                var apn = this._attrPairNames;
3925
                if(apn[name]){ return apn[name]; }
3926
                var uc = name.charAt(0).toUpperCase() + name.substr(1);
3927
                return (apn[name] = {
3928
                        n: name+"Node",
3929
                        s: "_set"+uc+"Attr",
3930
                        g: "_get"+uc+"Attr"
3931
                });
3932
        },
3933
3934
        toString: function(){
3935
                // summary:
3936
                //                Returns a string that represents the widget
3937
                // description:
3938
                //                When a widget is cast to a string, this method will be used to generate the
3939
                //                output. Currently, it does not implement any sort of reversible
3940
                //                serialization.
3941
                return '[Widget ' + this.declaredClass + ', ' + (this.id || 'NO ID') + ']'; // String
3942
        },
3943
3944
        getDescendants: function(){
3945
                // summary:
3946
                //                Returns all the widgets contained by this, i.e., all widgets underneath this.containerNode.
3947
                //                This method should generally be avoided as it returns widgets declared in templates, which are
3948
                //                supposed to be internal/hidden, but it's left here for back-compat reasons.
3949
3950
                return this.containerNode ? dojo.query('[widgetId]', this.containerNode).map(dijit.byNode) : []; // dijit._Widget[]
3951
        },
3952
3953
        getChildren: function(){
3954
                // summary:
3955
                //                Returns all the widgets contained by this, i.e., all widgets underneath this.containerNode.
3956
                //                Does not return nested widgets, nor widgets that are part of this widget's template.
3957
                return this.containerNode ? dijit.findWidgets(this.containerNode) : []; // dijit._Widget[]
3958
        },
3959
3960
        // nodesWithKeyClick: [private] String[]
3961
        //                List of nodes that correctly handle click events via native browser support,
3962
        //                and don't need dijit's help
3963
        nodesWithKeyClick: ["input", "button"],
3964
3965
        connect: function(
3966
                        /*Object|null*/ obj,
3967
                        /*String|Function*/ event,
3968
                        /*String|Function*/ method){
3969
                // summary:
3970
                //                Connects specified obj/event to specified method of this object
3971
                //                and registers for disconnect() on widget destroy.
3972
                // description:
3973
                //                Provide widget-specific analog to dojo.connect, except with the
3974
                //                implicit use of this widget as the target object.
3975
                //                This version of connect also provides a special "ondijitclick"
3976
                //                event which triggers on a click or space or enter keyup
3977
                // returns:
3978
                //                A handle that can be passed to `disconnect` in order to disconnect before
3979
                //                the widget is destroyed.
3980
                // example:
3981
                //        |        var btn = new dijit.form.Button();
3982
                //        |        // when foo.bar() is called, call the listener we're going to
3983
                //        |        // provide in the scope of btn
3984
                //        |        btn.connect(foo, "bar", function(){
3985
                //        |                console.debug(this.toString());
3986
                //        |        });
3987
                // tags:
3988
                //                protected
3989
3990
                var d = dojo,
3991
                        dc = d._connect,
3992
                        handles = [];
3993
                if(event == "ondijitclick"){
3994
                        // add key based click activation for unsupported nodes.
3995
                        // do all processing onkey up to prevent spurious clicks
3996
                        // for details see comments at top of this file where _lastKeyDownNode is defined
3997
                        if(dojo.indexOf(this.nodesWithKeyClick, obj.nodeName.toLowerCase()) == -1){ // is NOT input or button
3998
                                var m = d.hitch(this, method);
3999
                                handles.push(
4000
                                        dc(obj, "onkeydown", this, function(e){
4001
                                                //console.log(this.id + ": onkeydown, e.target = ", e.target, ", lastKeyDownNode was ", dijit._lastKeyDownNode, ", equality is ", (e.target === dijit._lastKeyDownNode));
4002
                                                if((e.keyCode == d.keys.ENTER || e.keyCode == d.keys.SPACE) &&
4003
                                                        !e.ctrlKey && !e.shiftKey && !e.altKey && !e.metaKey){
4004
                                                        // needed on IE for when focus changes between keydown and keyup - otherwise dropdown menus do not work
4005
                                                        dijit._lastKeyDownNode = e.target;
4006
                                                        e.preventDefault();                // stop event to prevent scrolling on space key in IE
4007
                                                }
4008
                                         }),
4009
                                        dc(obj, "onkeyup", this, function(e){
4010
                                                //console.log(this.id + ": onkeyup, e.target = ", e.target, ", lastKeyDownNode was ", dijit._lastKeyDownNode, ", equality is ", (e.target === dijit._lastKeyDownNode));
4011
                                                if( (e.keyCode == d.keys.ENTER || e.keyCode == d.keys.SPACE) &&
4012
                                                        e.target === dijit._lastKeyDownNode &&
4013
                                                        !e.ctrlKey && !e.shiftKey && !e.altKey && !e.metaKey){
4014
                                                                //need reset here or have problems in FF when focus returns to trigger element after closing popup/alert
4015
                                                                dijit._lastKeyDownNode = null;
4016
                                                                return m(e);
4017
                                                }
4018
                                        })
4019
                                );
4020
                        }
4021
                        event = "onclick";
4022
                }
4023
                handles.push(dc(obj, event, this, method));
4024
4025
                this._connects.push(handles);
4026
                return handles;                // _Widget.Handle
4027
        },
4028
4029
        disconnect: function(/* _Widget.Handle */ handles){
4030
                // summary:
4031
                //                Disconnects handle created by `connect`.
4032
                //                Also removes handle from this widget's list of connects.
4033
                // tags:
4034
                //                protected
4035
                for(var i=0; i<this._connects.length; i++){
4036
                        if(this._connects[i] == handles){
4037
                                dojo.forEach(handles, dojo.disconnect);
4038
                                this._connects.splice(i, 1);
4039
                                return;
4040
                        }
4041
                }
4042
        },
4043
4044
        subscribe: function(
4045
                        /*String*/ topic,
4046
                        /*String|Function*/ method){
4047
                // summary:
4048
                //                Subscribes to the specified topic and calls the specified method
4049
                //                of this object and registers for unsubscribe() on widget destroy.
4050
                // description:
4051
                //                Provide widget-specific analog to dojo.subscribe, except with the
4052
                //                implicit use of this widget as the target object.
4053
                // example:
4054
                //        |        var btn = new dijit.form.Button();
4055
                //        |        // when /my/topic is published, this button changes its label to
4056
                //        |   // be the parameter of the topic.
4057
                //        |        btn.subscribe("/my/topic", function(v){
4058
                //        |                this.set("label", v);
4059
                //        |        });
4060
                var d = dojo,
4061
                        handle = d.subscribe(topic, this, method);
4062
4063
                // return handles for Any widget that may need them
4064
                this._subscribes.push(handle);
4065
                return handle;
4066
        },
4067
4068
        unsubscribe: function(/*Object*/ handle){
4069
                // summary:
4070
                //                Unsubscribes handle created by this.subscribe.
4071
                //                Also removes handle from this widget's list of subscriptions
4072
                for(var i=0; i<this._subscribes.length; i++){
4073
                        if(this._subscribes[i] == handle){
4074
                                dojo.unsubscribe(handle);
4075
                                this._subscribes.splice(i, 1);
4076
                                return;
4077
                        }
4078
                }
4079
        },
4080
4081
        isLeftToRight: function(){
4082
                // summary:
4083
                //                Return this widget's explicit or implicit orientation (true for LTR, false for RTL)
4084
                // tags:
4085
                //                protected
4086
                return this.dir ? (this.dir == "ltr") : dojo._isBodyLtr(); //Boolean
4087
        },
4088
4089
        isFocusable: function(){
4090
                // summary:
4091
                //                Return true if this widget can currently be focused
4092
                //                and false if not
4093
                return this.focus && (dojo.style(this.domNode, "display") != "none");
4094
        },
4095
4096
        placeAt: function(/* String|DomNode|_Widget */reference, /* String?|Int? */position){
4097
                // summary:
4098
                //                Place this widget's domNode reference somewhere in the DOM based
4099
                //                on standard dojo.place conventions, or passing a Widget reference that
4100
                //                contains and addChild member.
4101
                //
4102
                // description:
4103
                //                A convenience function provided in all _Widgets, providing a simple
4104
                //                shorthand mechanism to put an existing (or newly created) Widget
4105
                //                somewhere in the dom, and allow chaining.
4106
                //
4107
                // reference:
4108
                //                The String id of a domNode, a domNode reference, or a reference to a Widget posessing
4109
                //                an addChild method.
4110
                //
4111
                // position:
4112
                //                If passed a string or domNode reference, the position argument
4113
                //                accepts a string just as dojo.place does, one of: "first", "last",
4114
                //                "before", or "after".
4115
                //
4116
                //                If passed a _Widget reference, and that widget reference has an ".addChild" method,
4117
                //                it will be called passing this widget instance into that method, supplying the optional
4118
                //                position index passed.
4119
                //
4120
                // returns:
4121
                //                dijit._Widget
4122
                //                Provides a useful return of the newly created dijit._Widget instance so you
4123
                //                can "chain" this function by instantiating, placing, then saving the return value
4124
                //                to a variable.
4125
                //
4126
                // example:
4127
                // |         // create a Button with no srcNodeRef, and place it in the body:
4128
                // |         var button = new dijit.form.Button({ label:"click" }).placeAt(dojo.body());
4129
                // |         // now, 'button' is still the widget reference to the newly created button
4130
                // |         dojo.connect(button, "onClick", function(e){ console.log('click'); });
4131
                //
4132
                // example:
4133
                // |        // create a button out of a node with id="src" and append it to id="wrapper":
4134
                // |         var button = new dijit.form.Button({},"src").placeAt("wrapper");
4135
                //
4136
                // example:
4137
                // |        // place a new button as the first element of some div
4138
                // |        var button = new dijit.form.Button({ label:"click" }).placeAt("wrapper","first");
4139
                //
4140
                // example:
4141
                // |        // create a contentpane and add it to a TabContainer
4142
                // |        var tc = dijit.byId("myTabs");
4143
                // |        new dijit.layout.ContentPane({ href:"foo.html", title:"Wow!" }).placeAt(tc)
4144
4145
                if(reference.declaredClass && reference.addChild){
4146
                        reference.addChild(this, position);
4147
                }else{
4148
                        dojo.place(this.domNode, reference, position);
4149
                }
4150
                return this;
4151
        },
4152
4153
        _onShow: function(){
4154
                // summary:
4155
                //                Internal method called when this widget is made visible.
4156
                //                See `onShow` for details.
4157
                this.onShow();
4158
        },
4159
4160
        onShow: function(){
4161
                // summary:
4162
                //                Called when this widget becomes the selected pane in a
4163
                //                `dijit.layout.TabContainer`, `dijit.layout.StackContainer`,
4164
                //                `dijit.layout.AccordionContainer`, etc.
4165
                //
4166
                //                Also called to indicate display of a `dijit.Dialog`, `dijit.TooltipDialog`, or `dijit.TitlePane`.
4167
                // tags:
4168
                //                callback
4169
        },
4170
4171
        onHide: function(){
4172
                // summary:
4173
                        //                Called when another widget becomes the selected pane in a
4174
                        //                `dijit.layout.TabContainer`, `dijit.layout.StackContainer`,
4175
                        //                `dijit.layout.AccordionContainer`, etc.
4176
                        //
4177
                        //                Also called to indicate hide of a `dijit.Dialog`, `dijit.TooltipDialog`, or `dijit.TitlePane`.
4178
                        // tags:
4179
                        //                callback
4180
        },
4181
4182
        onClose: function(){
4183
                // summary:
4184
                //                Called when this widget is being displayed as a popup (ex: a Calendar popped
4185
                //                up from a DateTextBox), and it is hidden.
4186
                //                This is called from the dijit.popup code, and should not be called directly.
4187
                //
4188
                //                Also used as a parameter for children of `dijit.layout.StackContainer` or subclasses.
4189
                //                Callback if a user tries to close the child.   Child will be closed if this function returns true.
4190
                // tags:
4191
                //                extension
4192
4193
                return true;                // Boolean
4194
        }
4195
});
4196
4197
})();
4198
4199
}
4200
4201
if(!dojo._hasResource["dojo.string"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
4202
dojo._hasResource["dojo.string"] = true;
4203
dojo.provide("dojo.string");
4204
4205
/*=====
4206
dojo.string = {
4207
        // summary: String utilities for Dojo
4208
};
4209
=====*/
4210
4211
dojo.string.rep = function(/*String*/str, /*Integer*/num){
4212
        //        summary:
4213
        //                Efficiently replicate a string `n` times.
4214
        //        str:
4215
        //                the string to replicate
4216
        //        num:
4217
        //                number of times to replicate the string
4218
4219
        if(num <= 0 || !str){ return ""; }
4220
4221
        var buf = [];
4222
        for(;;){
4223
                if(num & 1){
4224
                        buf.push(str);
4225
                }
4226
                if(!(num >>= 1)){ break; }
4227
                str += str;
4228
        }
4229
        return buf.join("");        // String
4230
};
4231
4232
dojo.string.pad = function(/*String*/text, /*Integer*/size, /*String?*/ch, /*Boolean?*/end){
4233
        //        summary:
4234
        //                Pad a string to guarantee that it is at least `size` length by
4235
        //                filling with the character `ch` at either the start or end of the
4236
        //                string. Pads at the start, by default.
4237
        //        text:
4238
        //                the string to pad
4239
        //        size:
4240
        //                length to provide padding
4241
        //        ch:
4242
        //                character to pad, defaults to '0'
4243
        //        end:
4244
        //                adds padding at the end if true, otherwise pads at start
4245
        //        example:
4246
        //        |        // Fill the string to length 10 with "+" characters on the right.  Yields "Dojo++++++".
4247
        //        |        dojo.string.pad("Dojo", 10, "+", true);
4248
4249
        if(!ch){
4250
                ch = '0';
4251
        }
4252
        var out = String(text),
4253
                pad = dojo.string.rep(ch, Math.ceil((size - out.length) / ch.length));
4254
        return end ? out + pad : pad + out;        // String
4255
};
4256
4257
dojo.string.substitute = function(        /*String*/                template,
4258
                                                                        /*Object|Array*/map,
4259
                                                                        /*Function?*/        transform,
4260
                                                                        /*Object?*/                thisObject){
4261
        //        summary:
4262
        //                Performs parameterized substitutions on a string. Throws an
4263
        //                exception if any parameter is unmatched.
4264
        //        template:
4265
        //                a string with expressions in the form `${key}` to be replaced or
4266
        //                `${key:format}` which specifies a format function. keys are case-sensitive.
4267
        //        map:
4268
        //                hash to search for substitutions
4269
        //        transform:
4270
        //                a function to process all parameters before substitution takes
4271
        //                place, e.g. mylib.encodeXML
4272
        //        thisObject:
4273
        //                where to look for optional format function; default to the global
4274
        //                namespace
4275
        //        example:
4276
        //                Substitutes two expressions in a string from an Array or Object
4277
        //        |        // returns "File 'foo.html' is not found in directory '/temp'."
4278
        //        |        // by providing substitution data in an Array
4279
        //        |        dojo.string.substitute(
4280
        //        |                "File '${0}' is not found in directory '${1}'.",
4281
        //        |                ["foo.html","/temp"]
4282
        //        |        );
4283
        //        |
4284
        //        |        // also returns "File 'foo.html' is not found in directory '/temp'."
4285
        //        |        // but provides substitution data in an Object structure.  Dotted
4286
        //        |        // notation may be used to traverse the structure.
4287
        //        |        dojo.string.substitute(
4288
        //        |                "File '${name}' is not found in directory '${info.dir}'.",
4289
        //        |                { name: "foo.html", info: { dir: "/temp" } }
4290
        //        |        );
4291
        //        example:
4292
        //                Use a transform function to modify the values:
4293
        //        |        // returns "file 'foo.html' is not found in directory '/temp'."
4294
        //        |        dojo.string.substitute(
4295
        //        |                "${0} is not found in ${1}.",
4296
        //        |                ["foo.html","/temp"],
4297
        //        |                function(str){
4298
        //        |                        // try to figure out the type
4299
        //        |                        var prefix = (str.charAt(0) == "/") ? "directory": "file";
4300
        //        |                        return prefix + " '" + str + "'";
4301
        //        |                }
4302
        //        |        );
4303
        //        example:
4304
        //                Use a formatter
4305
        //        |        // returns "thinger -- howdy"
4306
        //        |        dojo.string.substitute(
4307
        //        |                "${0:postfix}", ["thinger"], null, {
4308
        //        |                        postfix: function(value, key){
4309
        //        |                                return value + " -- howdy";
4310
        //        |                        }
4311
        //        |                }
4312
        //        |        );
4313
4314
        thisObject = thisObject || dojo.global;
4315
        transform = transform ?
4316
                dojo.hitch(thisObject, transform) : function(v){ return v; };
4317
4318
        return template.replace(/\$\{([^\s\:\}]+)(?:\:([^\s\:\}]+))?\}/g,
4319
                function(match, key, format){
4320
                        var value = dojo.getObject(key, false, map);
4321
                        if(format){
4322
                                value = dojo.getObject(format, false, thisObject).call(thisObject, value, key);
4323
                        }
4324
                        return transform(value, key).toString();
4325
                }); // String
4326
};
4327
4328
/*=====
4329
dojo.string.trim = function(str){
4330
        //        summary:
4331
        //                Trims whitespace from both sides of the string
4332
        //        str: String
4333
        //                String to be trimmed
4334
        //        returns: String
4335
        //                Returns the trimmed string
4336
        //        description:
4337
        //                This version of trim() was taken from [Steven Levithan's blog](http://blog.stevenlevithan.com/archives/faster-trim-javascript).
4338
        //                The short yet performant version of this function is dojo.trim(),
4339
        //                which is part of Dojo base.  Uses String.prototype.trim instead, if available.
4340
        return "";        // String
4341
}
4342
=====*/
4343
4344
dojo.string.trim = String.prototype.trim ?
4345
        dojo.trim : // aliasing to the native function
4346
        function(str){
4347
                str = str.replace(/^\s+/, '');
4348
                for(var i = str.length - 1; i >= 0; i--){
4349
                        if(/\S/.test(str.charAt(i))){
4350
                                str = str.substring(0, i + 1);
4351
                                break;
4352
                        }
4353
                }
4354
                return str;
4355
        };
4356
4357
}
4358
4359
if(!dojo._hasResource["dojo.cache"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
4360
dojo._hasResource["dojo.cache"] = true;
4361
dojo.provide("dojo.cache");
4362
4363
/*=====
4364
dojo.cache = {
4365
        // summary:
4366
        //                 A way to cache string content that is fetchable via `dojo.moduleUrl`.
4367
};
4368
=====*/
4369
4370
(function(){
4371
        var cache = {};
4372
        dojo.cache = function(/*String||Object*/module, /*String*/url, /*String||Object?*/value){
4373
                // summary:
4374
                //                 A getter and setter for storing the string content associated with the
4375
                //                 module and url arguments.
4376
                // description:
4377
                //                 module and url are used to call `dojo.moduleUrl()` to generate a module URL.
4378
                //                 If value is specified, the cache value for the moduleUrl will be set to
4379
                //                 that value. Otherwise, dojo.cache will fetch the moduleUrl and store it
4380
                //                 in its internal cache and return that cached value for the URL. To clear
4381
                //                 a cache value pass null for value. Since XMLHttpRequest (XHR) is used to fetch the
4382
                //                 the URL contents, only modules on the same domain of the page can use this capability.
4383
                //                 The build system can inline the cache values though, to allow for xdomain hosting.
4384
                // module: String||Object
4385
                //                 If a String, the module name to use for the base part of the URL, similar to module argument
4386
                //                 to `dojo.moduleUrl`. If an Object, something that has a .toString() method that
4387
                //                 generates a valid path for the cache item. For example, a dojo._Url object.
4388
                // url: String
4389
                //                 The rest of the path to append to the path derived from the module argument. If
4390
                //                 module is an object, then this second argument should be the "value" argument instead.
4391
                // value: String||Object?
4392
                //                 If a String, the value to use in the cache for the module/url combination.
4393
                //                 If an Object, it can have two properties: value and sanitize. The value property
4394
                //                 should be the value to use in the cache, and sanitize can be set to true or false,
4395
                //                 to indicate if XML declarations should be removed from the value and if the HTML
4396
                //                 inside a body tag in the value should be extracted as the real value. The value argument
4397
                //                 or the value property on the value argument are usually only used by the build system
4398
                //                 as it inlines cache content.
4399
                //        example:
4400
                //                To ask dojo.cache to fetch content and store it in the cache (the dojo["cache"] style
4401
                //                 of call is used to avoid an issue with the build system erroneously trying to intern
4402
                //                 this example. To get the build system to intern your dojo.cache calls, use the
4403
                //                 "dojo.cache" style of call):
4404
                //                 |        //If template.html contains "<h1>Hello</h1>" that will be
4405
                //                 |        //the value for the text variable.
4406
                //                |        var text = dojo["cache"]("my.module", "template.html");
4407
                //        example:
4408
                //                To ask dojo.cache to fetch content and store it in the cache, and sanitize the input
4409
                //                  (the dojo["cache"] style of call is used to avoid an issue with the build system
4410
                //                 erroneously trying to intern this example. To get the build system to intern your
4411
                //                 dojo.cache calls, use the "dojo.cache" style of call):
4412
                //                 |        //If template.html contains "<html><body><h1>Hello</h1></body></html>", the
4413
                //                 |        //text variable will contain just "<h1>Hello</h1>".
4414
                //                |        var text = dojo["cache"]("my.module", "template.html", {sanitize: true});
4415
                //        example:
4416
                //                Same example as previous, but demostrates how an object can be passed in as
4417
                //                the first argument, then the value argument can then be the second argument.
4418
                //                 |        //If template.html contains "<html><body><h1>Hello</h1></body></html>", the
4419
                //                 |        //text variable will contain just "<h1>Hello</h1>".
4420
                //                |        var text = dojo["cache"](new dojo._Url("my/module/template.html"), {sanitize: true});
4421
4422
                //Module could be a string, or an object that has a toString() method
4423
                //that will return a useful path. If it is an object, then the "url" argument
4424
                //will actually be the value argument.
4425
                if(typeof module == "string"){
4426
                        var pathObj = dojo.moduleUrl(module, url);
4427
                }else{
4428
                        pathObj = module;
4429
                        value = url;
4430
                }
4431
                var key = pathObj.toString();
4432
4433
                var val = value;
4434
                if(value != undefined && !dojo.isString(value)){
4435
                        val = ("value" in value ? value.value : undefined);
4436
                }
4437
4438
                var sanitize = value && value.sanitize ? true : false;
4439
4440
                if(typeof val == "string"){
4441
                        //We have a string, set cache value
4442
                        val = cache[key] = sanitize ? dojo.cache._sanitize(val) : val;
4443
                }else if(val === null){
4444
                        //Remove cached value
4445
                        delete cache[key];
4446
                }else{
4447
                        //Allow cache values to be empty strings. If key property does
4448
                        //not exist, fetch it.
4449
                        if(!(key in cache)){
4450
                                val = dojo._getText(key);
4451
                                cache[key] = sanitize ? dojo.cache._sanitize(val) : val;
4452
                        }
4453
                        val = cache[key];
4454
                }
4455
                return val; //String
4456
        };
4457
4458
        dojo.cache._sanitize = function(/*String*/val){
4459
                // summary:
4460
                //                Strips <?xml ...?> declarations so that external SVG and XML
4461
                //                 documents can be added to a document without worry. Also, if the string
4462
                //                is an HTML document, only the part inside the body tag is returned.
4463
                // description:
4464
                //                 Copied from dijit._Templated._sanitizeTemplateString.
4465
                if(val){
4466
                        val = val.replace(/^\s*<\?xml(\s)+version=[\'\"](\d)*.(\d)*[\'\"](\s)*\?>/im, "");
4467
                        var matches = val.match(/<body[^>]*>\s*([\s\S]+)\s*<\/body>/im);
4468
                        if(matches){
4469
                                val = matches[1];
4470
                        }
4471
                }else{
4472
                        val = "";
4473
                }
4474
                return val; //String
4475
        };
4476
})();
4477
4478
}
4479
4480
if(!dojo._hasResource["dijit._Templated"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
4481
dojo._hasResource["dijit._Templated"] = true;
4482
dojo.provide("dijit._Templated");
4483
4484
4485
4486
4487
4488
4489
dojo.declare("dijit._Templated",
4490
        null,
4491
        {
4492
                // summary:
4493
                //                Mixin for widgets that are instantiated from a template
4494
4495
                // templateString: [protected] String
4496
                //                A string that represents the widget template. Pre-empts the
4497
                //                templatePath. In builds that have their strings "interned", the
4498
                //                templatePath is converted to an inline templateString, thereby
4499
                //                preventing a synchronous network call.
4500
                //
4501
                //                Use in conjunction with dojo.cache() to load from a file.
4502
                templateString: null,
4503
4504
                // templatePath: [protected deprecated] String
4505
                //                Path to template (HTML file) for this widget relative to dojo.baseUrl.
4506
                //                Deprecated: use templateString with dojo.cache() instead.
4507
                templatePath: null,
4508
4509
                // widgetsInTemplate: [protected] Boolean
4510
                //                Should we parse the template to find widgets that might be
4511
                //                declared in markup inside it?  False by default.
4512
                widgetsInTemplate: false,
4513
4514
                // skipNodeCache: [protected] Boolean
4515
                //                If using a cached widget template node poses issues for a
4516
                //                particular widget class, it can set this property to ensure
4517
                //                that its template is always re-built from a string
4518
                _skipNodeCache: false,
4519
4520
                // _earlyTemplatedStartup: Boolean
4521
                //                A fallback to preserve the 1.0 - 1.3 behavior of children in
4522
                //                templates having their startup called before the parent widget
4523
                //                fires postCreate. Defaults to 'false', causing child widgets to
4524
                //                have their .startup() called immediately before a parent widget
4525
                //                .startup(), but always after the parent .postCreate(). Set to
4526
                //                'true' to re-enable to previous, arguably broken, behavior.
4527
                _earlyTemplatedStartup: false,
4528
4529
                // _attachPoints: [private] String[]
4530
                //                List of widget attribute names associated with dojoAttachPoint=... in the
4531
                //                template, ex: ["containerNode", "labelNode"]
4532
/*=====
4533
                 _attachPoints: [],
4534
 =====*/
4535
4536
                constructor: function(){
4537
                        this._attachPoints = [];
4538
                },
4539
4540
                _stringRepl: function(tmpl){
4541
                        // summary:
4542
                        //                Does substitution of ${foo} type properties in template string
4543
                        // tags:
4544
                        //                private
4545
                        var className = this.declaredClass, _this = this;
4546
                        // Cache contains a string because we need to do property replacement
4547
                        // do the property replacement
4548
                        return dojo.string.substitute(tmpl, this, function(value, key){
4549
                                if(key.charAt(0) == '!'){ value = dojo.getObject(key.substr(1), false, _this); }
4550
                                if(typeof value == "undefined"){ throw new Error(className+" template:"+key); } // a debugging aide
4551
                                if(value == null){ return ""; }
4552
4553
                                // Substitution keys beginning with ! will skip the transform step,
4554
                                // in case a user wishes to insert unescaped markup, e.g. ${!foo}
4555
                                return key.charAt(0) == "!" ? value :
4556
                                        // Safer substitution, see heading "Attribute values" in
4557
                                        // http://www.w3.org/TR/REC-html40/appendix/notes.html#h-B.3.2
4558
                                        value.toString().replace(/"/g,"&quot;"); //TODO: add &amp? use encodeXML method?
4559
                        }, this);
4560
                },
4561
4562
                // method over-ride
4563
                buildRendering: function(){
4564
                        // summary:
4565
                        //                Construct the UI for this widget from a template, setting this.domNode.
4566
                        // tags:
4567
                        //                protected
4568
4569
                        // Lookup cached version of template, and download to cache if it
4570
                        // isn't there already.  Returns either a DomNode or a string, depending on
4571
                        // whether or not the template contains ${foo} replacement parameters.
4572
                        var cached = dijit._Templated.getCachedTemplate(this.templatePath, this.templateString, this._skipNodeCache);
4573
4574
                        var node;
4575
                        if(dojo.isString(cached)){
4576
                                node = dojo._toDom(this._stringRepl(cached));
4577
                                if(node.nodeType != 1){
4578
                                        // Flag common problems such as templates with multiple top level nodes (nodeType == 11)
4579
                                        throw new Error("Invalid template: " + cached);
4580
                                }
4581
                        }else{
4582
                                // if it's a node, all we have to do is clone it
4583
                                node = cached.cloneNode(true);
4584
                        }
4585
4586
                        this.domNode = node;
4587
4588
                        // recurse through the node, looking for, and attaching to, our
4589
                        // attachment points and events, which should be defined on the template node.
4590
                        this._attachTemplateNodes(node);
4591
4592
                        if(this.widgetsInTemplate){
4593
                                // Make sure dojoType is used for parsing widgets in template.
4594
                                // The dojo.parser.query could be changed from multiversion support.
4595
                                var parser = dojo.parser, qry, attr;
4596
                                if(parser._query != "[dojoType]"){
4597
                                        qry = parser._query;
4598
                                        attr = parser._attrName;
4599
                                        parser._query = "[dojoType]";
4600
                                        parser._attrName = "dojoType";
4601
                                }
4602
4603
                                // Store widgets that we need to start at a later point in time
4604
                                var cw = (this._startupWidgets = dojo.parser.parse(node, {
4605
                                        noStart: !this._earlyTemplatedStartup,
4606
                                        inherited: {dir: this.dir, lang: this.lang}
4607
                                }));
4608
4609
                                // Restore the query.
4610
                                if(qry){
4611
                                        parser._query = qry;
4612
                                        parser._attrName = attr;
4613
                                }
4614
4615
                                this._supportingWidgets = dijit.findWidgets(node);
4616
4617
                                this._attachTemplateNodes(cw, function(n,p){
4618
                                        return n[p];
4619
                                });
4620
                        }
4621
4622
                        this._fillContent(this.srcNodeRef);
4623
                },
4624
4625
                _fillContent: function(/*DomNode*/ source){
4626
                        // summary:
4627
                        //                Relocate source contents to templated container node.
4628
                        //                this.containerNode must be able to receive children, or exceptions will be thrown.
4629
                        // tags:
4630
                        //                protected
4631
                        var dest = this.containerNode;
4632
                        if(source && dest){
4633
                                while(source.hasChildNodes()){
4634
                                        dest.appendChild(source.firstChild);
4635
                                }
4636
                        }
4637
                },
4638
4639
                _attachTemplateNodes: function(rootNode, getAttrFunc){
4640
                        // summary:
4641
                        //                Iterate through the template and attach functions and nodes accordingly.
4642
                        // description:
4643
                        //                Map widget properties and functions to the handlers specified in
4644
                        //                the dom node and it's descendants. This function iterates over all
4645
                        //                nodes and looks for these properties:
4646
                        //                        * dojoAttachPoint
4647
                        //                        * dojoAttachEvent
4648
                        //                        * waiRole
4649
                        //                        * waiState
4650
                        // rootNode: DomNode|Array[Widgets]
4651
                        //                the node to search for properties. All children will be searched.
4652
                        // getAttrFunc: Function?
4653
                        //                a function which will be used to obtain property for a given
4654
                        //                DomNode/Widget
4655
                        // tags:
4656
                        //                private
4657
4658
                        getAttrFunc = getAttrFunc || function(n,p){ return n.getAttribute(p); };
4659
4660
                        var nodes = dojo.isArray(rootNode) ? rootNode : (rootNode.all || rootNode.getElementsByTagName("*"));
4661
                        var x = dojo.isArray(rootNode) ? 0 : -1;
4662
                        for(; x<nodes.length; x++){
4663
                                var baseNode = (x == -1) ? rootNode : nodes[x];
4664
                                if(this.widgetsInTemplate && getAttrFunc(baseNode, "dojoType")){
4665
                                        continue;
4666
                                }
4667
                                // Process dojoAttachPoint
4668
                                var attachPoint = getAttrFunc(baseNode, "dojoAttachPoint");
4669
                                if(attachPoint){
4670
                                        var point, points = attachPoint.split(/\s*,\s*/);
4671
                                        while((point = points.shift())){
4672
                                                if(dojo.isArray(this[point])){
4673
                                                        this[point].push(baseNode);
4674
                                                }else{
4675
                                                        this[point]=baseNode;
4676
                                                }
4677
                                                this._attachPoints.push(point);
4678
                                        }
4679
                                }
4680
4681
                                // Process dojoAttachEvent
4682
                                var attachEvent = getAttrFunc(baseNode, "dojoAttachEvent");
4683
                                if(attachEvent){
4684
                                        // NOTE: we want to support attributes that have the form
4685
                                        // "domEvent: nativeEvent; ..."
4686
                                        var event, events = attachEvent.split(/\s*,\s*/);
4687
                                        var trim = dojo.trim;
4688
                                        while((event = events.shift())){
4689
                                                if(event){
4690
                                                        var thisFunc = null;
4691
                                                        if(event.indexOf(":") != -1){
4692
                                                                // oh, if only JS had tuple assignment
4693
                                                                var funcNameArr = event.split(":");
4694
                                                                event = trim(funcNameArr[0]);
4695
                                                                thisFunc = trim(funcNameArr[1]);
4696
                                                        }else{
4697
                                                                event = trim(event);
4698
                                                        }
4699
                                                        if(!thisFunc){
4700
                                                                thisFunc = event;
4701
                                                        }
4702
                                                        this.connect(baseNode, event, thisFunc);
4703
                                                }
4704
                                        }
4705
                                }
4706
4707
                                // waiRole, waiState
4708
                                var role = getAttrFunc(baseNode, "waiRole");
4709
                                if(role){
4710
                                        dijit.setWaiRole(baseNode, role);
4711
                                }
4712
                                var values = getAttrFunc(baseNode, "waiState");
4713
                                if(values){
4714
                                        dojo.forEach(values.split(/\s*,\s*/), function(stateValue){
4715
                                                if(stateValue.indexOf('-') != -1){
4716
                                                        var pair = stateValue.split('-');
4717
                                                        dijit.setWaiState(baseNode, pair[0], pair[1]);
4718
                                                }
4719
                                        });
4720
                                }
4721
                        }
4722
                },
4723
4724
                startup: function(){
4725
                        dojo.forEach(this._startupWidgets, function(w){
4726
                                if(w && !w._started && w.startup){
4727
                                        w.startup();
4728
                                }
4729
                        });
4730
                        this.inherited(arguments);
4731
                },
4732
4733
                destroyRendering: function(){
4734
                        // Delete all attach points to prevent IE6 memory leaks.
4735
                        dojo.forEach(this._attachPoints, function(point){
4736
                                delete this[point];
4737
                        }, this);
4738
                        this._attachPoints = [];
4739
4740
                        this.inherited(arguments);
4741
                }
4742
        }
4743
);
4744
4745
// key is either templatePath or templateString; object is either string or DOM tree
4746
dijit._Templated._templateCache = {};
4747
4748
dijit._Templated.getCachedTemplate = function(templatePath, templateString, alwaysUseString){
4749
        // summary:
4750
        //                Static method to get a template based on the templatePath or
4751
        //                templateString key
4752
        // templatePath: String||dojo.uri.Uri
4753
        //                The URL to get the template from.
4754
        // templateString: String?
4755
        //                a string to use in lieu of fetching the template from a URL. Takes precedence
4756
        //                over templatePath
4757
        // returns: Mixed
4758
        //                Either string (if there are ${} variables that need to be replaced) or just
4759
        //                a DOM tree (if the node can be cloned directly)
4760
4761
        // is it already cached?
4762
        var tmplts = dijit._Templated._templateCache;
4763
        var key = templateString || templatePath;
4764
        var cached = tmplts[key];
4765
        if(cached){
4766
                try{
4767
                        // if the cached value is an innerHTML string (no ownerDocument) or a DOM tree created within the current document, then use the current cached value
4768
                        if(!cached.ownerDocument || cached.ownerDocument == dojo.doc){
4769
                                // string or node of the same document
4770
                                return cached;
4771
                        }
4772
                }catch(e){ /* squelch */ } // IE can throw an exception if cached.ownerDocument was reloaded
4773
                dojo.destroy(cached);
4774
        }
4775
4776
        // If necessary, load template string from template path
4777
        if(!templateString){
4778
                templateString = dojo.cache(templatePath, {sanitize: true});
4779
        }
4780
        templateString = dojo.string.trim(templateString);
4781
4782
        if(alwaysUseString || templateString.match(/\$\{([^\}]+)\}/g)){
4783
                // there are variables in the template so all we can do is cache the string
4784
                return (tmplts[key] = templateString); //String
4785
        }else{
4786
                // there are no variables in the template so we can cache the DOM tree
4787
                var node = dojo._toDom(templateString);
4788
                if(node.nodeType != 1){
4789
                        throw new Error("Invalid template: " + templateString);
4790
                }
4791
                return (tmplts[key] = node); //Node
4792
        }
4793
};
4794
4795
if(dojo.isIE){
4796
        dojo.addOnWindowUnload(function(){
4797
                var cache = dijit._Templated._templateCache;
4798
                for(var key in cache){
4799
                        var value = cache[key];
4800
                        if(typeof value == "object"){ // value is either a string or a DOM node template
4801
                                dojo.destroy(value);
4802
                        }
4803
                        delete cache[key];
4804
                }
4805
        });
4806
}
4807
4808
// These arguments can be specified for widgets which are used in templates.
4809
// Since any widget can be specified as sub widgets in template, mix it
4810
// into the base widget class.  (This is a hack, but it's effective.)
4811
dojo.extend(dijit._Widget,{
4812
        dojoAttachEvent: "",
4813
        dojoAttachPoint: "",
4814
        waiRole: "",
4815
        waiState:""
4816
});
4817
4818
}
4819
4820
if(!dojo._hasResource["dijit._Container"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
4821
dojo._hasResource["dijit._Container"] = true;
4822
dojo.provide("dijit._Container");
4823
4824
dojo.declare("dijit._Container",
4825
        null,
4826
        {
4827
                // summary:
4828
                //                Mixin for widgets that contain a set of widget children.
4829
                // description:
4830
                //                Use this mixin for widgets that needs to know about and
4831
                //                keep track of their widget children. Suitable for widgets like BorderContainer
4832
                //                and TabContainer which contain (only) a set of child widgets.
4833
                //
4834
                //                It's not suitable for widgets like ContentPane
4835
                //                which contains mixed HTML (plain DOM nodes in addition to widgets),
4836
                //                and where contained widgets are not necessarily directly below
4837
                //                this.containerNode.   In that case calls like addChild(node, position)
4838
                //                wouldn't make sense.
4839
4840
                // isContainer: [protected] Boolean
4841
                //                Indicates that this widget acts as a "parent" to the descendant widgets.
4842
                //                When the parent is started it will call startup() on the child widgets.
4843
                //                See also `isLayoutContainer`.
4844
                isContainer: true,
4845
4846
                buildRendering: function(){
4847
                        this.inherited(arguments);
4848
                        if(!this.containerNode){
4849
                                // all widgets with descendants must set containerNode
4850
                                         this.containerNode = this.domNode;
4851
                        }
4852
                },
4853
4854
                addChild: function(/*dijit._Widget*/ widget, /*int?*/ insertIndex){
4855
                        // summary:
4856
                        //                Makes the given widget a child of this widget.
4857
                        // description:
4858
                        //                Inserts specified child widget's dom node as a child of this widget's
4859
                        //                container node, and possibly does other processing (such as layout).
4860
4861
                        var refNode = this.containerNode;
4862
                        if(insertIndex && typeof insertIndex == "number"){
4863
                                var children = this.getChildren();
4864
                                if(children && children.length >= insertIndex){
4865
                                        refNode = children[insertIndex-1].domNode;
4866
                                        insertIndex = "after";
4867
                                }
4868
                        }
4869
                        dojo.place(widget.domNode, refNode, insertIndex);
4870
4871
                        // If I've been started but the child widget hasn't been started,
4872
                        // start it now.  Make sure to do this after widget has been
4873
                        // inserted into the DOM tree, so it can see that it's being controlled by me,
4874
                        // so it doesn't try to size itself.
4875
                        if(this._started && !widget._started){
4876
                                widget.startup();
4877
                        }
4878
                },
4879
4880
                removeChild: function(/*Widget or int*/ widget){
4881
                        // summary:
4882
                        //                Removes the passed widget instance from this widget but does
4883
                        //                not destroy it.  You can also pass in an integer indicating
4884
                        //                the index within the container to remove
4885
4886
                        if(typeof widget == "number" && widget > 0){
4887
                                widget = this.getChildren()[widget];
4888
                        }
4889
4890
                        if(widget){
4891
                                var node = widget.domNode;
4892
                                if(node && node.parentNode){
4893
                                        node.parentNode.removeChild(node); // detach but don't destroy
4894
                                }
4895
                        }
4896
                },
4897
4898
                hasChildren: function(){
4899
                        // summary:
4900
                        //                Returns true if widget has children, i.e. if this.containerNode contains something.
4901
                        return this.getChildren().length > 0;        // Boolean
4902
                },
4903
4904
                destroyDescendants: function(/*Boolean*/ preserveDom){
4905
                        // summary:
4906
                        //      Destroys all the widgets inside this.containerNode,
4907
                        //      but not this widget itself
4908
                        dojo.forEach(this.getChildren(), function(child){ child.destroyRecursive(preserveDom); });
4909
                },
4910
4911
                _getSiblingOfChild: function(/*dijit._Widget*/ child, /*int*/ dir){
4912
                        // summary:
4913
                        //                Get the next or previous widget sibling of child
4914
                        // dir:
4915
                        //                if 1, get the next sibling
4916
                        //                if -1, get the previous sibling
4917
                        // tags:
4918
                        //      private
4919
                        var node = child.domNode,
4920
                                which = (dir>0 ? "nextSibling" : "previousSibling");
4921
                        do{
4922
                                node = node[which];
4923
                        }while(node && (node.nodeType != 1 || !dijit.byNode(node)));
4924
                        return node && dijit.byNode(node);        // dijit._Widget
4925
                },
4926
4927
                getIndexOfChild: function(/*dijit._Widget*/ child){
4928
                        // summary:
4929
                        //                Gets the index of the child in this container or -1 if not found
4930
                        return dojo.indexOf(this.getChildren(), child);        // int
4931
                },
4932
4933
                startup: function(){
4934
                        // summary:
4935
                        //                Called after all the widgets have been instantiated and their
4936
                        //                dom nodes have been inserted somewhere under dojo.doc.body.
4937
                        //
4938
                        //                Widgets should override this method to do any initialization
4939
                        //                dependent on other widgets existing, and then call
4940
                        //                this superclass method to finish things off.
4941
                        //
4942
                        //                startup() in subclasses shouldn't do anything
4943
                        //                size related because the size of the widget hasn't been set yet.
4944
4945
                        if(this._started){ return; }
4946
4947
                        // Startup all children of this widget
4948
                        dojo.forEach(this.getChildren(), function(child){ child.startup(); });
4949
4950
                        this.inherited(arguments);
4951
                }
4952
        }
4953
);
4954
4955
}
4956
4957
if(!dojo._hasResource["dijit._Contained"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
4958
dojo._hasResource["dijit._Contained"] = true;
4959
dojo.provide("dijit._Contained");
4960
4961
dojo.declare("dijit._Contained",
4962
                null,
4963
                {
4964
                        // summary:
4965
                        //                Mixin for widgets that are children of a container widget
4966
                        //
4967
                        // example:
4968
                        // |         // make a basic custom widget that knows about it's parents
4969
                        // |        dojo.declare("my.customClass",[dijit._Widget,dijit._Contained],{});
4970
4971
                        getParent: function(){
4972
                                // summary:
4973
                                //                Returns the parent widget of this widget, assuming the parent
4974
                                //                specifies isContainer
4975
                                var parent = dijit.getEnclosingWidget(this.domNode.parentNode);
4976
                                return parent && parent.isContainer ? parent : null;
4977
                        },
4978
4979
                        _getSibling: function(/*String*/ which){
4980
                                // summary:
4981
                                //      Returns next or previous sibling
4982
                                // which:
4983
                                //      Either "next" or "previous"
4984
                                // tags:
4985
                                //      private
4986
                                var node = this.domNode;
4987
                                do{
4988
                                        node = node[which+"Sibling"];
4989
                                }while(node && node.nodeType != 1);
4990
                                return node && dijit.byNode(node);        // dijit._Widget
4991
                        },
4992
4993
                        getPreviousSibling: function(){
4994
                                // summary:
4995
                                //                Returns null if this is the first child of the parent,
4996
                                //                otherwise returns the next element sibling to the "left".
4997
4998
                                return this._getSibling("previous"); // dijit._Widget
4999
                        },
5000
5001
                        getNextSibling: function(){
5002
                                // summary:
5003
                                //                Returns null if this is the last child of the parent,
5004
                                //                otherwise returns the next element sibling to the "right".
5005
5006
                                return this._getSibling("next"); // dijit._Widget
5007
                        },
5008
5009
                        getIndexInParent: function(){
5010
                                // summary:
5011
                                //                Returns the index of this widget within its container parent.
5012
                                //                It returns -1 if the parent does not exist, or if the parent
5013
                                //                is not a dijit._Container
5014
5015
                                var p = this.getParent();
5016
                                if(!p || !p.getIndexOfChild){
5017
                                        return -1; // int
5018
                                }
5019
                                return p.getIndexOfChild(this); // int
5020
                        }
5021
                }
5022
        );
5023
5024
5025
}
5026
5027
if(!dojo._hasResource["dijit.layout._LayoutWidget"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
5028
dojo._hasResource["dijit.layout._LayoutWidget"] = true;
5029
dojo.provide("dijit.layout._LayoutWidget");
5030
5031
5032
5033
5034
5035
dojo.declare("dijit.layout._LayoutWidget",
5036
        [dijit._Widget, dijit._Container, dijit._Contained],
5037
        {
5038
                // summary:
5039
                //                Base class for a _Container widget which is responsible for laying out its children.
5040
                //                Widgets which mixin this code must define layout() to manage placement and sizing of the children.
5041
5042
                // baseClass: [protected extension] String
5043
                //                This class name is applied to the widget's domNode
5044
                //                and also may be used to generate names for sub nodes,
5045
                //                for example dijitTabContainer-content.
5046
                baseClass: "dijitLayoutContainer",
5047
5048
                // isLayoutContainer: [protected] Boolean
5049
                //                Indicates that this widget is going to call resize() on its
5050
                //                children widgets, setting their size, when they become visible.
5051
                isLayoutContainer: true,
5052
5053
                postCreate: function(){
5054
                        dojo.addClass(this.domNode, "dijitContainer");
5055
5056
                        this.inherited(arguments);
5057
                },
5058
5059
                startup: function(){
5060
                        // summary:
5061
                        //                Called after all the widgets have been instantiated and their
5062
                        //                dom nodes have been inserted somewhere under dojo.doc.body.
5063
                        //
5064
                        //                Widgets should override this method to do any initialization
5065
                        //                dependent on other widgets existing, and then call
5066
                        //                this superclass method to finish things off.
5067
                        //
5068
                        //                startup() in subclasses shouldn't do anything
5069
                        //                size related because the size of the widget hasn't been set yet.
5070
5071
                        if(this._started){ return; }
5072
5073
                        // Need to call inherited first - so that child widgets get started
5074
                        // up correctly
5075
                        this.inherited(arguments);
5076
5077
                        // If I am a not being controlled by a parent layout widget...
5078
                        var parent = this.getParent && this.getParent()
5079
                        if(!(parent && parent.isLayoutContainer)){
5080
                                // Do recursive sizing and layout of all my descendants
5081
                                // (passing in no argument to resize means that it has to glean the size itself)
5082
                                this.resize();
5083
5084
                                // Since my parent isn't a layout container, and my style *may be* width=height=100%
5085
                                // or something similar (either set directly or via a CSS class),
5086
                                // monitor when my size changes so that I can re-layout.
5087
                                // For browsers where I can't directly monitor when my size changes,
5088
                                // monitor when the viewport changes size, which *may* indicate a size change for me.
5089
                                this.connect(dojo.isIE ? this.domNode : dojo.global, 'onresize', function(){
5090
                                        // Using function(){} closure to ensure no arguments to resize.
5091
                                        this.resize();
5092
                                });
5093
                        }
5094
                },
5095
5096
                resize: function(changeSize, resultSize){
5097
                        // summary:
5098
                        //                Call this to resize a widget, or after its size has changed.
5099
                        // description:
5100
                        //                Change size mode:
5101
                        //                        When changeSize is specified, changes the marginBox of this widget
5102
                        //                        and forces it to relayout its contents accordingly.
5103
                        //                        changeSize may specify height, width, or both.
5104
                        //
5105
                        //                        If resultSize is specified it indicates the size the widget will
5106
                        //                        become after changeSize has been applied.
5107
                        //
5108
                        //                Notification mode:
5109
                        //                        When changeSize is null, indicates that the caller has already changed
5110
                        //                        the size of the widget, or perhaps it changed because the browser
5111
                        //                        window was resized.  Tells widget to relayout its contents accordingly.
5112
                        //
5113
                        //                        If resultSize is also specified it indicates the size the widget has
5114
                        //                        become.
5115
                        //
5116
                        //                In either mode, this method also:
5117
                        //                        1. Sets this._borderBox and this._contentBox to the new size of
5118
                        //                                the widget.  Queries the current domNode size if necessary.
5119
                        //                        2. Calls layout() to resize contents (and maybe adjust child widgets).
5120
                        //
5121
                        // changeSize: Object?
5122
                        //                Sets the widget to this margin-box size and position.
5123
                        //                May include any/all of the following properties:
5124
                        //        |        {w: int, h: int, l: int, t: int}
5125
                        //
5126
                        // resultSize: Object?
5127
                        //                The margin-box size of this widget after applying changeSize (if
5128
                        //                changeSize is specified).  If caller knows this size and
5129
                        //                passes it in, we don't need to query the browser to get the size.
5130
                        //        |        {w: int, h: int}
5131
5132
                        var node = this.domNode;
5133
5134
                        // set margin box size, unless it wasn't specified, in which case use current size
5135
                        if(changeSize){
5136
                                dojo.marginBox(node, changeSize);
5137
5138
                                // set offset of the node
5139
                                if(changeSize.t){ node.style.top = changeSize.t + "px"; }
5140
                                if(changeSize.l){ node.style.left = changeSize.l + "px"; }
5141
                        }
5142
5143
                        // If either height or width wasn't specified by the user, then query node for it.
5144
                        // But note that setting the margin box and then immediately querying dimensions may return
5145
                        // inaccurate results, so try not to depend on it.
5146
                        var mb = resultSize || {};
5147
                        dojo.mixin(mb, changeSize || {});        // changeSize overrides resultSize
5148
                        if( !("h" in mb) || !("w" in mb) ){
5149
                                mb = dojo.mixin(dojo.marginBox(node), mb);        // just use dojo.marginBox() to fill in missing values
5150
                        }
5151
5152
                        // Compute and save the size of my border box and content box
5153
                        // (w/out calling dojo.contentBox() since that may fail if size was recently set)
5154
                        var cs = dojo.getComputedStyle(node);
5155
                        var me = dojo._getMarginExtents(node, cs);
5156
                        var be = dojo._getBorderExtents(node, cs);
5157
                        var bb = (this._borderBox = {
5158
                                w: mb.w - (me.w + be.w),
5159
                                h: mb.h - (me.h + be.h)
5160
                        });
5161
                        var pe = dojo._getPadExtents(node, cs);
5162
                        this._contentBox = {
5163
                                l: dojo._toPixelValue(node, cs.paddingLeft),
5164
                                t: dojo._toPixelValue(node, cs.paddingTop),
5165
                                w: bb.w - pe.w,
5166
                                h: bb.h - pe.h
5167
                        };
5168
5169
                        // Callback for widget to adjust size of its children
5170
                        this.layout();
5171
                },
5172
5173
                layout: function(){
5174
                        // summary:
5175
                        //                Widgets override this method to size and position their contents/children.
5176
                        //                When this is called this._contentBox is guaranteed to be set (see resize()).
5177
                        //
5178
                        //                This is called after startup(), and also when the widget's size has been
5179
                        //                changed.
5180
                        // tags:
5181
                        //                protected extension
5182
                },
5183
5184
                _setupChild: function(/*dijit._Widget*/child){
5185
                        // summary:
5186
                        //                Common setup for initial children and children which are added after startup
5187
                        // tags:
5188
                        //                protected extension
5189
5190
                        dojo.addClass(child.domNode, this.baseClass+"-child");
5191
                        if(child.baseClass){
5192
                                dojo.addClass(child.domNode, this.baseClass+"-"+child.baseClass);
5193
                        }
5194
                },
5195
5196
                addChild: function(/*dijit._Widget*/ child, /*Integer?*/ insertIndex){
5197
                        // Overrides _Container.addChild() to call _setupChild()
5198
                        this.inherited(arguments);
5199
                        if(this._started){
5200
                                this._setupChild(child);
5201
                        }
5202
                },
5203
5204
                removeChild: function(/*dijit._Widget*/ child){
5205
                        // Overrides _Container.removeChild() to remove class added by _setupChild()
5206
                        dojo.removeClass(child.domNode, this.baseClass+"-child");
5207
                        if(child.baseClass){
5208
                                dojo.removeClass(child.domNode, this.baseClass+"-"+child.baseClass);
5209
                        }
5210
                        this.inherited(arguments);
5211
                }
5212
        }
5213
);
5214
5215
dijit.layout.marginBox2contentBox = function(/*DomNode*/ node, /*Object*/ mb){
5216
        // summary:
5217
        //                Given the margin-box size of a node, return its content box size.
5218
        //                Functions like dojo.contentBox() but is more reliable since it doesn't have
5219
        //                to wait for the browser to compute sizes.
5220
        var cs = dojo.getComputedStyle(node);
5221
        var me = dojo._getMarginExtents(node, cs);
5222
        var pb = dojo._getPadBorderExtents(node, cs);
5223
        return {
5224
                l: dojo._toPixelValue(node, cs.paddingLeft),
5225
                t: dojo._toPixelValue(node, cs.paddingTop),
5226
                w: mb.w - (me.w + pb.w),
5227
                h: mb.h - (me.h + pb.h)
5228
        };
5229
};
5230
5231
(function(){
5232
        var capitalize = function(word){
5233
                return word.substring(0,1).toUpperCase() + word.substring(1);
5234
        };
5235
5236
        var size = function(widget, dim){
5237
                // size the child
5238
                widget.resize ? widget.resize(dim) : dojo.marginBox(widget.domNode, dim);
5239
5240
                // record child's size, but favor our own numbers when we have them.
5241
                // the browser lies sometimes
5242
                dojo.mixin(widget, dojo.marginBox(widget.domNode));
5243
                dojo.mixin(widget, dim);
5244
        };
5245
5246
        dijit.layout.layoutChildren = function(/*DomNode*/ container, /*Object*/ dim, /*Object[]*/ children){
5247
                // summary
5248
                //                Layout a bunch of child dom nodes within a parent dom node
5249
                // container:
5250
                //                parent node
5251
                // dim:
5252
                //                {l, t, w, h} object specifying dimensions of container into which to place children
5253
                // children:
5254
                //                an array like [ {domNode: foo, layoutAlign: "bottom" }, {domNode: bar, layoutAlign: "client"} ]
5255
5256
                // copy dim because we are going to modify it
5257
                dim = dojo.mixin({}, dim);
5258
5259
                dojo.addClass(container, "dijitLayoutContainer");
5260
5261
                // Move "client" elements to the end of the array for layout.  a11y dictates that the author
5262
                // needs to be able to put them in the document in tab-order, but this algorithm requires that
5263
                // client be last.
5264
                children = dojo.filter(children, function(item){ return item.layoutAlign != "client"; })
5265
                        .concat(dojo.filter(children, function(item){ return item.layoutAlign == "client"; }));
5266
5267
                // set positions/sizes
5268
                dojo.forEach(children, function(child){
5269
                        var elm = child.domNode,
5270
                                pos = child.layoutAlign;
5271
5272
                        // set elem to upper left corner of unused space; may move it later
5273
                        var elmStyle = elm.style;
5274
                        elmStyle.left = dim.l+"px";
5275
                        elmStyle.top = dim.t+"px";
5276
                        elmStyle.bottom = elmStyle.right = "auto";
5277
5278
                        dojo.addClass(elm, "dijitAlign" + capitalize(pos));
5279
5280
                        // set size && adjust record of remaining space.
5281
                        // note that setting the width of a <div> may affect its height.
5282
                        if(pos == "top" || pos == "bottom"){
5283
                                size(child, { w: dim.w });
5284
                                dim.h -= child.h;
5285
                                if(pos == "top"){
5286
                                        dim.t += child.h;
5287
                                }else{
5288
                                        elmStyle.top = dim.t + dim.h + "px";
5289
                                }
5290
                        }else if(pos == "left" || pos == "right"){
5291
                                size(child, { h: dim.h });
5292
                                dim.w -= child.w;
5293
                                if(pos == "left"){
5294
                                        dim.l += child.w;
5295
                                }else{
5296
                                        elmStyle.left = dim.l + dim.w + "px";
5297
                                }
5298
                        }else if(pos == "client"){
5299
                                size(child, dim);
5300
                        }
5301
                });
5302
        };
5303
5304
})();
5305
5306
}
5307
5308
if(!dojo._hasResource["dijit._CssStateMixin"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
5309
dojo._hasResource["dijit._CssStateMixin"] = true;
5310
dojo.provide("dijit._CssStateMixin");
5311
5312
5313
dojo.declare("dijit._CssStateMixin", [], {
5314
        // summary:
5315
        //                Mixin for widgets to set CSS classes on the widget DOM nodes depending on hover/mouse press/focus
5316
        //                state changes, and also higher-level state changes such becoming disabled or selected.
5317
        //
5318
        // description:
5319
        //                By mixing this class into your widget, and setting the this.baseClass attribute, it will automatically
5320
        //                maintain CSS classes on the widget root node (this.domNode) depending on hover,
5321
        //                active, focus, etc. state.   Ex: with a baseClass of dijitButton, it will apply the classes
5322
        //                dijitButtonHovered and dijitButtonActive, as the user moves the mouse over the widget and clicks it.
5323
        //
5324
        //                It also sets CSS like dijitButtonDisabled based on widget semantic state.
5325
        //
5326
        //                By setting the cssStateNodes attribute, a widget can also track events on subnodes (like buttons
5327
        //                within the widget).
5328
5329
        // cssStateNodes: [protected] Object
5330
        //                List of sub-nodes within the widget that need CSS classes applied on mouse hover/press and focus
5331
        //.
5332
        //                Each entry in the hash is a an attachpoint names (like "upArrowButton") mapped to a CSS class names
5333
        //                (like "dijitUpArrowButton"). Example:
5334
        //        |                {
5335
        //        |                        "upArrowButton": "dijitUpArrowButton",
5336
        //        |                        "downArrowButton": "dijitDownArrowButton"
5337
        //        |                }
5338
        //                The above will set the CSS class dijitUpArrowButton to the this.upArrowButton DOMNode when it
5339
        //                is hovered, etc.
5340
        cssStateNodes: {},
5341
5342
        postCreate: function(){
5343
                this.inherited(arguments);
5344
5345
                // Automatically monitor mouse events (essentially :hover and :active) on this.domNode
5346
                dojo.forEach(["onmouseenter", "onmouseleave", "onmousedown"], function(e){
5347
                        this.connect(this.domNode, e, "_cssMouseEvent");
5348
                }, this);
5349
5350
                // Monitoring changes to disabled, readonly, etc. state, and update CSS class of root node
5351
                this.connect(this, "set", function(name, value){
5352
                        if(arguments.length >= 2 && {disabled: true, readOnly: true, checked:true, selected:true}[name]){
5353
                                this._setStateClass();
5354
                        }
5355
                });
5356
5357
                // The widget coming in/out of the focus change affects it's state
5358
                dojo.forEach(["_onFocus", "_onBlur"], function(ap){
5359
                        this.connect(this, ap, "_setStateClass");
5360
                }, this);
5361
5362
                // Events on sub nodes within the widget
5363
                for(var ap in this.cssStateNodes){
5364
                        this._trackMouseState(this[ap], this.cssStateNodes[ap]);
5365
                }
5366
                // Set state initially; there's probably no hover/active/focus state but widget might be
5367
                // disabled/readonly so we want to set CSS classes for those conditions.
5368
                this._setStateClass();
5369
        },
5370
5371
        _cssMouseEvent: function(/*Event*/ event){
5372
                // summary:
5373
                //        Sets _hovering and _active properties depending on mouse state,
5374
                //        then calls _setStateClass() to set appropriate CSS classes for this.domNode.
5375
5376
                if(!this.disabled){
5377
                        switch(event.type){
5378
                                case "mouseenter":
5379
                                case "mouseover":        // generated on non-IE browsers even though we connected to mouseenter
5380
                                        this._hovering = true;
5381
                                        this._active = this._mouseDown;
5382
                                        break;
5383
5384
                                case "mouseleave":
5385
                                case "mouseout":        // generated on non-IE browsers even though we connected to mouseleave
5386
                                        this._hovering = false;
5387
                                        this._active = false;
5388
                                        break;
5389
5390
                                case "mousedown" :
5391
                                        this._active = true;
5392
                                        this._mouseDown = true;
5393
                                        // Set a global event to handle mouseup, so it fires properly
5394
                                        // even if the cursor leaves this.domNode before the mouse up event.
5395
                                        // Alternately could set active=false on mouseout.
5396
                                        var mouseUpConnector = this.connect(dojo.body(), "onmouseup", function(){
5397
                                                this._active = false;
5398
                                                this._mouseDown = false;
5399
                                                this._setStateClass();
5400
                                                this.disconnect(mouseUpConnector);
5401
                                        });
5402
                                        break;
5403
                        }
5404
                        this._setStateClass();
5405
                }
5406
        },
5407
5408
        _setStateClass: function(){
5409
                // summary:
5410
                //                Update the visual state of the widget by setting the css classes on this.domNode
5411
                //                (or this.stateNode if defined) by combining this.baseClass with
5412
                //                various suffixes that represent the current widget state(s).
5413
                //
5414
                // description:
5415
                //                In the case where a widget has multiple
5416
                //                states, it sets the class based on all possible
5417
                //                 combinations.  For example, an invalid form widget that is being hovered
5418
                //                will be "dijitInput dijitInputInvalid dijitInputHover dijitInputInvalidHover".
5419
                //
5420
                //                The widget may have one or more of the following states, determined
5421
                //                by this.state, this.checked, this.valid, and this.selected:
5422
                //                        - Error - ValidationTextBox sets this.state to "Error" if the current input value is invalid
5423
                //                        - Checked - ex: a checkmark or a ToggleButton in a checked state, will have this.checked==true
5424
                //                        - Selected - ex: currently selected tab will have this.selected==true
5425
                //
5426
                //                In addition, it may have one or more of the following states,
5427
                //                based on this.disabled and flags set in _onMouse (this._active, this._hovering, this._focused):
5428
                //                        - Disabled        - if the widget is disabled
5429
                //                        - Active                - if the mouse (or space/enter key?) is being pressed down
5430
                //                        - Focused                - if the widget has focus
5431
                //                        - Hover                - if the mouse is over the widget
5432
5433
                // Compute new set of classes
5434
                var newStateClasses = this.baseClass.split(" ");
5435
5436
                function multiply(modifier){
5437
                        newStateClasses = newStateClasses.concat(dojo.map(newStateClasses, function(c){ return c+modifier; }), "dijit"+modifier);
5438
                }
5439
5440
                if(!this.isLeftToRight()){
5441
                        // For RTL mode we need to set an addition class like dijitTextBoxRtl.
5442
                        multiply("Rtl");
5443
                }
5444
5445
                if(this.checked){
5446
                        multiply("Checked");
5447
                }
5448
                if(this.state){
5449
                        multiply(this.state);
5450
                }
5451
                if(this.selected){
5452
                        multiply("Selected");
5453
                }
5454
5455
                if(this.disabled){
5456
                        multiply("Disabled");
5457
                }else if(this.readOnly){
5458
                        multiply("ReadOnly");
5459
                }else{
5460
                        if(this._active){
5461
                                multiply("Active");
5462
                        }else if(this._hovering){
5463
                                multiply("Hover");
5464
                        }
5465
                }
5466
5467
                if(this._focused){
5468
                        multiply("Focused");
5469
                }
5470
5471
                // Remove old state classes and add new ones.
5472
                // For performance concerns we only write into domNode.className once.
5473
                var tn = this.stateNode || this.domNode,
5474
                        classHash = {};        // set of all classes (state and otherwise) for node
5475
5476
                dojo.forEach(tn.className.split(" "), function(c){ classHash[c] = true; });
5477
5478
                if("_stateClasses" in this){
5479
                        dojo.forEach(this._stateClasses, function(c){ delete classHash[c]; });
5480
                }
5481
5482
                dojo.forEach(newStateClasses, function(c){ classHash[c] = true; });
5483
5484
                var newClasses = [];
5485
                for(var c in classHash){
5486
                        newClasses.push(c);
5487
                }
5488
                tn.className = newClasses.join(" ");
5489
5490
                this._stateClasses = newStateClasses;
5491
        },
5492
5493
        _trackMouseState: function(/*DomNode*/ node, /*String*/ clazz){
5494
                // summary:
5495
                //                Track mouse/focus events on specified node and set CSS class on that node to indicate
5496
                //                current state.   Usually not called directly, but via cssStateNodes attribute.
5497
                // description:
5498
                //                Given class=foo, will set the following CSS class on the node
5499
                //                        - fooActive: if the user is currently pressing down the mouse button while over the node
5500
                //                        - fooHover: if the user is hovering the mouse over the node, but not pressing down a button
5501
                //                        - fooFocus: if the node is focused
5502
                //
5503
                //                Note that it won't set any classes if the widget is disabled.
5504
                // node: DomNode
5505
                //                Should be a sub-node of the widget, not the top node (this.domNode), since the top node
5506
                //                is handled specially and automatically just by mixing in this class.
5507
                // clazz: String
5508
                //                CSS class name (ex: dijitSliderUpArrow).
5509
5510
                // Current state of node (initially false)
5511
                // NB: setting specifically to false because dojo.toggleClass() needs true boolean as third arg
5512
                var hovering=false, active=false, focused=false;
5513
5514
                var self = this,
5515
                        cn = dojo.hitch(this, "connect", node);
5516
5517
                function setClass(){
5518
                        var disabled = ("disabled" in self && self.disabled) || ("readonly" in self && self.readonly);
5519
                        dojo.toggleClass(node, clazz+"Hover", hovering && !active && !disabled);
5520
                        dojo.toggleClass(node, clazz+"Active", active && !disabled);
5521
                        dojo.toggleClass(node, clazz+"Focused", focused && !disabled);
5522
                }
5523
5524
                // Mouse
5525
                cn("onmouseenter", function(){
5526
                        hovering = true;
5527
                        setClass();
5528
                });
5529
                cn("onmouseleave", function(){
5530
                        hovering = false;
5531
                        active = false;
5532
                        setClass();
5533
                });
5534
                cn("onmousedown", function(){
5535
                        active = true;
5536
                        setClass();
5537
                });
5538
                cn("onmouseup", function(){
5539
                        active = false;
5540
                        setClass();
5541
                });
5542
5543
                // Focus
5544
                cn("onfocus", function(){
5545
                        focused = true;
5546
                        setClass();
5547
                });
5548
                cn("onblur", function(){
5549
                        focused = false;
5550
                        setClass();
5551
                });
5552
5553
                // Just in case widget is enabled/disabled while it has focus/hover/active state.
5554
                // Maybe this is overkill.
5555
                this.connect(this, "set", function(name, value){
5556
                        if(name == "disabled" || name == "readOnly"){
5557
                                setClass();
5558
                        }
5559
                });
5560
        }
5561
});
5562
5563
}
5564
5565
if(!dojo._hasResource["dijit.form._FormWidget"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
5566
dojo._hasResource["dijit.form._FormWidget"] = true;
5567
dojo.provide("dijit.form._FormWidget");
5568
5569
5570
5571
5572
5573
5574
5575
dojo.declare("dijit.form._FormWidget", [dijit._Widget, dijit._Templated, dijit._CssStateMixin],
5576
        {
5577
        // summary:
5578
        //                Base class for widgets corresponding to native HTML elements such as <checkbox> or <button>,
5579
        //                which can be children of a <form> node or a `dijit.form.Form` widget.
5580
        //
5581
        // description:
5582
        //                Represents a single HTML element.
5583
        //                All these widgets should have these attributes just like native HTML input elements.
5584
        //                You can set them during widget construction or afterwards, via `dijit._Widget.attr`.
5585
        //
5586
        //                They also share some common methods.
5587
5588
        // name: String
5589
        //                Name used when submitting form; same as "name" attribute or plain HTML elements
5590
        name: "",
5591
5592
        // alt: String
5593
        //                Corresponds to the native HTML <input> element's attribute.
5594
        alt: "",
5595
5596
        // value: String
5597
        //                Corresponds to the native HTML <input> element's attribute.
5598
        value: "",
5599
5600
        // type: String
5601
        //                Corresponds to the native HTML <input> element's attribute.
5602
        type: "text",
5603
5604
        // tabIndex: Integer
5605
        //                Order fields are traversed when user hits the tab key
5606
        tabIndex: "0",
5607
5608
        // disabled: Boolean
5609
        //                Should this widget respond to user input?
5610
        //                In markup, this is specified as "disabled='disabled'", or just "disabled".
5611
        disabled: false,
5612
5613
        // intermediateChanges: Boolean
5614
        //                Fires onChange for each value change or only on demand
5615
        intermediateChanges: false,
5616
5617
        // scrollOnFocus: Boolean
5618
        //                On focus, should this widget scroll into view?
5619
        scrollOnFocus: true,
5620
5621
        // These mixins assume that the focus node is an INPUT, as many but not all _FormWidgets are.
5622
        attributeMap: dojo.delegate(dijit._Widget.prototype.attributeMap, {
5623
                value: "focusNode",
5624
                id: "focusNode",
5625
                tabIndex: "focusNode",
5626
                alt: "focusNode",
5627
                title: "focusNode"
5628
        }),
5629
5630
        postMixInProperties: function(){
5631
                // Setup name=foo string to be referenced from the template (but only if a name has been specified)
5632
                // Unfortunately we can't use attributeMap to set the name due to IE limitations, see #8660
5633
                // Regarding escaping, see heading "Attribute values" in
5634
                // http://www.w3.org/TR/REC-html40/appendix/notes.html#h-B.3.2
5635
                this.nameAttrSetting = this.name ? ('name="' + this.name.replace(/'/g, "&quot;") + '"') : '';
5636
                this.inherited(arguments);
5637
        },
5638
5639
        postCreate: function(){
5640
                this.inherited(arguments);
5641
                this.connect(this.domNode, "onmousedown", "_onMouseDown");
5642
        },
5643
5644
        _setDisabledAttr: function(/*Boolean*/ value){
5645
                this.disabled = value;
5646
                dojo.attr(this.focusNode, 'disabled', value);
5647
                if(this.valueNode){
5648
                        dojo.attr(this.valueNode, 'disabled', value);
5649
                }
5650
                dijit.setWaiState(this.focusNode, "disabled", value);
5651
5652
                if(value){
5653
                        // reset these, because after the domNode is disabled, we can no longer receive
5654
                        // mouse related events, see #4200
5655
                        this._hovering = false;
5656
                        this._active = false;
5657
5658
                        // clear tab stop(s) on this widget's focusable node(s)  (ComboBox has two focusable nodes)
5659
                        var attachPointNames = "tabIndex" in this.attributeMap ? this.attributeMap.tabIndex : "focusNode";
5660
                        dojo.forEach(dojo.isArray(attachPointNames) ? attachPointNames : [attachPointNames], function(attachPointName){
5661
                                var node = this[attachPointName];
5662
                                // complex code because tabIndex=-1 on a <div> doesn't work on FF
5663
                                if(dojo.isWebKit || dijit.hasDefaultTabStop(node)){        // see #11064 about webkit bug
5664
                                        node.setAttribute('tabIndex', "-1");
5665
                                }else{
5666
                                        node.removeAttribute('tabIndex');
5667
                                }
5668
                        }, this);
5669
                }else{
5670
                        this.focusNode.setAttribute('tabIndex', this.tabIndex);
5671
                }
5672
        },
5673
5674
        setDisabled: function(/*Boolean*/ disabled){
5675
                // summary:
5676
                //                Deprecated.   Use set('disabled', ...) instead.
5677
                dojo.deprecated("setDisabled("+disabled+") is deprecated. Use set('disabled',"+disabled+") instead.", "", "2.0");
5678
                this.set('disabled', disabled);
5679
        },
5680
5681
        _onFocus: function(e){
5682
                if(this.scrollOnFocus){
5683
                        dojo.window.scrollIntoView(this.domNode);
5684
                }
5685
                this.inherited(arguments);
5686
        },
5687
5688
        isFocusable: function(){
5689
                // summary:
5690
                //                Tells if this widget is focusable or not.   Used internally by dijit.
5691
                // tags:
5692
                //                protected
5693
                return !this.disabled && !this.readOnly && this.focusNode && (dojo.style(this.domNode, "display") != "none");
5694
        },
5695
5696
        focus: function(){
5697
                // summary:
5698
                //                Put focus on this widget
5699
                dijit.focus(this.focusNode);
5700
        },
5701
5702
        compare: function(/*anything*/val1, /*anything*/val2){
5703
                // summary:
5704
                //                Compare 2 values (as returned by attr('value') for this widget).
5705
                // tags:
5706
                //                protected
5707
                if(typeof val1 == "number" && typeof val2 == "number"){
5708
                        return (isNaN(val1) && isNaN(val2)) ? 0 : val1 - val2;
5709
                }else if(val1 > val2){
5710
                        return 1;
5711
                }else if(val1 < val2){
5712
                        return -1;
5713
                }else{
5714
                        return 0;
5715
                }
5716
        },
5717
5718
        onChange: function(newValue){
5719
                // summary:
5720
                //                Callback when this widget's value is changed.
5721
                // tags:
5722
                //                callback
5723
        },
5724
5725
        // _onChangeActive: [private] Boolean
5726
        //                Indicates that changes to the value should call onChange() callback.
5727
        //                This is false during widget initialization, to avoid calling onChange()
5728
        //                when the initial value is set.
5729
        _onChangeActive: false,
5730
5731
        _handleOnChange: function(/*anything*/ newValue, /* Boolean? */ priorityChange){
5732
                // summary:
5733
                //                Called when the value of the widget is set.  Calls onChange() if appropriate
5734
                // newValue:
5735
                //                the new value
5736
                // priorityChange:
5737
                //                For a slider, for example, dragging the slider is priorityChange==false,
5738
                //                but on mouse up, it's priorityChange==true.  If intermediateChanges==true,
5739
                //                onChange is only called form priorityChange=true events.
5740
                // tags:
5741
                //                private
5742
                this._lastValue = newValue;
5743
                if(this._lastValueReported == undefined && (priorityChange === null || !this._onChangeActive)){
5744
                        // this block executes not for a change, but during initialization,
5745
                        // and is used to store away the original value (or for ToggleButton, the original checked state)
5746
                        this._resetValue = this._lastValueReported = newValue;
5747
                }
5748
                if((this.intermediateChanges || priorityChange || priorityChange === undefined) &&
5749
                        ((typeof newValue != typeof this._lastValueReported) ||
5750
                                this.compare(newValue, this._lastValueReported) != 0)){
5751
                        this._lastValueReported = newValue;
5752
                        if(this._onChangeActive){
5753
                                if(this._onChangeHandle){
5754
                                        clearTimeout(this._onChangeHandle);
5755
                                }
5756
                                // setTimout allows hidden value processing to run and
5757
                                // also the onChange handler can safely adjust focus, etc
5758
                                this._onChangeHandle = setTimeout(dojo.hitch(this,
5759
                                        function(){
5760
                                                this._onChangeHandle = null;
5761
                                                this.onChange(newValue);
5762
                                        }), 0); // try to collapse multiple onChange's fired faster than can be processed
5763
                        }
5764
                }
5765
        },
5766
5767
        create: function(){
5768
                // Overrides _Widget.create()
5769
                this.inherited(arguments);
5770
                this._onChangeActive = true;
5771
        },
5772
5773
        destroy: function(){
5774
                if(this._onChangeHandle){ // destroy called before last onChange has fired
5775
                        clearTimeout(this._onChangeHandle);
5776
                        this.onChange(this._lastValueReported);
5777
                }
5778
                this.inherited(arguments);
5779
        },
5780
5781
        setValue: function(/*String*/ value){
5782
                // summary:
5783
                //                Deprecated.   Use set('value', ...) instead.
5784
                dojo.deprecated("dijit.form._FormWidget:setValue("+value+") is deprecated.  Use set('value',"+value+") instead.", "", "2.0");
5785
                this.set('value', value);
5786
        },
5787
5788
        getValue: function(){
5789
                // summary:
5790
                //                Deprecated.   Use get('value') instead.
5791
                dojo.deprecated(this.declaredClass+"::getValue() is deprecated. Use get('value') instead.", "", "2.0");
5792
                return this.get('value');
5793
        },
5794
5795
        _onMouseDown: function(e){
5796
                // If user clicks on the button, even if the mouse is released outside of it,
5797
                // this button should get focus (to mimics native browser buttons).
5798
                // This is also needed on chrome because otherwise buttons won't get focus at all,
5799
                // which leads to bizarre focus restore on Dialog close etc.
5800
                if(!e.ctrlKey && this.isFocusable()){ // !e.ctrlKey to ignore right-click on mac
5801
                        // Set a global event to handle mouseup, so it fires properly
5802
                        // even if the cursor leaves this.domNode before the mouse up event.
5803
                        var mouseUpConnector = this.connect(dojo.body(), "onmouseup", function(){
5804
                                if (this.isFocusable()) {
5805
                                        this.focus();
5806
                                }
5807
                                this.disconnect(mouseUpConnector);
5808
                        });
5809
                }
5810
        }
5811
});
5812
5813
dojo.declare("dijit.form._FormValueWidget", dijit.form._FormWidget,
5814
{
5815
        // summary:
5816
        //                Base class for widgets corresponding to native HTML elements such as <input> or <select> that have user changeable values.
5817
        // description:
5818
        //                Each _FormValueWidget represents a single input value, and has a (possibly hidden) <input> element,
5819
        //                to which it serializes it's input value, so that form submission (either normal submission or via FormBind?)
5820
        //                works as expected.
5821
5822
        // Don't attempt to mixin the 'type', 'name' attributes here programatically -- they must be declared
5823
        // directly in the template as read by the parser in order to function. IE is known to specifically
5824
        // require the 'name' attribute at element creation time.   See #8484, #8660.
5825
        // TODO: unclear what that {value: ""} is for; FormWidget.attributeMap copies value to focusNode,
5826
        // so maybe {value: ""} is so the value *doesn't* get copied to focusNode?
5827
        // Seems like we really want value removed from attributeMap altogether
5828
        // (although there's no easy way to do that now)
5829
5830
        // readOnly: Boolean
5831
        //                Should this widget respond to user input?
5832
        //                In markup, this is specified as "readOnly".
5833
        //                Similar to disabled except readOnly form values are submitted.
5834
        readOnly: false,
5835
5836
        attributeMap: dojo.delegate(dijit.form._FormWidget.prototype.attributeMap, {
5837
                value: "",
5838
                readOnly: "focusNode"
5839
        }),
5840
5841
        _setReadOnlyAttr: function(/*Boolean*/ value){
5842
                this.readOnly = value;
5843
                dojo.attr(this.focusNode, 'readOnly', value);
5844
                dijit.setWaiState(this.focusNode, "readonly", value);
5845
        },
5846
5847
        postCreate: function(){
5848
                this.inherited(arguments);
5849
5850
                if(dojo.isIE){ // IE won't stop the event with keypress
5851
                        this.connect(this.focusNode || this.domNode, "onkeydown", this._onKeyDown);
5852
                }
5853
                // Update our reset value if it hasn't yet been set (because this.set()
5854
                // is only called when there *is* a value)
5855
                if(this._resetValue === undefined){
5856
                        this._resetValue = this.value;
5857
                }
5858
        },
5859
5860
        _setValueAttr: function(/*anything*/ newValue, /*Boolean, optional*/ priorityChange){
5861
                // summary:
5862
                //                Hook so attr('value', value) works.
5863
                // description:
5864
                //                Sets the value of the widget.
5865
                //                If the value has changed, then fire onChange event, unless priorityChange
5866
                //                is specified as null (or false?)
5867
                this.value = newValue;
5868
                this._handleOnChange(newValue, priorityChange);
5869
        },
5870
5871
        _getValueAttr: function(){
5872
                // summary:
5873
                //                Hook so attr('value') works.
5874
                return this._lastValue;
5875
        },
5876
5877
        undo: function(){
5878
                // summary:
5879
                //                Restore the value to the last value passed to onChange
5880
                this._setValueAttr(this._lastValueReported, false);
5881
        },
5882
5883
        reset: function(){
5884
                // summary:
5885
                //                Reset the widget's value to what it was at initialization time
5886
                this._hasBeenBlurred = false;
5887
                this._setValueAttr(this._resetValue, true);
5888
        },
5889
5890
        _onKeyDown: function(e){
5891
                if(e.keyCode == dojo.keys.ESCAPE && !(e.ctrlKey || e.altKey || e.metaKey)){
5892
                        var te;
5893
                        if(dojo.isIE){
5894
                                e.preventDefault(); // default behavior needs to be stopped here since keypress is too late
5895
                                te = document.createEventObject();
5896
                                te.keyCode = dojo.keys.ESCAPE;
5897
                                te.shiftKey = e.shiftKey;
5898
                                e.srcElement.fireEvent('onkeypress', te);
5899
                        }
5900
                }
5901
        },
5902
5903
        _layoutHackIE7: function(){
5904
                // summary:
5905
                //                Work around table sizing bugs on IE7 by forcing redraw
5906
5907
                if(dojo.isIE == 7){ // fix IE7 layout bug when the widget is scrolled out of sight
5908
                        var domNode = this.domNode;
5909
                        var parent = domNode.parentNode;
5910
                        var pingNode = domNode.firstChild || domNode; // target node most unlikely to have a custom filter
5911
                        var origFilter = pingNode.style.filter; // save custom filter, most likely nothing
5912
                        var _this = this;
5913
                        while(parent && parent.clientHeight == 0){ // search for parents that haven't rendered yet
5914
                                (function ping(){
5915
                                        var disconnectHandle = _this.connect(parent, "onscroll",
5916
                                                function(e){
5917
                                                        _this.disconnect(disconnectHandle); // only call once
5918
                                                        pingNode.style.filter = (new Date()).getMilliseconds(); // set to anything that's unique
5919
                                                        setTimeout(function(){ pingNode.style.filter = origFilter }, 0); // restore custom filter, if any
5920
                                                }
5921
                                        );
5922
                                })();
5923
                                parent = parent.parentNode;
5924
                        }
5925
                }
5926
        }
5927
});
5928
5929
}
5930
5931
if(!dojo._hasResource["dijit.dijit"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code.
5932
dojo._hasResource["dijit.dijit"] = true;
5933
dojo.provide("dijit.dijit");
5934
5935
/*=====
5936
dijit.dijit = {
5937
        // summary:
5938
        //                A roll-up for common dijit methods
5939
        // description:
5940
        //        A rollup file for the build system including the core and common
5941
        //        dijit files.
5942
        //
5943
        // example:
5944
        // | <script type="text/javascript" src="js/dojo/dijit/dijit.js"></script>
5945
        //
5946
};
5947
=====*/
5948
5949
// All the stuff in _base (these are the function that are guaranteed available without an explicit dojo.require)
5950
5951
5952
// And some other stuff that we tend to pull in all the time anyway
5953
5954
5955
5956
5957
5958
5959
5960
}