Roo/MessageBox.js
[roojs1] / Roo / MessageBox.js
1 /*
2  * Based on:
3  * Ext JS Library 1.1.1
4  * Copyright(c) 2006-2007, Ext JS, LLC.
5  *
6  * Originally Released Under LGPL - original licence link has changed is not relivant.
7  *
8  * Fork - LGPL
9  * <script type="text/javascript">
10  */
11  
12 /**
13  * @class Roo.MessageBox
14  * Utility class for generating different styles of message boxes.  The alias Roo.Msg can also be used.
15  * Example usage:
16  *<pre><code>
17 // Basic alert:
18 Roo.Msg.alert('Status', 'Changes saved successfully.');
19
20 // Prompt for user data:
21 Roo.Msg.prompt('Name', 'Please enter your name:', function(btn, text){
22     if (btn == 'ok'){
23         // process text value...
24     }
25 });
26
27 // Show a dialog using config options:
28 Roo.Msg.show({
29    title:'Save Changes?',
30    msg: 'Your are closing a tab that has unsaved changes. Would you like to save your changes?',
31    buttons: Roo.Msg.YESNOCANCEL,
32    fn: processResult,
33    animEl: 'elId'
34 });
35 </code></pre>
36  * @singleton
37  */
38 Roo.MessageBox = function(){
39     var dlg, opt, mask, waitTimer;
40     var bodyEl, msgEl, textboxEl, textareaEl, progressEl, pp;
41     var buttons, activeTextEl, bwidth;
42
43     // private
44     var handleButton = function(button){
45         dlg.hide();
46         Roo.callback(opt.fn, opt.scope||window, [button, activeTextEl.dom.value], 1);
47     };
48
49     // private
50     var handleHide = function(){
51         if(opt && opt.cls){
52             dlg.el.removeClass(opt.cls);
53         }
54         if(waitTimer){
55             Roo.TaskMgr.stop(waitTimer);
56             waitTimer = null;
57         }
58     };
59
60     // private
61     var updateButtons = function(b){
62         var width = 0;
63         if(!b){
64             buttons["ok"].hide();
65             buttons["cancel"].hide();
66             buttons["yes"].hide();
67             buttons["no"].hide();
68             dlg.footer.dom.style.display = 'none';
69             return width;
70         }
71         dlg.footer.dom.style.display = '';
72         for(var k in buttons){
73             if(typeof buttons[k] != "function"){
74                 if(b[k]){
75                     buttons[k].show();
76                     buttons[k].setText(typeof b[k] == "string" ? b[k] : Roo.MessageBox.buttonText[k]);
77                     width += buttons[k].el.getWidth()+15;
78                 }else{
79                     buttons[k].hide();
80                 }
81             }
82         }
83         return width;
84     };
85
86     // private
87     var handleEsc = function(d, k, e){
88         if(opt && opt.closable !== false){
89             dlg.hide();
90         }
91         if(e){
92             e.stopEvent();
93         }
94     };
95
96     return {
97         /**
98          * Returns a reference to the underlying {@link Roo.BasicDialog} element
99          * @return {Roo.BasicDialog} The BasicDialog element
100          */
101         getDialog : function(){
102            if(!dlg){
103                 dlg = new Roo.BasicDialog("x-msg-box", {
104                     autoCreate : true,
105                     shadow: true,
106                     draggable: true,
107                     resizable:false,
108                     constraintoviewport:false,
109                     fixedcenter:true,
110                     collapsible : false,
111                     shim:true,
112                     modal: true,
113                     width:400, height:100,
114                     buttonAlign:"center",
115                     closeClick : function(){
116                         if(opt && opt.buttons && opt.buttons.no && !opt.buttons.cancel){
117                             handleButton("no");
118                         }else{
119                             handleButton("cancel");
120                         }
121                     }
122                 });
123                 dlg.on("hide", handleHide);
124                 mask = dlg.mask;
125                 dlg.addKeyListener(27, handleEsc);
126                 buttons = {};
127                 var bt = this.buttonText;
128                 buttons["ok"] = dlg.addButton(bt["ok"], handleButton.createCallback("ok"));
129                 buttons["yes"] = dlg.addButton(bt["yes"], handleButton.createCallback("yes"));
130                 buttons["no"] = dlg.addButton(bt["no"], handleButton.createCallback("no"));
131                 buttons["cancel"] = dlg.addButton(bt["cancel"], handleButton.createCallback("cancel"));
132                 bodyEl = dlg.body.createChild({
133
134                     html:'<span class="roo-mb-text"></span><br /><input type="text" class="roo-mb-input" /><textarea class="roo-mb-textarea"></textarea><div class="roo-mb-progress-wrap"><div class="roo-mb-progress"><div class="roo-mb-progress-bar">&#160;</div></div></div>'
135                 });
136                 msgEl = bodyEl.dom.firstChild;
137                 textboxEl = Roo.get(bodyEl.dom.childNodes[2]);
138                 textboxEl.enableDisplayMode();
139                 textboxEl.addKeyListener([10,13], function(){
140                     if(dlg.isVisible() && opt && opt.buttons){
141                         if(opt.buttons.ok){
142                             handleButton("ok");
143                         }else if(opt.buttons.yes){
144                             handleButton("yes");
145                         }
146                     }
147                 });
148                 textareaEl = Roo.get(bodyEl.dom.childNodes[3]);
149                 textareaEl.enableDisplayMode();
150                 progressEl = Roo.get(bodyEl.dom.childNodes[4]);
151                 progressEl.enableDisplayMode();
152                 var pf = progressEl.dom.firstChild;
153                 if (pf) {
154                     pp = Roo.get(pf.firstChild);
155                     pp.setHeight(pf.offsetHeight);
156                 }
157                 
158             }
159             return dlg;
160         },
161
162         /**
163          * Updates the message box body text
164          * @param {String} text (optional) Replaces the message box element's innerHTML with the specified string (defaults to
165          * the XHTML-compliant non-breaking space character '&amp;#160;')
166          * @return {Roo.MessageBox} This message box
167          */
168         updateText : function(text){
169             if(!dlg.isVisible() && !opt.width){
170                 dlg.resizeTo(this.maxWidth, 100); // resize first so content is never clipped from previous shows
171             }
172             msgEl.innerHTML = text || '&#160;';
173       
174             var cw =  Math.max(msgEl.offsetWidth, msgEl.parentNode.scrollWidth);
175             //Roo.log("guesed size: " + JSON.stringify([cw,msgEl.offsetWidth, msgEl.parentNode.scrollWidth]));
176             var w = Math.max(
177                     Math.min(opt.width || cw , this.maxWidth), 
178                     Math.max(opt.minWidth || this.minWidth, bwidth)
179             );
180             if(opt.prompt){
181                 activeTextEl.setWidth(w);
182             }
183             if(dlg.isVisible()){
184                 dlg.fixedcenter = false;
185             }
186             // to big, make it scroll. = But as usual stupid IE does not support
187             // !important..
188             
189             if ( bodyEl.getHeight() > (Roo.lib.Dom.getViewHeight() - 100)) {
190                 bodyEl.setHeight ( Roo.lib.Dom.getViewHeight() - 100 );
191                 bodyEl.dom.style.overflowY = 'auto' + ( Roo.isIE ? '' : ' !important');
192             } else {
193                 bodyEl.dom.style.height = '';
194                 bodyEl.dom.style.overflowY = '';
195             }
196             if (cw > w) {
197                 bodyEl.dom.style.get = 'auto' + ( Roo.isIE ? '' : ' !important');
198             } else {
199                 bodyEl.dom.style.overflowX = '';
200             }
201             
202             dlg.setContentSize(w, bodyEl.getHeight());
203             if(dlg.isVisible()){
204                 dlg.fixedcenter = true;
205             }
206             return this;
207         },
208
209         /**
210          * Updates a progress-style message box's text and progress bar.  Only relevant on message boxes
211          * initiated via {@link Roo.MessageBox#progress} or by calling {@link Roo.MessageBox#show} with progress: true.
212          * @param {Number} value Any number between 0 and 1 (e.g., .5)
213          * @param {String} text (optional) If defined, the message box's body text is replaced with the specified string (defaults to undefined)
214          * @return {Roo.MessageBox} This message box
215          */
216         updateProgress : function(value, text){
217             if(text){
218                 this.updateText(text);
219             }
220             if (pp) { // weird bug on my firefox - for some reason this is not defined
221                 pp.setWidth(Math.floor(value*progressEl.dom.firstChild.offsetWidth));
222             }
223             return this;
224         },        
225
226         /**
227          * Returns true if the message box is currently displayed
228          * @return {Boolean} True if the message box is visible, else false
229          */
230         isVisible : function(){
231             return dlg && dlg.isVisible();  
232         },
233
234         /**
235          * Hides the message box if it is displayed
236          */
237         hide : function(){
238             if(this.isVisible()){
239                 dlg.hide();
240             }  
241         },
242
243         /**
244          * Displays a new message box, or reinitializes an existing message box, based on the config options
245          * passed in. All functions (e.g. prompt, alert, etc) on MessageBox call this function internally.
246          * The following config object properties are supported:
247          * <pre>
248 Property    Type             Description
249 ----------  ---------------  ------------------------------------------------------------------------------------
250 animEl            String/Element   An id or Element from which the message box should animate as it opens and
251                                    closes (defaults to undefined)
252 buttons           Object/Boolean   A button config object (e.g., Roo.MessageBox.OKCANCEL or {ok:'Foo',
253                                    cancel:'Bar'}), or false to not show any buttons (defaults to false)
254 closable          Boolean          False to hide the top-right close button (defaults to true).  Note that
255                                    progress and wait dialogs will ignore this property and always hide the
256                                    close button as they can only be closed programmatically.
257 cls               String           A custom CSS class to apply to the message box element
258 defaultTextHeight Number           The default height in pixels of the message box's multiline textarea if
259                                    displayed (defaults to 75)
260 fn                Function         A callback function to execute after closing the dialog.  The arguments to the
261                                    function will be btn (the name of the button that was clicked, if applicable,
262                                    e.g. "ok"), and text (the value of the active text field, if applicable).
263                                    Progress and wait dialogs will ignore this option since they do not respond to
264                                    user actions and can only be closed programmatically, so any required function
265                                    should be called by the same code after it closes the dialog.
266 icon              String           A CSS class that provides a background image to be used as an icon for
267                                    the dialog (e.g., Roo.MessageBox.WARNING or 'custom-class', defaults to '')
268 maxWidth          Number           The maximum width in pixels of the message box (defaults to 600)
269 minWidth          Number           The minimum width in pixels of the message box (defaults to 100)
270 modal             Boolean          False to allow user interaction with the page while the message box is
271                                    displayed (defaults to true)
272 msg               String           A string that will replace the existing message box body text (defaults
273                                    to the XHTML-compliant non-breaking space character '&#160;')
274 multiline         Boolean          True to prompt the user to enter multi-line text (defaults to false)
275 progress          Boolean          True to display a progress bar (defaults to false)
276 progressText      String           The text to display inside the progress bar if progress = true (defaults to '')
277 prompt            Boolean          True to prompt the user to enter single-line text (defaults to false)
278 proxyDrag         Boolean          True to display a lightweight proxy while dragging (defaults to false)
279 title             String           The title text
280 value             String           The string value to set into the active textbox element if displayed
281 wait              Boolean          True to display a progress bar (defaults to false)
282 width             Number           The width of the dialog in pixels
283 </pre>
284          *
285          * Example usage:
286          * <pre><code>
287 Roo.Msg.show({
288    title: 'Address',
289    msg: 'Please enter your address:',
290    width: 300,
291    buttons: Roo.MessageBox.OKCANCEL,
292    multiline: true,
293    fn: saveAddress,
294    animEl: 'addAddressBtn'
295 });
296 </code></pre>
297          * @param {Object} config Configuration options
298          * @return {Roo.MessageBox} This message box
299          */
300         show : function(options)
301         {
302             
303             // this causes nightmares if you show one dialog after another
304             // especially on callbacks..
305              
306             if(this.isVisible()){
307                 
308                 this.hide();
309                 Roo.log("Old Dialog Message:" +  msgEl.innerHTML )
310                 //this.alert("ERROR", "Multiple dialogs where displayed at the same time");
311                 //throw "Roo.MessageBox ERROR : Multiple dialogs where displayed at the same time";
312                 
313             }
314             var d = this.getDialog();
315             opt = options;
316             d.setTitle(opt.title || "&#160;");
317             d.close.setDisplayed(opt.closable !== false);
318             activeTextEl = textboxEl;
319             opt.prompt = opt.prompt || (opt.multiline ? true : false);
320             if(opt.prompt){
321                 if(opt.multiline){
322                     textboxEl.hide();
323                     textareaEl.show();
324                     textareaEl.setHeight(typeof opt.multiline == "number" ?
325                         opt.multiline : this.defaultTextHeight);
326                     activeTextEl = textareaEl;
327                 }else{
328                     textboxEl.show();
329                     textareaEl.hide();
330                 }
331             }else{
332                 textboxEl.hide();
333                 textareaEl.hide();
334             }
335             progressEl.setDisplayed(opt.progress === true);
336             this.updateProgress(0);
337             activeTextEl.dom.value = opt.value || "";
338             if(opt.prompt){
339                 dlg.setDefaultButton(activeTextEl);
340             }else{
341                 var bs = opt.buttons;
342                 var db = null;
343                 if(bs && bs.ok){
344                     db = buttons["ok"];
345                 }else if(bs && bs.yes){
346                     db = buttons["yes"];
347                 }
348                 dlg.setDefaultButton(db);
349             }
350             bwidth = updateButtons(opt.buttons);
351             this.updateText(opt.msg);
352             if(opt.cls){
353                 d.el.addClass(opt.cls);
354             }
355             d.proxyDrag = opt.proxyDrag === true;
356             d.modal = opt.modal !== false;
357             d.mask = opt.modal !== false ? mask : false;
358             if(!d.isVisible()){
359                 // force it to the end of the z-index stack so it gets a cursor in FF
360                 document.body.appendChild(dlg.el.dom);
361                 d.animateTarget = null;
362                 d.show(options.animEl);
363             }
364             return this;
365         },
366
367         /**
368          * Displays a message box with a progress bar.  This message box has no buttons and is not closeable by
369          * the user.  You are responsible for updating the progress bar as needed via {@link Roo.MessageBox#updateProgress}
370          * and closing the message box when the process is complete.
371          * @param {String} title The title bar text
372          * @param {String} msg The message box body text
373          * @return {Roo.MessageBox} This message box
374          */
375         progress : function(title, msg){
376             this.show({
377                 title : title,
378                 msg : msg,
379                 buttons: false,
380                 progress:true,
381                 closable:false,
382                 minWidth: this.minProgressWidth,
383                 modal : true
384             });
385             return this;
386         },
387
388         /**
389          * Displays a standard read-only message box with an OK button (comparable to the basic JavaScript Window.alert).
390          * If a callback function is passed it will be called after the user clicks the button, and the
391          * id of the button that was clicked will be passed as the only parameter to the callback
392          * (could also be the top-right close button).
393          * @param {String} title The title bar text
394          * @param {String} msg The message box body text
395          * @param {Function} fn (optional) The callback function invoked after the message box is closed
396          * @param {Object} scope (optional) The scope of the callback function
397          * @return {Roo.MessageBox} This message box
398          */
399         alert : function(title, msg, fn, scope){
400             this.show({
401                 title : title,
402                 msg : msg,
403                 buttons: this.OK,
404                 fn: fn,
405                 scope : scope,
406                 modal : true
407             });
408             return this;
409         },
410
411         /**
412          * Displays a message box with an infinitely auto-updating progress bar.  This can be used to block user
413          * interaction while waiting for a long-running process to complete that does not have defined intervals.
414          * You are responsible for closing the message box when the process is complete.
415          * @param {String} msg The message box body text
416          * @param {String} title (optional) The title bar text
417          * @return {Roo.MessageBox} This message box
418          */
419         wait : function(msg, title){
420             this.show({
421                 title : title,
422                 msg : msg,
423                 buttons: false,
424                 closable:false,
425                 progress:true,
426                 modal:true,
427                 width:300,
428                 wait:true
429             });
430             waitTimer = Roo.TaskMgr.start({
431                 run: function(i){
432                     Roo.MessageBox.updateProgress(((((i+20)%20)+1)*5)*.01);
433                 },
434                 interval: 1000
435             });
436             return this;
437         },
438
439         /**
440          * Displays a confirmation message box with Yes and No buttons (comparable to JavaScript's Window.confirm).
441          * If a callback function is passed it will be called after the user clicks either button, and the id of the
442          * button that was clicked will be passed as the only parameter to the callback (could also be the top-right close button).
443          * @param {String} title The title bar text
444          * @param {String} msg The message box body text
445          * @param {Function} fn (optional) The callback function invoked after the message box is closed
446          * @param {Object} scope (optional) The scope of the callback function
447          * @return {Roo.MessageBox} This message box
448          */
449         confirm : function(title, msg, fn, scope){
450             this.show({
451                 title : title,
452                 msg : msg,
453                 buttons: this.YESNO,
454                 fn: fn,
455                 scope : scope,
456                 modal : true
457             });
458             return this;
459         },
460
461         /**
462          * Displays a message box with OK and Cancel buttons prompting the user to enter some text (comparable to
463          * JavaScript's Window.prompt).  The prompt can be a single-line or multi-line textbox.  If a callback function
464          * is passed it will be called after the user clicks either button, and the id of the button that was clicked
465          * (could also be the top-right close button) and the text that was entered will be passed as the two
466          * parameters to the callback.
467          * @param {String} title The title bar text
468          * @param {String} msg The message box body text
469          * @param {Function} fn (optional) The callback function invoked after the message box is closed
470          * @param {Object} scope (optional) The scope of the callback function
471          * @param {Boolean/Number} multiline (optional) True to create a multiline textbox using the defaultTextHeight
472          * property, or the height in pixels to create the textbox (defaults to false / single-line)
473          * @return {Roo.MessageBox} This message box
474          */
475         prompt : function(title, msg, fn, scope, multiline){
476             this.show({
477                 title : title,
478                 msg : msg,
479                 buttons: this.OKCANCEL,
480                 fn: fn,
481                 minWidth:250,
482                 scope : scope,
483                 prompt:true,
484                 multiline: multiline,
485                 modal : true
486             });
487             return this;
488         },
489
490         /**
491          * Button config that displays a single OK button
492          * @type Object
493          */
494         OK : {ok:true},
495         /**
496          * Button config that displays Yes and No buttons
497          * @type Object
498          */
499         YESNO : {yes:true, no:true},
500         /**
501          * Button config that displays OK and Cancel buttons
502          * @type Object
503          */
504         OKCANCEL : {ok:true, cancel:true},
505         /**
506          * Button config that displays Yes, No and Cancel buttons
507          * @type Object
508          */
509         YESNOCANCEL : {yes:true, no:true, cancel:true},
510
511         /**
512          * The default height in pixels of the message box's multiline textarea if displayed (defaults to 75)
513          * @type Number
514          */
515         defaultTextHeight : 75,
516         /**
517          * The maximum width in pixels of the message box (defaults to 600)
518          * @type Number
519          */
520         maxWidth : 600,
521         /**
522          * The minimum width in pixels of the message box (defaults to 100)
523          * @type Number
524          */
525         minWidth : 100,
526         /**
527          * The minimum width in pixels of the message box if it is a progress-style dialog.  This is useful
528          * for setting a different minimum width than text-only dialogs may need (defaults to 250)
529          * @type Number
530          */
531         minProgressWidth : 250,
532         /**
533          * An object containing the default button text strings that can be overriden for localized language support.
534          * Supported properties are: ok, cancel, yes and no.
535          * Customize the default text like so: Roo.MessageBox.buttonText.yes = "S?";
536          * @type Object
537          */
538         buttonText : {
539             ok : "OK",
540             cancel : "Cancel",
541             yes : "Yes",
542             no : "No"
543         }
544     };
545 }();
546
547 /**
548  * Shorthand for {@link Roo.MessageBox}
549  */
550 Roo.Msg = Roo.MessageBox;