123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530 |
- /**
- * @license Copyright (c) 2003-2015, CKSource - Frederico Knabben. All rights reserved.
- * For licensing, see LICENSE.md or http://ckeditor.com/license
- */
- /**
- * @fileOverview The Dialog User Interface plugin.
- */
- CKEDITOR.plugins.add( 'dialogui', {
- onLoad: function() {
- var initPrivateObject = function( elementDefinition ) {
- this._ || ( this._ = {} );
- this._[ 'default' ] = this._.initValue = elementDefinition[ 'default' ] || '';
- this._.required = elementDefinition.required || false;
- var args = [ this._ ];
- for ( var i = 1; i < arguments.length; i++ )
- args.push( arguments[ i ] );
- args.push( true );
- CKEDITOR.tools.extend.apply( CKEDITOR.tools, args );
- return this._;
- },
- textBuilder = {
- build: function( dialog, elementDefinition, output ) {
- return new CKEDITOR.ui.dialog.textInput( dialog, elementDefinition, output );
- }
- },
- commonBuilder = {
- build: function( dialog, elementDefinition, output ) {
- return new CKEDITOR.ui.dialog[ elementDefinition.type ]( dialog, elementDefinition, output );
- }
- },
- containerBuilder = {
- build: function( dialog, elementDefinition, output ) {
- var children = elementDefinition.children,
- child,
- childHtmlList = [],
- childObjList = [];
- for ( var i = 0;
- ( i < children.length && ( child = children[ i ] ) ); i++ ) {
- var childHtml = [];
- childHtmlList.push( childHtml );
- childObjList.push( CKEDITOR.dialog._.uiElementBuilders[ child.type ].build( dialog, child, childHtml ) );
- }
- return new CKEDITOR.ui.dialog[ elementDefinition.type ]( dialog, childObjList, childHtmlList, output, elementDefinition );
- }
- },
- commonPrototype = {
- isChanged: function() {
- return this.getValue() != this.getInitValue();
- },
- reset: function( noChangeEvent ) {
- this.setValue( this.getInitValue(), noChangeEvent );
- },
- setInitValue: function() {
- this._.initValue = this.getValue();
- },
- resetInitValue: function() {
- this._.initValue = this._[ 'default' ];
- },
- getInitValue: function() {
- return this._.initValue;
- }
- },
- commonEventProcessors = CKEDITOR.tools.extend( {}, CKEDITOR.ui.dialog.uiElement.prototype.eventProcessors, {
- onChange: function( dialog, func ) {
- if ( !this._.domOnChangeRegistered ) {
- dialog.on( 'load', function() {
- this.getInputElement().on( 'change', function() {
- // Make sure 'onchange' doesn't get fired after dialog closed. (#5719)
- if ( !dialog.parts.dialog.isVisible() )
- return;
- this.fire( 'change', { value: this.getValue() } );
- }, this );
- }, this );
- this._.domOnChangeRegistered = true;
- }
- this.on( 'change', func );
- }
- }, true ),
- eventRegex = /^on([A-Z]\w+)/,
- cleanInnerDefinition = function( def ) {
- // An inner UI element should not have the parent's type, title or events.
- for ( var i in def ) {
- if ( eventRegex.test( i ) || i == 'title' || i == 'type' )
- delete def[ i ];
- }
- return def;
- },
- // @context {CKEDITOR.dialog.uiElement} UI element (textarea or textInput)
- // @param {CKEDITOR.dom.event} evt
- toggleBidiKeyUpHandler = function( evt ) {
- var keystroke = evt.data.getKeystroke();
- // ALT + SHIFT + Home for LTR direction.
- if ( keystroke == CKEDITOR.SHIFT + CKEDITOR.ALT + 36 )
- this.setDirectionMarker( 'ltr' );
- // ALT + SHIFT + End for RTL direction.
- else if ( keystroke == CKEDITOR.SHIFT + CKEDITOR.ALT + 35 )
- this.setDirectionMarker( 'rtl' );
- };
- CKEDITOR.tools.extend( CKEDITOR.ui.dialog, {
- /**
- * Base class for all dialog window elements with a textual label on the left.
- *
- * @class CKEDITOR.ui.dialog.labeledElement
- * @extends CKEDITOR.ui.dialog.uiElement
- * @constructor Creates a labeledElement class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `label` (Required) The label string.
- * * `labelLayout` (Optional) Put 'horizontal' here if the
- * label element is to be laid out horizontally. Otherwise a vertical
- * layout will be used.
- * * `widths` (Optional) This applies only to horizontal
- * layouts — a two-element array of lengths to specify the widths of the
- * label and the content element.
- * * `role` (Optional) Value for the `role` attribute.
- * * `includeLabel` (Optional) If set to `true`, the `aria-labelledby` attribute
- * will be included.
- *
- * @param {Array} htmlList The list of HTML code to output to.
- * @param {Function} contentHtml
- * A function returning the HTML code string to be added inside the content
- * cell.
- */
- labeledElement: function( dialog, elementDefinition, htmlList, contentHtml ) {
- if ( arguments.length < 4 )
- return;
- var _ = initPrivateObject.call( this, elementDefinition );
- _.labelId = CKEDITOR.tools.getNextId() + '_label';
- this._.children = [];
- var innerHTML = function() {
- var html = [],
- requiredClass = elementDefinition.required ? ' cke_required' : '';
- if ( elementDefinition.labelLayout != 'horizontal' ) {
- html.push(
- '<label class="cke_dialog_ui_labeled_label' + requiredClass + '" ', ' id="' + _.labelId + '"',
- ( _.inputId ? ' for="' + _.inputId + '"' : '' ),
- ( elementDefinition.labelStyle ? ' style="' + elementDefinition.labelStyle + '"' : '' ) + '>',
- elementDefinition.label,
- '</label>',
- '<div class="cke_dialog_ui_labeled_content"',
- ( elementDefinition.controlStyle ? ' style="' + elementDefinition.controlStyle + '"' : '' ),
- ' role="presentation">',
- contentHtml.call( this, dialog, elementDefinition ),
- '</div>' );
- } else {
- var hboxDefinition = {
- type: 'hbox',
- widths: elementDefinition.widths,
- padding: 0,
- children: [ {
- type: 'html',
- html: '<label class="cke_dialog_ui_labeled_label' + requiredClass + '"' +
- ' id="' + _.labelId + '"' +
- ' for="' + _.inputId + '"' +
- ( elementDefinition.labelStyle ? ' style="' + elementDefinition.labelStyle + '"' : '' ) + '>' +
- CKEDITOR.tools.htmlEncode( elementDefinition.label ) +
- '</label>'
- },
- {
- type: 'html',
- html: '<span class="cke_dialog_ui_labeled_content"' + ( elementDefinition.controlStyle ? ' style="' + elementDefinition.controlStyle + '"' : '' ) + '>' +
- contentHtml.call( this, dialog, elementDefinition ) +
- '</span>'
- } ]
- };
- CKEDITOR.dialog._.uiElementBuilders.hbox.build( dialog, hboxDefinition, html );
- }
- return html.join( '' );
- };
- var attributes = { role: elementDefinition.role || 'presentation' };
- if ( elementDefinition.includeLabel )
- attributes[ 'aria-labelledby' ] = _.labelId;
- CKEDITOR.ui.dialog.uiElement.call( this, dialog, elementDefinition, htmlList, 'div', null, attributes, innerHTML );
- },
- /**
- * A text input with a label. This UI element class represents both the
- * single-line text inputs and password inputs in dialog boxes.
- *
- * @class CKEDITOR.ui.dialog.textInput
- * @extends CKEDITOR.ui.dialog.labeledElement
- * @constructor Creates a textInput class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `default` (Optional) The default value.
- * * `validate` (Optional) The validation function.
- * * `maxLength` (Optional) The maximum length of text box contents.
- * * `size` (Optional) The size of the text box. This is
- * usually overridden by the size defined by the skin, though.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- textInput: function( dialog, elementDefinition, htmlList ) {
- if ( arguments.length < 3 )
- return;
- initPrivateObject.call( this, elementDefinition );
- var domId = this._.inputId = CKEDITOR.tools.getNextId() + '_textInput',
- attributes = { 'class': 'cke_dialog_ui_input_' + elementDefinition.type, id: domId, type: elementDefinition.type };
- // Set the validator, if any.
- if ( elementDefinition.validate )
- this.validate = elementDefinition.validate;
- // Set the max length and size.
- if ( elementDefinition.maxLength )
- attributes.maxlength = elementDefinition.maxLength;
- if ( elementDefinition.size )
- attributes.size = elementDefinition.size;
- if ( elementDefinition.inputStyle )
- attributes.style = elementDefinition.inputStyle;
- // If user presses Enter in a text box, it implies clicking OK for the dialog.
- var me = this,
- keyPressedOnMe = false;
- dialog.on( 'load', function() {
- me.getInputElement().on( 'keydown', function( evt ) {
- if ( evt.data.getKeystroke() == 13 )
- keyPressedOnMe = true;
- } );
- // Lower the priority this 'keyup' since 'ok' will close the dialog.(#3749)
- me.getInputElement().on( 'keyup', function( evt ) {
- if ( evt.data.getKeystroke() == 13 && keyPressedOnMe ) {
- dialog.getButton( 'ok' ) && setTimeout( function() {
- dialog.getButton( 'ok' ).click();
- }, 0 );
- keyPressedOnMe = false;
- }
- if ( me.bidi )
- toggleBidiKeyUpHandler.call( me, evt );
- }, null, null, 1000 );
- } );
- var innerHTML = function() {
- // IE BUG: Text input fields in IE at 100% would exceed a <td> or inline
- // container's width, so need to wrap it inside a <div>.
- var html = [ '<div class="cke_dialog_ui_input_', elementDefinition.type, '" role="presentation"' ];
- if ( elementDefinition.width )
- html.push( 'style="width:' + elementDefinition.width + '" ' );
- html.push( '><input ' );
- attributes[ 'aria-labelledby' ] = this._.labelId;
- this._.required && ( attributes[ 'aria-required' ] = this._.required );
- for ( var i in attributes )
- html.push( i + '="' + attributes[ i ] + '" ' );
- html.push( ' /></div>' );
- return html.join( '' );
- };
- CKEDITOR.ui.dialog.labeledElement.call( this, dialog, elementDefinition, htmlList, innerHTML );
- },
- /**
- * A text area with a label at the top or on the left.
- *
- * @class CKEDITOR.ui.dialog.textarea
- * @extends CKEDITOR.ui.dialog.labeledElement
- * @constructor Creates a textarea class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- *
- * The element definition. Accepted fields:
- *
- * * `rows` (Optional) The number of rows displayed.
- * Defaults to 5 if not defined.
- * * `cols` (Optional) The number of cols displayed.
- * Defaults to 20 if not defined. Usually overridden by skins.
- * * `default` (Optional) The default value.
- * * `validate` (Optional) The validation function.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- textarea: function( dialog, elementDefinition, htmlList ) {
- if ( arguments.length < 3 )
- return;
- initPrivateObject.call( this, elementDefinition );
- var me = this,
- domId = this._.inputId = CKEDITOR.tools.getNextId() + '_textarea',
- attributes = {};
- if ( elementDefinition.validate )
- this.validate = elementDefinition.validate;
- // Generates the essential attributes for the textarea tag.
- attributes.rows = elementDefinition.rows || 5;
- attributes.cols = elementDefinition.cols || 20;
- attributes[ 'class' ] = 'cke_dialog_ui_input_textarea ' + ( elementDefinition[ 'class' ] || '' );
- if ( typeof elementDefinition.inputStyle != 'undefined' )
- attributes.style = elementDefinition.inputStyle;
- if ( elementDefinition.dir )
- attributes.dir = elementDefinition.dir;
- if ( me.bidi ) {
- dialog.on( 'load', function() {
- me.getInputElement().on( 'keyup', toggleBidiKeyUpHandler );
- }, me );
- }
- var innerHTML = function() {
- attributes[ 'aria-labelledby' ] = this._.labelId;
- this._.required && ( attributes[ 'aria-required' ] = this._.required );
- var html = [ '<div class="cke_dialog_ui_input_textarea" role="presentation"><textarea id="', domId, '" ' ];
- for ( var i in attributes )
- html.push( i + '="' + CKEDITOR.tools.htmlEncode( attributes[ i ] ) + '" ' );
- html.push( '>', CKEDITOR.tools.htmlEncode( me._[ 'default' ] ), '</textarea></div>' );
- return html.join( '' );
- };
- CKEDITOR.ui.dialog.labeledElement.call( this, dialog, elementDefinition, htmlList, innerHTML );
- },
- /**
- * A single checkbox with a label on the right.
- *
- * @class CKEDITOR.ui.dialog.checkbox
- * @extends CKEDITOR.ui.dialog.uiElement
- * @constructor Creates a checkbox class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `checked` (Optional) Whether the checkbox is checked
- * on instantiation. Defaults to `false`.
- * * `validate` (Optional) The validation function.
- * * `label` (Optional) The checkbox label.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- checkbox: function( dialog, elementDefinition, htmlList ) {
- if ( arguments.length < 3 )
- return;
- var _ = initPrivateObject.call( this, elementDefinition, { 'default': !!elementDefinition[ 'default' ] } );
- if ( elementDefinition.validate )
- this.validate = elementDefinition.validate;
- var innerHTML = function() {
- var myDefinition = CKEDITOR.tools.extend(
- {},
- elementDefinition,
- {
- id: elementDefinition.id ? elementDefinition.id + '_checkbox' : CKEDITOR.tools.getNextId() + '_checkbox'
- },
- true
- ),
- html = [];
- var labelId = CKEDITOR.tools.getNextId() + '_label';
- var attributes = { 'class': 'cke_dialog_ui_checkbox_input', type: 'checkbox', 'aria-labelledby': labelId };
- cleanInnerDefinition( myDefinition );
- if ( elementDefinition[ 'default' ] )
- attributes.checked = 'checked';
- if ( typeof myDefinition.inputStyle != 'undefined' )
- myDefinition.style = myDefinition.inputStyle;
- _.checkbox = new CKEDITOR.ui.dialog.uiElement( dialog, myDefinition, html, 'input', null, attributes );
- html.push(
- ' <label id="',
- labelId,
- '" for="',
- attributes.id,
- '"' + ( elementDefinition.labelStyle ? ' style="' + elementDefinition.labelStyle + '"' : '' ) + '>',
- CKEDITOR.tools.htmlEncode( elementDefinition.label ),
- '</label>'
- );
- return html.join( '' );
- };
- CKEDITOR.ui.dialog.uiElement.call( this, dialog, elementDefinition, htmlList, 'span', null, null, innerHTML );
- },
- /**
- * A group of radio buttons.
- *
- * @class CKEDITOR.ui.dialog.radio
- * @extends CKEDITOR.ui.dialog.labeledElement
- * @constructor Creates a radio class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `default` (Required) The default value.
- * * `validate` (Optional) The validation function.
- * * `items` (Required) An array of options. Each option
- * is a one- or two-item array of format `[ 'Description', 'Value' ]`. If `'Value'`
- * is missing, then the value would be assumed to be the same as the description.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- radio: function( dialog, elementDefinition, htmlList ) {
- if ( arguments.length < 3 )
- return;
- initPrivateObject.call( this, elementDefinition );
- if ( !this._[ 'default' ] )
- this._[ 'default' ] = this._.initValue = elementDefinition.items[ 0 ][ 1 ];
- if ( elementDefinition.validate )
- this.validate = elementDefinition.validate;
- var children = [],
- me = this;
- var innerHTML = function() {
- var inputHtmlList = [],
- html = [],
- commonName = ( elementDefinition.id ? elementDefinition.id : CKEDITOR.tools.getNextId() ) + '_radio';
- for ( var i = 0; i < elementDefinition.items.length; i++ ) {
- var item = elementDefinition.items[ i ],
- title = item[ 2 ] !== undefined ? item[ 2 ] : item[ 0 ],
- value = item[ 1 ] !== undefined ? item[ 1 ] : item[ 0 ],
- inputId = CKEDITOR.tools.getNextId() + '_radio_input',
- labelId = inputId + '_label',
- inputDefinition = CKEDITOR.tools.extend( {}, elementDefinition, {
- id: inputId,
- title: null,
- type: null
- }, true ),
- labelDefinition = CKEDITOR.tools.extend( {}, inputDefinition, {
- title: title
- }, true ),
- inputAttributes = {
- type: 'radio',
- 'class': 'cke_dialog_ui_radio_input',
- name: commonName,
- value: value,
- 'aria-labelledby': labelId
- },
- inputHtml = [];
- if ( me._[ 'default' ] == value )
- inputAttributes.checked = 'checked';
- cleanInnerDefinition( inputDefinition );
- cleanInnerDefinition( labelDefinition );
- if ( typeof inputDefinition.inputStyle != 'undefined' )
- inputDefinition.style = inputDefinition.inputStyle;
- // Make inputs of radio type focusable (#10866).
- inputDefinition.keyboardFocusable = true;
- children.push( new CKEDITOR.ui.dialog.uiElement( dialog, inputDefinition, inputHtml, 'input', null, inputAttributes ) );
- inputHtml.push( ' ' );
- new CKEDITOR.ui.dialog.uiElement( dialog, labelDefinition, inputHtml, 'label', null, {
- id: labelId,
- 'for': inputAttributes.id
- }, item[ 0 ] );
- inputHtmlList.push( inputHtml.join( '' ) );
- }
- new CKEDITOR.ui.dialog.hbox( dialog, children, inputHtmlList, html );
- return html.join( '' );
- };
- // Adding a role="radiogroup" to definition used for wrapper.
- elementDefinition.role = 'radiogroup';
- elementDefinition.includeLabel = true;
- CKEDITOR.ui.dialog.labeledElement.call( this, dialog, elementDefinition, htmlList, innerHTML );
- this._.children = children;
- },
- /**
- * A button with a label inside.
- *
- * @class CKEDITOR.ui.dialog.button
- * @extends CKEDITOR.ui.dialog.uiElement
- * @constructor Creates a button class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `label` (Required) The button label.
- * * `disabled` (Optional) Set to `true` if you want the
- * button to appear in the disabled state.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- button: function( dialog, elementDefinition, htmlList ) {
- if ( !arguments.length )
- return;
- if ( typeof elementDefinition == 'function' )
- elementDefinition = elementDefinition( dialog.getParentEditor() );
- initPrivateObject.call( this, elementDefinition, { disabled: elementDefinition.disabled || false } );
- // Add OnClick event to this input.
- CKEDITOR.event.implementOn( this );
- var me = this;
- // Register an event handler for processing button clicks.
- dialog.on( 'load', function() {
- var element = this.getElement();
- ( function() {
- element.on( 'click', function( evt ) {
- me.click();
- // #9958
- evt.data.preventDefault();
- } );
- element.on( 'keydown', function( evt ) {
- if ( evt.data.getKeystroke() in { 32: 1 } ) {
- me.click();
- evt.data.preventDefault();
- }
- } );
- } )();
- element.unselectable();
- }, this );
- var outerDefinition = CKEDITOR.tools.extend( {}, elementDefinition );
- delete outerDefinition.style;
- var labelId = CKEDITOR.tools.getNextId() + '_label';
- CKEDITOR.ui.dialog.uiElement.call( this, dialog, outerDefinition, htmlList, 'a', null, {
- style: elementDefinition.style,
- href: 'javascript:void(0)', // jshint ignore:line
- title: elementDefinition.label,
- hidefocus: 'true',
- 'class': elementDefinition[ 'class' ],
- role: 'button',
- 'aria-labelledby': labelId
- }, '<span id="' + labelId + '" class="cke_dialog_ui_button">' +
- CKEDITOR.tools.htmlEncode( elementDefinition.label ) +
- '</span>' );
- },
- /**
- * A select box.
- *
- * @class CKEDITOR.ui.dialog.select
- * @extends CKEDITOR.ui.dialog.uiElement
- * @constructor Creates a button class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `default` (Required) The default value.
- * * `validate` (Optional) The validation function.
- * * `items` (Required) An array of options. Each option
- * is a one- or two-item array of format `[ 'Description', 'Value' ]`. If `'Value'`
- * is missing, then the value would be assumed to be the same as the
- * description.
- * * `multiple` (Optional) Set this to `true` if you would like
- * to have a multiple-choice select box.
- * * `size` (Optional) The number of items to display in
- * the select box.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- select: function( dialog, elementDefinition, htmlList ) {
- if ( arguments.length < 3 )
- return;
- var _ = initPrivateObject.call( this, elementDefinition );
- if ( elementDefinition.validate )
- this.validate = elementDefinition.validate;
- _.inputId = CKEDITOR.tools.getNextId() + '_select';
- var innerHTML = function() {
- var myDefinition = CKEDITOR.tools.extend(
- {},
- elementDefinition,
- {
- id: ( elementDefinition.id ? elementDefinition.id + '_select' : CKEDITOR.tools.getNextId() + '_select' )
- },
- true
- ),
- html = [],
- innerHTML = [],
- attributes = { 'id': _.inputId, 'class': 'cke_dialog_ui_input_select', 'aria-labelledby': this._.labelId };
- html.push( '<div class="cke_dialog_ui_input_', elementDefinition.type, '" role="presentation"' );
- if ( elementDefinition.width )
- html.push( 'style="width:' + elementDefinition.width + '" ' );
- html.push( '>' );
- // Add multiple and size attributes from element definition.
- if ( elementDefinition.size !== undefined )
- attributes.size = elementDefinition.size;
- if ( elementDefinition.multiple !== undefined )
- attributes.multiple = elementDefinition.multiple;
- cleanInnerDefinition( myDefinition );
- for ( var i = 0, item; i < elementDefinition.items.length && ( item = elementDefinition.items[ i ] ); i++ ) {
- innerHTML.push( '<option value="', CKEDITOR.tools.htmlEncode( item[ 1 ] !== undefined ? item[ 1 ] : item[ 0 ] ).replace( /"/g, '"' ), '" /> ', CKEDITOR.tools.htmlEncode( item[ 0 ] ) );
- }
- if ( typeof myDefinition.inputStyle != 'undefined' )
- myDefinition.style = myDefinition.inputStyle;
- _.select = new CKEDITOR.ui.dialog.uiElement( dialog, myDefinition, html, 'select', null, attributes, innerHTML.join( '' ) );
- html.push( '</div>' );
- return html.join( '' );
- };
- CKEDITOR.ui.dialog.labeledElement.call( this, dialog, elementDefinition, htmlList, innerHTML );
- },
- /**
- * A file upload input.
- *
- * @class CKEDITOR.ui.dialog.file
- * @extends CKEDITOR.ui.dialog.labeledElement
- * @constructor Creates a file class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `validate` (Optional) The validation function.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- file: function( dialog, elementDefinition, htmlList ) {
- if ( arguments.length < 3 )
- return;
- if ( elementDefinition[ 'default' ] === undefined )
- elementDefinition[ 'default' ] = '';
- var _ = CKEDITOR.tools.extend( initPrivateObject.call( this, elementDefinition ), { definition: elementDefinition, buttons: [] } );
- if ( elementDefinition.validate )
- this.validate = elementDefinition.validate;
- /** @ignore */
- var innerHTML = function() {
- _.frameId = CKEDITOR.tools.getNextId() + '_fileInput';
- var html = [
- '<iframe' +
- ' frameborder="0"' +
- ' allowtransparency="0"' +
- ' class="cke_dialog_ui_input_file"' +
- ' role="presentation"' +
- ' id="', _.frameId, '"' +
- ' title="', elementDefinition.label, '"' +
- ' src="javascript:void('
- ];
- // Support for custom document.domain on IE. (#10165)
- html.push( CKEDITOR.env.ie ?
- '(function(){' + encodeURIComponent(
- 'document.open();' +
- '(' + CKEDITOR.tools.fixDomain + ')();' +
- 'document.close();'
- ) + '})()'
- :
- '0'
- );
- html.push( ')"></iframe>' );
- return html.join( '' );
- };
- // IE BUG: Parent container does not resize to contain the iframe automatically.
- dialog.on( 'load', function() {
- var iframe = CKEDITOR.document.getById( _.frameId ),
- contentDiv = iframe.getParent();
- contentDiv.addClass( 'cke_dialog_ui_input_file' );
- } );
- CKEDITOR.ui.dialog.labeledElement.call( this, dialog, elementDefinition, htmlList, innerHTML );
- },
- /**
- * A button for submitting the file in a file upload input.
- *
- * @class CKEDITOR.ui.dialog.fileButton
- * @extends CKEDITOR.ui.dialog.button
- * @constructor Creates a fileButton class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `for` (Required) The file input's page and element ID
- * to associate with, in a two-item array format: `[ 'page_id', 'element_id' ]`.
- * * `validate` (Optional) The validation function.
- *
- * @param {Array} htmlList List of HTML code to output to.
- */
- fileButton: function( dialog, elementDefinition, htmlList ) {
- var me = this;
- if ( arguments.length < 3 )
- return;
- initPrivateObject.call( this, elementDefinition );
- if ( elementDefinition.validate )
- this.validate = elementDefinition.validate;
- var myDefinition = CKEDITOR.tools.extend( {}, elementDefinition );
- var onClick = myDefinition.onClick;
- myDefinition.className = ( myDefinition.className ? myDefinition.className + ' ' : '' ) + 'cke_dialog_ui_button';
- myDefinition.onClick = function( evt ) {
- var target = elementDefinition[ 'for' ]; // [ pageId, elementId ]
- if ( !onClick || onClick.call( this, evt ) !== false ) {
- dialog.getContentElement( target[ 0 ], target[ 1 ] ).submit();
- this.disable();
- }
- };
- dialog.on( 'load', function() {
- dialog.getContentElement( elementDefinition[ 'for' ][ 0 ], elementDefinition[ 'for' ][ 1 ] )._.buttons.push( me );
- } );
- CKEDITOR.ui.dialog.button.call( this, dialog, myDefinition, htmlList );
- },
- html: ( function() {
- var myHtmlRe = /^\s*<[\w:]+\s+([^>]*)?>/,
- theirHtmlRe = /^(\s*<[\w:]+(?:\s+[^>]*)?)((?:.|\r|\n)+)$/,
- emptyTagRe = /\/$/;
- /**
- * A dialog window element made from raw HTML code.
- *
- * @class CKEDITOR.ui.dialog.html
- * @extends CKEDITOR.ui.dialog.uiElement
- * @constructor Creates a html class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition Element definition.
- * Accepted fields:
- *
- * * `html` (Required) HTML code of this element.
- *
- * @param {Array} htmlList List of HTML code to be added to the dialog's content area.
- */
- return function( dialog, elementDefinition, htmlList ) {
- if ( arguments.length < 3 )
- return;
- var myHtmlList = [],
- myHtml,
- theirHtml = elementDefinition.html,
- myMatch, theirMatch;
- // If the HTML input doesn't contain any tags at the beginning, add a <span> tag around it.
- if ( theirHtml.charAt( 0 ) != '<' )
- theirHtml = '<span>' + theirHtml + '</span>';
- // Look for focus function in definition.
- var focus = elementDefinition.focus;
- if ( focus ) {
- var oldFocus = this.focus;
- this.focus = function() {
- ( typeof focus == 'function' ? focus : oldFocus ).call( this );
- this.fire( 'focus' );
- };
- if ( elementDefinition.isFocusable ) {
- var oldIsFocusable = this.isFocusable;
- this.isFocusable = oldIsFocusable;
- }
- this.keyboardFocusable = true;
- }
- CKEDITOR.ui.dialog.uiElement.call( this, dialog, elementDefinition, myHtmlList, 'span', null, null, '' );
- // Append the attributes created by the uiElement call to the real HTML.
- myHtml = myHtmlList.join( '' );
- myMatch = myHtml.match( myHtmlRe );
- theirMatch = theirHtml.match( theirHtmlRe ) || [ '', '', '' ];
- if ( emptyTagRe.test( theirMatch[ 1 ] ) ) {
- theirMatch[ 1 ] = theirMatch[ 1 ].slice( 0, -1 );
- theirMatch[ 2 ] = '/' + theirMatch[ 2 ];
- }
- htmlList.push( [ theirMatch[ 1 ], ' ', myMatch[ 1 ] || '', theirMatch[ 2 ] ].join( '' ) );
- };
- } )(),
- /**
- * Form fieldset for grouping dialog UI elements.
- *
- * @class CKEDITOR.ui.dialog.fieldset
- * @extends CKEDITOR.ui.dialog.uiElement
- * @constructor Creates a fieldset class instance.
- * @param {CKEDITOR.dialog} dialog Parent dialog window object.
- * @param {Array} childObjList
- * Array of {@link CKEDITOR.ui.dialog.uiElement} objects inside this container.
- * @param {Array} childHtmlList Array of HTML code that corresponds to the HTML output of all the
- * objects in childObjList.
- * @param {Array} htmlList Array of HTML code that this element will output to.
- * @param {CKEDITOR.dialog.definition.uiElement} elementDefinition
- * The element definition. Accepted fields:
- *
- * * `label` (Optional) The legend of the this fieldset.
- * * `children` (Required) An array of dialog window field definitions which will be grouped inside this fieldset.
- *
- */
- fieldset: function( dialog, childObjList, childHtmlList, htmlList, elementDefinition ) {
- var legendLabel = elementDefinition.label;
- /** @ignore */
- var innerHTML = function() {
- var html = [];
- legendLabel && html.push( '<legend' +
- ( elementDefinition.labelStyle ? ' style="' + elementDefinition.labelStyle + '"' : '' ) +
- '>' + legendLabel + '</legend>' );
- for ( var i = 0; i < childHtmlList.length; i++ )
- html.push( childHtmlList[ i ] );
- return html.join( '' );
- };
- this._ = { children: childObjList };
- CKEDITOR.ui.dialog.uiElement.call( this, dialog, elementDefinition, htmlList, 'fieldset', null, null, innerHTML );
- }
- }, true );
- CKEDITOR.ui.dialog.html.prototype = new CKEDITOR.ui.dialog.uiElement();
- /** @class CKEDITOR.ui.dialog.labeledElement */
- CKEDITOR.ui.dialog.labeledElement.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.uiElement(), {
- /**
- * Sets the label text of the element.
- *
- * @param {String} label The new label text.
- * @returns {CKEDITOR.ui.dialog.labeledElement} The current labeled element.
- */
- setLabel: function( label ) {
- var node = CKEDITOR.document.getById( this._.labelId );
- if ( node.getChildCount() < 1 )
- ( new CKEDITOR.dom.text( label, CKEDITOR.document ) ).appendTo( node );
- else
- node.getChild( 0 ).$.nodeValue = label;
- return this;
- },
- /**
- * Retrieves the current label text of the elment.
- *
- * @returns {String} The current label text.
- */
- getLabel: function() {
- var node = CKEDITOR.document.getById( this._.labelId );
- if ( !node || node.getChildCount() < 1 )
- return '';
- else
- return node.getChild( 0 ).getText();
- },
- /**
- * Defines the `onChange` event for UI element definitions.
- * @property {Object}
- */
- eventProcessors: commonEventProcessors
- }, true );
- /** @class CKEDITOR.ui.dialog.button */
- CKEDITOR.ui.dialog.button.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.uiElement(), {
- /**
- * Simulates a click to the button.
- *
- * @returns {Object} Return value of the `click` event.
- */
- click: function() {
- if ( !this._.disabled )
- return this.fire( 'click', { dialog: this._.dialog } );
- return false;
- },
- /**
- * Enables the button.
- */
- enable: function() {
- this._.disabled = false;
- var element = this.getElement();
- element && element.removeClass( 'cke_disabled' );
- },
- /**
- * Disables the button.
- */
- disable: function() {
- this._.disabled = true;
- this.getElement().addClass( 'cke_disabled' );
- },
- /**
- * Checks whether a field is visible.
- *
- * @returns {Boolean}
- */
- isVisible: function() {
- return this.getElement().getFirst().isVisible();
- },
- /**
- * Checks whether a field is enabled. Fields can be disabled by using the
- * {@link #disable} method and enabled by using the {@link #enable} method.
- *
- * @returns {Boolean}
- */
- isEnabled: function() {
- return !this._.disabled;
- },
- /**
- * Defines the `onChange` event and `onClick` for button element definitions.
- *
- * @property {Object}
- */
- eventProcessors: CKEDITOR.tools.extend( {}, CKEDITOR.ui.dialog.uiElement.prototype.eventProcessors, {
- onClick: function( dialog, func ) {
- this.on( 'click', function() {
- func.apply( this, arguments );
- } );
- }
- }, true ),
- /**
- * Handler for the element's access key up event. Simulates a click to
- * the button.
- */
- accessKeyUp: function() {
- this.click();
- },
- /**
- * Handler for the element's access key down event. Simulates a mouse
- * down to the button.
- */
- accessKeyDown: function() {
- this.focus();
- },
- keyboardFocusable: true
- }, true );
- /** @class CKEDITOR.ui.dialog.textInput */
- CKEDITOR.ui.dialog.textInput.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.labeledElement(), {
- /**
- * Gets the text input DOM element under this UI object.
- *
- * @returns {CKEDITOR.dom.element} The DOM element of the text input.
- */
- getInputElement: function() {
- return CKEDITOR.document.getById( this._.inputId );
- },
- /**
- * Puts focus into the text input.
- */
- focus: function() {
- var me = this.selectParentTab();
- // GECKO BUG: setTimeout() is needed to workaround invisible selections.
- setTimeout( function() {
- var element = me.getInputElement();
- element && element.$.focus();
- }, 0 );
- },
- /**
- * Selects all the text in the text input.
- */
- select: function() {
- var me = this.selectParentTab();
- // GECKO BUG: setTimeout() is needed to workaround invisible selections.
- setTimeout( function() {
- var e = me.getInputElement();
- if ( e ) {
- e.$.focus();
- e.$.select();
- }
- }, 0 );
- },
- /**
- * Handler for the text input's access key up event. Makes a `select()`
- * call to the text input.
- */
- accessKeyUp: function() {
- this.select();
- },
- /**
- * Sets the value of this text input object.
- *
- * uiElement.setValue( 'Blamo' );
- *
- * @param {Object} value The new value.
- * @returns {CKEDITOR.ui.dialog.textInput} The current UI element.
- */
- setValue: function( value ) {
- if ( this.bidi ) {
- var marker = value && value.charAt( 0 ),
- dir = ( marker == '\u202A' ? 'ltr' : marker == '\u202B' ? 'rtl' : null );
- if ( dir ) {
- value = value.slice( 1 );
- }
- // Set the marker or reset it (if dir==null).
- this.setDirectionMarker( dir );
- }
- if ( !value ) {
- value = '';
- }
- return CKEDITOR.ui.dialog.uiElement.prototype.setValue.apply( this, arguments );
- },
- /**
- * Gets the value of this text input object.
- *
- * @returns {String} The value.
- */
- getValue: function() {
- var value = CKEDITOR.ui.dialog.uiElement.prototype.getValue.call( this );
- if ( this.bidi && value ) {
- var dir = this.getDirectionMarker();
- if ( dir ) {
- value = ( dir == 'ltr' ? '\u202A' : '\u202B' ) + value;
- }
- }
- return value;
- },
- /**
- * Sets the text direction marker and the `dir` attribute of the input element.
- *
- * @since 4.5
- * @param {String} dir The text direction. Pass `null` to reset.
- */
- setDirectionMarker: function( dir ) {
- var inputElement = this.getInputElement();
- if ( dir ) {
- inputElement.setAttributes( {
- dir: dir,
- 'data-cke-dir-marker': dir
- } );
- // Don't remove the dir attribute if this field hasn't got the marker,
- // because the dir attribute could be set independently.
- } else if ( this.getDirectionMarker() ) {
- inputElement.removeAttributes( [ 'dir', 'data-cke-dir-marker' ] );
- }
- },
- /**
- * Gets the value of the text direction marker.
- *
- * @since 4.5
- * @returns {String} `'ltr'`, `'rtl'` or `null` if the marker is not set.
- */
- getDirectionMarker: function() {
- return this.getInputElement().data( 'cke-dir-marker' );
- },
- keyboardFocusable: true
- }, commonPrototype, true );
- CKEDITOR.ui.dialog.textarea.prototype = new CKEDITOR.ui.dialog.textInput();
- /** @class CKEDITOR.ui.dialog.select */
- CKEDITOR.ui.dialog.select.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.labeledElement(), {
- /**
- * Gets the DOM element of the select box.
- *
- * @returns {CKEDITOR.dom.element} The `<select>` element of this UI element.
- */
- getInputElement: function() {
- return this._.select.getElement();
- },
- /**
- * Adds an option to the select box.
- *
- * @param {String} label Option label.
- * @param {String} value (Optional) Option value, if not defined it will be
- * assumed to be the same as the label.
- * @param {Number} index (Optional) Position of the option to be inserted
- * to. If not defined, the new option will be inserted to the end of list.
- * @returns {CKEDITOR.ui.dialog.select} The current select UI element.
- */
- add: function( label, value, index ) {
- var option = new CKEDITOR.dom.element( 'option', this.getDialog().getParentEditor().document ),
- selectElement = this.getInputElement().$;
- option.$.text = label;
- option.$.value = ( value === undefined || value === null ) ? label : value;
- if ( index === undefined || index === null ) {
- if ( CKEDITOR.env.ie ) {
- selectElement.add( option.$ );
- } else {
- selectElement.add( option.$, null );
- }
- } else {
- selectElement.add( option.$, index );
- }
- return this;
- },
- /**
- * Removes an option from the selection list.
- *
- * @param {Number} index Index of the option to be removed.
- * @returns {CKEDITOR.ui.dialog.select} The current select UI element.
- */
- remove: function( index ) {
- var selectElement = this.getInputElement().$;
- selectElement.remove( index );
- return this;
- },
- /**
- * Clears all options out of the selection list.
- *
- * @returns {CKEDITOR.ui.dialog.select} The current select UI element.
- */
- clear: function() {
- var selectElement = this.getInputElement().$;
- while ( selectElement.length > 0 )
- selectElement.remove( 0 );
- return this;
- },
- keyboardFocusable: true
- }, commonPrototype, true );
- /** @class CKEDITOR.ui.dialog.checkbox */
- CKEDITOR.ui.dialog.checkbox.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.uiElement(), {
- /**
- * Gets the checkbox DOM element.
- *
- * @returns {CKEDITOR.dom.element} The DOM element of the checkbox.
- */
- getInputElement: function() {
- return this._.checkbox.getElement();
- },
- /**
- * Sets the state of the checkbox.
- *
- * @param {Boolean} checked `true` to tick the checkbox, `false` to untick it.
- * @param {Boolean} noChangeEvent Internal commit, to supress `change` event on this element.
- */
- setValue: function( checked, noChangeEvent ) {
- this.getInputElement().$.checked = checked;
- !noChangeEvent && this.fire( 'change', { value: checked } );
- },
- /**
- * Gets the state of the checkbox.
- *
- * @returns {Boolean} `true` means that the checkbox is ticked, `false` means it is not ticked.
- */
- getValue: function() {
- return this.getInputElement().$.checked;
- },
- /**
- * Handler for the access key up event. Toggles the checkbox.
- */
- accessKeyUp: function() {
- this.setValue( !this.getValue() );
- },
- /**
- * Defines the `onChange` event for UI element definitions.
- *
- * @property {Object}
- */
- eventProcessors: {
- onChange: function( dialog, func ) {
- if ( !CKEDITOR.env.ie || ( CKEDITOR.env.version > 8 ) )
- return commonEventProcessors.onChange.apply( this, arguments );
- else {
- dialog.on( 'load', function() {
- var element = this._.checkbox.getElement();
- element.on( 'propertychange', function( evt ) {
- evt = evt.data.$;
- if ( evt.propertyName == 'checked' )
- this.fire( 'change', { value: element.$.checked } );
- }, this );
- }, this );
- this.on( 'change', func );
- }
- return null;
- }
- },
- keyboardFocusable: true
- }, commonPrototype, true );
- /** @class CKEDITOR.ui.dialog.radio */
- CKEDITOR.ui.dialog.radio.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.uiElement(), {
- /**
- * Selects one of the radio buttons in this button group.
- *
- * @param {String} value The value of the button to be chcked.
- * @param {Boolean} noChangeEvent Internal commit, to supress the `change` event on this element.
- */
- setValue: function( value, noChangeEvent ) {
- var children = this._.children,
- item;
- for ( var i = 0;
- ( i < children.length ) && ( item = children[ i ] ); i++ )
- item.getElement().$.checked = ( item.getValue() == value );
- !noChangeEvent && this.fire( 'change', { value: value } );
- },
- /**
- * Gets the value of the currently selected radio button.
- *
- * @returns {String} The currently selected button's value.
- */
- getValue: function() {
- var children = this._.children;
- for ( var i = 0; i < children.length; i++ ) {
- if ( children[ i ].getElement().$.checked )
- return children[ i ].getValue();
- }
- return null;
- },
- /**
- * Handler for the access key up event. Focuses the currently
- * selected radio button, or the first radio button if none is selected.
- */
- accessKeyUp: function() {
- var children = this._.children,
- i;
- for ( i = 0; i < children.length; i++ ) {
- if ( children[ i ].getElement().$.checked ) {
- children[ i ].getElement().focus();
- return;
- }
- }
- children[ 0 ].getElement().focus();
- },
- /**
- * Defines the `onChange` event for UI element definitions.
- *
- * @property {Object}
- */
- eventProcessors: {
- onChange: function( dialog, func ) {
- if ( !CKEDITOR.env.ie )
- return commonEventProcessors.onChange.apply( this, arguments );
- else {
- dialog.on( 'load', function() {
- var children = this._.children,
- me = this;
- for ( var i = 0; i < children.length; i++ ) {
- var element = children[ i ].getElement();
- element.on( 'propertychange', function( evt ) {
- evt = evt.data.$;
- if ( evt.propertyName == 'checked' && this.$.checked )
- me.fire( 'change', { value: this.getAttribute( 'value' ) } );
- } );
- }
- }, this );
- this.on( 'change', func );
- }
- return null;
- }
- }
- }, commonPrototype, true );
- /** @class CKEDITOR.ui.dialog.file */
- CKEDITOR.ui.dialog.file.prototype = CKEDITOR.tools.extend( new CKEDITOR.ui.dialog.labeledElement(), commonPrototype, {
- /**
- * Gets the `<input>` element of this file input.
- *
- * @returns {CKEDITOR.dom.element} The file input element.
- */
- getInputElement: function() {
- var frameDocument = CKEDITOR.document.getById( this._.frameId ).getFrameDocument();
- return frameDocument.$.forms.length > 0 ? new CKEDITOR.dom.element( frameDocument.$.forms[ 0 ].elements[ 0 ] ) : this.getElement();
- },
- /**
- * Uploads the file in the file input.
- *
- * @returns {CKEDITOR.ui.dialog.file} This object.
- */
- submit: function() {
- this.getInputElement().getParent().$.submit();
- return this;
- },
- /**
- * Gets the action assigned to the form.
- *
- * @returns {String} The value of the action.
- */
- getAction: function() {
- return this.getInputElement().getParent().$.action;
- },
- /**
- * The events must be applied to the inner input element, and
- * this must be done when the iframe and form have been loaded.
- */
- registerEvents: function( definition ) {
- var regex = /^on([A-Z]\w+)/,
- match;
- var registerDomEvent = function( uiElement, dialog, eventName, func ) {
- uiElement.on( 'formLoaded', function() {
- uiElement.getInputElement().on( eventName, func, uiElement );
- } );
- };
- for ( var i in definition ) {
- if ( !( match = i.match( regex ) ) )
- continue;
- if ( this.eventProcessors[ i ] )
- this.eventProcessors[ i ].call( this, this._.dialog, definition[ i ] );
- else
- registerDomEvent( this, this._.dialog, match[ 1 ].toLowerCase(), definition[ i ] );
- }
- return this;
- },
- /**
- * Redraws the file input and resets the file path in the file input.
- * The redrawing logic is necessary because non-IE browsers tend to clear
- * the `<iframe>` containing the file input after closing the dialog window.
- */
- reset: function() {
- var _ = this._,
- frameElement = CKEDITOR.document.getById( _.frameId ),
- frameDocument = frameElement.getFrameDocument(),
- elementDefinition = _.definition,
- buttons = _.buttons,
- callNumber = this.formLoadedNumber,
- unloadNumber = this.formUnloadNumber,
- langDir = _.dialog._.editor.lang.dir,
- langCode = _.dialog._.editor.langCode;
- // The callback function for the iframe, but we must call tools.addFunction only once
- // so we store the function number in this.formLoadedNumber
- if ( !callNumber ) {
- callNumber = this.formLoadedNumber = CKEDITOR.tools.addFunction( function() {
- // Now we can apply the events to the input type=file
- this.fire( 'formLoaded' );
- }, this );
- // Remove listeners attached to the content of the iframe (the file input)
- unloadNumber = this.formUnloadNumber = CKEDITOR.tools.addFunction( function() {
- this.getInputElement().clearCustomData();
- }, this );
- this.getDialog()._.editor.on( 'destroy', function() {
- CKEDITOR.tools.removeFunction( callNumber );
- CKEDITOR.tools.removeFunction( unloadNumber );
- } );
- }
- function generateFormField() {
- frameDocument.$.open();
- var size = '';
- if ( elementDefinition.size )
- size = elementDefinition.size - ( CKEDITOR.env.ie ? 7 : 0 ); // "Browse" button is bigger in IE.
- var inputId = _.frameId + '_input';
- frameDocument.$.write( [
- '<html dir="' + langDir + '" lang="' + langCode + '"><head><title></title></head><body style="margin: 0; overflow: hidden; background: transparent;">',
- '<form enctype="multipart/form-data" method="POST" dir="' + langDir + '" lang="' + langCode + '" action="',
- CKEDITOR.tools.htmlEncode( elementDefinition.action ),
- '">',
- // Replicate the field label inside of iframe.
- '<label id="', _.labelId, '" for="', inputId, '" style="display:none">',
- CKEDITOR.tools.htmlEncode( elementDefinition.label ),
- '</label>',
- // Set width to make sure that input is not clipped by the iframe (#11253).
- '<input style="width:100%" id="', inputId, '" aria-labelledby="', _.labelId, '" type="file" name="',
- CKEDITOR.tools.htmlEncode( elementDefinition.id || 'cke_upload' ),
- '" size="',
- CKEDITOR.tools.htmlEncode( size > 0 ? size : '' ),
- '" />',
- '</form>',
- '</body></html>',
- '<script>',
- // Support for custom document.domain in IE.
- CKEDITOR.env.ie ? '(' + CKEDITOR.tools.fixDomain + ')();' : '',
- 'window.parent.CKEDITOR.tools.callFunction(' + callNumber + ');',
- 'window.onbeforeunload = function() {window.parent.CKEDITOR.tools.callFunction(' + unloadNumber + ')}',
- '</script>'
- ].join( '' ) );
- frameDocument.$.close();
- for ( var i = 0; i < buttons.length; i++ )
- buttons[ i ].enable();
- }
- // #3465: Wait for the browser to finish rendering the dialog first.
- if ( CKEDITOR.env.gecko )
- setTimeout( generateFormField, 500 );
- else
- generateFormField();
- },
- getValue: function() {
- return this.getInputElement().$.value || '';
- },
- /**
- * The default value of input `type="file"` is an empty string, but during the initialization
- * of this UI element, the iframe still is not ready so it cannot be read from that object.
- * Setting it manually prevents later issues with the current value (`''`) being different
- * than the initial value (undefined as it asked for `.value` of a div).
- */
- setInitValue: function() {
- this._.initValue = '';
- },
- /**
- * Defines the `onChange` event for UI element definitions.
- *
- * @property {Object}
- */
- eventProcessors: {
- onChange: function( dialog, func ) {
- // If this method is called several times (I'm not sure about how this can happen but the default
- // onChange processor includes this protection)
- // In order to reapply to the new element, the property is deleted at the beggining of the registerEvents method
- if ( !this._.domOnChangeRegistered ) {
- // By listening for the formLoaded event, this handler will get reapplied when a new
- // form is created
- this.on( 'formLoaded', function() {
- this.getInputElement().on( 'change', function() {
- this.fire( 'change', { value: this.getValue() } );
- }, this );
- }, this );
- this._.domOnChangeRegistered = true;
- }
- this.on( 'change', func );
- }
- },
- keyboardFocusable: true
- }, true );
- CKEDITOR.ui.dialog.fileButton.prototype = new CKEDITOR.ui.dialog.button();
- CKEDITOR.ui.dialog.fieldset.prototype = CKEDITOR.tools.clone( CKEDITOR.ui.dialog.hbox.prototype );
- CKEDITOR.dialog.addUIElement( 'text', textBuilder );
- CKEDITOR.dialog.addUIElement( 'password', textBuilder );
- CKEDITOR.dialog.addUIElement( 'textarea', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'checkbox', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'radio', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'button', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'select', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'file', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'fileButton', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'html', commonBuilder );
- CKEDITOR.dialog.addUIElement( 'fieldset', containerBuilder );
- }
- } );
- /**
- * Fired when the value of the uiElement is changed.
- *
- * @event change
- * @member CKEDITOR.ui.dialog.uiElement
- */
- /**
- * Fired when the inner frame created by the element is ready.
- * Each time the button is used or the dialog window is loaded, a new
- * form might be created.
- *
- * @event formLoaded
- * @member CKEDITOR.ui.dialog.fileButton
- */
|