JasonWoof Got questions, comments, patches, etc.? Contact Jason Woofenden
vanilla ckeditor-3.3.2
[ckeditor.git] / _source / plugins / dialog / plugin.js
1 /*\r
2 Copyright (c) 2003-2010, CKSource - Frederico Knabben. All rights reserved.\r
3 For licensing, see LICENSE.html or http://ckeditor.com/license\r
4 */\r
5 \r
6 /**\r
7  * @fileOverview The floating dialog plugin.\r
8  */\r
9 \r
10 /**\r
11  * No resize for this dialog.\r
12  * @constant\r
13  */\r
14 CKEDITOR.DIALOG_RESIZE_NONE = 0;\r
15 \r
16 /**\r
17  * Only allow horizontal resizing for this dialog, disable vertical resizing.\r
18  * @constant\r
19  */\r
20 CKEDITOR.DIALOG_RESIZE_WIDTH = 1;\r
21 \r
22 /**\r
23  * Only allow vertical resizing for this dialog, disable horizontal resizing.\r
24  * @constant\r
25  */\r
26 CKEDITOR.DIALOG_RESIZE_HEIGHT = 2;\r
27 \r
28 /*\r
29  * Allow the dialog to be resized in both directions.\r
30  * @constant\r
31  */\r
32 CKEDITOR.DIALOG_RESIZE_BOTH = 3;\r
33 \r
34 (function()\r
35 {\r
36         function isTabVisible( tabId )\r
37         {\r
38                 return !!this._.tabs[ tabId ][ 0 ].$.offsetHeight;\r
39         }\r
40 \r
41         function getPreviousVisibleTab()\r
42         {\r
43                 var tabId = this._.currentTabId,\r
44                         length = this._.tabIdList.length,\r
45                         tabIndex = CKEDITOR.tools.indexOf( this._.tabIdList, tabId ) + length;\r
46 \r
47                 for ( var i = tabIndex - 1 ; i > tabIndex - length ; i-- )\r
48                 {\r
49                         if ( isTabVisible.call( this, this._.tabIdList[ i % length ] ) )\r
50                                 return this._.tabIdList[ i % length ];\r
51                 }\r
52 \r
53                 return null;\r
54         }\r
55 \r
56         function getNextVisibleTab()\r
57         {\r
58                 var tabId = this._.currentTabId,\r
59                         length = this._.tabIdList.length,\r
60                         tabIndex = CKEDITOR.tools.indexOf( this._.tabIdList, tabId );\r
61 \r
62                 for ( var i = tabIndex + 1 ; i < tabIndex + length ; i++ )\r
63                 {\r
64                         if ( isTabVisible.call( this, this._.tabIdList[ i % length ] ) )\r
65                                 return this._.tabIdList[ i % length ];\r
66                 }\r
67 \r
68                 return null;\r
69         }\r
70 \r
71         /**\r
72          * This is the base class for runtime dialog objects. An instance of this\r
73          * class represents a single named dialog for a single editor instance.\r
74          * @param {Object} editor The editor which created the dialog.\r
75          * @param {String} dialogName The dialog's registered name.\r
76          * @constructor\r
77          * @example\r
78          * var dialogObj = new CKEDITOR.dialog( editor, 'smiley' );\r
79          */\r
80         CKEDITOR.dialog = function( editor, dialogName )\r
81         {\r
82                 // Load the dialog definition.\r
83                 var definition = CKEDITOR.dialog._.dialogDefinitions[ dialogName ];\r
84 \r
85                 // Completes the definition with the default values.\r
86                 definition = CKEDITOR.tools.extend( definition( editor ), defaultDialogDefinition );\r
87 \r
88                 // Clone a functionally independent copy for this dialog.\r
89                 definition = CKEDITOR.tools.clone( definition );\r
90 \r
91                 // Create a complex definition object, extending it with the API\r
92                 // functions.\r
93                 definition = new definitionObject( this, definition );\r
94 \r
95 \r
96                 var doc = CKEDITOR.document;\r
97 \r
98                 var themeBuilt = editor.theme.buildDialog( editor );\r
99 \r
100                 // Initialize some basic parameters.\r
101                 this._ =\r
102                 {\r
103                         editor : editor,\r
104                         element : themeBuilt.element,\r
105                         name : dialogName,\r
106                         contentSize : { width : 0, height : 0 },\r
107                         size : { width : 0, height : 0 },\r
108                         updateSize : false,\r
109                         contents : {},\r
110                         buttons : {},\r
111                         accessKeyMap : {},\r
112 \r
113                         // Initialize the tab and page map.\r
114                         tabs : {},\r
115                         tabIdList : [],\r
116                         currentTabId : null,\r
117                         currentTabIndex : null,\r
118                         pageCount : 0,\r
119                         lastTab : null,\r
120                         tabBarMode : false,\r
121 \r
122                         // Initialize the tab order array for input widgets.\r
123                         focusList : [],\r
124                         currentFocusIndex : 0,\r
125                         hasFocus : false\r
126                 };\r
127 \r
128                 this.parts = themeBuilt.parts;\r
129 \r
130                 CKEDITOR.tools.setTimeout( function()\r
131                         {\r
132                                 editor.fire( 'ariaWidget', this.parts.contents );\r
133                         },\r
134                         0, this );\r
135 \r
136                 // Set the startup styles for the dialog, avoiding it enlarging the\r
137                 // page size on the dialog creation.\r
138                 this.parts.dialog.setStyles(\r
139                         {\r
140                                 position : CKEDITOR.env.ie6Compat ? 'absolute' : 'fixed',\r
141                                 top : 0,\r
142                                 left: 0,\r
143                                 visibility : 'hidden'\r
144                         });\r
145 \r
146                 // Call the CKEDITOR.event constructor to initialize this instance.\r
147                 CKEDITOR.event.call( this );\r
148 \r
149                 // Fire the "dialogDefinition" event, making it possible to customize\r
150                 // the dialog definition.\r
151                 this.definition = definition = CKEDITOR.fire( 'dialogDefinition',\r
152                         {\r
153                                 name : dialogName,\r
154                                 definition : definition\r
155                         }\r
156                         , editor ).definition;\r
157                 // Initialize load, show, hide, ok and cancel events.\r
158                 if ( definition.onLoad )\r
159                         this.on( 'load', definition.onLoad );\r
160 \r
161                 if ( definition.onShow )\r
162                         this.on( 'show', definition.onShow );\r
163 \r
164                 if ( definition.onHide )\r
165                         this.on( 'hide', definition.onHide );\r
166 \r
167                 if ( definition.onOk )\r
168                 {\r
169                         this.on( 'ok', function( evt )\r
170                                 {\r
171                                         // Dialog confirm might probably introduce content changes (#5415).\r
172                                         editor.fire( 'saveSnapshot' );\r
173                                         setTimeout( function () { editor.fire( 'saveSnapshot' ); }, 0 );\r
174                                         if ( definition.onOk.call( this, evt ) === false )\r
175                                                 evt.data.hide = false;\r
176                                 });\r
177                 }\r
178 \r
179                 if ( definition.onCancel )\r
180                 {\r
181                         this.on( 'cancel', function( evt )\r
182                                 {\r
183                                         if ( definition.onCancel.call( this, evt ) === false )\r
184                                                 evt.data.hide = false;\r
185                                 });\r
186                 }\r
187 \r
188                 var me = this;\r
189 \r
190                 // Iterates over all items inside all content in the dialog, calling a\r
191                 // function for each of them.\r
192                 var iterContents = function( func )\r
193                 {\r
194                         var contents = me._.contents,\r
195                                 stop = false;\r
196 \r
197                         for ( var i in contents )\r
198                         {\r
199                                 for ( var j in contents[i] )\r
200                                 {\r
201                                         stop = func.call( this, contents[i][j] );\r
202                                         if ( stop )\r
203                                                 return;\r
204                                 }\r
205                         }\r
206                 };\r
207 \r
208                 this.on( 'ok', function( evt )\r
209                         {\r
210                                 iterContents( function( item )\r
211                                         {\r
212                                                 if ( item.validate )\r
213                                                 {\r
214                                                         var isValid = item.validate( this );\r
215 \r
216                                                         if ( typeof isValid == 'string' )\r
217                                                         {\r
218                                                                 alert( isValid );\r
219                                                                 isValid = false;\r
220                                                         }\r
221 \r
222                                                         if ( isValid === false )\r
223                                                         {\r
224                                                                 if ( item.select )\r
225                                                                         item.select();\r
226                                                                 else\r
227                                                                         item.focus();\r
228 \r
229                                                                 evt.data.hide = false;\r
230                                                                 evt.stop();\r
231                                                                 return true;\r
232                                                         }\r
233                                                 }\r
234                                         });\r
235                         }, this, null, 0 );\r
236 \r
237                 this.on( 'cancel', function( evt )\r
238                         {\r
239                                 iterContents( function( item )\r
240                                         {\r
241                                                 if ( item.isChanged() )\r
242                                                 {\r
243                                                         if ( !confirm( editor.lang.common.confirmCancel ) )\r
244                                                                 evt.data.hide = false;\r
245                                                         return true;\r
246                                                 }\r
247                                         });\r
248                         }, this, null, 0 );\r
249 \r
250                 this.parts.close.on( 'click', function( evt )\r
251                                 {\r
252                                         if ( this.fire( 'cancel', { hide : true } ).hide !== false )\r
253                                                 this.hide();\r
254                                         evt.data.preventDefault();\r
255                                 }, this );\r
256 \r
257                 // Sort focus list according to tab order definitions.\r
258                 function setupFocus()\r
259                 {\r
260                         var focusList = me._.focusList;\r
261                         focusList.sort( function( a, b )\r
262                                 {\r
263                                         // Mimics browser tab order logics;\r
264                                         if ( a.tabIndex != b.tabIndex )\r
265                                                 return b.tabIndex - a.tabIndex;\r
266                                         //  Sort is not stable in some browsers,\r
267                                         // fall-back the comparator to 'focusIndex';\r
268                                         else\r
269                                                 return a.focusIndex - b.focusIndex;\r
270                                 });\r
271 \r
272                         var size = focusList.length;\r
273                         for ( var i = 0; i < size; i++ )\r
274                                 focusList[ i ].focusIndex = i;\r
275                 }\r
276 \r
277                 function changeFocus( forward )\r
278                 {\r
279                         var focusList = me._.focusList,\r
280                                 offset = forward ? 1 : -1;\r
281                         if ( focusList.length < 1 )\r
282                                 return;\r
283 \r
284                         var current = me._.currentFocusIndex;\r
285 \r
286                         // Trigger the 'blur' event of  any input element before anything,\r
287                         // since certain UI updates may depend on it.\r
288                         try\r
289                         {\r
290                                 focusList[ current ].getInputElement().$.blur();\r
291                         }\r
292                         catch( e ){}\r
293 \r
294                         var startIndex = ( current + offset + focusList.length ) % focusList.length,\r
295                                 currentIndex = startIndex;\r
296                         while ( !focusList[ currentIndex ].isFocusable() )\r
297                         {\r
298                                 currentIndex = ( currentIndex + offset + focusList.length ) % focusList.length;\r
299                                 if ( currentIndex == startIndex )\r
300                                         break;\r
301                         }\r
302                         focusList[ currentIndex ].focus();\r
303 \r
304                         // Select whole field content.\r
305                         if ( focusList[ currentIndex ].type == 'text' )\r
306                                 focusList[ currentIndex ].select();\r
307                 }\r
308 \r
309                 this.changeFocus = changeFocus;\r
310 \r
311                 var processed;\r
312 \r
313                 function focusKeydownHandler( evt )\r
314                 {\r
315                         // If I'm not the top dialog, ignore.\r
316                         if ( me != CKEDITOR.dialog._.currentTop )\r
317                                 return;\r
318 \r
319                         var keystroke = evt.data.getKeystroke(),\r
320                                 rtl = editor.lang.dir == 'rtl';\r
321 \r
322                         processed = 0;\r
323                         if ( keystroke == 9 || keystroke == CKEDITOR.SHIFT + 9 )\r
324                         {\r
325                                 var shiftPressed = ( keystroke == CKEDITOR.SHIFT + 9 );\r
326 \r
327                                 // Handling Tab and Shift-Tab.\r
328                                 if ( me._.tabBarMode )\r
329                                 {\r
330                                         // Change tabs.\r
331                                         var nextId = shiftPressed ? getPreviousVisibleTab.call( me ) : getNextVisibleTab.call( me );\r
332                                         me.selectPage( nextId );\r
333                                         me._.tabs[ nextId ][ 0 ].focus();\r
334                                 }\r
335                                 else\r
336                                 {\r
337                                         // Change the focus of inputs.\r
338                                         changeFocus( !shiftPressed );\r
339                                 }\r
340 \r
341                                 processed = 1;\r
342                         }\r
343                         else if ( keystroke == CKEDITOR.ALT + 121 && !me._.tabBarMode && me.getPageCount() > 1 )\r
344                         {\r
345                                 // Alt-F10 puts focus into the current tab item in the tab bar.\r
346                                 me._.tabBarMode = true;\r
347                                 me._.tabs[ me._.currentTabId ][ 0 ].focus();\r
348                                 processed = 1;\r
349                         }\r
350                         else if ( ( keystroke == 37 || keystroke == 39 ) && me._.tabBarMode )\r
351                         {\r
352                                 // Arrow keys - used for changing tabs.\r
353                                 nextId = ( keystroke == ( rtl ? 39 : 37 ) ? getPreviousVisibleTab.call( me ) : getNextVisibleTab.call( me ) );\r
354                                 me.selectPage( nextId );\r
355                                 me._.tabs[ nextId ][ 0 ].focus();\r
356                                 processed = 1;\r
357                         }\r
358                         else if ( ( keystroke == 13 || keystroke == 32 ) && me._.tabBarMode )\r
359                         {\r
360                                 this.selectPage( this._.currentTabId );\r
361                                 this._.tabBarMode = false;\r
362                                 this._.currentFocusIndex = -1;\r
363                                 changeFocus( true );\r
364                                 processed = 1;\r
365                         }\r
366 \r
367                         if ( processed )\r
368                         {\r
369                                 evt.stop();\r
370                                 evt.data.preventDefault();\r
371                         }\r
372                 }\r
373 \r
374                 function focusKeyPressHandler( evt )\r
375                 {\r
376                         processed && evt.data.preventDefault();\r
377                 }\r
378 \r
379                 var dialogElement = this._.element;\r
380                 // Add the dialog keyboard handlers.\r
381                 this.on( 'show', function()\r
382                         {\r
383                                 dialogElement.on( 'keydown', focusKeydownHandler, this, null, 0 );\r
384                                 // Some browsers instead, don't cancel key events in the keydown, but in the\r
385                                 // keypress. So we must do a longer trip in those cases. (#4531)\r
386                                 if ( CKEDITOR.env.opera || ( CKEDITOR.env.gecko && CKEDITOR.env.mac ) )\r
387                                         dialogElement.on( 'keypress', focusKeyPressHandler, this );\r
388 \r
389                         } );\r
390                 this.on( 'hide', function()\r
391                         {\r
392                                 dialogElement.removeListener( 'keydown', focusKeydownHandler );\r
393                                 if ( CKEDITOR.env.opera || ( CKEDITOR.env.gecko && CKEDITOR.env.mac ) )\r
394                                         dialogElement.removeListener( 'keypress', focusKeyPressHandler );\r
395                         } );\r
396                 this.on( 'iframeAdded', function( evt )\r
397                         {\r
398                                 var doc = new CKEDITOR.dom.document( evt.data.iframe.$.contentWindow.document );\r
399                                 doc.on( 'keydown', focusKeydownHandler, this, null, 0 );\r
400                         } );\r
401 \r
402                 // Auto-focus logic in dialog.\r
403                 this.on( 'show', function()\r
404                         {\r
405                                 // Setup tabIndex on showing the dialog instead of on loading\r
406                                 // to allow dynamic tab order happen in dialog definition.\r
407                                 setupFocus();\r
408 \r
409                                 if ( editor.config.dialog_startupFocusTab\r
410                                         && me._.pageCount > 1 )\r
411                                 {\r
412                                         me._.tabBarMode = true;\r
413                                         me._.tabs[ me._.currentTabId ][ 0 ].focus();\r
414                                 }\r
415                                 else if ( !this._.hasFocus )\r
416                                 {\r
417                                         this._.currentFocusIndex = -1;\r
418 \r
419                                         // Decide where to put the initial focus.\r
420                                         if ( definition.onFocus )\r
421                                         {\r
422                                                 var initialFocus = definition.onFocus.call( this );\r
423                                                 // Focus the field that the user specified.\r
424                                                 initialFocus && initialFocus.focus();\r
425                                         }\r
426                                         // Focus the first field in layout order.\r
427                                         else\r
428                                                 changeFocus( true );\r
429 \r
430                                         /*\r
431                                          * IE BUG: If the initial focus went into a non-text element (e.g. button),\r
432                                          * then IE would still leave the caret inside the editing area.\r
433                                          */\r
434                                         if ( this._.editor.mode == 'wysiwyg' && CKEDITOR.env.ie )\r
435                                         {\r
436                                                 var $selection = editor.document.$.selection,\r
437                                                         $range = $selection.createRange();\r
438 \r
439                                                 if ( $range )\r
440                                                 {\r
441                                                         if ( $range.parentElement && $range.parentElement().ownerDocument == editor.document.$\r
442                                                           || $range.item && $range.item( 0 ).ownerDocument == editor.document.$ )\r
443                                                         {\r
444                                                                 var $myRange = document.body.createTextRange();\r
445                                                                 $myRange.moveToElementText( this.getElement().getFirst().$ );\r
446                                                                 $myRange.collapse( true );\r
447                                                                 $myRange.select();\r
448                                                         }\r
449                                                 }\r
450                                         }\r
451                                 }\r
452                         }, this, null, 0xffffffff );\r
453 \r
454                 // IE6 BUG: Text fields and text areas are only half-rendered the first time the dialog appears in IE6 (#2661).\r
455                 // This is still needed after [2708] and [2709] because text fields in hidden TR tags are still broken.\r
456                 if ( CKEDITOR.env.ie6Compat )\r
457                 {\r
458                         this.on( 'load', function( evt )\r
459                                         {\r
460                                                 var outer = this.getElement(),\r
461                                                         inner = outer.getFirst();\r
462                                                 inner.remove();\r
463                                                 inner.appendTo( outer );\r
464                                         }, this );\r
465                 }\r
466 \r
467                 initDragAndDrop( this );\r
468                 initResizeHandles( this );\r
469 \r
470                 // Insert the title.\r
471                 ( new CKEDITOR.dom.text( definition.title, CKEDITOR.document ) ).appendTo( this.parts.title );\r
472 \r
473                 // Insert the tabs and contents.\r
474                 for ( var i = 0 ; i < definition.contents.length ; i++ )\r
475                         this.addPage( definition.contents[i] );\r
476 \r
477                 this.parts['tabs'].on( 'click', function( evt )\r
478                                 {\r
479                                         var target = evt.data.getTarget();\r
480                                         // If we aren't inside a tab, bail out.\r
481                                         if ( target.hasClass( 'cke_dialog_tab' ) )\r
482                                         {\r
483                                                 var id = target.$.id;\r
484                                                 this.selectPage( id.substr( 0, id.lastIndexOf( '_' ) ) );\r
485                                                 if ( this._.tabBarMode )\r
486                                                 {\r
487                                                         this._.tabBarMode = false;\r
488                                                         this._.currentFocusIndex = -1;\r
489                                                         changeFocus( true );\r
490                                                 }\r
491                                                 evt.data.preventDefault();\r
492                                         }\r
493                                 }, this );\r
494 \r
495                 // Insert buttons.\r
496                 var buttonsHtml = [],\r
497                         buttons = CKEDITOR.dialog._.uiElementBuilders.hbox.build( this,\r
498                                 {\r
499                                         type : 'hbox',\r
500                                         className : 'cke_dialog_footer_buttons',\r
501                                         widths : [],\r
502                                         children : definition.buttons\r
503                                 }, buttonsHtml ).getChild();\r
504                 this.parts.footer.setHtml( buttonsHtml.join( '' ) );\r
505 \r
506                 for ( i = 0 ; i < buttons.length ; i++ )\r
507                         this._.buttons[ buttons[i].id ] = buttons[i];\r
508         };\r
509 \r
510         // Focusable interface. Use it via dialog.addFocusable.\r
511         function Focusable( dialog, element, index )\r
512         {\r
513                 this.element = element;\r
514                 this.focusIndex = index;\r
515                 // TODO: support tabIndex for focusables.\r
516                 this.tabIndex = 0;\r
517                 this.isFocusable = function()\r
518                 {\r
519                         return !element.getAttribute( 'disabled' ) && element.isVisible();\r
520                 };\r
521                 this.focus = function()\r
522                 {\r
523                         dialog._.currentFocusIndex = this.focusIndex;\r
524                         this.element.focus();\r
525                 };\r
526                 // Bind events\r
527                 element.on( 'keydown', function( e )\r
528                         {\r
529                                 if ( e.data.getKeystroke() in { 32:1, 13:1 }  )\r
530                                         this.fire( 'click' );\r
531                         } );\r
532                 element.on( 'focus', function()\r
533                         {\r
534                                 this.fire( 'mouseover' );\r
535                         } );\r
536                 element.on( 'blur', function()\r
537                         {\r
538                                 this.fire( 'mouseout' );\r
539                         } );\r
540         }\r
541 \r
542         CKEDITOR.dialog.prototype =\r
543         {\r
544                 destroy : function()\r
545                 {\r
546                         this.hide();\r
547                         this._.element.remove();\r
548                 },\r
549 \r
550                 /**\r
551                  * Resizes the dialog.\r
552                  * @param {Number} width The width of the dialog in pixels.\r
553                  * @param {Number} height The height of the dialog in pixels.\r
554                  * @function\r
555                  * @example\r
556                  * dialogObj.resize( 800, 640 );\r
557                  */\r
558                 resize : (function()\r
559                 {\r
560                         return function( width, height )\r
561                         {\r
562                                 if ( this._.contentSize && this._.contentSize.width == width && this._.contentSize.height == height )\r
563                                         return;\r
564 \r
565                                 CKEDITOR.dialog.fire( 'resize',\r
566                                         {\r
567                                                 dialog : this,\r
568                                                 skin : this._.editor.skinName,\r
569                                                 width : width,\r
570                                                 height : height\r
571                                         }, this._.editor );\r
572 \r
573                                 this._.contentSize = { width : width, height : height };\r
574                                 this._.updateSize = true;\r
575                         };\r
576                 })(),\r
577 \r
578                 /**\r
579                  * Gets the current size of the dialog in pixels.\r
580                  * @returns {Object} An object with "width" and "height" properties.\r
581                  * @example\r
582                  * var width = dialogObj.getSize().width;\r
583                  */\r
584                 getSize : function()\r
585                 {\r
586                         if ( !this._.updateSize )\r
587                                 return this._.size;\r
588                         var element = this._.element.getFirst();\r
589                         var size = this._.size = { width : element.$.offsetWidth || 0, height : element.$.offsetHeight || 0};\r
590 \r
591                         // If either the offsetWidth or offsetHeight is 0, the element isn't visible.\r
592                         this._.updateSize = !size.width || !size.height;\r
593 \r
594                         return size;\r
595                 },\r
596 \r
597                 /**\r
598                  * Moves the dialog to an (x, y) coordinate relative to the window.\r
599                  * @function\r
600                  * @param {Number} x The target x-coordinate.\r
601                  * @param {Number} y The target y-coordinate.\r
602                  * @example\r
603                  * dialogObj.move( 10, 40 );\r
604                  */\r
605                 move : (function()\r
606                 {\r
607                         var isFixed;\r
608                         return function( x, y )\r
609                         {\r
610                                 // The dialog may be fixed positioned or absolute positioned. Ask the\r
611                                 // browser what is the current situation first.\r
612                                 var element = this._.element.getFirst();\r
613                                 if ( isFixed === undefined )\r
614                                         isFixed = element.getComputedStyle( 'position' ) == 'fixed';\r
615 \r
616                                 if ( isFixed && this._.position && this._.position.x == x && this._.position.y == y )\r
617                                         return;\r
618 \r
619                                 // Save the current position.\r
620                                 this._.position = { x : x, y : y };\r
621 \r
622                                 // If not fixed positioned, add scroll position to the coordinates.\r
623                                 if ( !isFixed )\r
624                                 {\r
625                                         var scrollPosition = CKEDITOR.document.getWindow().getScrollPosition();\r
626                                         x += scrollPosition.x;\r
627                                         y += scrollPosition.y;\r
628                                 }\r
629 \r
630                                 element.setStyles(\r
631                                                 {\r
632                                                         'left'  : ( x > 0 ? x : 0 ) + 'px',\r
633                                                         'top'   : ( y > 0 ? y : 0 ) + 'px'\r
634                                                 });\r
635                         };\r
636                 })(),\r
637 \r
638                 /**\r
639                  * Gets the dialog's position in the window.\r
640                  * @returns {Object} An object with "x" and "y" properties.\r
641                  * @example\r
642                  * var dialogX = dialogObj.getPosition().x;\r
643                  */\r
644                 getPosition : function(){ return CKEDITOR.tools.extend( {}, this._.position ); },\r
645 \r
646                 /**\r
647                  * Shows the dialog box.\r
648                  * @example\r
649                  * dialogObj.show();\r
650                  */\r
651                 show : function()\r
652                 {\r
653                         var editor = this._.editor;\r
654                         if ( editor.mode == 'wysiwyg' && CKEDITOR.env.ie )\r
655                         {\r
656                                 var selection = editor.getSelection();\r
657                                 selection && selection.lock();\r
658                         }\r
659 \r
660                         // Insert the dialog's element to the root document.\r
661                         var element = this._.element;\r
662                         var definition = this.definition;\r
663                         if ( !( element.getParent() && element.getParent().equals( CKEDITOR.document.getBody() ) ) )\r
664                                 element.appendTo( CKEDITOR.document.getBody() );\r
665                         else\r
666                                 element.setStyle( 'display', 'block' );\r
667 \r
668                         // FIREFOX BUG: Fix vanishing caret for Firefox 2 or Gecko 1.8.\r
669                         if ( CKEDITOR.env.gecko && CKEDITOR.env.version < 10900 )\r
670                         {\r
671                                 var dialogElement = this.parts.dialog;\r
672                                 dialogElement.setStyle( 'position', 'absolute' );\r
673                                 setTimeout( function()\r
674                                         {\r
675                                                 dialogElement.setStyle( 'position', 'fixed' );\r
676                                         }, 0 );\r
677                         }\r
678 \r
679 \r
680                         // First, set the dialog to an appropriate size.\r
681                         this.resize( definition.minWidth, definition.minHeight );\r
682 \r
683                         // Select the first tab by default.\r
684                         this.selectPage( this.definition.contents[0].id );\r
685 \r
686                         // Reset all inputs back to their default value.\r
687                         this.reset();\r
688 \r
689                         // Set z-index.\r
690                         if ( CKEDITOR.dialog._.currentZIndex === null )\r
691                                 CKEDITOR.dialog._.currentZIndex = this._.editor.config.baseFloatZIndex;\r
692                         this._.element.getFirst().setStyle( 'z-index', CKEDITOR.dialog._.currentZIndex += 10 );\r
693 \r
694                         // Maintain the dialog ordering and dialog cover.\r
695                         // Also register key handlers if first dialog.\r
696                         if ( CKEDITOR.dialog._.currentTop === null )\r
697                         {\r
698                                 CKEDITOR.dialog._.currentTop = this;\r
699                                 this._.parentDialog = null;\r
700                                 showCover( this._.editor );\r
701 \r
702                                 element.on( 'keydown', accessKeyDownHandler );\r
703                                 element.on( CKEDITOR.env.opera ? 'keypress' : 'keyup', accessKeyUpHandler );\r
704 \r
705                                 // Prevent some keys from bubbling up. (#4269)\r
706                                 for ( var event in { keyup :1, keydown :1, keypress :1 } )\r
707                                         element.on( event, preventKeyBubbling );\r
708                         }\r
709                         else\r
710                         {\r
711                                 this._.parentDialog = CKEDITOR.dialog._.currentTop;\r
712                                 var parentElement = this._.parentDialog.getElement().getFirst();\r
713                                 parentElement.$.style.zIndex  -= Math.floor( this._.editor.config.baseFloatZIndex / 2 );\r
714                                 CKEDITOR.dialog._.currentTop = this;\r
715                         }\r
716 \r
717                         // Register the Esc hotkeys.\r
718                         registerAccessKey( this, this, '\x1b', null, function()\r
719                                         {\r
720                                                 this.getButton( 'cancel' ) && this.getButton( 'cancel' ).click();\r
721                                         } );\r
722 \r
723                         // Reset the hasFocus state.\r
724                         this._.hasFocus = false;\r
725 \r
726                         // Rearrange the dialog to the middle of the window.\r
727                         CKEDITOR.tools.setTimeout( function()\r
728                                 {\r
729                                         var viewSize = CKEDITOR.document.getWindow().getViewPaneSize();\r
730                                         var dialogSize = this.getSize();\r
731 \r
732                                         // We're using definition size for initial position because of\r
733                                         // offten corrupted data in offsetWidth at this point. (#4084)\r
734                                         this.move( ( viewSize.width - definition.minWidth ) / 2, ( viewSize.height - dialogSize.height ) / 2 );\r
735 \r
736                                         this.parts.dialog.setStyle( 'visibility', '' );\r
737 \r
738                                         // Execute onLoad for the first show.\r
739                                         this.fireOnce( 'load', {} );\r
740                                         this.fire( 'show', {} );\r
741                                         this._.editor.fire( 'dialogShow', this );\r
742 \r
743                                         // Save the initial values of the dialog.\r
744                                         this.foreach( function( contentObj ) { contentObj.setInitValue && contentObj.setInitValue(); } );\r
745 \r
746                                 },\r
747                                 100, this );\r
748                 },\r
749 \r
750                 /**\r
751                  * Executes a function for each UI element.\r
752                  * @param {Function} fn Function to execute for each UI element.\r
753                  * @returns {CKEDITOR.dialog} The current dialog object.\r
754                  */\r
755                 foreach : function( fn )\r
756                 {\r
757                         for ( var i in this._.contents )\r
758                         {\r
759                                 for ( var j in this._.contents[i] )\r
760                                         fn( this._.contents[i][j]);\r
761                         }\r
762                         return this;\r
763                 },\r
764 \r
765                 /**\r
766                  * Resets all input values in the dialog.\r
767                  * @example\r
768                  * dialogObj.reset();\r
769                  * @returns {CKEDITOR.dialog} The current dialog object.\r
770                  */\r
771                 reset : (function()\r
772                 {\r
773                         var fn = function( widget ){ if ( widget.reset ) widget.reset(); };\r
774                         return function(){ this.foreach( fn ); return this; };\r
775                 })(),\r
776 \r
777                 setupContent : function()\r
778                 {\r
779                         var args = arguments;\r
780                         this.foreach( function( widget )\r
781                                 {\r
782                                         if ( widget.setup )\r
783                                                 widget.setup.apply( widget, args );\r
784                                 });\r
785                 },\r
786 \r
787                 commitContent : function()\r
788                 {\r
789                         var args = arguments;\r
790                         this.foreach( function( widget )\r
791                                 {\r
792                                         if ( widget.commit )\r
793                                                 widget.commit.apply( widget, args );\r
794                                 });\r
795                 },\r
796 \r
797                 /**\r
798                  * Hides the dialog box.\r
799                  * @example\r
800                  * dialogObj.hide();\r
801                  */\r
802                 hide : function()\r
803                 {\r
804                         if ( !this.parts.dialog.isVisible() )\r
805                                 return;\r
806 \r
807                         this.fire( 'hide', {} );\r
808                         this._.editor.fire( 'dialogHide', this );\r
809                         var element = this._.element;\r
810                         element.setStyle( 'display', 'none' );\r
811                         this.parts.dialog.setStyle( 'visibility', 'hidden' );\r
812                         // Unregister all access keys associated with this dialog.\r
813                         unregisterAccessKey( this );\r
814 \r
815                         // Close any child(top) dialogs first.\r
816                         while( CKEDITOR.dialog._.currentTop != this )\r
817                                 CKEDITOR.dialog._.currentTop.hide();\r
818 \r
819                         // Maintain dialog ordering and remove cover if needed.\r
820                         if ( !this._.parentDialog )\r
821                                 hideCover();\r
822                         else\r
823                         {\r
824                                 var parentElement = this._.parentDialog.getElement().getFirst();\r
825                                 parentElement.setStyle( 'z-index', parseInt( parentElement.$.style.zIndex, 10 ) + Math.floor( this._.editor.config.baseFloatZIndex / 2 ) );\r
826                         }\r
827                         CKEDITOR.dialog._.currentTop = this._.parentDialog;\r
828 \r
829                         // Deduct or clear the z-index.\r
830                         if ( !this._.parentDialog )\r
831                         {\r
832                                 CKEDITOR.dialog._.currentZIndex = null;\r
833 \r
834                                 // Remove access key handlers.\r
835                                 element.removeListener( 'keydown', accessKeyDownHandler );\r
836                                 element.removeListener( CKEDITOR.env.opera ? 'keypress' : 'keyup', accessKeyUpHandler );\r
837 \r
838                                 // Remove bubbling-prevention handler. (#4269)\r
839                                 for ( var event in { keyup :1, keydown :1, keypress :1 } )\r
840                                         element.removeListener( event, preventKeyBubbling );\r
841 \r
842                                 var editor = this._.editor;\r
843                                 editor.focus();\r
844 \r
845                                 if ( editor.mode == 'wysiwyg' && CKEDITOR.env.ie )\r
846                                 {\r
847                                         var selection = editor.getSelection();\r
848                                         selection && selection.unlock( true );\r
849                                 }\r
850                         }\r
851                         else\r
852                                 CKEDITOR.dialog._.currentZIndex -= 10;\r
853 \r
854                         delete this._.parentDialog;\r
855                         // Reset the initial values of the dialog.\r
856                         this.foreach( function( contentObj ) { contentObj.resetInitValue && contentObj.resetInitValue(); } );\r
857                 },\r
858 \r
859                 /**\r
860                  * Adds a tabbed page into the dialog.\r
861                  * @param {Object} contents Content definition.\r
862                  * @example\r
863                  */\r
864                 addPage : function( contents )\r
865                 {\r
866                         var pageHtml = [],\r
867                                 titleHtml = contents.label ? ' title="' + CKEDITOR.tools.htmlEncode( contents.label ) + '"' : '',\r
868                                 elements = contents.elements,\r
869                                 vbox = CKEDITOR.dialog._.uiElementBuilders.vbox.build( this,\r
870                                                 {\r
871                                                         type : 'vbox',\r
872                                                         className : 'cke_dialog_page_contents',\r
873                                                         children : contents.elements,\r
874                                                         expand : !!contents.expand,\r
875                                                         padding : contents.padding,\r
876                                                         style : contents.style || 'width: 100%; height: 100%;'\r
877                                                 }, pageHtml );\r
878 \r
879                         // Create the HTML for the tab and the content block.\r
880                         var page = CKEDITOR.dom.element.createFromHtml( pageHtml.join( '' ) );\r
881                         page.setAttribute( 'role', 'tabpanel' );\r
882 \r
883                         var env = CKEDITOR.env;\r
884                         var tabId = contents.id + '_' + CKEDITOR.tools.getNextNumber(),\r
885                                  tab = CKEDITOR.dom.element.createFromHtml( [\r
886                                         '<a class="cke_dialog_tab"',\r
887                                                 ( this._.pageCount > 0 ? ' cke_last' : 'cke_first' ),\r
888                                                 titleHtml,\r
889                                                 ( !!contents.hidden ? ' style="display:none"' : '' ),\r
890                                                 ' id="', tabId, '"',\r
891                                                 env.gecko && env.version >= 10900 && !env.hc ? '' : ' href="javascript:void(0)"',\r
892                                                 ' tabIndex="-1"',\r
893                                                 ' hidefocus="true"',\r
894                                                 ' role="tab">',\r
895                                                         contents.label,\r
896                                         '</a>'\r
897                                 ].join( '' ) );\r
898 \r
899                         page.setAttribute( 'aria-labelledby', tabId );\r
900 \r
901                         // Take records for the tabs and elements created.\r
902                         this._.tabs[ contents.id ] = [ tab, page ];\r
903                         this._.tabIdList.push( contents.id );\r
904                         !contents.hidden && this._.pageCount++;\r
905                         this._.lastTab = tab;\r
906                         this.updateStyle();\r
907 \r
908                         var contentMap = this._.contents[ contents.id ] = {},\r
909                                 cursor,\r
910                                 children = vbox.getChild();\r
911 \r
912                         while ( ( cursor = children.shift() ) )\r
913                         {\r
914                                 contentMap[ cursor.id ] = cursor;\r
915                                 if ( typeof( cursor.getChild ) == 'function' )\r
916                                         children.push.apply( children, cursor.getChild() );\r
917                         }\r
918 \r
919                         // Attach the DOM nodes.\r
920 \r
921                         page.setAttribute( 'name', contents.id );\r
922                         page.appendTo( this.parts.contents );\r
923 \r
924                         tab.unselectable();\r
925                         this.parts.tabs.append( tab );\r
926 \r
927                         // Add access key handlers if access key is defined.\r
928                         if ( contents.accessKey )\r
929                         {\r
930                                 registerAccessKey( this, this, 'CTRL+' + contents.accessKey,\r
931                                         tabAccessKeyDown, tabAccessKeyUp );\r
932                                 this._.accessKeyMap[ 'CTRL+' + contents.accessKey ] = contents.id;\r
933                         }\r
934                 },\r
935 \r
936                 /**\r
937                  * Activates a tab page in the dialog by its id.\r
938                  * @param {String} id The id of the dialog tab to be activated.\r
939                  * @example\r
940                  * dialogObj.selectPage( 'tab_1' );\r
941                  */\r
942                 selectPage : function( id )\r
943                 {\r
944                         if ( this._.currentTabId == id )\r
945                                 return;\r
946 \r
947                         // Returning true means that the event has been canceled\r
948                         if ( this.fire( 'selectPage', { page : id, currentPage : this._.currentTabId } ) === true )\r
949                                 return;\r
950 \r
951                         // Hide the non-selected tabs and pages.\r
952                         for ( var i in this._.tabs )\r
953                         {\r
954                                 var tab = this._.tabs[i][0],\r
955                                         page = this._.tabs[i][1];\r
956                                 if ( i != id )\r
957                                 {\r
958                                         tab.removeClass( 'cke_dialog_tab_selected' );\r
959                                         page.hide();\r
960                                 }\r
961                                 page.setAttribute( 'aria-hidden', i != id );\r
962                         }\r
963 \r
964                         var selected = this._.tabs[id];\r
965                         selected[0].addClass( 'cke_dialog_tab_selected' );\r
966                         selected[1].show();\r
967                         this._.currentTabId = id;\r
968                         this._.currentTabIndex = CKEDITOR.tools.indexOf( this._.tabIdList, id );\r
969                 },\r
970 \r
971                 // Dialog state-specific style updates.\r
972                 updateStyle : function()\r
973                 {\r
974                         // If only a single page shown, a different style is used in the central pane.\r
975                         this.parts.dialog[ ( this._.pageCount === 1 ? 'add' : 'remove' ) + 'Class' ]( 'cke_single_page' );\r
976                 },\r
977 \r
978                 /**\r
979                  * Hides a page's tab away from the dialog.\r
980                  * @param {String} id The page's Id.\r
981                  * @example\r
982                  * dialog.hidePage( 'tab_3' );\r
983                  */\r
984                 hidePage : function( id )\r
985                 {\r
986                         var tab = this._.tabs[id] && this._.tabs[id][0];\r
987                         if ( !tab || this._.pageCount == 1 )\r
988                                 return;\r
989                         // Switch to other tab first when we're hiding the active tab.\r
990                         else if ( id == this._.currentTabId )\r
991                                 this.selectPage( getPreviousVisibleTab.call( this ) );\r
992 \r
993                         tab.hide();\r
994                         this._.pageCount--;\r
995                         this.updateStyle();\r
996                 },\r
997 \r
998                 /**\r
999                  * Unhides a page's tab.\r
1000                  * @param {String} id The page's Id.\r
1001                  * @example\r
1002                  * dialog.showPage( 'tab_2' );\r
1003                  */\r
1004                 showPage : function( id )\r
1005                 {\r
1006                         var tab = this._.tabs[id] && this._.tabs[id][0];\r
1007                         if ( !tab )\r
1008                                 return;\r
1009                         tab.show();\r
1010                         this._.pageCount++;\r
1011                         this.updateStyle();\r
1012                 },\r
1013 \r
1014                 /**\r
1015                  * Gets the root DOM element of the dialog.\r
1016                  * @returns {CKEDITOR.dom.element} The &lt;span&gt; element containing this dialog.\r
1017                  * @example\r
1018                  * var dialogElement = dialogObj.getElement().getFirst();\r
1019                  * dialogElement.setStyle( 'padding', '5px' );\r
1020                  */\r
1021                 getElement : function()\r
1022                 {\r
1023                         return this._.element;\r
1024                 },\r
1025 \r
1026                 /**\r
1027                  * Gets the name of the dialog.\r
1028                  * @returns {String} The name of this dialog.\r
1029                  * @example\r
1030                  * var dialogName = dialogObj.getName();\r
1031                  */\r
1032                 getName : function()\r
1033                 {\r
1034                         return this._.name;\r
1035                 },\r
1036 \r
1037                 /**\r
1038                  * Gets a dialog UI element object from a dialog page.\r
1039                  * @param {String} pageId id of dialog page.\r
1040                  * @param {String} elementId id of UI element.\r
1041                  * @example\r
1042                  * @returns {CKEDITOR.ui.dialog.uiElement} The dialog UI element.\r
1043                  */\r
1044                 getContentElement : function( pageId, elementId )\r
1045                 {\r
1046                         var page = this._.contents[ pageId ];\r
1047                         return page && page[ elementId ];\r
1048                 },\r
1049 \r
1050                 /**\r
1051                  * Gets the value of a dialog UI element.\r
1052                  * @param {String} pageId id of dialog page.\r
1053                  * @param {String} elementId id of UI element.\r
1054                  * @example\r
1055                  * @returns {Object} The value of the UI element.\r
1056                  */\r
1057                 getValueOf : function( pageId, elementId )\r
1058                 {\r
1059                         return this.getContentElement( pageId, elementId ).getValue();\r
1060                 },\r
1061 \r
1062                 /**\r
1063                  * Sets the value of a dialog UI element.\r
1064                  * @param {String} pageId id of the dialog page.\r
1065                  * @param {String} elementId id of the UI element.\r
1066                  * @param {Object} value The new value of the UI element.\r
1067                  * @example\r
1068                  */\r
1069                 setValueOf : function( pageId, elementId, value )\r
1070                 {\r
1071                         return this.getContentElement( pageId, elementId ).setValue( value );\r
1072                 },\r
1073 \r
1074                 /**\r
1075                  * Gets the UI element of a button in the dialog's button row.\r
1076                  * @param {String} id The id of the button.\r
1077                  * @example\r
1078                  * @returns {CKEDITOR.ui.dialog.button} The button object.\r
1079                  */\r
1080                 getButton : function( id )\r
1081                 {\r
1082                         return this._.buttons[ id ];\r
1083                 },\r
1084 \r
1085                 /**\r
1086                  * Simulates a click to a dialog button in the dialog's button row.\r
1087                  * @param {String} id The id of the button.\r
1088                  * @example\r
1089                  * @returns The return value of the dialog's "click" event.\r
1090                  */\r
1091                 click : function( id )\r
1092                 {\r
1093                         return this._.buttons[ id ].click();\r
1094                 },\r
1095 \r
1096                 /**\r
1097                  * Disables a dialog button.\r
1098                  * @param {String} id The id of the button.\r
1099                  * @example\r
1100                  */\r
1101                 disableButton : function( id )\r
1102                 {\r
1103                         return this._.buttons[ id ].disable();\r
1104                 },\r
1105 \r
1106                 /**\r
1107                  * Enables a dialog button.\r
1108                  * @param {String} id The id of the button.\r
1109                  * @example\r
1110                  */\r
1111                 enableButton : function( id )\r
1112                 {\r
1113                         return this._.buttons[ id ].enable();\r
1114                 },\r
1115 \r
1116                 /**\r
1117                  * Gets the number of pages in the dialog.\r
1118                  * @returns {Number} Page count.\r
1119                  */\r
1120                 getPageCount : function()\r
1121                 {\r
1122                         return this._.pageCount;\r
1123                 },\r
1124 \r
1125                 /**\r
1126                  * Gets the editor instance which opened this dialog.\r
1127                  * @returns {CKEDITOR.editor} Parent editor instances.\r
1128                  */\r
1129                 getParentEditor : function()\r
1130                 {\r
1131                         return this._.editor;\r
1132                 },\r
1133 \r
1134                 /**\r
1135                  * Gets the element that was selected when opening the dialog, if any.\r
1136                  * @returns {CKEDITOR.dom.element} The element that was selected, or null.\r
1137                  */\r
1138                 getSelectedElement : function()\r
1139                 {\r
1140                         return this.getParentEditor().getSelection().getSelectedElement();\r
1141                 },\r
1142 \r
1143                 /**\r
1144                  * Adds element to dialog's focusable list.\r
1145                  *\r
1146                  * @param {CKEDITOR.dom.element} element\r
1147                  * @param {Number} [index]\r
1148                  */\r
1149                 addFocusable: function( element, index ) {\r
1150                         if ( typeof index == 'undefined' )\r
1151                         {\r
1152                                 index = this._.focusList.length;\r
1153                                 this._.focusList.push( new Focusable( this, element, index ) );\r
1154                         }\r
1155                         else\r
1156                         {\r
1157                                 this._.focusList.splice( index, 0, new Focusable( this, element, index ) );\r
1158                                 for ( var i = index + 1 ; i < this._.focusList.length ; i++ )\r
1159                                         this._.focusList[ i ].focusIndex++;\r
1160                         }\r
1161                 }\r
1162         };\r
1163 \r
1164         CKEDITOR.tools.extend( CKEDITOR.dialog,\r
1165                 /**\r
1166                  * @lends CKEDITOR.dialog\r
1167                  */\r
1168                 {\r
1169                         /**\r
1170                          * Registers a dialog.\r
1171                          * @param {String} name The dialog's name.\r
1172                          * @param {Function|String} dialogDefinition\r
1173                          * A function returning the dialog's definition, or the URL to the .js file holding the function.\r
1174                          * The function should accept an argument "editor" which is the current editor instance, and\r
1175                          * return an object conforming to {@link CKEDITOR.dialog.dialogDefinition}.\r
1176                          * @example\r
1177                          * @see CKEDITOR.dialog.dialogDefinition\r
1178                          */\r
1179                         add : function( name, dialogDefinition )\r
1180                         {\r
1181                                 // Avoid path registration from multiple instances override definition.\r
1182                                 if ( !this._.dialogDefinitions[name]\r
1183                                         || typeof  dialogDefinition == 'function' )\r
1184                                         this._.dialogDefinitions[name] = dialogDefinition;\r
1185                         },\r
1186 \r
1187                         exists : function( name )\r
1188                         {\r
1189                                 return !!this._.dialogDefinitions[ name ];\r
1190                         },\r
1191 \r
1192                         getCurrent : function()\r
1193                         {\r
1194                                 return CKEDITOR.dialog._.currentTop;\r
1195                         },\r
1196 \r
1197                         /**\r
1198                          * The default OK button for dialogs. Fires the "ok" event and closes the dialog if the event succeeds.\r
1199                          * @static\r
1200                          * @field\r
1201                          * @example\r
1202                          * @type Function\r
1203                          */\r
1204                         okButton : (function()\r
1205                         {\r
1206                                 var retval = function( editor, override )\r
1207                                 {\r
1208                                         override = override || {};\r
1209                                         return CKEDITOR.tools.extend( {\r
1210                                                 id : 'ok',\r
1211                                                 type : 'button',\r
1212                                                 label : editor.lang.common.ok,\r
1213                                                 'class' : 'cke_dialog_ui_button_ok',\r
1214                                                 onClick : function( evt )\r
1215                                                 {\r
1216                                                         var dialog = evt.data.dialog;\r
1217                                                         if ( dialog.fire( 'ok', { hide : true } ).hide !== false )\r
1218                                                                 dialog.hide();\r
1219                                                 }\r
1220                                         }, override, true );\r
1221                                 };\r
1222                                 retval.type = 'button';\r
1223                                 retval.override = function( override )\r
1224                                 {\r
1225                                         return CKEDITOR.tools.extend( function( editor ){ return retval( editor, override ); },\r
1226                                                         { type : 'button' }, true );\r
1227                                 };\r
1228                                 return retval;\r
1229                         })(),\r
1230 \r
1231                         /**\r
1232                          * The default cancel button for dialogs. Fires the "cancel" event and closes the dialog if no UI element value changed.\r
1233                          * @static\r
1234                          * @field\r
1235                          * @example\r
1236                          * @type Function\r
1237                          */\r
1238                         cancelButton : (function()\r
1239                         {\r
1240                                 var retval = function( editor, override )\r
1241                                 {\r
1242                                         override = override || {};\r
1243                                         return CKEDITOR.tools.extend( {\r
1244                                                 id : 'cancel',\r
1245                                                 type : 'button',\r
1246                                                 label : editor.lang.common.cancel,\r
1247                                                 'class' : 'cke_dialog_ui_button_cancel',\r
1248                                                 onClick : function( evt )\r
1249                                                 {\r
1250                                                         var dialog = evt.data.dialog;\r
1251                                                         if ( dialog.fire( 'cancel', { hide : true } ).hide !== false )\r
1252                                                                 dialog.hide();\r
1253                                                 }\r
1254                                         }, override, true );\r
1255                                 };\r
1256                                 retval.type = 'button';\r
1257                                 retval.override = function( override )\r
1258                                 {\r
1259                                         return CKEDITOR.tools.extend( function( editor ){ return retval( editor, override ); },\r
1260                                                         { type : 'button' }, true );\r
1261                                 };\r
1262                                 return retval;\r
1263                         })(),\r
1264 \r
1265                         /**\r
1266                          * Registers a dialog UI element.\r
1267                          * @param {String} typeName The name of the UI element.\r
1268                          * @param {Function} builder The function to build the UI element.\r
1269                          * @example\r
1270                          */\r
1271                         addUIElement : function( typeName, builder )\r
1272                         {\r
1273                                 this._.uiElementBuilders[ typeName ] = builder;\r
1274                         }\r
1275                 });\r
1276 \r
1277         CKEDITOR.dialog._ =\r
1278         {\r
1279                 uiElementBuilders : {},\r
1280 \r
1281                 dialogDefinitions : {},\r
1282 \r
1283                 currentTop : null,\r
1284 \r
1285                 currentZIndex : null\r
1286         };\r
1287 \r
1288         // "Inherit" (copy actually) from CKEDITOR.event.\r
1289         CKEDITOR.event.implementOn( CKEDITOR.dialog );\r
1290         CKEDITOR.event.implementOn( CKEDITOR.dialog.prototype, true );\r
1291 \r
1292         var defaultDialogDefinition =\r
1293         {\r
1294                 resizable : CKEDITOR.DIALOG_RESIZE_BOTH,\r
1295                 minWidth : 600,\r
1296                 minHeight : 400,\r
1297                 buttons : [ CKEDITOR.dialog.okButton, CKEDITOR.dialog.cancelButton ]\r
1298         };\r
1299 \r
1300         // The buttons in MacOS Apps are in reverse order #4750\r
1301         CKEDITOR.env.mac && defaultDialogDefinition.buttons.reverse();\r
1302 \r
1303         // Tool function used to return an item from an array based on its id\r
1304         // property.\r
1305         var getById = function( array, id, recurse )\r
1306         {\r
1307                 for ( var i = 0, item ; ( item = array[ i ] ) ; i++ )\r
1308                 {\r
1309                         if ( item.id == id )\r
1310                                 return item;\r
1311                         if ( recurse && item[ recurse ] )\r
1312                         {\r
1313                                 var retval = getById( item[ recurse ], id, recurse ) ;\r
1314                                 if ( retval )\r
1315                                         return retval;\r
1316                         }\r
1317                 }\r
1318                 return null;\r
1319         };\r
1320 \r
1321         // Tool function used to add an item into an array.\r
1322         var addById = function( array, newItem, nextSiblingId, recurse, nullIfNotFound )\r
1323         {\r
1324                 if ( nextSiblingId )\r
1325                 {\r
1326                         for ( var i = 0, item ; ( item = array[ i ] ) ; i++ )\r
1327                         {\r
1328                                 if ( item.id == nextSiblingId )\r
1329                                 {\r
1330                                         array.splice( i, 0, newItem );\r
1331                                         return newItem;\r
1332                                 }\r
1333 \r
1334                                 if ( recurse && item[ recurse ] )\r
1335                                 {\r
1336                                         var retval = addById( item[ recurse ], newItem, nextSiblingId, recurse, true );\r
1337                                         if ( retval )\r
1338                                                 return retval;\r
1339                                 }\r
1340                         }\r
1341 \r
1342                         if ( nullIfNotFound )\r
1343                                 return null;\r
1344                 }\r
1345 \r
1346                 array.push( newItem );\r
1347                 return newItem;\r
1348         };\r
1349 \r
1350         // Tool function used to remove an item from an array based on its id.\r
1351         var removeById = function( array, id, recurse )\r
1352         {\r
1353                 for ( var i = 0, item ; ( item = array[ i ] ) ; i++ )\r
1354                 {\r
1355                         if ( item.id == id )\r
1356                                 return array.splice( i, 1 );\r
1357                         if ( recurse && item[ recurse ] )\r
1358                         {\r
1359                                 var retval = removeById( item[ recurse ], id, recurse );\r
1360                                 if ( retval )\r
1361                                         return retval;\r
1362                         }\r
1363                 }\r
1364                 return null;\r
1365         };\r
1366 \r
1367         /**\r
1368          * This class is not really part of the API. It is the "definition" property value\r
1369          * passed to "dialogDefinition" event handlers.\r
1370          * @constructor\r
1371          * @name CKEDITOR.dialog.dialogDefinitionObject\r
1372          * @extends CKEDITOR.dialog.dialogDefinition\r
1373          * @example\r
1374          * CKEDITOR.on( 'dialogDefinition', function( evt )\r
1375          *      {\r
1376          *              var definition = evt.data.definition;\r
1377          *              var content = definition.getContents( 'page1' );\r
1378          *              ...\r
1379          *      } );\r
1380          */\r
1381         var definitionObject = function( dialog, dialogDefinition )\r
1382         {\r
1383                 // TODO : Check if needed.\r
1384                 this.dialog = dialog;\r
1385 \r
1386                 // Transform the contents entries in contentObjects.\r
1387                 var contents = dialogDefinition.contents;\r
1388                 for ( var i = 0, content ; ( content = contents[i] ) ; i++ )\r
1389                         contents[ i ] = new contentObject( dialog, content );\r
1390 \r
1391                 CKEDITOR.tools.extend( this, dialogDefinition );\r
1392         };\r
1393 \r
1394         definitionObject.prototype =\r
1395         /** @lends CKEDITOR.dialog.dialogDefinitionObject.prototype */\r
1396         {\r
1397                 /**\r
1398                  * Gets a content definition.\r
1399                  * @param {String} id The id of the content definition.\r
1400                  * @returns {CKEDITOR.dialog.contentDefinition} The content definition\r
1401                  *              matching id.\r
1402                  */\r
1403                 getContents : function( id )\r
1404                 {\r
1405                         return getById( this.contents, id );\r
1406                 },\r
1407 \r
1408                 /**\r
1409                  * Gets a button definition.\r
1410                  * @param {String} id The id of the button definition.\r
1411                  * @returns {CKEDITOR.dialog.buttonDefinition} The button definition\r
1412                  *              matching id.\r
1413                  */\r
1414                 getButton : function( id )\r
1415                 {\r
1416                         return getById( this.buttons, id );\r
1417                 },\r
1418 \r
1419                 /**\r
1420                  * Adds a content definition object under this dialog definition.\r
1421                  * @param {CKEDITOR.dialog.contentDefinition} contentDefinition The\r
1422                  *              content definition.\r
1423                  * @param {String} [nextSiblingId] The id of an existing content\r
1424                  *              definition which the new content definition will be inserted\r
1425                  *              before. Omit if the new content definition is to be inserted as\r
1426                  *              the last item.\r
1427                  * @returns {CKEDITOR.dialog.contentDefinition} The inserted content\r
1428                  *              definition.\r
1429                  */\r
1430                 addContents : function( contentDefinition, nextSiblingId )\r
1431                 {\r
1432                         return addById( this.contents, contentDefinition, nextSiblingId );\r
1433                 },\r
1434 \r
1435                 /**\r
1436                  * Adds a button definition object under this dialog definition.\r
1437                  * @param {CKEDITOR.dialog.buttonDefinition} buttonDefinition The\r
1438                  *              button definition.\r
1439                  * @param {String} [nextSiblingId] The id of an existing button\r
1440                  *              definition which the new button definition will be inserted\r
1441                  *              before. Omit if the new button definition is to be inserted as\r
1442                  *              the last item.\r
1443                  * @returns {CKEDITOR.dialog.buttonDefinition} The inserted button\r
1444                  *              definition.\r
1445                  */\r
1446                 addButton : function( buttonDefinition, nextSiblingId )\r
1447                 {\r
1448                         return addById( this.buttons, buttonDefinition, nextSiblingId );\r
1449                 },\r
1450 \r
1451                 /**\r
1452                  * Removes a content definition from this dialog definition.\r
1453                  * @param {String} id The id of the content definition to be removed.\r
1454                  * @returns {CKEDITOR.dialog.contentDefinition} The removed content\r
1455                  *              definition.\r
1456                  */\r
1457                 removeContents : function( id )\r
1458                 {\r
1459                         removeById( this.contents, id );\r
1460                 },\r
1461 \r
1462                 /**\r
1463                  * Removes a button definition from the dialog definition.\r
1464                  * @param {String} id The id of the button definition to be removed.\r
1465                  * @returns {CKEDITOR.dialog.buttonDefinition} The removed button\r
1466                  *              definition.\r
1467                  */\r
1468                 removeButton : function( id )\r
1469                 {\r
1470                         removeById( this.buttons, id );\r
1471                 }\r
1472         };\r
1473 \r
1474         /**\r
1475          * This class is not really part of the API. It is the template of the\r
1476          * objects representing content pages inside the\r
1477          * CKEDITOR.dialog.dialogDefinitionObject.\r
1478          * @constructor\r
1479          * @name CKEDITOR.dialog.contentDefinitionObject\r
1480          * @example\r
1481          * CKEDITOR.on( 'dialogDefinition', function( evt )\r
1482          *      {\r
1483          *              var definition = evt.data.definition;\r
1484          *              var content = definition.getContents( 'page1' );\r
1485          *              content.remove( 'textInput1' );\r
1486          *              ...\r
1487          *      } );\r
1488          */\r
1489         function contentObject( dialog, contentDefinition )\r
1490         {\r
1491                 this._ =\r
1492                 {\r
1493                         dialog : dialog\r
1494                 };\r
1495 \r
1496                 CKEDITOR.tools.extend( this, contentDefinition );\r
1497         }\r
1498 \r
1499         contentObject.prototype =\r
1500         /** @lends CKEDITOR.dialog.contentDefinitionObject.prototype */\r
1501         {\r
1502                 /**\r
1503                  * Gets a UI element definition under the content definition.\r
1504                  * @param {String} id The id of the UI element definition.\r
1505                  * @returns {CKEDITOR.dialog.uiElementDefinition}\r
1506                  */\r
1507                 get : function( id )\r
1508                 {\r
1509                         return getById( this.elements, id, 'children' );\r
1510                 },\r
1511 \r
1512                 /**\r
1513                  * Adds a UI element definition to the content definition.\r
1514                  * @param {CKEDITOR.dialog.uiElementDefinition} elementDefinition The\r
1515                  *              UI elemnet definition to be added.\r
1516                  * @param {String} nextSiblingId The id of an existing UI element\r
1517                  *              definition which the new UI element definition will be inserted\r
1518                  *              before. Omit if the new button definition is to be inserted as\r
1519                  *              the last item.\r
1520                  * @returns {CKEDITOR.dialog.uiElementDefinition} The element\r
1521                  *              definition inserted.\r
1522                  */\r
1523                 add : function( elementDefinition, nextSiblingId )\r
1524                 {\r
1525                         return addById( this.elements, elementDefinition, nextSiblingId, 'children' );\r
1526                 },\r
1527 \r
1528                 /**\r
1529                  * Removes a UI element definition from the content definition.\r
1530                  * @param {String} id The id of the UI element definition to be\r
1531                  *              removed.\r
1532                  * @returns {CKEDITOR.dialog.uiElementDefinition} The element\r
1533                  *              definition removed.\r
1534                  * @example\r
1535                  */\r
1536                 remove : function( id )\r
1537                 {\r
1538                         removeById( this.elements, id, 'children' );\r
1539                 }\r
1540         };\r
1541 \r
1542         function initDragAndDrop( dialog )\r
1543         {\r
1544                 var lastCoords = null,\r
1545                         abstractDialogCoords = null,\r
1546                         element = dialog.getElement().getFirst(),\r
1547                         editor = dialog.getParentEditor(),\r
1548                         magnetDistance = editor.config.dialog_magnetDistance,\r
1549                         margins = editor.skin.margins || [ 0, 0, 0, 0 ];\r
1550 \r
1551                 if ( typeof magnetDistance == 'undefined' )\r
1552                         magnetDistance = 20;\r
1553 \r
1554                 function mouseMoveHandler( evt )\r
1555                 {\r
1556                         var dialogSize = dialog.getSize(),\r
1557                                 viewPaneSize = CKEDITOR.document.getWindow().getViewPaneSize(),\r
1558                                 x = evt.data.$.screenX,\r
1559                                 y = evt.data.$.screenY,\r
1560                                 dx = x - lastCoords.x,\r
1561                                 dy = y - lastCoords.y,\r
1562                                 realX, realY;\r
1563 \r
1564                         lastCoords = { x : x, y : y };\r
1565                         abstractDialogCoords.x += dx;\r
1566                         abstractDialogCoords.y += dy;\r
1567 \r
1568                         if ( abstractDialogCoords.x + margins[3] < magnetDistance )\r
1569                                 realX = - margins[3];\r
1570                         else if ( abstractDialogCoords.x - margins[1] > viewPaneSize.width - dialogSize.width - magnetDistance )\r
1571                                 realX = viewPaneSize.width - dialogSize.width + margins[1];\r
1572                         else\r
1573                                 realX = abstractDialogCoords.x;\r
1574 \r
1575                         if ( abstractDialogCoords.y + margins[0] < magnetDistance )\r
1576                                 realY = - margins[0];\r
1577                         else if ( abstractDialogCoords.y - margins[2] > viewPaneSize.height - dialogSize.height - magnetDistance )\r
1578                                 realY = viewPaneSize.height - dialogSize.height + margins[2];\r
1579                         else\r
1580                                 realY = abstractDialogCoords.y;\r
1581 \r
1582                         dialog.move( realX, realY );\r
1583 \r
1584                         evt.data.preventDefault();\r
1585                 }\r
1586 \r
1587                 function mouseUpHandler( evt )\r
1588                 {\r
1589                         CKEDITOR.document.removeListener( 'mousemove', mouseMoveHandler );\r
1590                         CKEDITOR.document.removeListener( 'mouseup', mouseUpHandler );\r
1591 \r
1592                         if ( CKEDITOR.env.ie6Compat )\r
1593                         {\r
1594                                 var coverDoc = currentCover.getChild( 0 ).getFrameDocument();\r
1595                                 coverDoc.removeListener( 'mousemove', mouseMoveHandler );\r
1596                                 coverDoc.removeListener( 'mouseup', mouseUpHandler );\r
1597                         }\r
1598                 }\r
1599 \r
1600                 dialog.parts.title.on( 'mousedown', function( evt )\r
1601                         {\r
1602                                 dialog._.updateSize = true;\r
1603 \r
1604                                 lastCoords = { x : evt.data.$.screenX, y : evt.data.$.screenY };\r
1605 \r
1606                                 CKEDITOR.document.on( 'mousemove', mouseMoveHandler );\r
1607                                 CKEDITOR.document.on( 'mouseup', mouseUpHandler );\r
1608                                 abstractDialogCoords = dialog.getPosition();\r
1609 \r
1610                                 if ( CKEDITOR.env.ie6Compat )\r
1611                                 {\r
1612                                         var coverDoc = currentCover.getChild( 0 ).getFrameDocument();\r
1613                                         coverDoc.on( 'mousemove', mouseMoveHandler );\r
1614                                         coverDoc.on( 'mouseup', mouseUpHandler );\r
1615                                 }\r
1616 \r
1617                                 evt.data.preventDefault();\r
1618                         }, dialog );\r
1619         }\r
1620 \r
1621         function initResizeHandles( dialog )\r
1622         {\r
1623                 var definition = dialog.definition,\r
1624                         minWidth = definition.minWidth || 0,\r
1625                         minHeight = definition.minHeight || 0,\r
1626                         resizable = definition.resizable,\r
1627                         margins = dialog.getParentEditor().skin.margins || [ 0, 0, 0, 0 ];\r
1628 \r
1629                 function topSizer( coords, dy )\r
1630                 {\r
1631                         coords.y += dy;\r
1632                 }\r
1633 \r
1634                 function rightSizer( coords, dx )\r
1635                 {\r
1636                         coords.x2 += dx;\r
1637                 }\r
1638 \r
1639                 function bottomSizer( coords, dy )\r
1640                 {\r
1641                         coords.y2 += dy;\r
1642                 }\r
1643 \r
1644                 function leftSizer( coords, dx )\r
1645                 {\r
1646                         coords.x += dx;\r
1647                 }\r
1648 \r
1649                 var lastCoords = null,\r
1650                         abstractDialogCoords = null,\r
1651                         magnetDistance = dialog._.editor.config.magnetDistance,\r
1652                         parts = [ 'tl', 't', 'tr', 'l', 'r', 'bl', 'b', 'br' ];\r
1653 \r
1654                 function mouseDownHandler( evt )\r
1655                 {\r
1656                         var partName = evt.listenerData.part, size = dialog.getSize();\r
1657                         abstractDialogCoords = dialog.getPosition();\r
1658                         CKEDITOR.tools.extend( abstractDialogCoords,\r
1659                                 {\r
1660                                         x2 : abstractDialogCoords.x + size.width,\r
1661                                         y2 : abstractDialogCoords.y + size.height\r
1662                                 } );\r
1663                         lastCoords = { x : evt.data.$.screenX, y : evt.data.$.screenY };\r
1664 \r
1665                         CKEDITOR.document.on( 'mousemove', mouseMoveHandler, dialog, { part : partName } );\r
1666                         CKEDITOR.document.on( 'mouseup', mouseUpHandler, dialog, { part : partName } );\r
1667 \r
1668                         if ( CKEDITOR.env.ie6Compat )\r
1669                         {\r
1670                                 var coverDoc = currentCover.getChild( 0 ).getFrameDocument();\r
1671                                 coverDoc.on( 'mousemove', mouseMoveHandler, dialog, { part : partName } );\r
1672                                 coverDoc.on( 'mouseup', mouseUpHandler, dialog, { part : partName } );\r
1673                         }\r
1674 \r
1675                         evt.data.preventDefault();\r
1676                 }\r
1677 \r
1678                 function mouseMoveHandler( evt )\r
1679                 {\r
1680                         var x = evt.data.$.screenX,\r
1681                                 y = evt.data.$.screenY,\r
1682                                 dx = x - lastCoords.x,\r
1683                                 dy = y - lastCoords.y,\r
1684                                 viewPaneSize = CKEDITOR.document.getWindow().getViewPaneSize(),\r
1685                                 partName = evt.listenerData.part;\r
1686 \r
1687                         if ( partName.search( 't' ) != -1 )\r
1688                                 topSizer( abstractDialogCoords, dy );\r
1689                         if ( partName.search( 'l' ) != -1 )\r
1690                                 leftSizer( abstractDialogCoords, dx );\r
1691                         if ( partName.search( 'b' ) != -1 )\r
1692                                 bottomSizer( abstractDialogCoords, dy );\r
1693                         if ( partName.search( 'r' ) != -1 )\r
1694                                 rightSizer( abstractDialogCoords, dx );\r
1695 \r
1696                         lastCoords = { x : x, y : y };\r
1697 \r
1698                         var realX, realY, realX2, realY2;\r
1699 \r
1700                         if ( abstractDialogCoords.x + margins[3] < magnetDistance )\r
1701                                 realX = - margins[3];\r
1702                         else if ( partName.search( 'l' ) != -1 && abstractDialogCoords.x2 - abstractDialogCoords.x < minWidth + magnetDistance )\r
1703                                 realX = abstractDialogCoords.x2 - minWidth;\r
1704                         else\r
1705                                 realX = abstractDialogCoords.x;\r
1706 \r
1707                         if ( abstractDialogCoords.y + margins[0] < magnetDistance )\r
1708                                 realY = - margins[0];\r
1709                         else if ( partName.search( 't' ) != -1 && abstractDialogCoords.y2 - abstractDialogCoords.y < minHeight + magnetDistance )\r
1710                                 realY = abstractDialogCoords.y2 - minHeight;\r
1711                         else\r
1712                                 realY = abstractDialogCoords.y;\r
1713 \r
1714                         if ( abstractDialogCoords.x2 - margins[1] > viewPaneSize.width - magnetDistance )\r
1715                                 realX2 = viewPaneSize.width + margins[1] ;\r
1716                         else if ( partName.search( 'r' ) != -1 && abstractDialogCoords.x2 - abstractDialogCoords.x < minWidth + magnetDistance )\r
1717                                 realX2 = abstractDialogCoords.x + minWidth;\r
1718                         else\r
1719                                 realX2 = abstractDialogCoords.x2;\r
1720 \r
1721                         if ( abstractDialogCoords.y2 - margins[2] > viewPaneSize.height - magnetDistance )\r
1722                                 realY2= viewPaneSize.height + margins[2] ;\r
1723                         else if ( partName.search( 'b' ) != -1 && abstractDialogCoords.y2 - abstractDialogCoords.y < minHeight + magnetDistance )\r
1724                                 realY2 = abstractDialogCoords.y + minHeight;\r
1725                         else\r
1726                                 realY2 = abstractDialogCoords.y2 ;\r
1727 \r
1728                         dialog.move( realX, realY );\r
1729                         dialog.resize( realX2 - realX, realY2 - realY );\r
1730 \r
1731                         evt.data.preventDefault();\r
1732                 }\r
1733 \r
1734                 function mouseUpHandler( evt )\r
1735                 {\r
1736                         CKEDITOR.document.removeListener( 'mouseup', mouseUpHandler );\r
1737                         CKEDITOR.document.removeListener( 'mousemove', mouseMoveHandler );\r
1738 \r
1739                         if ( CKEDITOR.env.ie6Compat )\r
1740                         {\r
1741                                 var coverDoc = currentCover.getChild( 0 ).getFrameDocument();\r
1742                                 coverDoc.removeListener( 'mouseup', mouseUpHandler );\r
1743                                 coverDoc.removeListener( 'mousemove', mouseMoveHandler );\r
1744                         }\r
1745                 }\r
1746 \r
1747 // TODO : Simplify the resize logic, having just a single resize grip <div>.\r
1748 //              var widthTest = /[lr]/,\r
1749 //                      heightTest = /[tb]/;\r
1750 //              for ( var i = 0 ; i < parts.length ; i++ )\r
1751 //              {\r
1752 //                      var element = dialog.parts[ parts[i] + '_resize' ];\r
1753 //                      if ( resizable == CKEDITOR.DIALOG_RESIZE_NONE ||\r
1754 //                                      resizable == CKEDITOR.DIALOG_RESIZE_HEIGHT && widthTest.test( parts[i] ) ||\r
1755 //                                      resizable == CKEDITOR.DIALOG_RESIZE_WIDTH && heightTest.test( parts[i] ) )\r
1756 //                      {\r
1757 //                              element.hide();\r
1758 //                              continue;\r
1759 //                      }\r
1760 //                      element.on( 'mousedown', mouseDownHandler, dialog, { part : parts[i] } );\r
1761 //              }\r
1762         }\r
1763 \r
1764         var resizeCover;\r
1765         // Caching resuable covers and allowing only one cover\r
1766         // on screen.\r
1767         var covers = {},\r
1768                 currentCover;\r
1769 \r
1770         function showCover( editor )\r
1771         {\r
1772                 var win = CKEDITOR.document.getWindow();\r
1773                 var backgroundColorStyle = editor.config.dialog_backgroundCoverColor || 'white',\r
1774                         backgroundCoverOpacity = editor.config.dialog_backgroundCoverOpacity,\r
1775                         baseFloatZIndex = editor.config.baseFloatZIndex,\r
1776                         coverKey = CKEDITOR.tools.genKey(\r
1777                                         backgroundColorStyle,\r
1778                                         backgroundCoverOpacity,\r
1779                                         baseFloatZIndex ),\r
1780                         coverElement = covers[ coverKey ];\r
1781 \r
1782                 if ( !coverElement )\r
1783                 {\r
1784                         var html = [\r
1785                                         '<div style="position: ', ( CKEDITOR.env.ie6Compat ? 'absolute' : 'fixed' ),\r
1786                                         '; z-index: ', baseFloatZIndex,\r
1787                                         '; top: 0px; left: 0px; ',\r
1788                                         ( !CKEDITOR.env.ie6Compat ? 'background-color: ' + backgroundColorStyle : '' ),\r
1789                                         '" class="cke_dialog_background_cover">'\r
1790                                 ];\r
1791 \r
1792                         if ( CKEDITOR.env.ie6Compat )\r
1793                         {\r
1794                                 // Support for custom document.domain in IE.\r
1795                                 var isCustomDomain = CKEDITOR.env.isCustomDomain(),\r
1796                                         iframeHtml = '<html><body style=\\\'background-color:' + backgroundColorStyle + ';\\\'></body></html>';\r
1797 \r
1798                                 html.push(\r
1799                                         '<iframe' +\r
1800                                                 ' hidefocus="true"' +\r
1801                                                 ' frameborder="0"' +\r
1802                                                 ' id="cke_dialog_background_iframe"' +\r
1803                                                 ' src="javascript:' );\r
1804 \r
1805                                 html.push( 'void((function(){' +\r
1806                                                                 'document.open();' +\r
1807                                                                 ( isCustomDomain ? 'document.domain=\'' + document.domain + '\';' : '' ) +\r
1808                                                                 'document.write( \'' + iframeHtml + '\' );' +\r
1809                                                                 'document.close();' +\r
1810                                                         '})())' );\r
1811 \r
1812                                 html.push(\r
1813                                                 '"' +\r
1814                                                 ' style="' +\r
1815                                                         'position:absolute;' +\r
1816                                                         'left:0;' +\r
1817                                                         'top:0;' +\r
1818                                                         'width:100%;' +\r
1819                                                         'height: 100%;' +\r
1820                                                         'progid:DXImageTransform.Microsoft.Alpha(opacity=0)">' +\r
1821                                         '</iframe>' );\r
1822                         }\r
1823 \r
1824                         html.push( '</div>' );\r
1825 \r
1826                         coverElement = CKEDITOR.dom.element.createFromHtml( html.join( '' ) );\r
1827                         coverElement.setOpacity( backgroundCoverOpacity != undefined ? backgroundCoverOpacity : 0.5 );\r
1828 \r
1829                         coverElement.appendTo( CKEDITOR.document.getBody() );\r
1830                         covers[ coverKey ] = coverElement;\r
1831                 }\r
1832                 else\r
1833                         coverElement.   show();\r
1834 \r
1835                 currentCover = coverElement;\r
1836                 var resizeFunc = function()\r
1837                 {\r
1838                         var size = win.getViewPaneSize();\r
1839                         coverElement.setStyles(\r
1840                                 {\r
1841                                         width : size.width + 'px',\r
1842                                         height : size.height + 'px'\r
1843                                 } );\r
1844                 };\r
1845 \r
1846                 var scrollFunc = function()\r
1847                 {\r
1848                         var pos = win.getScrollPosition(),\r
1849                                 cursor = CKEDITOR.dialog._.currentTop;\r
1850                         coverElement.setStyles(\r
1851                                         {\r
1852                                                 left : pos.x + 'px',\r
1853                                                 top : pos.y + 'px'\r
1854                                         });\r
1855 \r
1856                         do\r
1857                         {\r
1858                                 var dialogPos = cursor.getPosition();\r
1859                                 cursor.move( dialogPos.x, dialogPos.y );\r
1860                         } while ( ( cursor = cursor._.parentDialog ) );\r
1861                 };\r
1862 \r
1863                 resizeCover = resizeFunc;\r
1864                 win.on( 'resize', resizeFunc );\r
1865                 resizeFunc();\r
1866                 if ( CKEDITOR.env.ie6Compat )\r
1867                 {\r
1868                         // IE BUG: win.$.onscroll assignment doesn't work.. it must be window.onscroll.\r
1869                         // So we need to invent a really funny way to make it work.\r
1870                         var myScrollHandler = function()\r
1871                                 {\r
1872                                         scrollFunc();\r
1873                                         arguments.callee.prevScrollHandler.apply( this, arguments );\r
1874                                 };\r
1875                         win.$.setTimeout( function()\r
1876                                 {\r
1877                                         myScrollHandler.prevScrollHandler = window.onscroll || function(){};\r
1878                                         window.onscroll = myScrollHandler;\r
1879                                 }, 0 );\r
1880                         scrollFunc();\r
1881                 }\r
1882         }\r
1883 \r
1884         function hideCover()\r
1885         {\r
1886                 if ( !currentCover )\r
1887                         return;\r
1888 \r
1889                 var win = CKEDITOR.document.getWindow();\r
1890                 currentCover.hide();\r
1891                 win.removeListener( 'resize', resizeCover );\r
1892 \r
1893                 if ( CKEDITOR.env.ie6Compat )\r
1894                 {\r
1895                         win.$.setTimeout( function()\r
1896                                 {\r
1897                                         var prevScrollHandler = window.onscroll && window.onscroll.prevScrollHandler;\r
1898                                         window.onscroll = prevScrollHandler || null;\r
1899                                 }, 0 );\r
1900                 }\r
1901                 resizeCover = null;\r
1902         }\r
1903 \r
1904         function removeCovers()\r
1905         {\r
1906                 for ( var coverId in covers )\r
1907                         covers[ coverId ].remove();\r
1908                 covers = {};\r
1909         }\r
1910 \r
1911         var accessKeyProcessors = {};\r
1912 \r
1913         var accessKeyDownHandler = function( evt )\r
1914         {\r
1915                 var ctrl = evt.data.$.ctrlKey || evt.data.$.metaKey,\r
1916                         alt = evt.data.$.altKey,\r
1917                         shift = evt.data.$.shiftKey,\r
1918                         key = String.fromCharCode( evt.data.$.keyCode ),\r
1919                         keyProcessor = accessKeyProcessors[( ctrl ? 'CTRL+' : '' ) + ( alt ? 'ALT+' : '') + ( shift ? 'SHIFT+' : '' ) + key];\r
1920 \r
1921                 if ( !keyProcessor || !keyProcessor.length )\r
1922                         return;\r
1923 \r
1924                 keyProcessor = keyProcessor[keyProcessor.length - 1];\r
1925                 keyProcessor.keydown && keyProcessor.keydown.call( keyProcessor.uiElement, keyProcessor.dialog, keyProcessor.key );\r
1926                 evt.data.preventDefault();\r
1927         };\r
1928 \r
1929         var accessKeyUpHandler = function( evt )\r
1930         {\r
1931                 var ctrl = evt.data.$.ctrlKey || evt.data.$.metaKey,\r
1932                         alt = evt.data.$.altKey,\r
1933                         shift = evt.data.$.shiftKey,\r
1934                         key = String.fromCharCode( evt.data.$.keyCode ),\r
1935                         keyProcessor = accessKeyProcessors[( ctrl ? 'CTRL+' : '' ) + ( alt ? 'ALT+' : '') + ( shift ? 'SHIFT+' : '' ) + key];\r
1936 \r
1937                 if ( !keyProcessor || !keyProcessor.length )\r
1938                         return;\r
1939 \r
1940                 keyProcessor = keyProcessor[keyProcessor.length - 1];\r
1941                 if ( keyProcessor.keyup )\r
1942                 {\r
1943                         keyProcessor.keyup.call( keyProcessor.uiElement, keyProcessor.dialog, keyProcessor.key );\r
1944                         evt.data.preventDefault();\r
1945                 }\r
1946         };\r
1947 \r
1948         var registerAccessKey = function( uiElement, dialog, key, downFunc, upFunc )\r
1949         {\r
1950                 var procList = accessKeyProcessors[key] || ( accessKeyProcessors[key] = [] );\r
1951                 procList.push( {\r
1952                                 uiElement : uiElement,\r
1953                                 dialog : dialog,\r
1954                                 key : key,\r
1955                                 keyup : upFunc || uiElement.accessKeyUp,\r
1956                                 keydown : downFunc || uiElement.accessKeyDown\r
1957                         } );\r
1958         };\r
1959 \r
1960         var unregisterAccessKey = function( obj )\r
1961         {\r
1962                 for ( var i in accessKeyProcessors )\r
1963                 {\r
1964                         var list = accessKeyProcessors[i];\r
1965                         for ( var j = list.length - 1 ; j >= 0 ; j-- )\r
1966                         {\r
1967                                 if ( list[j].dialog == obj || list[j].uiElement == obj )\r
1968                                         list.splice( j, 1 );\r
1969                         }\r
1970                         if ( list.length === 0 )\r
1971                                 delete accessKeyProcessors[i];\r
1972                 }\r
1973         };\r
1974 \r
1975         var tabAccessKeyUp = function( dialog, key )\r
1976         {\r
1977                 if ( dialog._.accessKeyMap[key] )\r
1978                         dialog.selectPage( dialog._.accessKeyMap[key] );\r
1979         };\r
1980 \r
1981         var tabAccessKeyDown = function( dialog, key )\r
1982         {\r
1983         };\r
1984 \r
1985         // ESC, ENTER\r
1986         var preventKeyBubblingKeys = { 27 :1, 13 :1 };\r
1987         var preventKeyBubbling = function( e )\r
1988         {\r
1989                 if ( e.data.getKeystroke() in preventKeyBubblingKeys )\r
1990                         e.data.stopPropagation();\r
1991         };\r
1992 \r
1993         (function()\r
1994         {\r
1995                 CKEDITOR.ui.dialog =\r
1996                 {\r
1997                         /**\r
1998                          * The base class of all dialog UI elements.\r
1999                          * @constructor\r
2000                          * @param {CKEDITOR.dialog} dialog Parent dialog object.\r
2001                          * @param {CKEDITOR.dialog.uiElementDefinition} elementDefinition Element\r
2002                          * definition. Accepted fields:\r
2003                          * <ul>\r
2004                          *      <li><strong>id</strong> (Required) The id of the UI element. See {@link\r
2005                          *      CKEDITOR.dialog#getContentElement}</li>\r
2006                          *      <li><strong>type</strong> (Required) The type of the UI element. The\r
2007                          *      value to this field specifies which UI element class will be used to\r
2008                          *      generate the final widget.</li>\r
2009                          *      <li><strong>title</strong> (Optional) The popup tooltip for the UI\r
2010                          *      element.</li>\r
2011                          *      <li><strong>hidden</strong> (Optional) A flag that tells if the element\r
2012                          *      should be initially visible.</li>\r
2013                          *      <li><strong>className</strong> (Optional) Additional CSS class names\r
2014                          *      to add to the UI element. Separated by space.</li>\r
2015                          *      <li><strong>style</strong> (Optional) Additional CSS inline styles\r
2016                          *      to add to the UI element. A semicolon (;) is required after the last\r
2017                          *      style declaration.</li>\r
2018                          *      <li><strong>accessKey</strong> (Optional) The alphanumeric access key\r
2019                          *      for this element. Access keys are automatically prefixed by CTRL.</li>\r
2020                          *      <li><strong>on*</strong> (Optional) Any UI element definition field that\r
2021                          *      starts with <em>on</em> followed immediately by a capital letter and\r
2022                          *      probably more letters is an event handler. Event handlers may be further\r
2023                          *      divided into registered event handlers and DOM event handlers. Please\r
2024                          *      refer to {@link CKEDITOR.ui.dialog.uiElement#registerEvents} and\r
2025                          *      {@link CKEDITOR.ui.dialog.uiElement#eventProcessors} for more\r
2026                          *      information.</li>\r
2027                          * </ul>\r
2028                          * @param {Array} htmlList\r
2029                          * List of HTML code to be added to the dialog's content area.\r
2030                          * @param {Function|String} nodeNameArg\r
2031                          * A function returning a string, or a simple string for the node name for\r
2032                          * the root DOM node. Default is 'div'.\r
2033                          * @param {Function|Object} stylesArg\r
2034                          * A function returning an object, or a simple object for CSS styles applied\r
2035                          * to the DOM node. Default is empty object.\r
2036                          * @param {Function|Object} attributesArg\r
2037                          * A fucntion returning an object, or a simple object for attributes applied\r
2038                          * to the DOM node. Default is empty object.\r
2039                          * @param {Function|String} contentsArg\r
2040                          * A function returning a string, or a simple string for the HTML code inside\r
2041                          * the root DOM node. Default is empty string.\r
2042                          * @example\r
2043                          */\r
2044                         uiElement : function( dialog, elementDefinition, htmlList, nodeNameArg, stylesArg, attributesArg, contentsArg )\r
2045                         {\r
2046                                 if ( arguments.length < 4 )\r
2047                                         return;\r
2048 \r
2049                                 var nodeName = ( nodeNameArg.call ? nodeNameArg( elementDefinition ) : nodeNameArg ) || 'div',\r
2050                                         html = [ '<', nodeName, ' ' ],\r
2051                                         styles = ( stylesArg && stylesArg.call ? stylesArg( elementDefinition ) : stylesArg ) || {},\r
2052                                         attributes = ( attributesArg && attributesArg.call ? attributesArg( elementDefinition ) : attributesArg ) || {},\r
2053                                         innerHTML = ( contentsArg && contentsArg.call ? contentsArg.call( this, dialog, elementDefinition ) : contentsArg ) || '',\r
2054                                         domId = this.domId = attributes.id || CKEDITOR.tools.getNextNumber() + '_uiElement',\r
2055                                         id = this.id = elementDefinition.id,\r
2056                                         i;\r
2057 \r
2058                                 // Set the id, a unique id is required for getElement() to work.\r
2059                                 attributes.id = domId;\r
2060 \r
2061                                 // Set the type and definition CSS class names.\r
2062                                 var classes = {};\r
2063                                 if ( elementDefinition.type )\r
2064                                         classes[ 'cke_dialog_ui_' + elementDefinition.type ] = 1;\r
2065                                 if ( elementDefinition.className )\r
2066                                         classes[ elementDefinition.className ] = 1;\r
2067                                 var attributeClasses = ( attributes['class'] && attributes['class'].split ) ? attributes['class'].split( ' ' ) : [];\r
2068                                 for ( i = 0 ; i < attributeClasses.length ; i++ )\r
2069                                 {\r
2070                                         if ( attributeClasses[i] )\r
2071                                                 classes[ attributeClasses[i] ] = 1;\r
2072                                 }\r
2073                                 var finalClasses = [];\r
2074                                 for ( i in classes )\r
2075                                         finalClasses.push( i );\r
2076                                 attributes['class'] = finalClasses.join( ' ' );\r
2077 \r
2078                                 // Set the popup tooltop.\r
2079                                 if ( elementDefinition.title )\r
2080                                         attributes.title = elementDefinition.title;\r
2081 \r
2082                                 // Write the inline CSS styles.\r
2083                                 var styleStr = ( elementDefinition.style || '' ).split( ';' );\r
2084                                 for ( i in styles )\r
2085                                         styleStr.push( i + ':' + styles[i] );\r
2086                                 if ( elementDefinition.hidden )\r
2087                                         styleStr.push( 'display:none' );\r
2088                                 for ( i = styleStr.length - 1 ; i >= 0 ; i-- )\r
2089                                 {\r
2090                                         if ( styleStr[i] === '' )\r
2091                                                 styleStr.splice( i, 1 );\r
2092                                 }\r
2093                                 if ( styleStr.length > 0 )\r
2094                                         attributes.style = ( attributes.style ? ( attributes.style + '; ' ) : '' ) + styleStr.join( '; ' );\r
2095 \r
2096                                 // Write the attributes.\r
2097                                 for ( i in attributes )\r
2098                                         html.push( i + '="' + CKEDITOR.tools.htmlEncode( attributes[i] ) + '" ');\r
2099 \r
2100                                 // Write the content HTML.\r
2101                                 html.push( '>', innerHTML, '</', nodeName, '>' );\r
2102 \r
2103                                 // Add contents to the parent HTML array.\r
2104                                 htmlList.push( html.join( '' ) );\r
2105 \r
2106                                 ( this._ || ( this._ = {} ) ).dialog = dialog;\r
2107 \r
2108                                 // Override isChanged if it is defined in element definition.\r
2109                                 if ( typeof( elementDefinition.isChanged ) == 'boolean' )\r
2110                                         this.isChanged = function(){ return elementDefinition.isChanged; };\r
2111                                 if ( typeof( elementDefinition.isChanged ) == 'function' )\r
2112                                         this.isChanged = elementDefinition.isChanged;\r
2113 \r
2114                                 // Add events.\r
2115                                 CKEDITOR.event.implementOn( this );\r
2116 \r
2117                                 this.registerEvents( elementDefinition );\r
2118                                 if ( this.accessKeyUp && this.accessKeyDown && elementDefinition.accessKey )\r
2119                                         registerAccessKey( this, dialog, 'CTRL+' + elementDefinition.accessKey );\r
2120 \r
2121                                 var me = this;\r
2122                                 dialog.on( 'load', function()\r
2123                                         {\r
2124                                                 if ( me.getInputElement() )\r
2125                                                 {\r
2126                                                         me.getInputElement().on( 'focus', function()\r
2127                                                                 {\r
2128                                                                         dialog._.tabBarMode = false;\r
2129                                                                         dialog._.hasFocus = true;\r
2130                                                                         me.fire( 'focus' );\r
2131                                                                 }, me );\r
2132                                                 }\r
2133                                         } );\r
2134 \r
2135                                 // Register the object as a tab focus if it can be included.\r
2136                                 if ( this.keyboardFocusable )\r
2137                                 {\r
2138                                         this.tabIndex = elementDefinition.tabIndex || 0;\r
2139 \r
2140                                         this.focusIndex = dialog._.focusList.push( this ) - 1;\r
2141                                         this.on( 'focus', function()\r
2142                                                 {\r
2143                                                         dialog._.currentFocusIndex = me.focusIndex;\r
2144                                                 } );\r
2145                                 }\r
2146 \r
2147                                 // Completes this object with everything we have in the\r
2148                                 // definition.\r
2149                                 CKEDITOR.tools.extend( this, elementDefinition );\r
2150                         },\r
2151 \r
2152                         /**\r
2153                          * Horizontal layout box for dialog UI elements, auto-expends to available width of container.\r
2154                          * @constructor\r
2155                          * @extends CKEDITOR.ui.dialog.uiElement\r
2156                          * @param {CKEDITOR.dialog} dialog\r
2157                          * Parent dialog object.\r
2158                          * @param {Array} childObjList\r
2159                          * Array of {@link CKEDITOR.ui.dialog.uiElement} objects inside this\r
2160                          * container.\r
2161                          * @param {Array} childHtmlList\r
2162                          * Array of HTML code that correspond to the HTML output of all the\r
2163                          * objects in childObjList.\r
2164                          * @param {Array} htmlList\r
2165                          * Array of HTML code that this element will output to.\r
2166                          * @param {CKEDITOR.dialog.uiElementDefinition} elementDefinition\r
2167                          * The element definition. Accepted fields:\r
2168                          * <ul>\r
2169                          *      <li><strong>widths</strong> (Optional) The widths of child cells.</li>\r
2170                          *      <li><strong>height</strong> (Optional) The height of the layout.</li>\r
2171                          *      <li><strong>padding</strong> (Optional) The padding width inside child\r
2172                          *       cells.</li>\r
2173                          *      <li><strong>align</strong> (Optional) The alignment of the whole layout\r
2174                          *      </li>\r
2175                          * </ul>\r
2176                          * @example\r
2177                          */\r
2178                         hbox : function( dialog, childObjList, childHtmlList, htmlList, elementDefinition )\r
2179                         {\r
2180                                 if ( arguments.length < 4 )\r
2181                                         return;\r
2182 \r
2183                                 this._ || ( this._ = {} );\r
2184 \r
2185                                 var children = this._.children = childObjList,\r
2186                                         widths = elementDefinition && elementDefinition.widths || null,\r
2187                                         height = elementDefinition && elementDefinition.height || null,\r
2188                                         styles = {},\r
2189                                         i;\r
2190                                 /** @ignore */\r
2191                                 var innerHTML = function()\r
2192                                 {\r
2193                                         var html = [ '<tbody><tr class="cke_dialog_ui_hbox">' ];\r
2194                                         for ( i = 0 ; i < childHtmlList.length ; i++ )\r
2195                                         {\r
2196                                                 var className = 'cke_dialog_ui_hbox_child',\r
2197                                                         styles = [];\r
2198                                                 if ( i === 0 )\r
2199                                                         className = 'cke_dialog_ui_hbox_first';\r
2200                                                 if ( i == childHtmlList.length - 1 )\r
2201                                                         className = 'cke_dialog_ui_hbox_last';\r
2202                                                 html.push( '<td class="', className, '" role="presentation" ' );\r
2203                                                 if ( widths )\r
2204                                                 {\r
2205                                                         if ( widths[i] )\r
2206                                                                 styles.push( 'width:' + CKEDITOR.tools.cssLength( widths[i] ) );\r
2207                                                 }\r
2208                                                 else\r
2209                                                         styles.push( 'width:' + Math.floor( 100 / childHtmlList.length ) + '%' );\r
2210                                                 if ( height )\r
2211                                                         styles.push( 'height:' + CKEDITOR.tools.cssLength( height ) );\r
2212                                                 if ( elementDefinition && elementDefinition.padding != undefined )\r
2213                                                         styles.push( 'padding:' + CKEDITOR.tools.cssLength( elementDefinition.padding ) );\r
2214                                                 if ( styles.length > 0 )\r
2215                                                         html.push( 'style="' + styles.join('; ') + '" ' );\r
2216                                                 html.push( '>', childHtmlList[i], '</td>' );\r
2217                                         }\r
2218                                         html.push( '</tr></tbody>' );\r
2219                                         return html.join( '' );\r
2220                                 };\r
2221 \r
2222                                 var attribs = { role : 'presentation' };\r
2223                                 elementDefinition && elementDefinition.align && ( attribs.align = elementDefinition.align );\r
2224 \r
2225                                 CKEDITOR.ui.dialog.uiElement.call(\r
2226                                         this,\r
2227                                         dialog,\r
2228                                         elementDefinition || { type : 'hbox' },\r
2229                                         htmlList,\r
2230                                         'table',\r
2231                                         styles,\r
2232                                         attribs,\r
2233                                         innerHTML );\r
2234                         },\r
2235 \r
2236                         /**\r
2237                          * Vertical layout box for dialog UI elements.\r
2238                          * @constructor\r
2239                          * @extends CKEDITOR.ui.dialog.hbox\r
2240                          * @param {CKEDITOR.dialog} dialog\r
2241                          * Parent dialog object.\r
2242                          * @param {Array} childObjList\r
2243                          * Array of {@link CKEDITOR.ui.dialog.uiElement} objects inside this\r
2244                          * container.\r
2245                          * @param {Array} childHtmlList\r
2246                          * Array of HTML code that correspond to the HTML output of all the\r
2247                          * objects in childObjList.\r
2248                          * @param {Array} htmlList\r
2249                          * Array of HTML code that this element will output to.\r
2250                          * @param {CKEDITOR.dialog.uiElementDefinition} elementDefinition\r
2251                          * The element definition. Accepted fields:\r
2252                          * <ul>\r
2253                          *      <li><strong>width</strong> (Optional) The width of the layout.</li>\r
2254                          *      <li><strong>heights</strong> (Optional) The heights of individual cells.\r
2255                          *      </li>\r
2256                          *      <li><strong>align</strong> (Optional) The alignment of the layout.</li>\r
2257                          *      <li><strong>padding</strong> (Optional) The padding width inside child\r
2258                          *      cells.</li>\r
2259                          *      <li><strong>expand</strong> (Optional) Whether the layout should expand\r
2260                          *      vertically to fill its container.</li>\r
2261                          * </ul>\r
2262                          * @example\r
2263                          */\r
2264                         vbox : function( dialog, childObjList, childHtmlList, htmlList, elementDefinition )\r
2265                         {\r
2266                                 if (arguments.length < 3 )\r
2267                                         return;\r
2268 \r
2269                                 this._ || ( this._ = {} );\r
2270 \r
2271                                 var children = this._.children = childObjList,\r
2272                                         width = elementDefinition && elementDefinition.width || null,\r
2273                                         heights = elementDefinition && elementDefinition.heights || null;\r
2274                                 /** @ignore */\r
2275                                 var innerHTML = function()\r
2276                                 {\r
2277                                         var html = [ '<table role="presentation" cellspacing="0" border="0" ' ];\r
2278                                         html.push( 'style="' );\r
2279                                         if ( elementDefinition && elementDefinition.expand )\r
2280                                                 html.push( 'height:100%;' );\r
2281                                         html.push( 'width:' + CKEDITOR.tools.cssLength( width || '100%' ), ';' );\r
2282                                         html.push( '"' );\r
2283                                         html.push( 'align="', CKEDITOR.tools.htmlEncode(\r
2284                                                 ( elementDefinition && elementDefinition.align ) || ( dialog.getParentEditor().lang.dir == 'ltr' ? 'left' : 'right' ) ), '" ' );\r
2285 \r
2286                                         html.push( '><tbody>' );\r
2287                                         for ( var i = 0 ; i < childHtmlList.length ; i++ )\r
2288                                         {\r
2289                                                 var styles = [];\r
2290                                                 html.push( '<tr><td role="presentation" ' );\r
2291                                                 if ( width )\r
2292                                                         styles.push( 'width:' + CKEDITOR.tools.cssLength( width || '100%' ) );\r
2293                                                 if ( heights )\r
2294                                                         styles.push( 'height:' + CKEDITOR.tools.cssLength( heights[i] ) );\r
2295                                                 else if ( elementDefinition && elementDefinition.expand )\r
2296                                                         styles.push( 'height:' + Math.floor( 100 / childHtmlList.length ) + '%' );\r
2297                                                 if ( elementDefinition && elementDefinition.padding != undefined )\r
2298                                                         styles.push( 'padding:' + CKEDITOR.tools.cssLength( elementDefinition.padding ) );\r
2299                                                 if ( styles.length > 0 )\r
2300                                                         html.push( 'style="', styles.join( '; ' ), '" ' );\r
2301                                                 html.push( ' class="cke_dialog_ui_vbox_child">', childHtmlList[i], '</td></tr>' );\r
2302                                         }\r
2303                                         html.push( '</tbody></table>' );\r
2304                                         return html.join( '' );\r
2305                                 };\r
2306                                 CKEDITOR.ui.dialog.uiElement.call( this, dialog, elementDefinition || { type : 'vbox' }, htmlList, 'div', null, { role : 'presentation' }, innerHTML );\r
2307                         }\r
2308                 };\r
2309         })();\r
2310 \r
2311         CKEDITOR.ui.dialog.uiElement.prototype =\r
2312         {\r
2313                 /**\r
2314                  * Gets the root DOM element of this dialog UI object.\r
2315                  * @returns {CKEDITOR.dom.element} Root DOM element of UI object.\r
2316                  * @example\r
2317                  * uiElement.getElement().hide();\r
2318                  */\r
2319                 getElement : function()\r
2320                 {\r
2321                         return CKEDITOR.document.getById( this.domId );\r
2322                 },\r
2323 \r
2324                 /**\r
2325                  * Gets the DOM element that the user inputs values.\r
2326                  * This function is used by setValue(), getValue() and focus(). It should\r
2327                  * be overrided in child classes where the input element isn't the root\r
2328                  * element.\r
2329                  * @returns {CKEDITOR.dom.element} The element where the user input values.\r
2330                  * @example\r
2331                  * var rawValue = textInput.getInputElement().$.value;\r
2332                  */\r
2333                 getInputElement : function()\r
2334                 {\r
2335                         return this.getElement();\r
2336                 },\r
2337 \r
2338                 /**\r
2339                  * Gets the parent dialog object containing this UI element.\r
2340                  * @returns {CKEDITOR.dialog} Parent dialog object.\r
2341                  * @example\r
2342                  * var dialog = uiElement.getDialog();\r
2343                  */\r
2344                 getDialog : function()\r
2345                 {\r
2346                         return this._.dialog;\r
2347                 },\r
2348 \r
2349                 /**\r
2350                  * Sets the value of this dialog UI object.\r
2351                  * @param {Object} value The new value.\r
2352                  * @returns {CKEDITOR.dialog.uiElement} The current UI element.\r
2353                  * @example\r
2354                  * uiElement.setValue( 'Dingo' );\r
2355                  */\r
2356                 setValue : function( value )\r
2357                 {\r
2358                         this.getInputElement().setValue( value );\r
2359                         this.fire( 'change', { value : value } );\r
2360                         return this;\r
2361                 },\r
2362 \r
2363                 /**\r
2364                  * Gets the current value of this dialog UI object.\r
2365                  * @returns {Object} The current value.\r
2366                  * @example\r
2367                  * var myValue = uiElement.getValue();\r
2368                  */\r
2369                 getValue : function()\r
2370                 {\r
2371                         return this.getInputElement().getValue();\r
2372                 },\r
2373 \r
2374                 /**\r
2375                  * Tells whether the UI object's value has changed.\r
2376                  * @returns {Boolean} true if changed, false if not changed.\r
2377                  * @example\r
2378                  * if ( uiElement.isChanged() )\r
2379                  * &nbsp;&nbsp;confirm( 'Value changed! Continue?' );\r
2380                  */\r
2381                 isChanged : function()\r
2382                 {\r
2383                         // Override in input classes.\r
2384                         return false;\r
2385                 },\r
2386 \r
2387                 /**\r
2388                  * Selects the parent tab of this element. Usually called by focus() or overridden focus() methods.\r
2389                  * @returns {CKEDITOR.dialog.uiElement} The current UI element.\r
2390                  * @example\r
2391                  * focus : function()\r
2392                  * {\r
2393                  *              this.selectParentTab();\r
2394                  *              // do something else.\r
2395                  * }\r
2396                  */\r
2397                 selectParentTab : function()\r
2398                 {\r
2399                         var element = this.getInputElement(),\r
2400                                 cursor = element,\r
2401                                 tabId;\r
2402                         while ( ( cursor = cursor.getParent() ) && cursor.$.className.search( 'cke_dialog_page_contents' ) == -1 )\r
2403                         { /*jsl:pass*/ }\r
2404 \r
2405                         // Some widgets don't have parent tabs (e.g. OK and Cancel buttons).\r
2406                         if ( !cursor )\r
2407                                 return this;\r
2408 \r
2409                         tabId = cursor.getAttribute( 'name' );\r
2410                         // Avoid duplicate select.\r
2411                         if ( this._.dialog._.currentTabId != tabId )\r
2412                                 this._.dialog.selectPage( tabId );\r
2413                         return this;\r
2414                 },\r
2415 \r
2416                 /**\r
2417                  * Puts the focus to the UI object. Switches tabs if the UI object isn't in the active tab page.\r
2418                  * @returns {CKEDITOR.dialog.uiElement} The current UI element.\r
2419                  * @example\r
2420                  * uiElement.focus();\r
2421                  */\r
2422                 focus : function()\r
2423                 {\r
2424                         this.selectParentTab().getInputElement().focus();\r
2425                         return this;\r
2426                 },\r
2427 \r
2428                 /**\r
2429                  * Registers the on* event handlers defined in the element definition.\r
2430                  * The default behavior of this function is:\r
2431                  * <ol>\r
2432                  *  <li>\r
2433                  *      If the on* event is defined in the class's eventProcesors list,\r
2434                  *      then the registration is delegated to the corresponding function\r
2435                  *      in the eventProcessors list.\r
2436                  *  </li>\r
2437                  *  <li>\r
2438                  *      If the on* event is not defined in the eventProcessors list, then\r
2439                  *      register the event handler under the corresponding DOM event of\r
2440                  *      the UI element's input DOM element (as defined by the return value\r
2441                  *      of {@link CKEDITOR.ui.dialog.uiElement#getInputElement}).\r
2442                  *  </li>\r
2443                  * </ol>\r
2444                  * This function is only called at UI element instantiation, but can\r
2445                  * be overridded in child classes if they require more flexibility.\r
2446                  * @param {CKEDITOR.dialog.uiElementDefinition} definition The UI element\r
2447                  * definition.\r
2448                  * @returns {CKEDITOR.dialog.uiElement} The current UI element.\r
2449                  * @example\r
2450                  */\r
2451                 registerEvents : function( definition )\r
2452                 {\r
2453                         var regex = /^on([A-Z]\w+)/,\r
2454                                 match;\r
2455 \r
2456                         var registerDomEvent = function( uiElement, dialog, eventName, func )\r
2457                         {\r
2458                                 dialog.on( 'load', function()\r
2459                                 {\r
2460                                         uiElement.getInputElement().on( eventName, func, uiElement );\r
2461                                 });\r
2462                         };\r
2463 \r
2464                         for ( var i in definition )\r
2465                         {\r
2466                                 if ( !( match = i.match( regex ) ) )\r
2467                                         continue;\r
2468                                 if ( this.eventProcessors[i] )\r
2469                                         this.eventProcessors[i].call( this, this._.dialog, definition[i] );\r
2470                                 else\r
2471                                         registerDomEvent( this, this._.dialog, match[1].toLowerCase(), definition[i] );\r
2472                         }\r
2473 \r
2474                         return this;\r
2475                 },\r
2476 \r
2477                 /**\r
2478                  * The event processor list used by\r
2479                  * {@link CKEDITOR.ui.dialog.uiElement#getInputElement} at UI element\r
2480                  * instantiation. The default list defines three on* events:\r
2481                  * <ol>\r
2482                  *  <li>onLoad - Called when the element's parent dialog opens for the\r
2483                  *  first time</li>\r
2484                  *  <li>onShow - Called whenever the element's parent dialog opens.</li>\r
2485                  *  <li>onHide - Called whenever the element's parent dialog closes.</li>\r
2486                  * </ol>\r
2487                  * @field\r
2488                  * @type Object\r
2489                  * @example\r
2490                  * // This connects the 'click' event in CKEDITOR.ui.dialog.button to onClick\r
2491                  * // handlers in the UI element's definitions.\r
2492                  * CKEDITOR.ui.dialog.button.eventProcessors = CKEDITOR.tools.extend( {},\r
2493                  * &nbsp;&nbsp;CKEDITOR.ui.dialog.uiElement.prototype.eventProcessors,\r
2494                  * &nbsp;&nbsp;{ onClick : function( dialog, func ) { this.on( 'click', func ); } },\r
2495                  * &nbsp;&nbsp;true );\r
2496                  */\r
2497                 eventProcessors :\r
2498                 {\r
2499                         onLoad : function( dialog, func )\r
2500                         {\r
2501                                 dialog.on( 'load', func, this );\r
2502                         },\r
2503 \r
2504                         onShow : function( dialog, func )\r
2505                         {\r
2506                                 dialog.on( 'show', func, this );\r
2507                         },\r
2508 \r
2509                         onHide : function( dialog, func )\r
2510                         {\r
2511                                 dialog.on( 'hide', func, this );\r
2512                         }\r
2513                 },\r
2514 \r
2515                 /**\r
2516                  * The default handler for a UI element's access key down event, which\r
2517                  * tries to put focus to the UI element.<br />\r
2518                  * Can be overridded in child classes for more sophisticaed behavior.\r
2519                  * @param {CKEDITOR.dialog} dialog The parent dialog object.\r
2520                  * @param {String} key The key combination pressed. Since access keys\r
2521                  * are defined to always include the CTRL key, its value should always\r
2522                  * include a 'CTRL+' prefix.\r
2523                  * @example\r
2524                  */\r
2525                 accessKeyDown : function( dialog, key )\r
2526                 {\r
2527                         this.focus();\r
2528                 },\r
2529 \r
2530                 /**\r
2531                  * The default handler for a UI element's access key up event, which\r
2532                  * does nothing.<br />\r
2533                  * Can be overridded in child classes for more sophisticated behavior.\r
2534                  * @param {CKEDITOR.dialog} dialog The parent dialog object.\r
2535                  * @param {String} key The key combination pressed. Since access keys\r
2536                  * are defined to always include the CTRL key, its value should always\r
2537                  * include a 'CTRL+' prefix.\r
2538                  * @example\r
2539                  */\r
2540                 accessKeyUp : function( dialog, key )\r
2541                 {\r
2542                 },\r
2543 \r
2544                 /**\r
2545                  * Disables a UI element.\r
2546                  * @example\r
2547                  */\r
2548                 disable : function()\r
2549                 {\r
2550                         var element = this.getInputElement();\r
2551                         element.setAttribute( 'disabled', 'true' );\r
2552                         element.addClass( 'cke_disabled' );\r
2553                 },\r
2554 \r
2555                 /**\r
2556                  * Enables a UI element.\r
2557                  * @example\r
2558                  */\r
2559                 enable : function()\r
2560                 {\r
2561                         var element = this.getInputElement();\r
2562                         element.removeAttribute( 'disabled' );\r
2563                         element.removeClass( 'cke_disabled' );\r
2564                 },\r
2565 \r
2566                 /**\r
2567                  * Determines whether an UI element is enabled or not.\r
2568                  * @returns {Boolean} Whether the UI element is enabled.\r
2569                  * @example\r
2570                  */\r
2571                 isEnabled : function()\r
2572                 {\r
2573                         return !this.getInputElement().getAttribute( 'disabled' );\r
2574                 },\r
2575 \r
2576                 /**\r
2577                  * Determines whether an UI element is visible or not.\r
2578                  * @returns {Boolean} Whether the UI element is visible.\r
2579                  * @example\r
2580                  */\r
2581                 isVisible : function()\r
2582                 {\r
2583                         return this.getInputElement().isVisible();\r
2584                 },\r
2585 \r
2586                 /**\r
2587                  * Determines whether an UI element is focus-able or not.\r
2588                  * Focus-able is defined as being both visible and enabled.\r
2589                  * @returns {Boolean} Whether the UI element can be focused.\r
2590                  * @example\r
2591                  */\r
2592                 isFocusable : function()\r
2593                 {\r
2594                         if ( !this.isEnabled() || !this.isVisible() )\r
2595                                 return false;\r
2596                         return true;\r
2597                 }\r
2598         };\r
2599 \r
2600         CKEDITOR.ui.dialog.hbox.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.uiElement,\r
2601                 /**\r
2602                  * @lends CKEDITOR.ui.dialog.hbox.prototype\r
2603                  */\r
2604                 {\r
2605                         /**\r
2606                          * Gets a child UI element inside this container.\r
2607                          * @param {Array|Number} indices An array or a single number to indicate the child's\r
2608                          * position in the container's descendant tree. Omit to get all the children in an array.\r
2609                          * @returns {Array|CKEDITOR.ui.dialog.uiElement} Array of all UI elements in the container\r
2610                          * if no argument given, or the specified UI element if indices is given.\r
2611                          * @example\r
2612                          * var checkbox = hbox.getChild( [0,1] );\r
2613                          * checkbox.setValue( true );\r
2614                          */\r
2615                         getChild : function( indices )\r
2616                         {\r
2617                                 // If no arguments, return a clone of the children array.\r
2618                                 if ( arguments.length < 1 )\r
2619                                         return this._.children.concat();\r
2620 \r
2621                                 // If indices isn't array, make it one.\r
2622                                 if ( !indices.splice )\r
2623                                         indices = [ indices ];\r
2624 \r
2625                                 // Retrieve the child element according to tree position.\r
2626                                 if ( indices.length < 2 )\r
2627                                         return this._.children[ indices[0] ];\r
2628                                 else\r
2629                                         return ( this._.children[ indices[0] ] && this._.children[ indices[0] ].getChild ) ?\r
2630                                                 this._.children[ indices[0] ].getChild( indices.slice( 1, indices.length ) ) :\r
2631                                                 null;\r
2632                         }\r
2633                 }, true );\r
2634 \r
2635         CKEDITOR.ui.dialog.vbox.prototype = new CKEDITOR.ui.dialog.hbox();\r
2636 \r
2637 \r
2638 \r
2639         (function()\r
2640         {\r
2641                 var commonBuilder = {\r
2642                         build : function( dialog, elementDefinition, output )\r
2643                         {\r
2644                                 var children = elementDefinition.children,\r
2645                                         child,\r
2646                                         childHtmlList = [],\r
2647                                         childObjList = [];\r
2648                                 for ( var i = 0 ; ( i < children.length && ( child = children[i] ) ) ; i++ )\r
2649                                 {\r
2650                                         var childHtml = [];\r
2651                                         childHtmlList.push( childHtml );\r
2652                                         childObjList.push( CKEDITOR.dialog._.uiElementBuilders[ child.type ].build( dialog, child, childHtml ) );\r
2653                                 }\r
2654                                 return new CKEDITOR.ui.dialog[elementDefinition.type]( dialog, childObjList, childHtmlList, output, elementDefinition );\r
2655                         }\r
2656                 };\r
2657 \r
2658                 CKEDITOR.dialog.addUIElement( 'hbox', commonBuilder );\r
2659                 CKEDITOR.dialog.addUIElement( 'vbox', commonBuilder );\r
2660         })();\r
2661 \r
2662         /**\r
2663          * Generic dialog command. It opens a specific dialog when executed.\r
2664          * @constructor\r
2665          * @augments CKEDITOR.commandDefinition\r
2666          * @param {string} dialogName The name of the dialog to open when executing\r
2667          *              this command.\r
2668          * @example\r
2669          * // Register the "link" command, which opens the "link" dialog.\r
2670          * editor.addCommand( 'link', <b>new CKEDITOR.dialogCommand( 'link' )</b> );\r
2671          */\r
2672         CKEDITOR.dialogCommand = function( dialogName )\r
2673         {\r
2674                 this.dialogName = dialogName;\r
2675         };\r
2676 \r
2677         CKEDITOR.dialogCommand.prototype =\r
2678         {\r
2679                 /** @ignore */\r
2680                 exec : function( editor )\r
2681                 {\r
2682                         editor.openDialog( this.dialogName );\r
2683                 },\r
2684 \r
2685                 // Dialog commands just open a dialog ui, thus require no undo logic,\r
2686                 // undo support should dedicate to specific dialog implementation.\r
2687                 canUndo: false,\r
2688 \r
2689                 editorFocus : CKEDITOR.env.ie || CKEDITOR.env.webkit\r
2690         };\r
2691 \r
2692         (function()\r
2693         {\r
2694                 var notEmptyRegex = /^([a]|[^a])+$/,\r
2695                         integerRegex = /^\d*$/,\r
2696                         numberRegex = /^\d*(?:\.\d+)?$/;\r
2697 \r
2698                 CKEDITOR.VALIDATE_OR = 1;\r
2699                 CKEDITOR.VALIDATE_AND = 2;\r
2700 \r
2701                 CKEDITOR.dialog.validate =\r
2702                 {\r
2703                         functions : function()\r
2704                         {\r
2705                                 return function()\r
2706                                 {\r
2707                                         /**\r
2708                                          * It's important for validate functions to be able to accept the value\r
2709                                          * as argument in addition to this.getValue(), so that it is possible to\r
2710                                          * combine validate functions together to make more sophisticated\r
2711                                          * validators.\r
2712                                          */\r
2713                                         var value = this && this.getValue ? this.getValue() : arguments[0];\r
2714 \r
2715                                         var msg = undefined,\r
2716                                                 relation = CKEDITOR.VALIDATE_AND,\r
2717                                                 functions = [], i;\r
2718 \r
2719                                         for ( i = 0 ; i < arguments.length ; i++ )\r
2720                                         {\r
2721                                                 if ( typeof( arguments[i] ) == 'function' )\r
2722                                                         functions.push( arguments[i] );\r
2723                                                 else\r
2724                                                         break;\r
2725                                         }\r
2726 \r
2727                                         if ( i < arguments.length && typeof( arguments[i] ) == 'string' )\r
2728                                         {\r
2729                                                 msg = arguments[i];\r
2730                                                 i++;\r
2731                                         }\r
2732 \r
2733                                         if ( i < arguments.length && typeof( arguments[i]) == 'number' )\r
2734                                                 relation = arguments[i];\r
2735 \r
2736                                         var passed = ( relation == CKEDITOR.VALIDATE_AND ? true : false );\r
2737                                         for ( i = 0 ; i < functions.length ; i++ )\r
2738                                         {\r
2739                                                 if ( relation == CKEDITOR.VALIDATE_AND )\r
2740                                                         passed = passed && functions[i]( value );\r
2741                                                 else\r
2742                                                         passed = passed || functions[i]( value );\r
2743                                         }\r
2744 \r
2745                                         if ( !passed )\r
2746                                         {\r
2747                                                 if ( msg !== undefined )\r
2748                                                         alert( msg );\r
2749                                                 if ( this && ( this.select || this.focus ) )\r
2750                                                         ( this.select || this.focus )();\r
2751                                                 return false;\r
2752                                         }\r
2753 \r
2754                                         return true;\r
2755                                 };\r
2756                         },\r
2757 \r
2758                         regex : function( regex, msg )\r
2759                         {\r
2760                                 /*\r
2761                                  * Can be greatly shortened by deriving from functions validator if code size\r
2762                                  * turns out to be more important than performance.\r
2763                                  */\r
2764                                 return function()\r
2765                                 {\r
2766                                         var value = this && this.getValue ? this.getValue() : arguments[0];\r
2767                                         if ( !regex.test( value ) )\r
2768                                         {\r
2769                                                 if ( msg !== undefined )\r
2770                                                         alert( msg );\r
2771                                                 if ( this && ( this.select || this.focus ) )\r
2772                                                 {\r
2773                                                         if ( this.select )\r
2774                                                                 this.select();\r
2775                                                         else\r
2776                                                                 this.focus();\r
2777                                                 }\r
2778                                                 return false;\r
2779                                         }\r
2780                                         return true;\r
2781                                 };\r
2782                         },\r
2783 \r
2784                         notEmpty : function( msg )\r
2785                         {\r
2786                                 return this.regex( notEmptyRegex, msg );\r
2787                         },\r
2788 \r
2789                         integer : function( msg )\r
2790                         {\r
2791                                 return this.regex( integerRegex, msg );\r
2792                         },\r
2793 \r
2794                         'number' : function( msg )\r
2795                         {\r
2796                                 return this.regex( numberRegex, msg );\r
2797                         },\r
2798 \r
2799                         equals : function( value, msg )\r
2800                         {\r
2801                                 return this.functions( function( val ){ return val == value; }, msg );\r
2802                         },\r
2803 \r
2804                         notEqual : function( value, msg )\r
2805                         {\r
2806                                 return this.functions( function( val ){ return val != value; }, msg );\r
2807                         }\r
2808                 };\r
2809 \r
2810         CKEDITOR.on( 'instanceDestroyed', function( evt )\r
2811         {\r
2812                 // Remove dialog cover on last instance destroy.\r
2813                 if ( CKEDITOR.tools.isEmpty( CKEDITOR.instances ) )\r
2814                 {\r
2815                         var currentTopDialog;\r
2816                         while ( ( currentTopDialog = CKEDITOR.dialog._.currentTop ) )\r
2817                                 currentTopDialog.hide();\r
2818                         removeCovers();\r
2819                 }\r
2820 \r
2821                 var dialogs = evt.editor._.storedDialogs;\r
2822                 for ( var name in dialogs )\r
2823                         dialogs[ name ].destroy();\r
2824 \r
2825         });\r
2826 \r
2827         })();\r
2828 })();\r
2829 \r
2830 // Extend the CKEDITOR.editor class with dialog specific functions.\r
2831 CKEDITOR.tools.extend( CKEDITOR.editor.prototype,\r
2832         /** @lends CKEDITOR.editor.prototype */\r
2833         {\r
2834                 /**\r
2835                  * Loads and opens a registered dialog.\r
2836                  * @param {String} dialogName The registered name of the dialog.\r
2837                  * @param {Function} callback The function to be invoked after dialog instance created.\r
2838                  * @see CKEDITOR.dialog.add\r
2839                  * @example\r
2840                  * CKEDITOR.instances.editor1.openDialog( 'smiley' );\r
2841                  * @returns {CKEDITOR.dialog} The dialog object corresponding to the dialog displayed. null if the dialog name is not registered.\r
2842                  */\r
2843                 openDialog : function( dialogName, callback )\r
2844                 {\r
2845                         var dialogDefinitions = CKEDITOR.dialog._.dialogDefinitions[ dialogName ],\r
2846                                         dialogSkin = this.skin.dialog;\r
2847 \r
2848                         // If the dialogDefinition is already loaded, open it immediately.\r
2849                         if ( typeof dialogDefinitions == 'function' && dialogSkin._isLoaded )\r
2850                         {\r
2851                                 var storedDialogs = this._.storedDialogs ||\r
2852                                         ( this._.storedDialogs = {} );\r
2853 \r
2854                                 var dialog = storedDialogs[ dialogName ] ||\r
2855                                         ( storedDialogs[ dialogName ] = new CKEDITOR.dialog( this, dialogName ) );\r
2856 \r
2857                                 callback && callback.call( dialog, dialog );\r
2858                                 dialog.show();\r
2859 \r
2860                                 return dialog;\r
2861                         }\r
2862                         else if ( dialogDefinitions == 'failed' )\r
2863                                 throw new Error( '[CKEDITOR.dialog.openDialog] Dialog "' + dialogName + '" failed when loading definition.' );\r
2864 \r
2865                         // Not loaded? Load the .js file first.\r
2866                         var body = CKEDITOR.document.getBody(),\r
2867                                 cursor = body.$.style.cursor,\r
2868                                 me = this;\r
2869 \r
2870                         body.setStyle( 'cursor', 'wait' );\r
2871 \r
2872                         function onDialogFileLoaded( success )\r
2873                         {\r
2874                                 var dialogDefinition = CKEDITOR.dialog._.dialogDefinitions[ dialogName ],\r
2875                                                 skin = me.skin.dialog;\r
2876 \r
2877                                 // Check if both skin part and definition is loaded.\r
2878                                 if ( !skin._isLoaded || loadDefinition && typeof success == 'undefined' )\r
2879                                         return;\r
2880 \r
2881                                 // In case of plugin error, mark it as loading failed.\r
2882                                 if ( typeof dialogDefinition != 'function' )\r
2883                                         CKEDITOR.dialog._.dialogDefinitions[ dialogName ] = 'failed';\r
2884 \r
2885                                 me.openDialog( dialogName, callback );\r
2886                                 body.setStyle( 'cursor', cursor );\r
2887                         }\r
2888 \r
2889                         if ( typeof dialogDefinitions == 'string' )\r
2890                         {\r
2891                                 var loadDefinition = 1;\r
2892                                 CKEDITOR.scriptLoader.load( CKEDITOR.getUrl( dialogDefinitions ), onDialogFileLoaded );\r
2893                         }\r
2894 \r
2895                         CKEDITOR.skins.load( this, 'dialog', onDialogFileLoaded );\r
2896 \r
2897                         return null;\r
2898                 }\r
2899         });\r
2900 \r
2901 CKEDITOR.plugins.add( 'dialog',\r
2902         {\r
2903                 requires : [ 'dialogui' ]\r
2904         });\r
2905 \r
2906 // Dialog related configurations.\r
2907 \r
2908 /**\r
2909  * The color of the dialog background cover. It should be a valid CSS color\r
2910  * string.\r
2911  * @name CKEDITOR.config.dialog_backgroundCoverColor\r
2912  * @type String\r
2913  * @default 'white'\r
2914  * @example\r
2915  * config.dialog_backgroundCoverColor = 'rgb(255, 254, 253)';\r
2916  */\r
2917 \r
2918 /**\r
2919  * The opacity of the dialog background cover. It should be a number within the\r
2920  * range [0.0, 1.0].\r
2921  * @name CKEDITOR.config.dialog_backgroundCoverOpacity\r
2922  * @type Number\r
2923  * @default 0.5\r
2924  * @example\r
2925  * config.dialog_backgroundCoverOpacity = 0.7;\r
2926  */\r
2927 \r
2928 /**\r
2929  * If the dialog has more than one tab, put focus into the first tab as soon as dialog is opened.\r
2930  * @name CKEDITOR.config.dialog_startupFocusTab\r
2931  * @type Boolean\r
2932  * @default false\r
2933  * @example\r
2934  * config.dialog_startupFocusTab = true;\r
2935  */\r
2936 \r
2937 /**\r
2938  * The distance of magnetic borders used in moving and resizing dialogs,\r
2939  * measured in pixels.\r
2940  * @name CKEDITOR.config.dialog_magnetDistance\r
2941  * @type Number\r
2942  * @default 20\r
2943  * @example\r
2944  * config.dialog_magnetDistance = 30;\r
2945  */\r
2946 \r
2947 /**\r
2948  * Fired when a dialog definition is about to be used to create a dialog into\r
2949  * an editor instance. This event makes it possible to customize the definition\r
2950  * before creating it.\r
2951  * <p>Note that this event is called only the first time a specific dialog is\r
2952  * opened. Successive openings will use the cached dialog, and this event will\r
2953  * not get fired.</p>\r
2954  * @name CKEDITOR#dialogDefinition\r
2955  * @event\r
2956  * @param {CKEDITOR.dialog.dialogDefinition} data The dialog defination that\r
2957  *              is being loaded.\r
2958  * @param {CKEDITOR.editor} editor The editor instance that will use the\r
2959  *              dialog.\r
2960  */\r
2961 \r
2962 /**\r
2963  * Fired when a tab is going to be selected in a dialog\r
2964  * @name dialog#selectPage\r
2965  * @event\r
2966  * @param String page The id of the page that it's gonna be selected.\r
2967  * @param String currentPage The id of the current page.\r
2968  */\r