initial commit
[namibia] / public / scripts / ckeditor / _source / plugins / bbcode / plugin.js
1 /*
2 Copyright (c) 2003-2011, CKSource - Frederico Knabben. All rights reserved.
3 For licensing, see LICENSE.html or http://ckeditor.com/license
4 */
5
6 (function()
7 {
8         CKEDITOR.on( 'dialogDefinition', function( ev )
9         {
10                 var tab, name = ev.data.name,
11                         definition = ev.data.definition;
12
13                 if ( name == 'link' )
14                 {
15                         definition.removeContents( 'target' );
16                         definition.removeContents( 'upload' );
17                         definition.removeContents( 'advanced' );
18                         tab = definition.getContents( 'info' );
19                         tab.remove( 'emailSubject' );
20                         tab.remove( 'emailBody' );
21                 }
22                 else if ( name == 'image' )
23                 {
24                         definition.removeContents( 'advanced' );
25                         tab = definition.getContents( 'Link' );
26                         tab.remove( 'cmbTarget' );
27                         tab = definition.getContents( 'info' );
28                         tab.remove( 'txtAlt' );
29                         tab.remove( 'basic' );
30                 }
31         });
32
33         var bbcodeMap = { 'b' : 'strong', 'u': 'u', 'i' : 'em', 'color' : 'span', 'size' : 'span', 'quote' : 'blockquote', 'code' : 'code', 'url' : 'a', 'email' : 'span', 'img' : 'span', '*' : 'li', 'list' : 'ol' },
34                         convertMap = { 'strong' : 'b' , 'b' : 'b', 'u': 'u', 'em' : 'i', 'i': 'i', 'code' : 'code', 'li' : '*' },
35                         tagnameMap = { 'strong' : 'b', 'em' : 'i', 'u' : 'u', 'li' : '*', 'ul' : 'list', 'ol' : 'list', 'code' : 'code', 'a' : 'link', 'img' : 'img', 'blockquote' : 'quote' },
36                         stylesMap = { 'color' : 'color', 'size' : 'font-size' },
37                         attributesMap = { 'url' : 'href', 'email' : 'mailhref', 'quote': 'cite', 'list' : 'listType' };
38
39         // List of block-like tags.
40         var dtd =  CKEDITOR.dtd,
41                 blockLikeTags = CKEDITOR.tools.extend( { table:1 }, dtd.$block, dtd.$listItem, dtd.$tableContent, dtd.$list );
42
43         var semicolonFixRegex = /\s*(?:;\s*|$)/;
44         function serializeStyleText( stylesObject )
45         {
46                 var styleText = '';
47                 for ( var style in stylesObject )
48                 {
49                         var styleVal = stylesObject[ style ],
50                                 text = ( style + ':' + styleVal ).replace( semicolonFixRegex, ';' );
51
52                         styleText += text;
53                 }
54                 return styleText;
55         }
56
57         function parseStyleText( styleText )
58         {
59                 var retval = {};
60                 ( styleText || '' )
61                                 .replace( /"/g, '"' )
62                                 .replace( /\s*([^ :;]+)\s*:\s*([^;]+)\s*(?=;|$)/g, function( match, name, value )
63                 {
64                         retval[ name.toLowerCase() ] = value;
65                 } );
66                 return retval;
67         }
68
69         function RGBToHex( cssStyle )
70         {
71                 return cssStyle.replace( /(?:rgb\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*\))/gi, function( match, red, green, blue )
72                         {
73                                 red = parseInt( red, 10 ).toString( 16 );
74                                 green = parseInt( green, 10 ).toString( 16 );
75                                 blue = parseInt( blue, 10 ).toString( 16 );
76                                 var color = [red, green, blue] ;
77
78                                 // Add padding zeros if the hex value is less than 0x10.
79                                 for ( var i = 0 ; i < color.length ; i++ )
80                                         color[i] = String( '0' + color[i] ).slice( -2 ) ;
81
82                                 return '#' + color.join( '' ) ;
83                          });
84         }
85
86         // Maintain the map of smiley-to-description.
87         var smileyMap = {"smiley":":)","sad":":(","wink":";)","laugh":":D","cheeky":":P","blush":":*)","surprise":":-o","indecision":":|","angry":">:(","angel":"o:)","cool":"8-)","devil":">:-)","crying":";(","kiss":":-*" },
88                 smileyReverseMap = {},
89                 smileyRegExp = [];
90
91         // Build regexp for the list of smiley text.
92         for ( var i in smileyMap )
93         {
94                 smileyReverseMap[ smileyMap[ i ] ] = i;
95                 smileyRegExp.push( smileyMap[ i ].replace( /\(|\)|\:|\/|\*|\-|\|/g, function( match ) { return '\\' + match; } ) );
96         }
97
98         smileyRegExp = new RegExp( smileyRegExp.join( '|' ), 'g' );
99
100         var decodeHtml = ( function ()
101         {
102                 var regex = [],
103                         entities =
104                         {
105                                 nbsp    : '\u00A0',             // IE | FF
106                                 shy             : '\u00AD',             // IE
107                                 gt              : '\u003E',             // IE | FF |   --   | Opera
108                                 lt              : '\u003C'              // IE | FF | Safari | Opera
109                         };
110
111                 for ( var entity in entities )
112                         regex.push( entity );
113
114                 regex = new RegExp( '&(' + regex.join( '|' ) + ');', 'g' );
115
116                 return function( html )
117                 {
118                         return html.replace( regex, function( match, entity )
119                         {
120                                 return entities[ entity ];
121                         });
122                 };
123         })();
124
125         CKEDITOR.BBCodeParser = function()
126         {
127                 this._ =
128                 {
129                         bbcPartsRegex : /(?:\[([^\/\]=]*?)(?:=([^\]]*?))?\])|(?:\[\/([a-z]{1,16})\])/ig
130                 };
131         };
132
133         CKEDITOR.BBCodeParser.prototype =
134         {
135                 parse : function( bbcode )
136                 {
137                         var parts,
138                                         part,
139                                         lastIndex = 0;
140
141                         while ( ( parts = this._.bbcPartsRegex.exec( bbcode ) ) )
142                         {
143                                 var tagIndex = parts.index;
144                                 if ( tagIndex > lastIndex )
145                                 {
146                                         var text = bbcode.substring( lastIndex, tagIndex );
147                                         this.onText( text, 1 );
148                                 }
149
150                                 lastIndex = this._.bbcPartsRegex.lastIndex;
151
152                                 /*
153                                  "parts" is an array with the following items:
154                                  0 : The entire match for opening/closing tags and line-break;
155                                  1 : line-break;
156                                  2 : open of tag excludes option;
157                                  3 : tag option;
158                                  4 : close of tag;
159                                  */
160
161                                 part = ( parts[ 1 ] || parts[ 3 ] || '' ).toLowerCase();
162                                 // Unrecognized tags should be delivered as a simple text (#7860).
163                                 if ( part && !bbcodeMap[ part ] )
164                                 {
165                                         this.onText( parts[ 0 ] );
166                                         continue;
167                                 }
168
169                                 // Opening tag
170                                 if ( parts[ 1 ] )
171                                 {
172                                         var tagName = bbcodeMap[ part ],
173                                                         attribs = {},
174                                                         styles = {},
175                                                         optionPart = parts[ 2 ];
176
177                                         if ( optionPart )
178                                         {
179                                                 if ( part == 'list' )
180                                                 {
181                                                         if ( !isNaN( optionPart ) )
182                                                                 optionPart = 'decimal';
183                                                         else if ( /^[a-z]+$/.test( optionPart ) )
184                                                                 optionPart = 'lower-alpha';
185                                                         else if ( /^[A-Z]+$/.test( optionPart ) )
186                                                                 optionPart = 'upper-alpha';
187                                                 }
188
189                                                 if ( stylesMap[ part ] )
190                                                 {
191                                                         // Font size represents percentage.
192                                                         if ( part == 'size' )
193                                                                 optionPart += '%';
194
195                                                         styles[ stylesMap[ part ] ] = optionPart;
196                                                         attribs.style = serializeStyleText( styles );
197                                                 }
198                                                 else if ( attributesMap[ part ] )
199                                                         attribs[ attributesMap[ part ] ] = optionPart;
200                                         }
201
202                                         // Two special handling - image and email, protect them
203                                         // as "span" with an attribute marker.
204                                         if ( part == 'email' || part == 'img' )
205                                                 attribs[ 'bbcode' ] = part;
206
207                                         this.onTagOpen( tagName, attribs, CKEDITOR.dtd.$empty[ tagName ] );
208                                 }
209                                 // Closing tag
210                                 else if ( parts[ 3 ] )
211                                         this.onTagClose( bbcodeMap[ part ] );
212                         }
213
214                         if ( bbcode.length > lastIndex )
215                                 this.onText( bbcode.substring( lastIndex, bbcode.length ), 1 );
216                 }
217         };
218
219         /**
220          * Creates a {@link CKEDITOR.htmlParser.fragment} from an HTML string.
221          * @param {String} source The HTML to be parsed, filling the fragment.
222          * @param {Number} [fixForBody=false] Wrap body with specified element if needed.
223          * @returns CKEDITOR.htmlParser.fragment The fragment created.
224          * @example
225          * var fragment = CKEDITOR.htmlParser.fragment.fromHtml( '<b>Sample</b> Text' );
226          * alert( fragment.children[0].name );  "b"
227          * alert( fragment.children[1].value );  " Text"
228          */
229         CKEDITOR.htmlParser.fragment.fromBBCode = function( source )
230         {
231                 var parser = new CKEDITOR.BBCodeParser(),
232                         fragment = new CKEDITOR.htmlParser.fragment(),
233                         pendingInline = [],
234                         pendingBrs = 0,
235                         currentNode = fragment,
236                         returnPoint;
237
238                 function checkPending( newTagName )
239                 {
240                         if ( pendingInline.length > 0 )
241                         {
242                                 for ( var i = 0 ; i < pendingInline.length ; i++ )
243                                 {
244                                         var pendingElement = pendingInline[ i ],
245                                                 pendingName = pendingElement.name,
246                                                 pendingDtd = CKEDITOR.dtd[ pendingName ],
247                                                 currentDtd = currentNode.name && CKEDITOR.dtd[ currentNode.name ];
248
249                                         if ( ( !currentDtd || currentDtd[ pendingName ] ) && ( !newTagName || !pendingDtd || pendingDtd[ newTagName ] || !CKEDITOR.dtd[ newTagName ] ) )
250                                         {
251                                                 // Get a clone for the pending element.
252                                                 pendingElement = pendingElement.clone();
253
254                                                 // Add it to the current node and make it the current,
255                                                 // so the new element will be added inside of it.
256                                                 pendingElement.parent = currentNode;
257                                                 currentNode = pendingElement;
258
259                                                 // Remove the pending element (back the index by one
260                                                 // to properly process the next entry).
261                                                 pendingInline.splice( i, 1 );
262                                                 i--;
263                                         }
264                                 }
265                         }
266                 }
267
268                 function checkPendingBrs( tagName, closing )
269                 {
270                         var len = currentNode.children.length,
271                                 previous = len > 0 && currentNode.children[ len - 1 ],
272                                 lineBreakParent = !previous && BBCodeWriter.getRule( tagnameMap[ currentNode.name ], 'breakAfterOpen' ),
273                                 lineBreakPrevious = previous && previous.type == CKEDITOR.NODE_ELEMENT && BBCodeWriter.getRule( tagnameMap[ previous.name ], 'breakAfterClose' ),
274                                 lineBreakCurrent = tagName && BBCodeWriter.getRule( tagnameMap[ tagName ], closing ? 'breakBeforeClose' : 'breakBeforeOpen' );
275
276                         if ( pendingBrs && ( lineBreakParent || lineBreakPrevious || lineBreakCurrent ) )
277                                 pendingBrs--;
278
279                         // 1. Either we're at the end of block, where it requires us to compensate the br filler
280                         // removing logic (from htmldataprocessor).
281                         // 2. Or we're at the end of pseudo block, where it requires us to compensate
282                         // the bogus br effect.
283                         if ( pendingBrs && tagName in blockLikeTags )
284                                 pendingBrs++;
285
286                         while ( pendingBrs && pendingBrs-- )
287                                 currentNode.children.push( previous = new CKEDITOR.htmlParser.element( 'br' ) );
288                 }
289
290                 function addElement( node, target )
291                 {
292                         checkPendingBrs( node.name, 1 );
293
294                         target = target || currentNode || fragment;
295
296                         var len = target.children.length,
297                                 previous = len > 0 && target.children[ len - 1 ] || null;
298
299                         node.previous = previous;
300                         node.parent = target;
301
302                         target.children.push( node );
303
304                         if ( node.returnPoint )
305                         {
306                                 currentNode = node.returnPoint;
307                                 delete node.returnPoint;
308                         }
309                 }
310
311                 parser.onTagOpen = function( tagName, attributes, selfClosing )
312                 {
313                         var element = new CKEDITOR.htmlParser.element( tagName, attributes );
314
315                         // This is a tag to be removed if empty, so do not add it immediately.
316                         if ( CKEDITOR.dtd.$removeEmpty[ tagName ] )
317                         {
318                                 pendingInline.push( element );
319                                 return;
320                         }
321
322                         var currentName = currentNode.name;
323
324                         var currentDtd = currentName
325                                 && ( CKEDITOR.dtd[ currentName ]
326                                         || ( currentNode._.isBlockLike ? CKEDITOR.dtd.div : CKEDITOR.dtd.span ) );
327
328                         // If the element cannot be child of the current element.
329                         if ( currentDtd && !currentDtd[ tagName ] )
330                         {
331                                 var reApply = false,
332                                         addPoint;   // New position to start adding nodes.
333
334                                 // If the element name is the same as the current element name,
335                                 // then just close the current one and append the new one to the
336                                 // parent. This situation usually happens with <p>, <li>, <dt> and
337                                 // <dd>, specially in IE. Do not enter in this if block in this case.
338                                 if ( tagName == currentName )
339                                         addElement( currentNode, currentNode.parent );
340                                 else if ( tagName in CKEDITOR.dtd.$listItem )
341                                 {
342                                         parser.onTagOpen( 'ul', {} );
343                                         addPoint = currentNode;
344                                         reApply = true;
345                                 }
346                                 else
347                                 {
348                                         addElement( currentNode, currentNode.parent );
349
350                                         // The current element is an inline element, which
351                                         // cannot hold the new one. Put it in the pending list,
352                                         // and try adding the new one after it.
353                                         pendingInline.unshift( currentNode );
354                                         reApply = true;
355                                 }
356
357                                 if ( addPoint )
358                                         currentNode = addPoint;
359                                 // Try adding it to the return point, or the parent element.
360                                 else
361                                         currentNode = currentNode.returnPoint || currentNode.parent;
362
363                                 if ( reApply )
364                                 {
365                                         parser.onTagOpen.apply( this, arguments );
366                                         return;
367                                 }
368                         }
369
370                         checkPending( tagName );
371                         checkPendingBrs( tagName );
372
373                         element.parent = currentNode;
374                         element.returnPoint = returnPoint;
375                         returnPoint = 0;
376
377                         if ( element.isEmpty )
378                                 addElement( element );
379                         else
380                                 currentNode = element;
381                 };
382
383                 parser.onTagClose = function( tagName )
384                 {
385                         // Check if there is any pending tag to be closed.
386                         for ( var i = pendingInline.length - 1 ; i >= 0 ; i-- )
387                         {
388                                 // If found, just remove it from the list.
389                                 if ( tagName == pendingInline[ i ].name )
390                                 {
391                                         pendingInline.splice( i, 1 );
392                                         return;
393                                 }
394                         }
395
396                         var pendingAdd = [],
397                                 newPendingInline = [],
398                                 candidate = currentNode;
399
400                         while ( candidate.type && candidate.name != tagName )
401                         {
402                                 // If this is an inline element, add it to the pending list, if we're
403                                 // really closing one of the parents element later, they will continue
404                                 // after it.
405                                 if ( !candidate._.isBlockLike )
406                                         newPendingInline.unshift( candidate );
407
408                                 // This node should be added to it's parent at this point. But,
409                                 // it should happen only if the closing tag is really closing
410                                 // one of the nodes. So, for now, we just cache it.
411                                 pendingAdd.push( candidate );
412
413                                 candidate = candidate.parent;
414                         }
415
416                         if ( candidate.type )
417                         {
418                                 // Add all elements that have been found in the above loop.
419                                 for ( i = 0 ; i < pendingAdd.length ; i++ )
420                                 {
421                                         var node = pendingAdd[ i ];
422                                         addElement( node, node.parent );
423                                 }
424
425                                 currentNode = candidate;
426
427
428                                 addElement( candidate, candidate.parent );
429
430                                 // The parent should start receiving new nodes now, except if
431                                 // addElement changed the currentNode.
432                                 if ( candidate == currentNode )
433                                         currentNode = currentNode.parent;
434
435                                 pendingInline = pendingInline.concat( newPendingInline );
436                         }
437                 };
438
439                 parser.onText = function( text )
440                 {
441                         var currentDtd = CKEDITOR.dtd[ currentNode.name ];
442                         if ( !currentDtd || currentDtd[ '#' ] )
443                         {
444                                 checkPendingBrs();
445                                 checkPending();
446
447                                 text.replace(/([\r\n])|[^\r\n]*/g, function( piece, lineBreak )
448                                 {
449                                         if ( lineBreak !== undefined && lineBreak.length )
450                                                 pendingBrs++;
451                                         else if ( piece.length )
452                                         {
453                                                 var lastIndex = 0;
454
455                                                 // Create smiley from text emotion.
456                                                 piece.replace( smileyRegExp, function( match, index )
457                                                 {
458                                                         addElement( new CKEDITOR.htmlParser.text( piece.substring( lastIndex, index ) ), currentNode );
459                                                         addElement( new CKEDITOR.htmlParser.element( 'smiley', { 'desc': smileyReverseMap[ match ] } ), currentNode );
460                                                         lastIndex = index + match.length;
461                                                 });
462
463                                                 if ( lastIndex != piece.length )
464                                                         addElement( new CKEDITOR.htmlParser.text( piece.substring( lastIndex, piece.length ) ), currentNode );
465                                         }
466                                 });
467                         }
468                 };
469
470                 // Parse it.
471                 parser.parse( CKEDITOR.tools.htmlEncode( source ) );
472
473                 // Close all hanging nodes.
474                 while ( currentNode.type )
475                 {
476                         var parent = currentNode.parent,
477                                 node = currentNode;
478
479                         addElement( node, parent );
480                         currentNode = parent;
481                 }
482
483                 return fragment;
484         };
485
486         CKEDITOR.htmlParser.BBCodeWriter = CKEDITOR.tools.createClass(
487         {
488                 $ : function()
489                 {
490                         this._ =
491                         {
492                                 output : [],
493                                 rules : []
494                         };
495
496                         // List and list item.
497                         this.setRules( 'list',
498                    {
499                            breakBeforeOpen : 1,
500                            breakAfterOpen : 1,
501                            breakBeforeClose : 1,
502                            breakAfterClose : 1
503                    } );
504
505                         this.setRules( '*',
506                    {
507                            breakBeforeOpen : 1,
508                            breakAfterOpen : 0,
509                            breakBeforeClose : 1,
510                            breakAfterClose : 0
511                    } );
512
513                         this.setRules( 'quote',
514                    {
515                            breakBeforeOpen : 1,
516                            breakAfterOpen : 0,
517                            breakBeforeClose : 0,
518                            breakAfterClose : 1
519                    } );
520                 },
521
522                 proto :
523                 {
524                         /**
525                          * Sets formatting rules for a given tag. The possible rules are:
526                          * <ul>
527                          *      <li><b>breakBeforeOpen</b>: break line before the opener tag for this element.</li>
528                          *      <li><b>breakAfterOpen</b>: break line after the opener tag for this element.</li>
529                          *      <li><b>breakBeforeClose</b>: break line before the closer tag for this element.</li>
530                          *      <li><b>breakAfterClose</b>: break line after the closer tag for this element.</li>
531                          * </ul>
532                          *
533                          * All rules default to "false". Each call to the function overrides
534                          * already present rules, leaving the undefined untouched.
535                          *
536                          * @param {String} tagName The tag name to which set the rules.
537                          * @param {Object} rules An object containing the element rules.
538                          * @example
539                          * // Break line before and after "img" tags.
540                          * writer.setRules( 'list',
541                          *     {
542                          *         breakBeforeOpen : true
543                          *         breakAfterOpen : true
544                          *     });
545                          */
546                         setRules : function( tagName, rules )
547                         {
548                                 var currentRules = this._.rules[ tagName ];
549
550                                 if ( currentRules )
551                                         CKEDITOR.tools.extend( currentRules, rules, true );
552                                 else
553                                         this._.rules[ tagName ] = rules;
554                         },
555
556                         getRule : function( tagName, ruleName )
557                         {
558                                 return this._.rules[ tagName ] && this._.rules[ tagName ][ ruleName ];
559                         },
560
561                         openTag : function( tag, attributes )
562                         {
563                                 if ( tag in bbcodeMap )
564                                 {
565                                         if ( this.getRule( tag, 'breakBeforeOpen' ) )
566                                                 this.lineBreak( 1 );
567
568                                         this.write( '[', tag );
569                                         var option = attributes.option;
570                                         option && this.write( '=', option );
571                                         this.write( ']' );
572
573                                         if ( this.getRule( tag, 'breakAfterOpen' ) )
574                                                 this.lineBreak( 1 );
575                                 }
576                                 else if ( tag == 'br' )
577                                         this._.output.push( '\n' );
578                         },
579
580                         openTagClose : function() { },
581                         attribute : function() { },
582
583                         closeTag : function( tag )
584                         {
585                                 if ( tag in bbcodeMap )
586                                 {
587                                         if ( this.getRule( tag, 'breakBeforeClose' ) )
588                                                 this.lineBreak( 1 );
589
590                                         tag != '*' && this.write( '[/', tag, ']' );
591
592                                         if ( this.getRule( tag, 'breakAfterClose' ) )
593                                                 this.lineBreak( 1 );
594                                 }
595                         },
596
597                         text : function( text )
598                         {
599                                 this.write( text );
600                         },
601
602                         /**
603                          * Writes a comment.
604                          * @param {String} comment The comment text.
605                          * @example
606                          * // Writes "&lt;!-- My comment --&gt;".
607                          * writer.comment( ' My comment ' );
608                          */
609                         comment : function() {},
610
611                         /*
612                         * Output line-break for formatting.
613                          */
614                         lineBreak : function()
615                         {
616                                 // Avoid line break when:
617                                 // 1) Previous tag already put one.
618                                 // 2) We're at output start.
619                                 if ( !this._.hasLineBreak && this._.output.length )
620                                 {
621                                         this.write( '\n' );
622                                         this._.hasLineBreak = 1;
623                                 }
624                         },
625
626                         write : function()
627                         {
628                                 this._.hasLineBreak = 0;
629                                 var data = Array.prototype.join.call( arguments, '' );
630                                 this._.output.push( data );
631                         },
632
633                         reset : function()
634                         {
635                                 this._.output = [];
636                                 this._.hasLineBreak = 0;
637                         },
638
639                         getHtml : function( reset )
640                         {
641                                 var bbcode = this._.output.join( '' );
642
643                                 if ( reset )
644                                         this.reset();
645
646                                 return decodeHtml ( bbcode );
647                         }
648                 }
649         });
650
651         var BBCodeWriter = new CKEDITOR.htmlParser.BBCodeWriter();
652
653         CKEDITOR.plugins.add( 'bbcode',
654           {
655                   requires : [ 'htmldataprocessor', 'entities' ],
656                   beforeInit : function( editor )
657                   {
658                           // Adapt some critical editor configuration for better support
659                           // of BBCode environment.
660                           var config = editor.config;
661                           CKEDITOR.tools.extend( config,
662                           {
663                                   enterMode : CKEDITOR.ENTER_BR,
664                                   basicEntities: false,
665                                   entities : false,
666                                   fillEmptyBlocks : false
667                           }, true );
668                   },
669                   init : function( editor )
670                   {
671                           var config = editor.config;
672
673                           function BBCodeToHtml( code )
674                           {
675                                   var fragment = CKEDITOR.htmlParser.fragment.fromBBCode( code ),
676                                                   writer = new CKEDITOR.htmlParser.basicWriter();
677
678                                   fragment.writeHtml( writer, dataFilter );
679                                   return writer.getHtml( true );
680                           }
681
682                           var dataFilter = new CKEDITOR.htmlParser.filter();
683                           dataFilter.addRules(
684                           {
685                                   elements :
686                                   {
687                                           'blockquote' : function( element )
688                                           {
689                                                   var quoted = new CKEDITOR.htmlParser.element( 'div' );
690                                                   quoted.children = element.children;
691                                                   element.children = [ quoted ];
692                                                   var citeText = element.attributes.cite;
693                                                   if ( citeText )
694                                                   {
695                                                           var cite = new CKEDITOR.htmlParser.element( 'cite' );
696                                                           cite.add( new CKEDITOR.htmlParser.text( citeText.replace( /^"|"$/g, '' ) ) );
697                                                           delete element.attributes.cite;
698                                                           element.children.unshift( cite );
699                                                   }
700                                           },
701                                           'span' : function( element )
702                                           {
703                                                   var bbcode;
704                                                   if ( ( bbcode = element.attributes.bbcode ) )
705                                                   {
706                                                           if ( bbcode == 'img' )
707                                                           {
708                                                                   element.name = 'img';
709                                                                   element.attributes.src = element.children[ 0 ].value;
710                                                                   element.children = [];
711                                                           }
712                                                           else if ( bbcode == 'email' )
713                                                           {
714                                                                   element.name = 'a';
715                                                                   element.attributes.href = 'mailto:' + element.children[ 0 ].value;
716                                                           }
717
718                                                           delete element.attributes.bbcode;
719                                                   }
720                                           },
721                                           'ol' : function ( element )
722                                           {
723                                                   if ( element.attributes.listType )
724                                                   {
725                                                           if ( element.attributes.listType != 'decimal' )
726                                                                   element.attributes.style = 'list-style-type:' + element.attributes.listType;
727                                                   }
728                                                   else
729                                                           element.name = 'ul';
730
731                                                   delete element.attributes.listType;
732                                           },
733                                           a : function( element )
734                                           {
735                                                   if ( !element.attributes.href )
736                                                           element.attributes.href = element.children[ 0 ].value;
737                                           },
738                                           'smiley' : function( element )
739                                           {
740                                                         element.name = 'img';
741
742                                                         var description = element.attributes.desc,
743                                                                 image = config.smiley_images[ CKEDITOR.tools.indexOf( config.smiley_descriptions, description ) ],
744                                                                 src = CKEDITOR.tools.htmlEncode( config.smiley_path + image );
745
746                                                   element.attributes =
747                                                   {
748                                                           src : src,
749                                                           'data-cke-saved-src' : src,
750                                                           title :  description,
751                                                           alt : description
752                                                   };
753                                           }
754                                   }
755                           } );
756
757                           editor.dataProcessor.htmlFilter.addRules(
758                           {
759                                 elements :
760                                 {
761                                         $ : function( element )
762                                         {
763                                                 var attributes = element.attributes,
764                                                                 style = parseStyleText( attributes.style ),
765                                                                 value;
766
767                                                 var tagName = element.name;
768                                                 if ( tagName in convertMap )
769                                                         tagName = convertMap[ tagName ];
770                                                 else if ( tagName == 'span' )
771                                                 {
772                                                         if ( ( value = style.color ) )
773                                                         {
774                                                                 tagName = 'color';
775                                                                 value = RGBToHex( value );
776                                                         }
777                                                         else if ( ( value = style[ 'font-size' ] ) )
778                                                         {
779                                                                 var percentValue = value.match( /(\d+)%$/ );
780                                                                 if ( percentValue )
781                                                                 {
782                                                                         value = percentValue[ 1 ];
783                                                                         tagName = 'size';
784                                                                 }
785                                                         }
786                                                 }
787                                                 else if ( tagName == 'ol' || tagName == 'ul' )
788                                                 {
789                                                         if ( ( value = style[ 'list-style-type'] ) )
790                                                         {
791                                                                 switch ( value )
792                                                                 {
793                                                                         case 'lower-alpha':
794                                                                                 value = 'a';
795                                                                                 break;
796                                                                         case 'upper-alpha':
797                                                                                 value = 'A';
798                                                                                 break;
799                                                                 }
800                                                         }
801                                                         else if ( tagName == 'ol' )
802                                                                 value = 1;
803
804                                                         tagName = 'list';
805                                                 }
806                                                 else if ( tagName == 'blockquote' )
807                                                 {
808                                                         try
809                                                         {
810                                                                 var cite = element.children[ 0 ],
811                                                                                 quoted = element.children[ 1 ],
812                                                                                 citeText = cite.name == 'cite' && cite.children[ 0 ].value;
813
814                                                                 if ( citeText )
815                                                                 {
816                                                                         value = '"' + citeText + '"';
817                                                                         element.children = quoted.children;
818                                                                 }
819
820                                                         }
821                                                         catch( er )
822                                                         {
823                                                         }
824
825                                                         tagName = 'quote';
826                                                 }
827                                                 else if ( tagName == 'a' )
828                                                 {
829                                                         if ( ( value = attributes.href ) )
830                                                         {
831                                                                 if ( value.indexOf( 'mailto:' ) !== -1 )
832                                                                 {
833                                                                         tagName = 'email';
834                                                                         // [email] should have a single text child with email address.
835                                                                         element.children = [ new CKEDITOR.htmlParser.text( value.replace( 'mailto:', '' ) ) ];
836                                                                         value = '';
837                                                                 }
838                                                                 else
839                                                                 {
840                                                                         var singleton = element.children.length == 1 && element.children[ 0 ];
841                                                                         if ( singleton
842                                                                                         && singleton.type == CKEDITOR.NODE_TEXT
843                                                                                         && singleton.value == value )
844                                                                                 value = '';
845
846                                                                         tagName = 'url';
847                                                                 }
848                                                         }
849                                                 }
850                                                 else if ( tagName == 'img' )
851                                                 {
852                                                         element.isEmpty = 0;
853
854                                                         // Translate smiley (image) to text emotion.
855                                                         var src = attributes[ 'data-cke-saved-src' ];
856                                                         if ( src && src.indexOf( editor.config.smiley_path ) != -1 )
857                                                                 return new CKEDITOR.htmlParser.text( smileyMap[ attributes.alt ] );
858                                                         else
859                                                                 element.children = [ new CKEDITOR.htmlParser.text( src ) ];
860                                                 }
861
862                                                 element.name = tagName;
863                                                 value && ( element.attributes.option = value );
864
865                                                 return null;
866                                         },
867
868                                         // Remove any bogus br from the end of a pseudo block,
869                                         // e.g. <div>some text<br /><p>paragraph</p></div>
870                                         br : function( element )
871                                         {
872                                                 var next = element.next;
873                                                 if ( next && next.name in blockLikeTags )
874                                                         return false;
875                                         }
876                                 }
877                           }, 1 );
878
879                           editor.dataProcessor.writer = BBCodeWriter;
880
881                           editor.on( 'beforeSetMode', function( evt )
882                           {
883                                   evt.removeListener();
884                                   var wysiwyg = editor._.modes[ 'wysiwyg' ];
885                                   wysiwyg.loadData = CKEDITOR.tools.override( wysiwyg.loadData, function( org )
886                                   {
887                                           return function( data )
888                                           {
889                                                   return ( org.call( this, BBCodeToHtml( data ) ) );
890                                           };
891                                   } );
892                           } );
893                   },
894
895                   afterInit : function( editor )
896                   {
897                           var filters;
898                           if ( editor._.elementsPath  )
899                           {
900                                   // Eliminate irrelevant elements from displaying, e.g body and p.
901                                   if ( ( filters = editor._.elementsPath.filters ) )
902                                         filters.push( function( element )
903                                                 {
904                                                         var htmlName = element.getName(),
905                                                                 name = tagnameMap[ htmlName ] || false;
906
907                                                         // Specialized anchor presents as email.
908                                                         if ( name == 'link' && element.getAttribute( 'href' ).indexOf( 'mailto:' ) === 0 )
909                                                                 name = 'email';
910                                                         // Styled span could be either size or color.
911                                                         else if ( htmlName == 'span' )
912                                                         {
913                                                                 if ( element.getStyle( 'font-size' ) )
914                                                                         name = 'size';
915                                                                 else if ( element.getStyle( 'color' ) )
916                                                                         name = 'color';
917                                                         }
918                                                         else if ( name == 'img' )
919                                                         {
920                                                                 var src = element.data( 'cke-saved-src' );
921                                                                 if ( src && src.indexOf( editor.config.smiley_path ) === 0 )
922                                                                         name = 'smiley';
923                                                         }
924
925                                                         return name;
926                                                 });
927                           }
928                   }
929           } );
930
931 })();