5d0573c40386caae1ee4834169816dc9c227fec7
2 Copyright (c) 2003-2011, CKSource - Frederico Knabben. All rights reserved.
3 For licensing, see LICENSE.html or http://ckeditor.com/license
7 * @fileOverview Defines the {@link CKEDITOR.tools} object, which contains
15 CKEDITOR
.on( 'reset', function()
28 * Compare the elements of two arrays.
29 * @param {Array} arrayA An array to be compared.
30 * @param {Array} arrayB The other array to be compared.
31 * @returns {Boolean} "true" is the arrays have the same lenght and
32 * their elements match.
34 * var a = [ 1, 'a', 3 ];
35 * var b = [ 1, 3, 'a' ];
36 * var c = [ 1, 'a', 3 ];
37 * var d = [ 1, 'a', 3, 4 ];
39 * alert( CKEDITOR.tools.arrayCompare( a, b ) ); // false
40 * alert( CKEDITOR.tools.arrayCompare( a, c ) ); // true
41 * alert( CKEDITOR.tools.arrayCompare( a, d ) ); // false
43 arrayCompare : function( arrayA
, arrayB
)
45 if ( !arrayA
&& !arrayB
)
48 if ( !arrayA
|| !arrayB
|| arrayA
.length
!= arrayB
.length
)
51 for ( var i
= 0 ; i
< arrayA
.length
; i
++ )
53 if ( arrayA
[ i
] != arrayB
[ i
] )
61 * Creates a deep copy of an object.
62 * Attention: there is no support for recursive references.
63 * @param {Object} object The object to be cloned.
64 * @returns {Object} The object clone.
71 * Mercedes : { color : 'blue' },
72 * Porsche : { color : 'red' }
75 * var clone = CKEDITOR.tools.clone( obj );
76 * clone.name = 'Paul';
77 * clone.cars.Porsche.color = 'silver';
78 * alert( obj.name ); // John
79 * alert( clone.name ); // Paul
80 * alert( obj.cars.Porsche.color ); // red
81 * alert( clone.cars.Porsche.color ); // silver
83 clone : function( obj
)
88 if ( obj
&& ( obj
instanceof Array
) )
92 for ( var i
= 0 ; i
< obj
.length
; i
++ )
93 clone
[ i
] = this.clone( obj
[ i
] );
100 || ( typeof( obj
) != 'object' )
101 || ( obj
instanceof String
)
102 || ( obj
instanceof Number
)
103 || ( obj
instanceof Boolean
)
104 || ( obj
instanceof Date
)
105 || ( obj
instanceof RegExp
) )
111 clone
= new obj
.constructor();
113 for ( var propertyName
in obj
)
115 var property
= obj
[ propertyName
];
116 clone
[ propertyName
] = this.clone( property
);
123 * Turn the first letter of string to upper-case.
124 * @param {String} str
126 capitalize: function( str
)
128 return str
.charAt( 0 ).toUpperCase() + str
.substring( 1 ).toLowerCase();
132 * Copy the properties from one object to another. By default, properties
133 * already present in the target object <strong>are not</strong> overwritten.
134 * @param {Object} target The object to be extended.
135 * @param {Object} source[,souce(n)] The objects from which copy
136 * properties. Any number of objects can be passed to this function.
137 * @param {Boolean} [overwrite] If 'true' is specified it indicates that
138 * properties already present in the target object could be
139 * overwritten by subsequent objects.
140 * @param {Object} [properties] Only properties within the specified names
141 * list will be received from the source object.
142 * @returns {Object} the extended object (target).
144 * // Create the sample object.
150 * // Extend the above object with two properties.
151 * CKEDITOR.tools.extend( myObject,
157 * // Alert "prop1", "prop2" and "prop3".
158 * for ( var p in myObject )
161 extend : function( target
)
163 var argsLength
= arguments
.length
,
164 overwrite
, propertiesList
;
166 if ( typeof ( overwrite
= arguments
[ argsLength
- 1 ] ) == 'boolean')
168 else if ( typeof ( overwrite
= arguments
[ argsLength
- 2 ] ) == 'boolean' )
170 propertiesList
= arguments
[ argsLength
-1 ];
173 for ( var i
= 1 ; i
< argsLength
; i
++ )
175 var source
= arguments
[ i
];
176 for ( var propertyName
in source
)
178 // Only copy existed fields if in overwrite mode.
179 if ( overwrite
=== true || target
[ propertyName
] == undefined )
181 // Only copy specified fields if list is provided.
182 if ( !propertiesList
|| ( propertyName
in propertiesList
) )
183 target
[ propertyName
] = source
[ propertyName
];
193 * Creates an object which is an instance of a class which prototype is a
194 * predefined object. All properties defined in the source object are
195 * automatically inherited by the resulting object, including future
197 * @param {Object} source The source object to be used as the prototype for
199 * @returns {Object} The resulting copy.
201 prototypedCopy : function( source
)
203 var copy = function()
205 copy
.prototype = source
;
210 * Checks if an object is an Array.
211 * @param {Object} object The object to be checked.
213 * @returns <i>true</i> if the object is an Array, otherwise <i>false</i>.
215 * alert( CKEDITOR.tools.isArray( [] ) ); // "true"
216 * alert( CKEDITOR.tools.isArray( 'Test' ) ); // "false"
218 isArray : function( object
)
220 return ( !!object
&& object
instanceof Array
);
224 * Whether the object contains no properties of it's own.
227 isEmpty : function ( object
)
229 for ( var i
in object
)
231 if ( object
.hasOwnProperty( i
) )
238 * Transforms a CSS property name to its relative DOM style name.
239 * @param {String} cssName The CSS property name.
240 * @returns {String} The transformed name.
242 * alert( CKEDITOR.tools.cssStyleToDomStyle( 'background-color' ) ); // "backgroundColor"
243 * alert( CKEDITOR.tools.cssStyleToDomStyle( 'float' ) ); // "cssFloat"
245 cssStyleToDomStyle
: ( function()
247 var test
= document
.createElement( 'div' ).style
;
249 var cssFloat
= ( typeof test
.cssFloat
!= 'undefined' ) ? 'cssFloat'
250 : ( typeof test
.styleFloat
!= 'undefined' ) ? 'styleFloat'
253 return function( cssName
)
255 if ( cssName
== 'float' )
259 return cssName
.replace( /-./g, function( match
)
261 return match
.substr( 1 ).toUpperCase();
268 * Build the HTML snippet of a set of <style>/<link>.
269 * @param css {String|Array} Each of which are url (absolute) of a CSS file or
270 * a trunk of style text.
272 buildStyleHtml : function ( css
)
274 css
= [].concat( css
);
275 var item
, retval
= [];
276 for ( var i
= 0; i
< css
.length
; i
++ )
279 // Is CSS style text ?
280 if ( /@import|[{}]/.test(item
) )
281 retval
.push('<style>' + item
+ '</style>');
283 retval
.push('<link type="text/css" rel=stylesheet href="' + item
+ '">');
285 return retval
.join( '' );
289 * Replace special HTML characters in a string with their relative HTML
291 * @param {String} text The string to be encoded.
292 * @returns {String} The encode string.
294 * alert( CKEDITOR.tools.htmlEncode( 'A > B & C < D' ) ); // "A &gt; B &amp; C &lt; D"
296 htmlEncode : function( text
)
298 var standard = function( text
)
300 var span
= new CKEDITOR
.dom
.element( 'span' );
301 span
.setText( text
);
302 return span
.getHtml();
305 var fix1
= ( standard( '\n' ).toLowerCase() == '<br>' ) ?
308 // #3874 IE and Safari encode line-break into <br>
309 return standard( text
).replace( /<br>/gi, '\n' );
313 var fix2
= ( standard( '>' ) == '>' ) ?
316 // WebKit does't encode the ">" character, which makes sense, but
317 // it's different than other browsers.
318 return fix1( text
).replace( />/g
, '>' );
322 var fix3
= ( standard( ' ' ) == ' ' ) ?
325 // #3785 IE8 changes spaces (>= 2) to
326 return fix2( text
).replace( / /g, ' ' );
330 this.htmlEncode
= fix3
;
332 return this.htmlEncode( text
);
336 * Replace special HTML characters in HTMLElement's attribute with their relative HTML entity values.
337 * @param {String} The attribute's value to be encoded.
338 * @returns {String} The encode value.
340 * element.setAttribute( 'title', '<a " b >' );
341 * alert( CKEDITOR.tools.htmlEncodeAttr( element.getAttribute( 'title' ) ); // ">a " b <"
343 htmlEncodeAttr : function( text
)
345 return text
.replace( /"/g, '"' ).replace( /</g, '<' ).replace( />/g
, '>' );
349 * Gets a unique number for this CKEDITOR execution session. It returns
350 * progressive numbers starting at 1.
352 * @returns {Number} A unique number.
354 * alert( CKEDITOR.tools.<b>getNextNumber()</b> ); // "1" (e.g.)
355 * alert( CKEDITOR.tools.<b>getNextNumber()</b> ); // "2"
357 getNextNumber
: (function()
367 * Gets a unique ID for CKEditor's interface elements. It returns a
368 * string with the "cke_" prefix and a progressive number.
370 * @returns {String} A unique ID.
372 * alert( CKEDITOR.tools.<b>getNextId()</b> ); // "cke_1" (e.g.)
373 * alert( CKEDITOR.tools.<b>getNextId()</b> ); // "cke_2"
375 getNextId : function()
377 return 'cke_' + this.getNextNumber();
381 * Creates a function override.
382 * @param {Function} originalFunction The function to be overridden.
383 * @param {Function} functionBuilder A function that returns the new
384 * function. The original function reference will be passed to this
386 * @returns {Function} The new function.
390 * myFunction : function( name )
392 * alert( 'Name: ' + name );
396 * example.myFunction = CKEDITOR.tools.override( example.myFunction, function( myFunctionOriginal )
398 * return function( name )
400 * alert( 'Override Name: ' + name );
401 * myFunctionOriginal.call( this, name );
405 override : function( originalFunction
, functionBuilder
)
407 return functionBuilder( originalFunction
);
411 * Executes a function after specified delay.
412 * @param {Function} func The function to be executed.
413 * @param {Number} [milliseconds] The amount of time (millisecods) to wait
414 * to fire the function execution. Defaults to zero.
415 * @param {Object} [scope] The object to hold the function execution scope
416 * (the "this" object). By default the "window" object.
417 * @param {Object|Array} [args] A single object, or an array of objects, to
418 * pass as arguments to the function.
419 * @param {Object} [ownerWindow] The window that will be used to set the
420 * timeout. By default the current "window".
421 * @returns {Object} A value that can be used to cancel the function execution.
423 * CKEDITOR.tools.<b>setTimeout(
426 * alert( 'Executed after 2 seconds' );
430 setTimeout : function( func
, milliseconds
, scope
, args
, ownerWindow
)
433 ownerWindow
= window
;
438 return ownerWindow
.setTimeout(
442 func
.apply( scope
, [].concat( args
) ) ;
444 func
.apply( scope
) ;
450 * Remove spaces from the start and the end of a string. The following
451 * characters are removed: space, tab, line break, line feed.
453 * @param {String} str The text from which remove the spaces.
454 * @returns {String} The modified string without the boundary spaces.
456 * alert( CKEDITOR.tools.trim( ' example ' ); // "example"
460 // We are not using \s because we don't want "non-breaking spaces" to be caught.
461 var trimRegex
= /(?:^[ \t\n\r]+)|(?:[ \t\n\r]+$)/g;
462 return function( str
)
464 return str
.replace( trimRegex
, '' ) ;
469 * Remove spaces from the start (left) of a string. The following
470 * characters are removed: space, tab, line break, line feed.
472 * @param {String} str The text from which remove the spaces.
473 * @returns {String} The modified string excluding the removed spaces.
475 * alert( CKEDITOR.tools.ltrim( ' example ' ); // "example "
479 // We are not using \s because we don't want "non-breaking spaces" to be caught.
480 var trimRegex
= /^[ \t\n\r]+/g;
481 return function( str
)
483 return str
.replace( trimRegex
, '' ) ;
488 * Remove spaces from the end (right) of a string. The following
489 * characters are removed: space, tab, line break, line feed.
491 * @param {String} str The text from which remove the spaces.
492 * @returns {String} The modified string excluding the removed spaces.
494 * alert( CKEDITOR.tools.ltrim( ' example ' ); // " example"
498 // We are not using \s because we don't want "non-breaking spaces" to be caught.
499 var trimRegex
= /[ \t\n\r]+$/g;
500 return function( str
)
502 return str
.replace( trimRegex
, '' ) ;
507 * Returns the index of an element in an array.
508 * @param {Array} array The array to be searched.
509 * @param {Object} entry The element to be found.
510 * @returns {Number} The (zero based) index of the first entry that matches
511 * the entry, or -1 if not found.
513 * var letters = [ 'a', 'b', 0, 'c', false ];
514 * alert( CKEDITOR.tools.indexOf( letters, '0' ) ); "-1" because 0 !== '0'
515 * alert( CKEDITOR.tools.indexOf( letters, false ) ); "4" because 0 !== false
518 // #2514: We should try to use Array.indexOf if it does exist.
519 ( Array
.prototype.indexOf
) ?
520 function( array
, entry
)
522 return array
.indexOf( entry
);
525 function( array
, entry
)
527 for ( var i
= 0, len
= array
.length
; i
< len
; i
++ )
529 if ( array
[ i
] === entry
)
536 * Creates a function that will always execute in the context of a
538 * @param {Function} func The function to be executed.
539 * @param {Object} obj The object to which bind the execution context.
540 * @returns {Function} The function that can be used to execute the
541 * "func" function in the context of "obj".
543 * var obj = { text : 'My Object' };
545 * function alertText()
547 * alert( this.text );
550 * var newFunc = <b>CKEDITOR.tools.bind( alertText, obj )</b>;
551 * newFunc(); // Alerts "My Object".
553 bind : function( func
, obj
)
555 return function() { return func
.apply( obj
, arguments
); };
559 * Class creation based on prototype inheritance, with supports of the
560 * following features:
562 * <li> Static fields </li>
563 * <li> Private fields </li>
564 * <li> Public (prototype) fields </li>
565 * <li> Chainable base class constructor </li>
567 * @param {Object} definition The class definition object.
568 * @returns {Function} A class-like JavaScript function.
570 createClass : function( definition
)
572 var $ = definition
.$,
573 baseClass
= definition
.base
,
574 privates
= definition
.privates
|| definition
._
,
575 proto
= definition
.proto
,
576 statics
= definition
.statics
;
580 var originalConstructor
= $;
583 // Create (and get) the private namespace.
584 var _
= this._
|| ( this._
= {} );
586 // Make some magic so "this" will refer to the main
587 // instance when coding private functions.
588 for ( var privateName
in privates
)
590 var priv
= privates
[ privateName
];
593 ( typeof priv
== 'function' ) ? CKEDITOR
.tools
.bind( priv
, this ) : priv
;
596 originalConstructor
.apply( this, arguments
);
602 $.prototype = this.prototypedCopy( baseClass
.prototype );
603 $.prototype.constructor = $;
604 $.prototype.base = function()
606 this.base
= baseClass
.prototype.base
;
607 baseClass
.apply( this, arguments
);
608 this.base
= arguments
.callee
;
613 this.extend( $.prototype, proto
, true );
616 this.extend( $, statics
, true );
622 * Creates a function reference that can be called later using
623 * CKEDITOR.tools.callFunction. This approach is specially useful to
624 * make DOM attribute function calls to JavaScript defined functions.
625 * @param {Function} fn The function to be executed on call.
626 * @param {Object} [scope] The object to have the context on "fn" execution.
627 * @returns {Number} A unique reference to be used in conjuction with
628 * CKEDITOR.tools.callFunction.
630 * var ref = <b>CKEDITOR.tools.addFunction</b>(
635 * CKEDITOR.tools.callFunction( ref ); // Hello!
637 addFunction : function( fn
, scope
)
639 return functions
.push( function()
641 return fn
.apply( scope
|| this, arguments
);
646 * Removes the function reference created with {@see CKEDITOR.tools.addFunction}.
647 * @param {Number} ref The function reference created with
648 * CKEDITOR.tools.addFunction.
650 removeFunction : function( ref
)
652 functions
[ ref
] = null;
656 * Executes a function based on the reference created with
657 * CKEDITOR.tools.addFunction.
658 * @param {Number} ref The function reference created with
659 * CKEDITOR.tools.addFunction.
660 * @param {[Any,[Any,...]} params Any number of parameters to be passed
661 * to the executed function.
662 * @returns {Any} The return value of the function.
664 * var ref = CKEDITOR.tools.addFunction(
669 * <b>CKEDITOR.tools.callFunction( ref )</b>; // Hello!
671 callFunction : function( ref
)
673 var fn
= functions
[ ref
];
674 return fn
&& fn
.apply( window
, Array
.prototype.slice
.call( arguments
, 1 ) );
678 * Append the 'px' length unit to the size if it's missing.
681 cssLength
: (function()
683 return function( length
)
685 return length
+ ( !length
|| isNaN( Number( length
) ) ? '' : 'px' );
690 * Convert the specified CSS length value to the calculated pixel length inside this page.
691 * <strong>Note:</strong> Percentage based value is left intact.
692 * @param {String} cssLength CSS length value.
694 convertToPx
: ( function ()
698 return function( cssLength
)
702 calculator
= CKEDITOR
.dom
.element
.createFromHtml(
703 '<div style="position:absolute;left:-9999px;' +
704 'top:-9999px;margin:0px;padding:0px;border:0px;"' +
705 '></div>', CKEDITOR
.document
);
706 CKEDITOR
.document
.getBody().append( calculator
);
709 if ( !(/%$/).test( cssLength
) )
711 calculator
.setStyle( 'width', cssLength
);
712 return calculator
.$.clientWidth
;
720 * String specified by {@param str} repeats {@param times} times.
724 repeat : function( str
, times
)
726 return new Array( times
+ 1 ).join( str
);
730 * Return the first successfully executed function's return value that
731 * doesn't throw any exception.
733 tryThese : function()
736 for ( var i
= 0, length
= arguments
.length
; i
< length
; i
++ )
738 var lambda
= arguments
[i
];
741 returnValue
= lambda();
750 * Generate a combined key from a series of params.
751 * @param {String} subKey One or more string used as sub keys.
753 * var key = CKEDITOR.tools.genKey( 'key1', 'key2', 'key3' );
754 * alert( key ); // "key1-key2-key3".
758 return Array
.prototype.slice
.call( arguments
).join( '-' );
763 // PACKAGER_RENAME( CKEDITOR.tools )