/** * @license Copyright (c) 2003-2017, CKSource - Frederico Knabben. All rights reserved. * For licensing, see LICENSE.md or http://ckeditor.com/license */ 'use strict'; ( function() { CKEDITOR.dialog.add( 'link', function( editor ) { var plugin = CKEDITOR.plugins.link, initialLinkText; function createRangeForLink( editor, link ) { var range = editor.createRange(); range.setStartBefore( link ); range.setEndAfter( link ); return range; } function insertLinksIntoSelection( editor, data ) { var attributes = plugin.getLinkAttributes( editor, data ), ranges = editor.getSelection().getRanges(), style = new CKEDITOR.style( { element: 'a', attributes: attributes.set } ), rangesToSelect = [], range, text, nestedLinks, i, j; style.type = CKEDITOR.STYLE_INLINE; // need to override... dunno why. for ( i = 0; i < ranges.length; i++ ) { range = ranges[ i ]; // Use link URL as text with a collapsed cursor. if ( range.collapsed ) { // Short mailto link text view (http://dev.ckeditor.com/ticket/5736). text = new CKEDITOR.dom.text( data.linkText || ( data.type == 'email' ? data.email.address : attributes.set[ 'data-cke-saved-href' ] ), editor.document ); range.insertNode( text ); range.selectNodeContents( text ); } else if ( initialLinkText !== data.linkText ) { text = new CKEDITOR.dom.text( data.linkText, editor.document ); // Shrink range to preserve block element. range.shrink( CKEDITOR.SHRINK_TEXT ); // Use extractHtmlFromRange to remove markup within the selection. Also this method is a little // smarter than range#deleteContents as it plays better e.g. with table cells. editor.editable().extractHtmlFromRange( range ); range.insertNode( text ); } // Editable links nested within current range should be removed, so that the link is applied to whole selection. nestedLinks = range._find( 'a' ); for ( j = 0; j < nestedLinks.length; j++ ) { nestedLinks[ j ].remove( true ); } // Apply style. style.applyToRange( range, editor ); rangesToSelect.push( range ); } editor.getSelection().selectRanges( rangesToSelect ); } function editLinksInSelection( editor, selectedElements, data ) { var attributes = plugin.getLinkAttributes( editor, data ), ranges = [], element, href, textView, newText, i; for ( i = 0; i < selectedElements.length; i++ ) { // We're only editing an existing link, so just overwrite the attributes. element = selectedElements[ i ]; href = element.data( 'cke-saved-href' ); textView = element.getHtml(); element.setAttributes( attributes.set ); element.removeAttributes( attributes.removed ); if ( data.linkText && initialLinkText != data.linkText ) { // Display text has been changed. newText = data.linkText; } else if ( href == textView || data.type == 'email' && textView.indexOf( '@' ) != -1 ) { // Update text view when user changes protocol (http://dev.ckeditor.com/ticket/4612). // Short mailto link text view (http://dev.ckeditor.com/ticket/5736). newText = data.type == 'email' ? data.email.address : attributes.set[ 'data-cke-saved-href' ]; } if ( newText ) { element.setText( newText ); } ranges.push( createRangeForLink( editor, element ) ); } // We changed the content, so need to select it again. editor.getSelection().selectRanges( ranges ); } // Handles the event when the "Target" selection box is changed. var targetChanged = function() { var dialog = this.getDialog(), popupFeatures = dialog.getContentElement( 'target', 'popupFeatures' ), targetName = dialog.getContentElement( 'target', 'linkTargetName' ), value = this.getValue(); if ( !popupFeatures || !targetName ) return; popupFeatures = popupFeatures.getElement(); popupFeatures.hide(); targetName.setValue( '' ); switch ( value ) { case 'frame': targetName.setLabel( editor.lang.link.targetFrameName ); targetName.getElement().show(); break; case 'popup': popupFeatures.show(); targetName.setLabel( editor.lang.link.targetPopupName ); targetName.getElement().show(); break; default: targetName.setValue( value ); targetName.getElement().hide(); break; } }; // Handles the event when the "Type" selection box is changed. var linkTypeChanged = function() { var dialog = this.getDialog(), partIds = [ 'urlOptions', 'localPageOptions', 'anchorOptions', 'emailOptions' ],// added by @simo - http://blog.xoundboy.com/?p=393 typeValue = this.getValue(), uploadTab = dialog.definition.getContents( 'upload' ), uploadInitiallyHidden = uploadTab && uploadTab.hidden; if ( typeValue == 'url' ) { if ( editor.config.linkShowTargetTab ) dialog.showPage( 'target' ); if ( !uploadInitiallyHidden ) dialog.showPage( 'upload' ); } else { dialog.hidePage( 'target' ); if ( !uploadInitiallyHidden ) dialog.hidePage( 'upload' ); } for ( var i = 0; i < partIds.length; i++ ) { var element = dialog.getContentElement( 'info', partIds[ i ] ); if ( !element ) continue; element = element.getElement().getParent().getParent(); if ( partIds[ i ] == typeValue + 'Options' ) element.show(); else element.hide(); } dialog.layout(); }; var setupParams = function( page, data ) { if ( data[ page ] ) this.setValue( data[ page ][ this.id ] || '' ); }; var setupPopupParams = function( data ) { return setupParams.call( this, 'target', data ); }; var setupAdvParams = function( data ) { return setupParams.call( this, 'advanced', data ); }; var commitParams = function( page, data ) { if ( !data[ page ] ) data[ page ] = {}; data[ page ][ this.id ] = this.getValue() || ''; }; var commitPopupParams = function( data ) { return commitParams.call( this, 'target', data ); }; var commitAdvParams = function( data ) { return commitParams.call( this, 'advanced', data ); }; var commonLang = editor.lang.common, linkLang = editor.lang.adv_link, // added by @simo - http://blog.xoundboy.com/?p=393 anchors; return { title: linkLang.title, minWidth: ( CKEDITOR.skinName || editor.config.skin ) == 'moono-lisa' ? 450 : 350, minHeight: 240, contents: [ { id: 'info', label: linkLang.info, title: linkLang.info, elements: [ { type: 'text', id: 'linkDisplayText', label: linkLang.displayText, setup: function() { this.enable(); this.setValue( editor.getSelection().getSelectedText() ); // Keep inner text so that it can be compared in commit function. By obtaining value from getData() // we get value stripped from new line chars which is important when comparing the value later on. initialLinkText = this.getValue(); }, commit: function( data ) { data.linkText = this.isEnabled() ? this.getValue() : ''; } }, { id: 'linkType', type: 'select', label: linkLang.type, 'default': 'url', items: [ [ linkLang.toUrl, 'url' ], [ linkLang.toAnchor, 'anchor' ], [ linkLang.localPages, 'localPage'], // added by @simo - http://blog.xoundboy.com/?p=393 [ linkLang.toEmail, 'email' ] ], onChange: linkTypeChanged, setup: function( data ) { this.setValue( data.type || 'url' ); }, commit: function( data ) { data.type = this.getValue(); } }, // added by @simo - http://blog.xoundboy.com/?p=393 // see also : http://docs.ckeditor.com/source/dialogDefinition.html#CKEDITOR-dialog-definition-uiElement-property-type // http://docs.ckeditor.com/#!/guide/plugin_sdk_sample_1 { type : 'vbox', id : 'localPageOptions', children : [ { type : 'select', label : linkLang.selectPageLabel, id : 'localPage', title : linkLang.selectPageTitle, items : [], onLoad : function(element) { var element_id = '#' + this.getInputElement().$.id; // ajax call inspired from http://stackoverflow.com/questions/5293920/ckeditor-dynamic-select-in-a-dialog $.ajax({ type: 'POST', url: '../admin/_link_to_pages.php', contentType: 'application/json; charset=utf-8', dataType: 'json', async: false, success: function(data) { $.each(data, function(index, item) { $(element_id).get(0).options[$(element_id).get(0).options.length] = new Option(decodeURIComponent(item[0]), item[1]); }); }, error:function (xhr, ajaxOptions, thrownError){ alert(xhr.status); alert(thrownError); } }); }, commit : function( data ) { if ( !data.localPage ) // console.log(data); data.localPage = {}; data.localPage = this.getValue(); } }] }, // added by @simo - end { type: 'vbox', id: 'urlOptions', children: [ { type: 'hbox', widths: [ '25%', '75%' ], children: [ { id: 'protocol', type: 'select', label: commonLang.protocol, 'default': 'http://', items: [ // Force 'ltr' for protocol names in BIDI. (http://dev.ckeditor.com/ticket/5433) [ 'http://\u200E', 'http://' ], [ 'https://\u200E', 'https://' ], [ 'ftp://\u200E', 'ftp://' ], [ 'news://\u200E', 'news://' ], [ linkLang.other, '' ] ], setup: function( data ) { if ( data.url ) this.setValue( data.url.protocol || '' ); }, commit: function( data ) { if ( !data.url ) data.url = {}; data.url.protocol = this.getValue(); } }, { type: 'text', id: 'url', label: commonLang.url, required: true, onLoad: function() { this.allowOnChange = true; }, onKeyUp: function() { this.allowOnChange = false; var protocolCmb = this.getDialog().getContentElement( 'info', 'protocol' ), url = this.getValue(), urlOnChangeProtocol = /^(http|https|ftp|news):\/\/(?=.)/i, urlOnChangeTestOther = /^((javascript:)|[#\/\.\?])/i; var protocol = urlOnChangeProtocol.exec( url ); if ( protocol ) { this.setValue( url.substr( protocol[ 0 ].length ) ); protocolCmb.setValue( protocol[ 0 ].toLowerCase() ); } else if ( urlOnChangeTestOther.test( url ) ) { protocolCmb.setValue( '' ); } this.allowOnChange = true; }, onChange: function() { if ( this.allowOnChange ) // Dont't call on dialog load. this.onKeyUp(); }, validate: function() { var dialog = this.getDialog(); if ( dialog.getContentElement( 'info', 'linkType' ) && dialog.getValueOf( 'info', 'linkType' ) != 'url' ) return true; if ( !editor.config.linkJavaScriptLinksAllowed && ( /javascript\:/ ).test( this.getValue() ) ) { alert( commonLang.invalidValue ); // jshint ignore:line return false; } if ( this.getDialog().fakeObj ) // Edit Anchor. return true; var func = CKEDITOR.dialog.validate.notEmpty( linkLang.noUrl ); return func.apply( this ); }, setup: function( data ) { this.allowOnChange = false; if ( data.url ) this.setValue( data.url.url ); this.allowOnChange = true; }, commit: function( data ) { // IE will not trigger the onChange event if the mouse has been used // to carry all the operations http://dev.ckeditor.com/ticket/4724 this.onChange(); if ( !data.url ) data.url = {}; data.url.url = this.getValue(); this.allowOnChange = false; } } ], setup: function() { if ( !this.getDialog().getContentElement( 'info', 'linkType' ) ) this.getElement().show(); } }, { type: 'button', id: 'browse', hidden: 'true', filebrowser: 'info:url', label: commonLang.browseServer } ] }, { type: 'vbox', id: 'anchorOptions', width: 260, align: 'center', padding: 0, children: [ { type: 'fieldset', id: 'selectAnchorText', label: linkLang.selectAnchor, setup: function() { anchors = plugin.getEditorAnchors( editor ); this.getElement()[ anchors && anchors.length ? 'show' : 'hide' ](); }, children: [ { type: 'hbox', id: 'selectAnchor', children: [ { type: 'select', id: 'anchorName', 'default': '', label: linkLang.anchorName, style: 'width: 100%;', items: [ [ '' ] ], setup: function( data ) { this.clear(); this.add( '' ); if ( anchors ) { for ( var i = 0; i < anchors.length; i++ ) { if ( anchors[ i ].name ) this.add( anchors[ i ].name ); } } if ( data.anchor ) this.setValue( data.anchor.name ); var linkType = this.getDialog().getContentElement( 'info', 'linkType' ); if ( linkType && linkType.getValue() == 'email' ) this.focus(); }, commit: function( data ) { if ( !data.anchor ) data.anchor = {}; data.anchor.name = this.getValue(); } }, { type: 'select', id: 'anchorId', 'default': '', label: linkLang.anchorId, style: 'width: 100%;', items: [ [ '' ] ], setup: function( data ) { this.clear(); this.add( '' ); if ( anchors ) { for ( var i = 0; i < anchors.length; i++ ) { if ( anchors[ i ].id ) this.add( anchors[ i ].id ); } } if ( data.anchor ) this.setValue( data.anchor.id ); }, commit: function( data ) { if ( !data.anchor ) data.anchor = {}; data.anchor.id = this.getValue(); } } ], setup: function() { this.getElement()[ anchors && anchors.length ? 'show' : 'hide' ](); } } ] }, { type: 'html', id: 'noAnchors', style: 'text-align: center;', html: '