2 Copyright (c) 2003-2011, CKSource - Frederico Knabben. All rights reserved.
3 For licensing, see LICENSE.html or http://ckeditor.com/license
6 CKEDITOR.dialog.add( 'specialchar', function( editor )
9 * Simulate "this" of a dialog for non-dialog events.
10 * @type {CKEDITOR.dialog}
13 lang = editor.lang.specialChar;
15 var onChoice = function( evt )
19 target = evt.data.getTarget();
21 target = new CKEDITOR.dom.element( evt );
23 if ( target.getName() == 'a' && ( value = target.getChild( 0 ).getHtml() ) )
25 target.removeClass( "cke_light_background" );
28 editor.insertHtml( value );
32 var onClick = CKEDITOR.tools.addFunction( onChoice );
36 var onFocus = function( evt, target )
39 target = target || evt.data.getTarget();
41 if ( target.getName() == 'span' )
42 target = target.getParent();
44 if ( target.getName() == 'a' && ( value = target.getChild( 0 ).getHtml() ) )
46 // Trigger blur manually if there is focused node.
48 onBlur( null, focusedNode );
50 var htmlPreview = dialog.getContentElement( 'info', 'htmlPreview' ).getElement();
52 dialog.getContentElement( 'info', 'charPreview' ).getElement().setHtml( value );
53 htmlPreview.setHtml( CKEDITOR.tools.htmlEncode( value ) );
54 target.getParent().addClass( "cke_light_background" );
56 // Memorize focused node.
61 var onBlur = function( evt, target )
63 target = target || evt.data.getTarget();
65 if ( target.getName() == 'span' )
66 target = target.getParent();
68 if ( target.getName() == 'a' )
70 dialog.getContentElement( 'info', 'charPreview' ).getElement().setHtml( ' ' );
71 dialog.getContentElement( 'info', 'htmlPreview' ).getElement().setHtml( ' ' );
72 target.getParent().removeClass( "cke_light_background" );
74 focusedNode = undefined;
78 var onKeydown = CKEDITOR.tools.addFunction( function( ev )
80 ev = new CKEDITOR.dom.event( ev );
82 // Get an Anchor element.
83 var element = ev.getTarget();
84 var relative, nodeToMove;
85 var keystroke = ev.getKeystroke(),
86 rtl = editor.lang.dir == 'rtl';
93 if ( ( relative = element.getParent().getParent().getPrevious() ) )
95 nodeToMove = relative.getChild( [element.getParent().getIndex(), 0] );
97 onBlur( null, element );
98 onFocus( null, nodeToMove );
105 if ( ( relative = element.getParent().getParent().getNext() ) )
107 nodeToMove = relative.getChild( [ element.getParent().getIndex(), 0 ] );
108 if ( nodeToMove && nodeToMove.type == 1 )
111 onBlur( null, element );
112 onFocus( null, nodeToMove );
118 // ENTER is already handled as onClick
120 onChoice( { data: ev } );
129 if ( ( relative = element.getParent().getNext() ) )
131 nodeToMove = relative.getChild( 0 );
132 if ( nodeToMove.type == 1 )
135 onBlur( null, element );
136 onFocus( null, nodeToMove );
137 ev.preventDefault( true );
140 onBlur( null, element );
143 else if ( ( relative = element.getParent().getParent().getNext() ) )
145 nodeToMove = relative.getChild( [ 0, 0 ] );
146 if ( nodeToMove && nodeToMove.type == 1 )
149 onBlur( null, element );
150 onFocus( null, nodeToMove );
151 ev.preventDefault( true );
154 onBlur( null, element );
161 case CKEDITOR.SHIFT + 9 :
163 if ( ( relative = element.getParent().getPrevious() ) )
165 nodeToMove = relative.getChild( 0 );
167 onBlur( null, element );
168 onFocus( null, nodeToMove );
169 ev.preventDefault( true );
172 else if ( ( relative = element.getParent().getParent().getPrevious() ) )
174 nodeToMove = relative.getLast().getChild( 0 );
176 onBlur( null, element );
177 onFocus( null, nodeToMove );
178 ev.preventDefault( true );
181 onBlur( null, element );
184 // Do not stop not handled events.
193 buttons : [ CKEDITOR.dialog.cancelButton ],
197 var columns = this.definition.charColumns,
198 extraChars = editor.config.extraSpecialChars,
199 chars = editor.config.specialChars;
201 var charsTableLabel = CKEDITOR.tools.getNextId() + '_specialchar_table_label';
202 var html = [ '<table role="listbox" aria-labelledby="' + charsTableLabel + '"' +
203 ' style="width: 320px; height: 100%; border-collapse: separate;"' +
204 ' align="center" cellspacing="2" cellpadding="2" border="0">' ];
213 html.push( '<tr>' ) ;
215 for ( var j = 0 ; j < columns ; j++, i++ )
217 if ( ( character = chars[ i ] ) )
221 if ( character instanceof Array )
223 charDesc = character[ 1 ];
224 character = character[ 0 ];
228 var _tmpName = character.toLowerCase().replace( '&', '' ).replace( ';', '' ).replace( '#', '' );
230 // Use character in case description unavailable.
231 charDesc = lang[ _tmpName ] || character;
234 var charLabelId = 'cke_specialchar_label_' + i + '_' + CKEDITOR.tools.getNextNumber();
237 '<td class="cke_dark_background" style="cursor: default" role="presentation">' +
238 '<a href="javascript: void(0);" role="option"' +
239 ' aria-posinset="' + ( i +1 ) + '"',
240 ' aria-setsize="' + size + '"',
241 ' aria-labelledby="' + charLabelId + '"',
242 ' style="cursor: inherit; display: block; height: 1.25em; margin-top: 0.25em; text-align: center;" title="', CKEDITOR.tools.htmlEncode( charDesc ), '"' +
243 ' onkeydown="CKEDITOR.tools.callFunction( ' + onKeydown + ', event, this )"' +
244 ' onclick="CKEDITOR.tools.callFunction(' + onClick + ', this); return false;"' +
246 '<span style="margin: 0 auto;cursor: inherit">' +
249 '<span class="cke_voice_label" id="' + charLabelId + '">' +
254 html.push( '<td class="cke_dark_background"> ' );
256 html.push( '</td>' );
258 html.push( '</tr>' );
261 html.push( '</tbody></table>', '<span id="' + charsTableLabel + '" class="cke_voice_label">' + lang.options +'</span>' );
263 this.getContentElement( 'info', 'charContainer' ).getElement().setHtml( html.join( '' ) );
268 label : editor.lang.common.generalTab,
269 title : editor.lang.common.generalTab,
276 widths : [ '320px', '90px' ],
281 id : 'charContainer',
283 onMouseover : onFocus,
287 var firstChar = this.getElement().getElementsByTag( 'a' ).getItem( 0 );
288 setTimeout( function()
291 onFocus( null, firstChar );
296 var firstChar = this.getElement().getChild( [ 0, 0, 0, 0, 0 ] );
297 setTimeout( function()
300 onFocus( null, firstChar );
303 onLoad : function( event )
305 dialog = event.sender;
326 className : 'cke_dark_background',
327 style : 'border:1px solid #eeeeee;font-size:28px;height:40px;width:70px;padding-top:9px;font-family:\'Microsoft Sans Serif\',Arial,Helvetica,Verdana;text-align:center;',
328 html : '<div> </div>'
333 className : 'cke_dark_background',
334 style : 'border:1px solid #eeeeee;font-size:14px;height:20px;width:70px;padding-top:2px;font-family:\'Microsoft Sans Serif\',Arial,Helvetica,Verdana;text-align:center;',
335 html : '<div> </div>'