Integration with CKEditor 4 (jQuery UI dialog mode) - Studio-42/elFinder GitHub Wiki
Demo site - http://hypweb.net/elFinder-nightly/demo/2.1/ckeditor.html
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>CKEditor + elFinder Integration (jQuery dialog version)</title>
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=2" />
<!-- Require JS (REQUIRED) -->
<script data-main="./main.ckedialog.js" src="//cdnjs.cloudflare.com/ajax/libs/require.js/2.3.2/require.min.js"></script>
</head>
<body style="margin:10px; padding:0;">
<h1>CKEditor + elFinder Integration (jQuery dialog version)</h1>
<h2 id="loading" style="text-align:center;">Now loading...</h2>
<form method="post">
<textarea id="mytextarea" style="display:none;"><h1>You can upload pictures directly to elFinder by "drag & drop" into this editting area.</h1></textarea>
</form>
</body>
</html>
main.ckedialog.js Example
"use strict";
/**
* elFinder client options and main script for RequireJS
*
* Rename "main.default.js" to "main.js" and edit it if you need configure elFInder options or any things. And use that in elfinder.html.
* e.g. `<script data-main="./main.js" src="./require.js"></script>`
**/
(function(){
var // jQuery and jQueryUI version
jqver = '3.2.1',
uiver = '1.12.1',
// Detect language (optional)
lang = (function() {
var locq = window.location.search,
fullLang, locm, lang;
if (locq && (locm = locq.match(/lang=([a-zA-Z_-]+)/))) {
// detection by url query (?lang=xx)
fullLang = locm[1];
} else {
// detection by browser language
fullLang = (navigator.browserLanguage || navigator.language || navigator.userLanguage);
}
lang = fullLang.substr(0,2);
if (lang === 'pt') lang = 'pt_BR';
else if (lang === 'ug') lang = 'ug_CN';
else if (lang === 'zh') lang = (fullLang.substr(0,5) === 'zh-TW')? 'zh_TW' : 'zh_CN';
return lang;
})(),
// Start CKEditor with elFinder
start = function(CKEDITOR, elFinder) {
// load jQueryUI CSS
elFinder.prototype.loadCss('//cdnjs.cloudflare.com/ajax/libs/jqueryui/'+uiver+'/themes/smoothness/jquery-ui.css');
$(function() {
var elfNode, elfInstance, dialogName,
elfUrl = './php/connector.minimal.php', // Your connector's URL
elfDirHashMap = { // Dialog name / elFinder holder hash Map
image : '',
flash : '',
files : '',
link : '',
fb : 'l1_Lw' // Fall back target : `/`
},
imgShowMaxSize = 400, // Max image size(px) to show
customData = {},
// Set image size to show
setShowImgSize = function(url, callback) {
$('<img/>').attr('src', url).on('load', function() {
var w = this.naturalWidth,
h = this.naturalHeight,
s = imgShowMaxSize;
if (w > s || h > s) {
if (w > h) {
h = Math.floor(h * (s / w));
w = s;
} else {
w = Math.floor(w * (s / h));
h = s;
}
}
callback({width: w, height: h});
});
},
// Set values to dialog of CKEditor
setDialogValue = function(file, fm) {
var url = fm.convAbsUrl(file.url),
dialog = CKEDITOR.dialog.getCurrent(),
dialogName = dialog._.name,
tabName = dialog._.currentTabId,
urlObj;
if (dialogName == 'image') {
urlObj = 'txtUrl';
} else if (dialogName == 'flash') {
urlObj = 'src';
} else if (dialogName == 'files' || dialogName == 'link') {
urlObj = 'url';
} else if (dialogName == 'image2') {
urlObj = 'src';
} else {
return;
}
if (tabName == 'Upload') {
tabName = 'info';
dialog.selectPage(tabName);
}
dialog.setValueOf(tabName, urlObj, url);
if (dialogName == 'image' && tabName == 'info') {
setShowImgSize(url, function(size) {
dialog.setValueOf('info', 'txtWidth', size.width);
dialog.setValueOf('info', 'txtHeight', size.height);
dialog.preview.$.style.width = size.width+'px';
dialog.preview.$.style.height = size.height+'px';
dialog.setValueOf('Link', 'txtUrl', url);
dialog.setValueOf('Link', 'cmbTarget', '_blank');
});
} else if (dialogName == 'image2' && tabName == 'info') {
dialog.setValueOf(tabName, 'alt', file.name + ' (' + elfInstance.formatSize(file.size) + ')');
setShowImgSize(url, function(size) {
setTimeout(function() {
dialog.setValueOf('info', 'width', size.width);
dialog.setValueOf('info', 'height', size.height);
}, 100);
});
} else if (dialogName == 'files' || dialogName == 'link') {
try {
dialog.setValueOf('info', 'linkDisplayText', file.name);
} catch(e) {}
}
};
// Setup upload tab in CKEditor dialog
CKEDITOR.on('dialogDefinition', function (event) {
var editor = event.editor,
dialogDefinition = event.data.definition,
tabCount = dialogDefinition.contents.length,
browseButton, uploadButton, submitButton, inputId;
for (var i = 0; i < tabCount; i++) {
try {
browseButton = dialogDefinition.contents[i].get('browse');
uploadButton = dialogDefinition.contents[i].get('upload');
submitButton = dialogDefinition.contents[i].get('uploadButton');
} catch(e) {
browseButton = uploadButton = null;
}
if (browseButton !== null) {
browseButton.hidden = false;
browseButton.onClick = function (dialog, i) {
dialogName = CKEDITOR.dialog.getCurrent()._.name;
if (dialogName === 'image2') {
dialogName = 'image';
}
if (elfNode) {
if (elfDirHashMap[dialogName] && elfDirHashMap[dialogName] != elfInstance.cwd().hash) {
elfInstance.request({
data : {cmd : 'open', target : elfDirHashMap[dialogName]},
notify : {type : 'open', cnt : 1, hideCnt : true},
syncOnFail : true
});
}
elfNode.dialog('open');
}
}
}
if (uploadButton !== null && submitButton !== null) {
uploadButton.hidden = false;
submitButton.hidden = false;
uploadButton.onChange = function() {
inputId = this.domId;
}
// upload a file to elFinder connector
submitButton.onClick = function(e) {
dialogName = CKEDITOR.dialog.getCurrent()._.name;
if (dialogName === 'image2') {
dialogName = 'image';
}
var target = elfDirHashMap[dialogName]? elfDirHashMap[dialogName] : elfDirHashMap['fb'],
name = $('#'+inputId),
input = name.find('iframe').contents().find('form').find('input:file'),
error = function(err) {
alert(elfInstance.i18n(err).replace('<br>', '\n'));
};
if (input.val()) {
var fd = new FormData();
fd.append('cmd', 'upload');
fd.append('target', target);
fd.append('overwrite', 0); // Instruction to save alias when same name file exists
$.each(customData, function(key, val) {
fd.append(key, val);
});
fd.append('upload[]', input[0].files[0]);
$.ajax({
url: editor.config.filebrowserUploadUrl,
type: "POST",
data: fd,
processData: false,
contentType: false,
dataType: 'json'
})
.done(function( data ) {
if (data.added && data.added[0]) {
elfInstance.exec('reload');
setDialogValue(data.added[0]);
} else {
error(data.error || data.warning || 'errUploadFile');
}
})
.fail(function() {
error('errUploadFile');
})
.always(function() {
input.val('');
});
}
return false;
}
}
}
});
// Create elFinder dialog for CKEditor
CKEDITOR.on('instanceReady', function(e) {
elfNode = $('<div style="padding:0;">');
elfNode.dialog({
autoOpen: false,
modal: true,
width: '80%',
title: 'Server File Manager',
create: function (event, ui) {
var startPathHash = (elfDirHashMap[dialogName] && elfDirHashMap[dialogName])? elfDirHashMap[dialogName] : '';
// elFinder configure
elfInstance = $(this).elfinder({
startPathHash: startPathHash,
useBrowserHistory: false,
resizable: false,
width: '100%',
url: elfUrl,
lang: lang,
dialogContained : true,
getFileCallback: function(file, fm) {
setDialogValue(file, fm);
elfNode.dialog('close');
}
}).elfinder('instance');
},
open: function() {
elfNode.find('div.elfinder-toolbar input').blur();
setTimeout(function(){
elfInstance.enable();
}, 100);
},
resizeStop: function() {
elfNode.trigger('resize');
}
}).parent().css({'zIndex':'11000'});
// CKEditor instance
var cke = e.editor;
// Setup the procedure when DnD image upload was completed
cke.widgets.registered.uploadimage.onUploaded = function(upload){
var self = this;
setShowImgSize(upload.url, function(size) {
self.replaceWith('<img src="'+encodeURI(upload.url)+'" width="'+size.width+'" height="'+size.height+'"></img>');
});
}
// Setup the procedure when send DnD image upload data to elFinder's connector
cke.on('fileUploadRequest', function(e){
var target = elfDirHashMap['image']? elfDirHashMap['image'] : elfDirHashMap['fb'],
fileLoader = e.data.fileLoader,
xhr = fileLoader.xhr,
formData = new FormData();
e.stop();
xhr.open('POST', fileLoader.uploadUrl, true);
formData.append('cmd', 'upload');
formData.append('target', target);
formData.append('upload[]', fileLoader.file, fileLoader.fileName);
xhr.send(formData);
}, null, null, 4);
// Setup the procedure when got DnD image upload response
cke.on('fileUploadResponse', function(e){
var file;
e.stop();
var data = e.data,
res = JSON.parse(data.fileLoader.xhr.responseText);
if (!res.added || res.added.length < 1) {
data.message = 'Can not upload.';
e.cancel();
} else {
elfInstance.exec('reload');
file = res.added[0];
if (file.url && file.url != '1') {
data.url = file.url;
try {
data.url = decodeURIComponent(data.url);
} catch(e) {}
} else {
data.url = elfInstance.options.url + ((elfInstance.options.url.indexOf('?') === -1)? '?' : '&') + 'cmd=file&target=' + file.hash;
}
data.url = elfInstance.convAbsUrl(data.url);
}
});
});
// setup CKEditor
CKEDITOR.replace('mytextarea', {
filebrowserBrowseUrl: '#',
extraPlugins: 'uploadimage,image2',
filebrowserUploadUrl: './php/connector.minimal.php',
imageUploadUrl: './php/connector.minimal.php'
});
// Remove "Now loading..."
$('#loading').remove();
// Show textarea
$('#mytextarea').show();
});
},
// JavaScript loader (REQUIRED)
load = function() {
require(
[
'ckeditor',
'elfinder'
, (lang !== 'en')? 'elfinder.lang' : null // load detected language
//, 'extras/quicklook.googledocs' // optional preview for GoogleApps contents on the GoogleDrive volume
//, (lang === 'jp')? 'extras/encoding-japanese.min' : null // optional Japanese decoder for archive preview
],
start,
function(error) {
alert(error.message);
}
);
},
// is IE8? for determine the jQuery version to use (optional)
ie8 = (typeof window.addEventListener === 'undefined' && typeof document.getElementsByClassName === 'undefined');
// config of RequireJS (REQUIRED)
require.config({
baseUrl : 'js',
paths : {
'ckeditor' : '//cdnjs.cloudflare.com/ajax/libs/ckeditor/4.7.3/ckeditor',
'jquery' : '//cdnjs.cloudflare.com/ajax/libs/jquery/'+(ie8? '1.12.4' : jqver)+'/jquery.min',
'jquery-ui': '//cdnjs.cloudflare.com/ajax/libs/jqueryui/'+uiver+'/jquery-ui.min',
'elfinder' : 'elfinder.min',
'elfinder.lang': [
'i18n/elfinder.'+lang,
'i18n/elfinder.fallback'
]
},
shim : {
'ckeditor' : { exports : 'CKEDITOR' }
},
waitSeconds : 10 // optional
});
// load JavaScripts (REQUIRED)
load();
})();