mirror of
https://github.com/silverstripe/silverstripe-framework
synced 2024-10-22 12:05:37 +00:00
Fix HtmlEditorField.js indentation
This commit is contained in:
parent
37cb351a66
commit
513c7aebc7
@ -19,1542 +19,1542 @@ var ss = typeof window.ss !== 'undefined' ? window.ss : {};
|
||||
ss.editorWrappers = {};
|
||||
ss.editorWrappers.tinyMCE = (function() {
|
||||
|
||||
// ID of editor this is assigned to
|
||||
var editorID;
|
||||
// ID of editor this is assigned to
|
||||
var editorID;
|
||||
|
||||
return {
|
||||
/**
|
||||
* Initialise the editor
|
||||
*
|
||||
* @param {String} ID of parent textarea domID
|
||||
return {
|
||||
/**
|
||||
* Initialise the editor
|
||||
*
|
||||
* @param {String} ID of parent textarea domID
|
||||
*/
|
||||
init: function(ID) {
|
||||
editorID = ID;
|
||||
init: function(ID) {
|
||||
editorID = ID;
|
||||
|
||||
this.create();
|
||||
},
|
||||
this.create();
|
||||
},
|
||||
|
||||
/**
|
||||
* Remove the editor and cleanup
|
||||
*/
|
||||
destroy: function() {
|
||||
tinymce.EditorManager.execCommand('mceRemoveEditor', false, editorID);
|
||||
},
|
||||
|
||||
/**
|
||||
* Get TinyMCE Editor instance
|
||||
*
|
||||
* @returns Editor
|
||||
/**
|
||||
* Remove the editor and cleanup
|
||||
*/
|
||||
getInstance: function() {
|
||||
return tinymce.EditorManager.get(editorID);
|
||||
},
|
||||
destroy: function() {
|
||||
tinymce.EditorManager.execCommand('mceRemoveEditor', false, editorID);
|
||||
},
|
||||
|
||||
/**(
|
||||
* Invoked when a content-modifying UI is opened.
|
||||
*/
|
||||
onopen: function() {
|
||||
// NOOP
|
||||
},
|
||||
|
||||
/**(
|
||||
* Invoked when a content-modifying UI is closed.
|
||||
*/
|
||||
onclose: function() {
|
||||
// NOOP
|
||||
},
|
||||
|
||||
/**
|
||||
* Get config for this data
|
||||
*
|
||||
* @returns array
|
||||
/**
|
||||
* Get TinyMCE Editor instance
|
||||
*
|
||||
* @returns Editor
|
||||
*/
|
||||
getConfig: function() {
|
||||
var selector = "#" + editorID,
|
||||
config = $(selector).data('config'),
|
||||
self = this;
|
||||
getInstance: function() {
|
||||
return tinymce.EditorManager.get(editorID);
|
||||
},
|
||||
|
||||
// Add instance specific data to config
|
||||
config.selector = selector;
|
||||
/**(
|
||||
* Invoked when a content-modifying UI is opened.
|
||||
*/
|
||||
onopen: function() {
|
||||
// NOOP
|
||||
},
|
||||
|
||||
// Ensure save events write back to textarea
|
||||
config.setup = function(ed) {
|
||||
ed.on('change', function() {
|
||||
self.save();
|
||||
});
|
||||
};
|
||||
return config;
|
||||
},
|
||||
/**(
|
||||
* Invoked when a content-modifying UI is closed.
|
||||
*/
|
||||
onclose: function() {
|
||||
// NOOP
|
||||
},
|
||||
|
||||
/**
|
||||
* Write the HTML back to the original text area field.
|
||||
*/
|
||||
save: function() {
|
||||
var instance = this.getInstance();
|
||||
instance.save();
|
||||
/**
|
||||
* Get config for this data
|
||||
*
|
||||
* @returns array
|
||||
*/
|
||||
getConfig: function() {
|
||||
var selector = "#" + editorID,
|
||||
config = $(selector).data('config'),
|
||||
self = this;
|
||||
|
||||
// Update change detection
|
||||
$(instance.getElement()).trigger("change");
|
||||
},
|
||||
// Add instance specific data to config
|
||||
config.selector = selector;
|
||||
|
||||
/**
|
||||
* Create a new instance based on a textarea field.
|
||||
*/
|
||||
create: function() {
|
||||
var config = this.getConfig();
|
||||
// hack to set baseURL safely
|
||||
if(typeof config.baseURL !== 'undefined') {
|
||||
tinymce.EditorManager.baseURL = config.baseURL;
|
||||
}
|
||||
tinymce.init(config);
|
||||
},
|
||||
// Ensure save events write back to textarea
|
||||
config.setup = function(ed) {
|
||||
ed.on('change', function() {
|
||||
self.save();
|
||||
});
|
||||
};
|
||||
return config;
|
||||
},
|
||||
|
||||
/**
|
||||
* Request an update to editor content
|
||||
*/
|
||||
repaint: function() {
|
||||
// NOOP
|
||||
},
|
||||
/**
|
||||
* Write the HTML back to the original text area field.
|
||||
*/
|
||||
save: function() {
|
||||
var instance = this.getInstance();
|
||||
instance.save();
|
||||
|
||||
/**
|
||||
* @return boolean
|
||||
*/
|
||||
isDirty: function() {
|
||||
return this.getInstance().isDirty();
|
||||
},
|
||||
// Update change detection
|
||||
$(instance.getElement()).trigger("change");
|
||||
},
|
||||
|
||||
/**
|
||||
* HTML representation of the edited content.
|
||||
*
|
||||
* Returns: {String}
|
||||
*/
|
||||
getContent: function() {
|
||||
return this.getInstance().getContent();
|
||||
},
|
||||
/**
|
||||
* Create a new instance based on a textarea field.
|
||||
*/
|
||||
create: function() {
|
||||
var config = this.getConfig();
|
||||
// hack to set baseURL safely
|
||||
if(typeof config.baseURL !== 'undefined') {
|
||||
tinymce.EditorManager.baseURL = config.baseURL;
|
||||
}
|
||||
tinymce.init(config);
|
||||
},
|
||||
|
||||
/**
|
||||
* DOM tree of the edited content
|
||||
*
|
||||
* Returns: DOMElement
|
||||
*/
|
||||
getDOM: function() {
|
||||
return this.getInstance().getElement();
|
||||
},
|
||||
/**
|
||||
* Request an update to editor content
|
||||
*/
|
||||
repaint: function() {
|
||||
// NOOP
|
||||
},
|
||||
|
||||
/**
|
||||
* Returns: DOMElement
|
||||
*/
|
||||
getContainer: function() {
|
||||
return this.getInstance().getContainer();
|
||||
},
|
||||
/**
|
||||
* @return boolean
|
||||
*/
|
||||
isDirty: function() {
|
||||
return this.getInstance().isDirty();
|
||||
},
|
||||
|
||||
/**
|
||||
* Get the closest node matching the current selection.
|
||||
*
|
||||
* Returns: {jQuery} DOMElement
|
||||
*/
|
||||
getSelectedNode: function() {
|
||||
return this.getInstance().selection.getNode();
|
||||
},
|
||||
/**
|
||||
* HTML representation of the edited content.
|
||||
*
|
||||
* Returns: {String}
|
||||
*/
|
||||
getContent: function() {
|
||||
return this.getInstance().getContent();
|
||||
},
|
||||
|
||||
/**
|
||||
* Select the given node within the editor DOM
|
||||
*
|
||||
* Parameters: {DOMElement}
|
||||
*/
|
||||
selectNode: function(node) {
|
||||
this.getInstance().selection.select(node);
|
||||
},
|
||||
/**
|
||||
* DOM tree of the edited content
|
||||
*
|
||||
* Returns: DOMElement
|
||||
*/
|
||||
getDOM: function() {
|
||||
return this.getInstance().getElement();
|
||||
},
|
||||
|
||||
/**
|
||||
* Replace entire content
|
||||
*
|
||||
* @param {String} html
|
||||
* @param {Object} opts
|
||||
*/
|
||||
setContent: function(html, opts) {
|
||||
this.getInstance().setContent(html, opts);
|
||||
},
|
||||
/**
|
||||
* Returns: DOMElement
|
||||
*/
|
||||
getContainer: function() {
|
||||
return this.getInstance().getContainer();
|
||||
},
|
||||
|
||||
/**
|
||||
* Insert content at the current caret position
|
||||
*
|
||||
* @param {String} html
|
||||
* @param {Object} opts
|
||||
*/
|
||||
insertContent: function(html, opts) {
|
||||
this.getInstance().insertContent(html, opts);
|
||||
},
|
||||
/**
|
||||
* Replace currently selected content
|
||||
*
|
||||
* @param {String} html
|
||||
*/
|
||||
replaceContent: function(html, opts) {
|
||||
this.getInstance().execCommand('mceReplaceContent', false, html, opts);
|
||||
},
|
||||
/**
|
||||
* Insert or update a link in the content area (based on current editor selection)
|
||||
*
|
||||
* Parameters: {Object} attrs
|
||||
*/
|
||||
insertLink: function(attrs, opts) {
|
||||
this.getInstance().execCommand("mceInsertLink", false, attrs, opts);
|
||||
},
|
||||
/**
|
||||
* Remove the link from the currently selected node (if any).
|
||||
*/
|
||||
removeLink: function() {
|
||||
this.getInstance().execCommand('unlink', false);
|
||||
},
|
||||
/**
|
||||
* Strip any editor-specific notation from link in order to make it presentable in the UI.
|
||||
*
|
||||
* Parameters:
|
||||
* {Object}
|
||||
* {DOMElement}
|
||||
*/
|
||||
cleanLink: function(href, node) {
|
||||
var settings = this.getConfig,
|
||||
cb = settings['urlconverter_callback'];
|
||||
if(cb) href = eval(cb + "(href, node, true);");
|
||||
/**
|
||||
* Get the closest node matching the current selection.
|
||||
*
|
||||
* Returns: {jQuery} DOMElement
|
||||
*/
|
||||
getSelectedNode: function() {
|
||||
return this.getInstance().selection.getNode();
|
||||
},
|
||||
|
||||
// Turn into relative
|
||||
if(href.match(new RegExp('^' + tinyMCE.settings['document_base_url'] + '(.*)$'))) {
|
||||
href = RegExp.$1;
|
||||
}
|
||||
/**
|
||||
* Select the given node within the editor DOM
|
||||
*
|
||||
* Parameters: {DOMElement}
|
||||
*/
|
||||
selectNode: function(node) {
|
||||
this.getInstance().selection.select(node);
|
||||
},
|
||||
|
||||
// Get rid of TinyMCE's temporary URLs
|
||||
if(href.match(/^javascript:\s*mctmp/)) href = '';
|
||||
/**
|
||||
* Replace entire content
|
||||
*
|
||||
* @param {String} html
|
||||
* @param {Object} opts
|
||||
*/
|
||||
setContent: function(html, opts) {
|
||||
this.getInstance().setContent(html, opts);
|
||||
},
|
||||
|
||||
return href;
|
||||
},
|
||||
/**
|
||||
* Creates a bookmark for the currently selected range,
|
||||
* which can be used to reselect this range at a later point.
|
||||
* @return {mixed}
|
||||
*/
|
||||
createBookmark: function() {
|
||||
return this.getInstance().selection.getBookmark();
|
||||
},
|
||||
/**
|
||||
* Selects a bookmarked range previously saved through createBookmark().
|
||||
* @param {mixed} bookmark
|
||||
*/
|
||||
moveToBookmark: function(bookmark) {
|
||||
this.getInstance().selection.moveToBookmark(bookmark);
|
||||
this.getInstance().focus();
|
||||
},
|
||||
/**
|
||||
* Removes any selection & de-focuses this editor
|
||||
*/
|
||||
blur: function() {
|
||||
this.getInstance().selection.collapse();
|
||||
},
|
||||
/**
|
||||
* Add new undo point with the current DOM content.
|
||||
*/
|
||||
addUndo: function() {
|
||||
this.getInstance().undoManager.add();
|
||||
}
|
||||
};
|
||||
/**
|
||||
* Insert content at the current caret position
|
||||
*
|
||||
* @param {String} html
|
||||
* @param {Object} opts
|
||||
*/
|
||||
insertContent: function(html, opts) {
|
||||
this.getInstance().insertContent(html, opts);
|
||||
},
|
||||
/**
|
||||
* Replace currently selected content
|
||||
*
|
||||
* @param {String} html
|
||||
*/
|
||||
replaceContent: function(html, opts) {
|
||||
this.getInstance().execCommand('mceReplaceContent', false, html, opts);
|
||||
},
|
||||
/**
|
||||
* Insert or update a link in the content area (based on current editor selection)
|
||||
*
|
||||
* Parameters: {Object} attrs
|
||||
*/
|
||||
insertLink: function(attrs, opts) {
|
||||
this.getInstance().execCommand("mceInsertLink", false, attrs, opts);
|
||||
},
|
||||
/**
|
||||
* Remove the link from the currently selected node (if any).
|
||||
*/
|
||||
removeLink: function() {
|
||||
this.getInstance().execCommand('unlink', false);
|
||||
},
|
||||
/**
|
||||
* Strip any editor-specific notation from link in order to make it presentable in the UI.
|
||||
*
|
||||
* Parameters:
|
||||
* {Object}
|
||||
* {DOMElement}
|
||||
*/
|
||||
cleanLink: function(href, node) {
|
||||
var settings = this.getConfig,
|
||||
cb = settings['urlconverter_callback'];
|
||||
if(cb) href = eval(cb + "(href, node, true);");
|
||||
|
||||
// Turn into relative
|
||||
if(href.match(new RegExp('^' + tinyMCE.settings['document_base_url'] + '(.*)$'))) {
|
||||
href = RegExp.$1;
|
||||
}
|
||||
|
||||
// Get rid of TinyMCE's temporary URLs
|
||||
if(href.match(/^javascript:\s*mctmp/)) href = '';
|
||||
|
||||
return href;
|
||||
},
|
||||
/**
|
||||
* Creates a bookmark for the currently selected range,
|
||||
* which can be used to reselect this range at a later point.
|
||||
* @return {mixed}
|
||||
*/
|
||||
createBookmark: function() {
|
||||
return this.getInstance().selection.getBookmark();
|
||||
},
|
||||
/**
|
||||
* Selects a bookmarked range previously saved through createBookmark().
|
||||
* @param {mixed} bookmark
|
||||
*/
|
||||
moveToBookmark: function(bookmark) {
|
||||
this.getInstance().selection.moveToBookmark(bookmark);
|
||||
this.getInstance().focus();
|
||||
},
|
||||
/**
|
||||
* Removes any selection & de-focuses this editor
|
||||
*/
|
||||
blur: function() {
|
||||
this.getInstance().selection.collapse();
|
||||
},
|
||||
/**
|
||||
* Add new undo point with the current DOM content.
|
||||
*/
|
||||
addUndo: function() {
|
||||
this.getInstance().undoManager.add();
|
||||
}
|
||||
};
|
||||
});
|
||||
// Override this to switch editor wrappers
|
||||
ss.editorWrappers['default'] = ss.editorWrappers.tinyMCE;
|
||||
|
||||
$.entwine('ss', function($) {
|
||||
|
||||
/**
|
||||
* Class: textarea.htmleditor
|
||||
*
|
||||
* Add tinymce to HtmlEditorFields within the CMS. Works in combination
|
||||
* with a TinyMCE.init() call which is prepopulated with the used HTMLEditorConfig settings,
|
||||
* and included in the page as an inline <script> tag.
|
||||
*/
|
||||
$('textarea.htmleditor').entwine({
|
||||
/**
|
||||
* Class: textarea.htmleditor
|
||||
*
|
||||
* Add tinymce to HtmlEditorFields within the CMS. Works in combination
|
||||
* with a TinyMCE.init() call which is prepopulated with the used HTMLEditorConfig settings,
|
||||
* and included in the page as an inline <script> tag.
|
||||
*/
|
||||
$('textarea.htmleditor').entwine({
|
||||
|
||||
Editor: null,
|
||||
Editor: null,
|
||||
|
||||
/**
|
||||
* Constructor: onmatch
|
||||
*/
|
||||
onadd: function() {
|
||||
var edClass = this.data('editor') || 'default',
|
||||
ed = ss.editorWrappers[edClass]();
|
||||
this.setEditor(ed);
|
||||
/**
|
||||
* Constructor: onmatch
|
||||
*/
|
||||
onadd: function() {
|
||||
var edClass = this.data('editor') || 'default',
|
||||
ed = ss.editorWrappers[edClass]();
|
||||
this.setEditor(ed);
|
||||
|
||||
ed.init(this.attr('id'));
|
||||
ed.init(this.attr('id'));
|
||||
|
||||
this._super();
|
||||
},
|
||||
this._super();
|
||||
},
|
||||
|
||||
/**
|
||||
* Destructor: onunmatch
|
||||
*/
|
||||
onremove: function() {
|
||||
this.getEditor().destroy();
|
||||
this._super();
|
||||
},
|
||||
/**
|
||||
* Destructor: onunmatch
|
||||
*/
|
||||
onremove: function() {
|
||||
this.getEditor().destroy();
|
||||
this._super();
|
||||
},
|
||||
|
||||
/**
|
||||
* Make sure the editor has flushed all it's buffers before the form is submitted.
|
||||
*/
|
||||
'from .cms-edit-form': {
|
||||
onbeforesubmitform: function() {
|
||||
this.getEditor().save();
|
||||
this._super();
|
||||
}
|
||||
},
|
||||
/**
|
||||
* Make sure the editor has flushed all it's buffers before the form is submitted.
|
||||
*/
|
||||
'from .cms-edit-form': {
|
||||
onbeforesubmitform: function() {
|
||||
this.getEditor().save();
|
||||
this._super();
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
* Triggers insert-link dialog
|
||||
* See editor_plugin_src.js
|
||||
*/
|
||||
openLinkDialog: function() {
|
||||
this.openDialog('link');
|
||||
},
|
||||
/**
|
||||
* Triggers insert-link dialog
|
||||
* See editor_plugin_src.js
|
||||
*/
|
||||
openLinkDialog: function() {
|
||||
this.openDialog('link');
|
||||
},
|
||||
|
||||
/**
|
||||
* Triggers insert-media dialog
|
||||
* See editor_plugin_src.js
|
||||
*/
|
||||
openMediaDialog: function() {
|
||||
this.openDialog('media');
|
||||
},
|
||||
/**
|
||||
* Triggers insert-media dialog
|
||||
* See editor_plugin_src.js
|
||||
*/
|
||||
openMediaDialog: function() {
|
||||
this.openDialog('media');
|
||||
},
|
||||
|
||||
openDialog: function(type) {
|
||||
var capitalize = function(text) {
|
||||
return text.charAt(0).toUpperCase() + text.slice(1).toLowerCase();
|
||||
};
|
||||
openDialog: function(type) {
|
||||
var capitalize = function(text) {
|
||||
return text.charAt(0).toUpperCase() + text.slice(1).toLowerCase();
|
||||
};
|
||||
|
||||
var self = this,
|
||||
url = $('#cms-editor-dialogs').data('url' + capitalize(type) + 'form'),
|
||||
dialog = $('.htmleditorfield-' + type + 'dialog');
|
||||
var self = this,
|
||||
url = $('#cms-editor-dialogs').data('url' + capitalize(type) + 'form'),
|
||||
dialog = $('.htmleditorfield-' + type + 'dialog');
|
||||
|
||||
if(dialog.length) {
|
||||
// Clean existing dialog for reload
|
||||
dialog.getForm().setElement(this);
|
||||
dialog.html('');
|
||||
dialog.addClass('loading');
|
||||
dialog.open();
|
||||
} else {
|
||||
// Show a placeholder for instant feedback. Will be replaced with actual
|
||||
// form dialog once its loaded.
|
||||
dialog = $('<div class="htmleditorfield-dialog htmleditorfield-' + type + 'dialog loading">');
|
||||
$('body').append(dialog);
|
||||
}
|
||||
if(dialog.length) {
|
||||
// Clean existing dialog for reload
|
||||
dialog.getForm().setElement(this);
|
||||
dialog.html('');
|
||||
dialog.addClass('loading');
|
||||
dialog.open();
|
||||
} else {
|
||||
// Show a placeholder for instant feedback. Will be replaced with actual
|
||||
// form dialog once its loaded.
|
||||
dialog = $('<div class="htmleditorfield-dialog htmleditorfield-' + type + 'dialog loading">');
|
||||
$('body').append(dialog);
|
||||
}
|
||||
|
||||
$.ajax({
|
||||
url: url,
|
||||
complete: function() {
|
||||
dialog.removeClass('loading');
|
||||
},
|
||||
success: function(html) {
|
||||
dialog.html(html);
|
||||
dialog.getForm().setElement(self);
|
||||
dialog.trigger('ssdialogopen');
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
$.ajax({
|
||||
url: url,
|
||||
complete: function() {
|
||||
dialog.removeClass('loading');
|
||||
},
|
||||
success: function(html) {
|
||||
dialog.html(html);
|
||||
dialog.getForm().setElement(self);
|
||||
dialog.trigger('ssdialogopen');
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
$('.htmleditorfield-dialog').entwine({
|
||||
onadd: function() {
|
||||
$('.htmleditorfield-dialog').entwine({
|
||||
onadd: function() {
|
||||
|
||||
// Create jQuery dialog
|
||||
if (!this.is('.ui-dialog-content')) {
|
||||
this.ssdialog({
|
||||
autoOpen: true,
|
||||
buttons: {
|
||||
'insert': {
|
||||
text: i18n._t(
|
||||
'HtmlEditorField.INSERT',
|
||||
'Insert'
|
||||
),
|
||||
'data-icon': 'accept',
|
||||
class: 'ss-ui-action-constructive media-insert',
|
||||
click: function() {
|
||||
$(this).find('form').submit();
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
// Create jQuery dialog
|
||||
if (!this.is('.ui-dialog-content')) {
|
||||
this.ssdialog({
|
||||
autoOpen: true,
|
||||
buttons: {
|
||||
'insert': {
|
||||
text: i18n._t(
|
||||
'HtmlEditorField.INSERT',
|
||||
'Insert'
|
||||
),
|
||||
'data-icon': 'accept',
|
||||
class: 'ss-ui-action-constructive media-insert',
|
||||
click: function() {
|
||||
$(this).find('form').submit();
|
||||
}
|
||||
}
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
this._super();
|
||||
},
|
||||
this._super();
|
||||
},
|
||||
|
||||
getForm: function() {
|
||||
return this.find('form');
|
||||
},
|
||||
open: function() {
|
||||
this.ssdialog('open');
|
||||
},
|
||||
close: function() {
|
||||
this.ssdialog('close');
|
||||
},
|
||||
toggle: function(bool) {
|
||||
if(this.is(':visible')) this.close();
|
||||
else this.open();
|
||||
},
|
||||
onscroll: function () {
|
||||
this.animate({
|
||||
scrollTop: this.find('form').height()
|
||||
}, 500);
|
||||
}
|
||||
});
|
||||
getForm: function() {
|
||||
return this.find('form');
|
||||
},
|
||||
open: function() {
|
||||
this.ssdialog('open');
|
||||
},
|
||||
close: function() {
|
||||
this.ssdialog('close');
|
||||
},
|
||||
toggle: function(bool) {
|
||||
if(this.is(':visible')) this.close();
|
||||
else this.open();
|
||||
},
|
||||
onscroll: function () {
|
||||
this.animate({
|
||||
scrollTop: this.find('form').height()
|
||||
}, 500);
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Base form implementation for interactions with an editor instance,
|
||||
* mostly geared towards modification and insertion of content.
|
||||
*/
|
||||
$('form.htmleditorfield-form').entwine({
|
||||
Selection: null,
|
||||
/**
|
||||
* Base form implementation for interactions with an editor instance,
|
||||
* mostly geared towards modification and insertion of content.
|
||||
*/
|
||||
$('form.htmleditorfield-form').entwine({
|
||||
Selection: null,
|
||||
|
||||
// Implementation-dependent serialization of the current editor selection state
|
||||
Bookmark: null,
|
||||
// Implementation-dependent serialization of the current editor selection state
|
||||
Bookmark: null,
|
||||
|
||||
// DOMElement pointing to the currently active textarea
|
||||
Element: null,
|
||||
// DOMElement pointing to the currently active textarea
|
||||
Element: null,
|
||||
|
||||
setSelection: function(node) {
|
||||
return this._super($(node));
|
||||
},
|
||||
setSelection: function(node) {
|
||||
return this._super($(node));
|
||||
},
|
||||
|
||||
onadd: function() {
|
||||
// Move title from headline to (jQuery compatible) title attribute
|
||||
var titleEl = this.find(':header:first');
|
||||
this.getDialog().attr('title', titleEl.text());
|
||||
onadd: function() {
|
||||
// Move title from headline to (jQuery compatible) title attribute
|
||||
var titleEl = this.find(':header:first');
|
||||
this.getDialog().attr('title', titleEl.text());
|
||||
|
||||
this._super();
|
||||
},
|
||||
onremove: function() {
|
||||
this.setSelection(null);
|
||||
this.setBookmark(null);
|
||||
this.setElement(null);
|
||||
this._super();
|
||||
},
|
||||
onremove: function() {
|
||||
this.setSelection(null);
|
||||
this.setBookmark(null);
|
||||
this.setElement(null);
|
||||
|
||||
this._super();
|
||||
},
|
||||
this._super();
|
||||
},
|
||||
|
||||
getDialog: function() {
|
||||
// TODO Refactor to listen to form events to remove two-way coupling
|
||||
return this.closest('.htmleditorfield-dialog');
|
||||
},
|
||||
getDialog: function() {
|
||||
// TODO Refactor to listen to form events to remove two-way coupling
|
||||
return this.closest('.htmleditorfield-dialog');
|
||||
},
|
||||
|
||||
fromDialog: {
|
||||
onssdialogopen: function(){
|
||||
var ed = this.getEditor();
|
||||
fromDialog: {
|
||||
onssdialogopen: function(){
|
||||
var ed = this.getEditor();
|
||||
|
||||
this.setSelection(ed.getSelectedNode());
|
||||
this.setBookmark(ed.createBookmark());
|
||||
this.setSelection(ed.getSelectedNode());
|
||||
this.setBookmark(ed.createBookmark());
|
||||
|
||||
ed.blur();
|
||||
ed.blur();
|
||||
|
||||
this.find(':input:not(:submit)[data-skip-autofocus!="true"]')
|
||||
.filter(':visible:enabled')
|
||||
.eq(0)
|
||||
.focus();
|
||||
this.find(':input:not(:submit)[data-skip-autofocus!="true"]')
|
||||
.filter(':visible:enabled')
|
||||
.eq(0)
|
||||
.focus();
|
||||
|
||||
this.redraw();
|
||||
this.updateFromEditor();
|
||||
},
|
||||
this.redraw();
|
||||
this.updateFromEditor();
|
||||
},
|
||||
|
||||
onssdialogclose: function(){
|
||||
var ed = this.getEditor();
|
||||
onssdialogclose: function(){
|
||||
var ed = this.getEditor();
|
||||
|
||||
ed.moveToBookmark(this.getBookmark());
|
||||
ed.moveToBookmark(this.getBookmark());
|
||||
|
||||
this.setSelection(null);
|
||||
this.setBookmark(null);
|
||||
this.setSelection(null);
|
||||
this.setBookmark(null);
|
||||
|
||||
this.resetFields();
|
||||
}
|
||||
},
|
||||
this.resetFields();
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
* @return Object ss.editorWrapper instance
|
||||
*/
|
||||
getEditor: function(){
|
||||
return this.getElement().getEditor();
|
||||
},
|
||||
/**
|
||||
* @return Object ss.editorWrapper instance
|
||||
*/
|
||||
getEditor: function(){
|
||||
return this.getElement().getEditor();
|
||||
},
|
||||
|
||||
modifySelection: function(callback) {
|
||||
var ed = this.getEditor();
|
||||
modifySelection: function(callback) {
|
||||
var ed = this.getEditor();
|
||||
|
||||
ed.moveToBookmark(this.getBookmark());
|
||||
callback.call(this, ed);
|
||||
ed.moveToBookmark(this.getBookmark());
|
||||
callback.call(this, ed);
|
||||
|
||||
this.setSelection(ed.getSelectedNode());
|
||||
this.setBookmark(ed.createBookmark());
|
||||
this.setSelection(ed.getSelectedNode());
|
||||
this.setBookmark(ed.createBookmark());
|
||||
|
||||
ed.blur();
|
||||
},
|
||||
ed.blur();
|
||||
},
|
||||
|
||||
updateFromEditor: function() {
|
||||
/* NOP */
|
||||
},
|
||||
redraw: function() {
|
||||
/* NOP */
|
||||
},
|
||||
resetFields: function() {
|
||||
// Flush the tree drop down fields, as their content might get changed in other parts of the CMS, ie in Files and images
|
||||
this.find('.tree-holder').empty();
|
||||
}
|
||||
});
|
||||
updateFromEditor: function() {
|
||||
/* NOP */
|
||||
},
|
||||
redraw: function() {
|
||||
/* NOP */
|
||||
},
|
||||
resetFields: function() {
|
||||
// Flush the tree drop down fields, as their content might get changed in other parts of the CMS, ie in Files and images
|
||||
this.find('.tree-holder').empty();
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Inserts and edits links in an html editor, including internal/external web links,
|
||||
* links to files on the webserver, email addresses, and anchors in the existing html content.
|
||||
* Every variation has its own fields (e.g. a "target" attribute doesn't make sense for an email link),
|
||||
* which are toggled through a type dropdown. Variations share fields, so there's only one "title" field in the form.
|
||||
*/
|
||||
$('form.htmleditorfield-linkform').entwine({
|
||||
/**
|
||||
* Inserts and edits links in an html editor, including internal/external web links,
|
||||
* links to files on the webserver, email addresses, and anchors in the existing html content.
|
||||
* Every variation has its own fields (e.g. a "target" attribute doesn't make sense for an email link),
|
||||
* which are toggled through a type dropdown. Variations share fields, so there's only one "title" field in the form.
|
||||
*/
|
||||
$('form.htmleditorfield-linkform').entwine({
|
||||
|
||||
// TODO Entwine doesn't respect submits triggered by ENTER key
|
||||
onsubmit: function(e) {
|
||||
this.insertLink();
|
||||
this.getDialog().close();
|
||||
return false;
|
||||
},
|
||||
resetFields: function() {
|
||||
this._super();
|
||||
// TODO Entwine doesn't respect submits triggered by ENTER key
|
||||
onsubmit: function(e) {
|
||||
this.insertLink();
|
||||
this.getDialog().close();
|
||||
return false;
|
||||
},
|
||||
resetFields: function() {
|
||||
this._super();
|
||||
|
||||
// Reset the form using a native call. This will also correctly reset checkboxes and radio buttons.
|
||||
this[0].reset();
|
||||
},
|
||||
redraw: function() {
|
||||
this._super();
|
||||
// Reset the form using a native call. This will also correctly reset checkboxes and radio buttons.
|
||||
this[0].reset();
|
||||
},
|
||||
redraw: function() {
|
||||
this._super();
|
||||
|
||||
var linkType = this.find(':input[name=LinkType]:checked').val();
|
||||
var linkType = this.find(':input[name=LinkType]:checked').val();
|
||||
|
||||
this.addAnchorSelector();
|
||||
this.addAnchorSelector();
|
||||
|
||||
this.resetFileField();
|
||||
this.resetFileField();
|
||||
|
||||
// Toggle field visibility depending on the link type.
|
||||
this.find('.step2').nextAll('.field').not('.field[id$="' + linkType +'_Holder"]').hide();
|
||||
this.find('.field[id$="LinkType_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
this.find('.field[id$="' + linkType +'_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
// Toggle field visibility depending on the link type.
|
||||
this.find('.step2').nextAll('.field').not('.field[id$="' + linkType +'_Holder"]').hide();
|
||||
this.find('.field[id$="LinkType_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
this.find('.field[id$="' + linkType +'_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
|
||||
if(linkType == 'internal' || linkType == 'anchor') {
|
||||
this.find('.field[id$="Anchor_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
}
|
||||
if(linkType == 'internal' || linkType == 'anchor') {
|
||||
this.find('.field[id$="Anchor_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
}
|
||||
|
||||
if(linkType == 'email') {
|
||||
this.find('.field[id$="Subject_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
} else {
|
||||
this.find('.field[id$="TargetBlank_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
}
|
||||
if(linkType == 'email') {
|
||||
this.find('.field[id$="Subject_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
} else {
|
||||
this.find('.field[id$="TargetBlank_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
}
|
||||
|
||||
if(linkType == 'anchor') {
|
||||
this.find('.field[id$="AnchorSelector_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
}
|
||||
this.find('.field[id$="Description_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
},
|
||||
/**
|
||||
* @return Object Keys: 'href', 'target', 'title'
|
||||
*/
|
||||
getLinkAttributes: function() {
|
||||
var href,
|
||||
target = null,
|
||||
subject = this.find(':input[name=Subject]').val(),
|
||||
anchor = this.find(':input[name=Anchor]').val();
|
||||
if(linkType == 'anchor') {
|
||||
this.find('.field[id$="AnchorSelector_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
}
|
||||
this.find('.field[id$="Description_Holder"]').attr('style', 'display: -webkit-flex; display: flex');
|
||||
},
|
||||
/**
|
||||
* @return Object Keys: 'href', 'target', 'title'
|
||||
*/
|
||||
getLinkAttributes: function() {
|
||||
var href,
|
||||
target = null,
|
||||
subject = this.find(':input[name=Subject]').val(),
|
||||
anchor = this.find(':input[name=Anchor]').val();
|
||||
|
||||
// Determine target
|
||||
if(this.find(':input[name=TargetBlank]').is(':checked')) {
|
||||
target = '_blank';
|
||||
}
|
||||
// Determine target
|
||||
if(this.find(':input[name=TargetBlank]').is(':checked')) {
|
||||
target = '_blank';
|
||||
}
|
||||
|
||||
// All other attributes
|
||||
switch(this.find(':input[name=LinkType]:checked').val()) {
|
||||
case 'internal':
|
||||
href = '[sitetree_link,id=' + this.find(':input[name=internal]').val() + ']';
|
||||
// All other attributes
|
||||
switch(this.find(':input[name=LinkType]:checked').val()) {
|
||||
case 'internal':
|
||||
href = '[sitetree_link,id=' + this.find(':input[name=internal]').val() + ']';
|
||||
|
||||
if(anchor) {
|
||||
href += '#' + anchor;
|
||||
}
|
||||
if(anchor) {
|
||||
href += '#' + anchor;
|
||||
}
|
||||
|
||||
break;
|
||||
break;
|
||||
|
||||
case 'anchor':
|
||||
href = '#' + anchor;
|
||||
break;
|
||||
case 'anchor':
|
||||
href = '#' + anchor;
|
||||
break;
|
||||
|
||||
case 'file':
|
||||
case 'file':
|
||||
var fileid = this.find('.ss-uploadfield .ss-uploadfield-item').attr('data-fileid');
|
||||
href = fileid ? '[file_link,id=' + fileid + ']' : '';
|
||||
break;
|
||||
break;
|
||||
|
||||
case 'email':
|
||||
href = 'mailto:' + this.find(':input[name=email]').val();
|
||||
if(subject) {
|
||||
href += '?subject=' + encodeURIComponent(subject);
|
||||
}
|
||||
target = null;
|
||||
break;
|
||||
case 'email':
|
||||
href = 'mailto:' + this.find(':input[name=email]').val();
|
||||
if(subject) {
|
||||
href += '?subject=' + encodeURIComponent(subject);
|
||||
}
|
||||
target = null;
|
||||
break;
|
||||
|
||||
// case 'external':
|
||||
default:
|
||||
href = this.find(':input[name=external]').val();
|
||||
// Prefix the URL with "http://" if no prefix is found
|
||||
if(href.indexOf('://') == -1) href = 'http://' + href;
|
||||
break;
|
||||
}
|
||||
// case 'external':
|
||||
default:
|
||||
href = this.find(':input[name=external]').val();
|
||||
// Prefix the URL with "http://" if no prefix is found
|
||||
if(href.indexOf('://') == -1) href = 'http://' + href;
|
||||
break;
|
||||
}
|
||||
|
||||
return {
|
||||
href : href,
|
||||
target : target,
|
||||
title : this.find(':input[name=Description]').val()
|
||||
};
|
||||
},
|
||||
insertLink: function() {
|
||||
this.modifySelection(function(ed){
|
||||
ed.insertLink(this.getLinkAttributes());
|
||||
});
|
||||
},
|
||||
removeLink: function() {
|
||||
this.modifySelection(function(ed){
|
||||
ed.removeLink();
|
||||
});
|
||||
return {
|
||||
href : href,
|
||||
target : target,
|
||||
title : this.find(':input[name=Description]').val()
|
||||
};
|
||||
},
|
||||
insertLink: function() {
|
||||
this.modifySelection(function(ed){
|
||||
ed.insertLink(this.getLinkAttributes());
|
||||
});
|
||||
},
|
||||
removeLink: function() {
|
||||
this.modifySelection(function(ed){
|
||||
ed.removeLink();
|
||||
});
|
||||
|
||||
this.resetFileField();
|
||||
this.close();
|
||||
},
|
||||
this.resetFileField();
|
||||
this.close();
|
||||
},
|
||||
|
||||
resetFileField: function() {
|
||||
// If there's an attached item, remove it
|
||||
var fileField = this.find('.ss-uploadfield[id$="file_Holder"]'),
|
||||
fileUpload = fileField.data('fileupload'),
|
||||
currentItem = fileField.find('.ss-uploadfield-item[data-fileid]');
|
||||
resetFileField: function() {
|
||||
// If there's an attached item, remove it
|
||||
var fileField = this.find('.ss-uploadfield[id$="file_Holder"]'),
|
||||
fileUpload = fileField.data('fileupload'),
|
||||
currentItem = fileField.find('.ss-uploadfield-item[data-fileid]');
|
||||
|
||||
if(currentItem.length) {
|
||||
fileUpload._trigger('destroy', null, {context: currentItem});
|
||||
fileField.find('.ss-uploadfield-addfile').removeClass('borderTop');
|
||||
}
|
||||
},
|
||||
if(currentItem.length) {
|
||||
fileUpload._trigger('destroy', null, {context: currentItem});
|
||||
fileField.find('.ss-uploadfield-addfile').removeClass('borderTop');
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
* Builds an anchor selector element and injects it into the DOM next to the anchor field.
|
||||
*/
|
||||
addAnchorSelector: function() {
|
||||
// Avoid adding twice
|
||||
if(this.find(':input[name=AnchorSelector]').length) return;
|
||||
/**
|
||||
* Builds an anchor selector element and injects it into the DOM next to the anchor field.
|
||||
*/
|
||||
addAnchorSelector: function() {
|
||||
// Avoid adding twice
|
||||
if(this.find(':input[name=AnchorSelector]').length) return;
|
||||
|
||||
var self = this;
|
||||
var anchorSelector = $(
|
||||
'<select id="Form_EditorToolbarLinkForm_AnchorSelector" name="AnchorSelector"></select>'
|
||||
);
|
||||
this.find(':input[name=Anchor]').parent().append(anchorSelector);
|
||||
var self = this;
|
||||
var anchorSelector = $(
|
||||
'<select id="Form_EditorToolbarLinkForm_AnchorSelector" name="AnchorSelector"></select>'
|
||||
);
|
||||
this.find(':input[name=Anchor]').parent().append(anchorSelector);
|
||||
|
||||
// Initialise the anchor dropdown.
|
||||
this.updateAnchorSelector();
|
||||
// Initialise the anchor dropdown.
|
||||
this.updateAnchorSelector();
|
||||
|
||||
// copy the value from dropdown to the text field
|
||||
anchorSelector.change(function(e) {
|
||||
self.find(':input[name="Anchor"]').val($(this).val());
|
||||
});
|
||||
},
|
||||
// copy the value from dropdown to the text field
|
||||
anchorSelector.change(function(e) {
|
||||
self.find(':input[name="Anchor"]').val($(this).val());
|
||||
});
|
||||
},
|
||||
|
||||
/**
|
||||
* Fetch relevant anchors, depending on the link type.
|
||||
*
|
||||
* @return $.Deferred A promise of an anchor array, or an error message.
|
||||
*/
|
||||
getAnchors: function() {
|
||||
var linkType = this.find(':input[name=LinkType]:checked').val();
|
||||
var dfdAnchors = $.Deferred();
|
||||
/**
|
||||
* Fetch relevant anchors, depending on the link type.
|
||||
*
|
||||
* @return $.Deferred A promise of an anchor array, or an error message.
|
||||
*/
|
||||
getAnchors: function() {
|
||||
var linkType = this.find(':input[name=LinkType]:checked').val();
|
||||
var dfdAnchors = $.Deferred();
|
||||
|
||||
switch (linkType) {
|
||||
case 'anchor':
|
||||
// Fetch from the local editor.
|
||||
var collectedAnchors = [];
|
||||
var ed = this.getEditor();
|
||||
// name attribute is defined as CDATA, should accept all characters and entities
|
||||
// http://www.w3.org/TR/1999/REC-html401-19991224/struct/links.html#h-12.2
|
||||
switch (linkType) {
|
||||
case 'anchor':
|
||||
// Fetch from the local editor.
|
||||
var collectedAnchors = [];
|
||||
var ed = this.getEditor();
|
||||
// name attribute is defined as CDATA, should accept all characters and entities
|
||||
// http://www.w3.org/TR/1999/REC-html401-19991224/struct/links.html#h-12.2
|
||||
|
||||
if(ed) {
|
||||
var raw = ed.getContent()
|
||||
.match(/\s+(name|id)\s*=\s*(["'])([^\2\s>]*?)\2|\s+(name|id)\s*=\s*([^"']+)[\s +>]/gim);
|
||||
if (raw && raw.length) {
|
||||
for(var i = 0; i < raw.length; i++) {
|
||||
var indexStart = (raw[i].indexOf('id=') == -1) ? 7 : 5;
|
||||
collectedAnchors.push(raw[i].substr(indexStart).replace(/"$/, ''));
|
||||
}
|
||||
}
|
||||
}
|
||||
if(ed) {
|
||||
var raw = ed.getContent()
|
||||
.match(/\s+(name|id)\s*=\s*(["'])([^\2\s>]*?)\2|\s+(name|id)\s*=\s*([^"']+)[\s +>]/gim);
|
||||
if (raw && raw.length) {
|
||||
for(var i = 0; i < raw.length; i++) {
|
||||
var indexStart = (raw[i].indexOf('id=') == -1) ? 7 : 5;
|
||||
collectedAnchors.push(raw[i].substr(indexStart).replace(/"$/, ''));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
dfdAnchors.resolve(collectedAnchors);
|
||||
break;
|
||||
dfdAnchors.resolve(collectedAnchors);
|
||||
break;
|
||||
|
||||
case 'internal':
|
||||
// Fetch available anchors from the target internal page.
|
||||
var pageId = this.find(':input[name=internal]').val();
|
||||
case 'internal':
|
||||
// Fetch available anchors from the target internal page.
|
||||
var pageId = this.find(':input[name=internal]').val();
|
||||
|
||||
if (pageId) {
|
||||
$.ajax({
|
||||
url: $.path.addSearchParams(
|
||||
this.attr('action').replace('LinkForm', 'getanchors'),
|
||||
{'PageID': parseInt(pageId)}
|
||||
),
|
||||
success: function(body, status, xhr) {
|
||||
dfdAnchors.resolve($.parseJSON(body));
|
||||
},
|
||||
error: function(xhr, status) {
|
||||
dfdAnchors.reject(xhr.responseText);
|
||||
}
|
||||
});
|
||||
} else {
|
||||
dfdAnchors.resolve([]);
|
||||
}
|
||||
break;
|
||||
if (pageId) {
|
||||
$.ajax({
|
||||
url: $.path.addSearchParams(
|
||||
this.attr('action').replace('LinkForm', 'getanchors'),
|
||||
{'PageID': parseInt(pageId)}
|
||||
),
|
||||
success: function(body, status, xhr) {
|
||||
dfdAnchors.resolve($.parseJSON(body));
|
||||
},
|
||||
error: function(xhr, status) {
|
||||
dfdAnchors.reject(xhr.responseText);
|
||||
}
|
||||
});
|
||||
} else {
|
||||
dfdAnchors.resolve([]);
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
// This type does not support anchors at all.
|
||||
dfdAnchors.reject(i18n._t(
|
||||
'HtmlEditorField.ANCHORSNOTSUPPORTED',
|
||||
'Anchors are not supported for this link type.'
|
||||
));
|
||||
break;
|
||||
}
|
||||
default:
|
||||
// This type does not support anchors at all.
|
||||
dfdAnchors.reject(i18n._t(
|
||||
'HtmlEditorField.ANCHORSNOTSUPPORTED',
|
||||
'Anchors are not supported for this link type.'
|
||||
));
|
||||
break;
|
||||
}
|
||||
|
||||
return dfdAnchors.promise();
|
||||
},
|
||||
return dfdAnchors.promise();
|
||||
},
|
||||
|
||||
/**
|
||||
* Update the anchor list in the dropdown.
|
||||
*/
|
||||
updateAnchorSelector: function() {
|
||||
var self = this;
|
||||
var selector = this.find(':input[name=AnchorSelector]');
|
||||
var dfdAnchors = this.getAnchors();
|
||||
/**
|
||||
* Update the anchor list in the dropdown.
|
||||
*/
|
||||
updateAnchorSelector: function() {
|
||||
var self = this;
|
||||
var selector = this.find(':input[name=AnchorSelector]');
|
||||
var dfdAnchors = this.getAnchors();
|
||||
|
||||
// Inform the user we are loading.
|
||||
selector.empty();
|
||||
selector.append($(
|
||||
'<option value="" selected="1">' +
|
||||
i18n._t('HtmlEditorField.LOOKINGFORANCHORS', 'Looking for anchors...') +
|
||||
'</option>'
|
||||
));
|
||||
// Inform the user we are loading.
|
||||
selector.empty();
|
||||
selector.append($(
|
||||
'<option value="" selected="1">' +
|
||||
i18n._t('HtmlEditorField.LOOKINGFORANCHORS', 'Looking for anchors...') +
|
||||
'</option>'
|
||||
));
|
||||
|
||||
dfdAnchors.done(function(anchors) {
|
||||
selector.empty();
|
||||
selector.append($(
|
||||
'<option value="" selected="1">' +
|
||||
i18n._t('HtmlEditorField.SelectAnchor') +
|
||||
'</option>'
|
||||
));
|
||||
dfdAnchors.done(function(anchors) {
|
||||
selector.empty();
|
||||
selector.append($(
|
||||
'<option value="" selected="1">' +
|
||||
i18n._t('HtmlEditorField.SelectAnchor') +
|
||||
'</option>'
|
||||
));
|
||||
|
||||
if (anchors) {
|
||||
for (var j = 0; j < anchors.length; j++) {
|
||||
selector.append($('<option value="'+anchors[j]+'">'+anchors[j]+'</option>'));
|
||||
}
|
||||
}
|
||||
if (anchors) {
|
||||
for (var j = 0; j < anchors.length; j++) {
|
||||
selector.append($('<option value="'+anchors[j]+'">'+anchors[j]+'</option>'));
|
||||
}
|
||||
}
|
||||
|
||||
}).fail(function(message) {
|
||||
selector.empty();
|
||||
selector.append($(
|
||||
'<option value="" selected="1">' +
|
||||
message +
|
||||
'</option>'
|
||||
));
|
||||
});
|
||||
}).fail(function(message) {
|
||||
selector.empty();
|
||||
selector.append($(
|
||||
'<option value="" selected="1">' +
|
||||
message +
|
||||
'</option>'
|
||||
));
|
||||
});
|
||||
|
||||
// Poke the selector for IE8, otherwise the changes won't be noticed.
|
||||
if ($.browser.msie) selector.hide().show();
|
||||
},
|
||||
// Poke the selector for IE8, otherwise the changes won't be noticed.
|
||||
if ($.browser.msie) selector.hide().show();
|
||||
},
|
||||
|
||||
/**
|
||||
* Updates the state of the dialog inputs to match the editor selection.
|
||||
* If selection does not contain a link, resets the fields.
|
||||
*/
|
||||
updateFromEditor: function() {
|
||||
var htmlTagPattern = /<\S[^><]*>/g, fieldName, data = this.getCurrentLink();
|
||||
/**
|
||||
* Updates the state of the dialog inputs to match the editor selection.
|
||||
* If selection does not contain a link, resets the fields.
|
||||
*/
|
||||
updateFromEditor: function() {
|
||||
var htmlTagPattern = /<\S[^><]*>/g, fieldName, data = this.getCurrentLink();
|
||||
|
||||
if(data) {
|
||||
for(fieldName in data) {
|
||||
var el = this.find(':input[name=' + fieldName + ']'), selected = data[fieldName];
|
||||
// Remove html tags in the selected text that occurs on IE browsers
|
||||
if(typeof(selected) == 'string') selected = selected.replace(htmlTagPattern, '');
|
||||
if(data) {
|
||||
for(fieldName in data) {
|
||||
var el = this.find(':input[name=' + fieldName + ']'), selected = data[fieldName];
|
||||
// Remove html tags in the selected text that occurs on IE browsers
|
||||
if(typeof(selected) == 'string') selected = selected.replace(htmlTagPattern, '');
|
||||
|
||||
// Set values and invoke the triggers (e.g. for TreeDropdownField).
|
||||
if(el.is(':checkbox')) {
|
||||
el.prop('checked', selected).change();
|
||||
} else if(el.is(':radio')) {
|
||||
el.val([selected]).change();
|
||||
} else if(fieldName == 'file') {
|
||||
// UploadField inputs have a slightly different naming convention
|
||||
el = this.find(':input[name="' + fieldName + '[Uploads][]"]');
|
||||
// We need the UploadField "field", not just the input
|
||||
el = el.parents('.ss-uploadfield');
|
||||
// Set values and invoke the triggers (e.g. for TreeDropdownField).
|
||||
if(el.is(':checkbox')) {
|
||||
el.prop('checked', selected).change();
|
||||
} else if(el.is(':radio')) {
|
||||
el.val([selected]).change();
|
||||
} else if(fieldName == 'file') {
|
||||
// UploadField inputs have a slightly different naming convention
|
||||
el = this.find(':input[name="' + fieldName + '[Uploads][]"]');
|
||||
// We need the UploadField "field", not just the input
|
||||
el = el.parents('.ss-uploadfield');
|
||||
|
||||
// We have to wait for the UploadField to initialise
|
||||
(function attach(el, selected) {
|
||||
if( ! el.getConfig()) {
|
||||
setTimeout(function(){ attach(el, selected); }, 50);
|
||||
} else {
|
||||
el.attachFiles([selected]);
|
||||
}
|
||||
})(el, selected);
|
||||
} else {
|
||||
el.val(selected).change();
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
// We have to wait for the UploadField to initialise
|
||||
(function attach(el, selected) {
|
||||
if( ! el.getConfig()) {
|
||||
setTimeout(function(){ attach(el, selected); }, 50);
|
||||
} else {
|
||||
el.attachFiles([selected]);
|
||||
}
|
||||
})(el, selected);
|
||||
} else {
|
||||
el.val(selected).change();
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
/**
|
||||
* Return information about the currently selected link, suitable for population of the link form.
|
||||
*
|
||||
* Returns null if no link was currently selected.
|
||||
*/
|
||||
getCurrentLink: function() {
|
||||
var selectedEl = this.getSelection(),
|
||||
href = "", target = "", title = "", action = "insert", style_class = "";
|
||||
/**
|
||||
* Return information about the currently selected link, suitable for population of the link form.
|
||||
*
|
||||
* Returns null if no link was currently selected.
|
||||
*/
|
||||
getCurrentLink: function() {
|
||||
var selectedEl = this.getSelection(),
|
||||
href = "", target = "", title = "", action = "insert", style_class = "";
|
||||
|
||||
// We use a separate field for linkDataSource from tinyMCE.linkElement.
|
||||
// If we have selected beyond the range of an <a> element, then use use that <a> element to get the link data source,
|
||||
// but we don't use it as the destination for the link insertion
|
||||
var linkDataSource = null;
|
||||
if(selectedEl.length) {
|
||||
if(selectedEl.is('a')) {
|
||||
// Element is a link
|
||||
linkDataSource = selectedEl;
|
||||
// TODO Limit to inline elements, otherwise will also apply to e.g. paragraphs which already contain one or more links
|
||||
// } else if((selectedEl.find('a').length)) {
|
||||
// // Element contains a link
|
||||
// var firstLinkEl = selectedEl.find('a:first');
|
||||
// if(firstLinkEl.length) linkDataSource = firstLinkEl;
|
||||
} else {
|
||||
// Element is a child of a link
|
||||
linkDataSource = selectedEl = selectedEl.parents('a:first');
|
||||
}
|
||||
}
|
||||
if(linkDataSource && linkDataSource.length) this.modifySelection(function(ed){
|
||||
ed.selectNode(linkDataSource[0]);
|
||||
});
|
||||
// We use a separate field for linkDataSource from tinyMCE.linkElement.
|
||||
// If we have selected beyond the range of an <a> element, then use use that <a> element to get the link data source,
|
||||
// but we don't use it as the destination for the link insertion
|
||||
var linkDataSource = null;
|
||||
if(selectedEl.length) {
|
||||
if(selectedEl.is('a')) {
|
||||
// Element is a link
|
||||
linkDataSource = selectedEl;
|
||||
// TODO Limit to inline elements, otherwise will also apply to e.g. paragraphs which already contain one or more links
|
||||
// } else if((selectedEl.find('a').length)) {
|
||||
// // Element contains a link
|
||||
// var firstLinkEl = selectedEl.find('a:first');
|
||||
// if(firstLinkEl.length) linkDataSource = firstLinkEl;
|
||||
} else {
|
||||
// Element is a child of a link
|
||||
linkDataSource = selectedEl = selectedEl.parents('a:first');
|
||||
}
|
||||
}
|
||||
if(linkDataSource && linkDataSource.length) this.modifySelection(function(ed){
|
||||
ed.selectNode(linkDataSource[0]);
|
||||
});
|
||||
|
||||
// Is anchor not a link
|
||||
if (!linkDataSource.attr('href')) linkDataSource = null;
|
||||
// Is anchor not a link
|
||||
if (!linkDataSource.attr('href')) linkDataSource = null;
|
||||
|
||||
if (linkDataSource) {
|
||||
href = linkDataSource.attr('href');
|
||||
target = linkDataSource.attr('target');
|
||||
title = linkDataSource.attr('title');
|
||||
style_class = linkDataSource.attr('class');
|
||||
href = this.getEditor().cleanLink(href, linkDataSource);
|
||||
action = "update";
|
||||
}
|
||||
if (linkDataSource) {
|
||||
href = linkDataSource.attr('href');
|
||||
target = linkDataSource.attr('target');
|
||||
title = linkDataSource.attr('title');
|
||||
style_class = linkDataSource.attr('class');
|
||||
href = this.getEditor().cleanLink(href, linkDataSource);
|
||||
action = "update";
|
||||
}
|
||||
|
||||
if(href.match(/^mailto:(.*)$/)) {
|
||||
return {
|
||||
LinkType: 'email',
|
||||
email: RegExp.$1,
|
||||
Description: title
|
||||
};
|
||||
} else if(href.match(/^(assets\/.*)$/) || href.match(/^\[file_link\s*(?:\s*|%20|,)?id=([0-9]+)\]?(#.*)?$/)) {
|
||||
return {
|
||||
LinkType: 'file',
|
||||
file: RegExp.$1,
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else if(href.match(/^#(.*)$/)) {
|
||||
return {
|
||||
LinkType: 'anchor',
|
||||
Anchor: RegExp.$1,
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else if(href.match(/^\[sitetree_link(?:\s*|%20|,)?id=([0-9]+)\]?(#.*)?$/i)) {
|
||||
return {
|
||||
LinkType: 'internal',
|
||||
internal: RegExp.$1,
|
||||
Anchor: RegExp.$2 ? RegExp.$2.substr(1) : '',
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else if(href) {
|
||||
return {
|
||||
LinkType: 'external',
|
||||
external: href,
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else {
|
||||
// No link/invalid link selected.
|
||||
return null;
|
||||
}
|
||||
}
|
||||
});
|
||||
if(href.match(/^mailto:(.*)$/)) {
|
||||
return {
|
||||
LinkType: 'email',
|
||||
email: RegExp.$1,
|
||||
Description: title
|
||||
};
|
||||
} else if(href.match(/^(assets\/.*)$/) || href.match(/^\[file_link\s*(?:\s*|%20|,)?id=([0-9]+)\]?(#.*)?$/)) {
|
||||
return {
|
||||
LinkType: 'file',
|
||||
file: RegExp.$1,
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else if(href.match(/^#(.*)$/)) {
|
||||
return {
|
||||
LinkType: 'anchor',
|
||||
Anchor: RegExp.$1,
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else if(href.match(/^\[sitetree_link(?:\s*|%20|,)?id=([0-9]+)\]?(#.*)?$/i)) {
|
||||
return {
|
||||
LinkType: 'internal',
|
||||
internal: RegExp.$1,
|
||||
Anchor: RegExp.$2 ? RegExp.$2.substr(1) : '',
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else if(href) {
|
||||
return {
|
||||
LinkType: 'external',
|
||||
external: href,
|
||||
Description: title,
|
||||
TargetBlank: target ? true : false
|
||||
};
|
||||
} else {
|
||||
// No link/invalid link selected.
|
||||
return null;
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
$('form.htmleditorfield-linkform input[name=LinkType]').entwine({
|
||||
onclick: function(e) {
|
||||
this.parents('form:first').redraw();
|
||||
this._super();
|
||||
},
|
||||
onchange: function() {
|
||||
this.parents('form:first').redraw();
|
||||
$('form.htmleditorfield-linkform input[name=LinkType]').entwine({
|
||||
onclick: function(e) {
|
||||
this.parents('form:first').redraw();
|
||||
this._super();
|
||||
},
|
||||
onchange: function() {
|
||||
this.parents('form:first').redraw();
|
||||
|
||||
// Update if a anchor-supporting link type is selected.
|
||||
var linkType = this.parent().find(':checked').val();
|
||||
if (linkType==='anchor' || linkType==='internal') {
|
||||
this.parents('form.htmleditorfield-linkform').updateAnchorSelector();
|
||||
}
|
||||
this._super();
|
||||
}
|
||||
});
|
||||
// Update if a anchor-supporting link type is selected.
|
||||
var linkType = this.parent().find(':checked').val();
|
||||
if (linkType==='anchor' || linkType==='internal') {
|
||||
this.parents('form.htmleditorfield-linkform').updateAnchorSelector();
|
||||
}
|
||||
this._super();
|
||||
}
|
||||
});
|
||||
|
||||
$('form.htmleditorfield-linkform input[name=internal]').entwine({
|
||||
/**
|
||||
* Update the anchor dropdown if a different page is selected in the "internal" dropdown.
|
||||
*/
|
||||
onvalueupdated: function() {
|
||||
this.parents('form.htmleditorfield-linkform').updateAnchorSelector();
|
||||
this._super();
|
||||
}
|
||||
});
|
||||
$('form.htmleditorfield-linkform input[name=internal]').entwine({
|
||||
/**
|
||||
* Update the anchor dropdown if a different page is selected in the "internal" dropdown.
|
||||
*/
|
||||
onvalueupdated: function() {
|
||||
this.parents('form.htmleditorfield-linkform').updateAnchorSelector();
|
||||
this._super();
|
||||
}
|
||||
});
|
||||
|
||||
$('form.htmleditorfield-linkform :submit[name=action_remove]').entwine({
|
||||
onclick: function(e) {
|
||||
this.parents('form:first').removeLink();
|
||||
this._super();
|
||||
return false;
|
||||
}
|
||||
});
|
||||
$('form.htmleditorfield-linkform :submit[name=action_remove]').entwine({
|
||||
onclick: function(e) {
|
||||
this.parents('form:first').removeLink();
|
||||
this._super();
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Responsible for inserting media files, although only images are supported so far.
|
||||
* Allows to select one or more files, and load form fields for each file via ajax.
|
||||
* This allows us to tailor the form fields to the file type (e.g. different ones for images and flash),
|
||||
* as well as add new form fields via framework extensions.
|
||||
* The inputs on each of those files are used for constructing the HTML to insert into
|
||||
* the rich text editor. Also allows editing the properties of existing files if any are selected in the editor.
|
||||
* Note: Not each file has a representation on the webserver filesystem, supports insertion and editing
|
||||
* of remove files as well.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform').entwine({
|
||||
toggleCloseButton: function(){
|
||||
var updateExisting = Boolean(this.find('.ss-htmleditorfield-file').length);
|
||||
this.find('.overview .action-delete')[updateExisting ? 'hide' : 'show']();
|
||||
},
|
||||
onsubmit: function() {
|
||||
this.modifySelection(function(ed){
|
||||
this.find('.ss-htmleditorfield-file').each(function() {
|
||||
$(this).insertHTML(ed);
|
||||
});
|
||||
});
|
||||
/**
|
||||
* Responsible for inserting media files, although only images are supported so far.
|
||||
* Allows to select one or more files, and load form fields for each file via ajax.
|
||||
* This allows us to tailor the form fields to the file type (e.g. different ones for images and flash),
|
||||
* as well as add new form fields via framework extensions.
|
||||
* The inputs on each of those files are used for constructing the HTML to insert into
|
||||
* the rich text editor. Also allows editing the properties of existing files if any are selected in the editor.
|
||||
* Note: Not each file has a representation on the webserver filesystem, supports insertion and editing
|
||||
* of remove files as well.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform').entwine({
|
||||
toggleCloseButton: function(){
|
||||
var updateExisting = Boolean(this.find('.ss-htmleditorfield-file').length);
|
||||
this.find('.overview .action-delete')[updateExisting ? 'hide' : 'show']();
|
||||
},
|
||||
onsubmit: function() {
|
||||
this.modifySelection(function(ed){
|
||||
this.find('.ss-htmleditorfield-file').each(function() {
|
||||
$(this).insertHTML(ed);
|
||||
});
|
||||
});
|
||||
|
||||
this.getDialog().close();
|
||||
return false;
|
||||
},
|
||||
updateFromEditor: function() {
|
||||
var self = this, node = this.getSelection();
|
||||
this.getDialog().close();
|
||||
return false;
|
||||
},
|
||||
updateFromEditor: function() {
|
||||
var self = this, node = this.getSelection();
|
||||
|
||||
// TODO Depends on managed mime type
|
||||
if(node.is('img')) {
|
||||
var idOrUrl = node.data('id') || node.data('url') || node.attr('src');
|
||||
this.showFileView(idOrUrl).done(function(filefield) {
|
||||
filefield.updateFromNode(node);
|
||||
self.toggleCloseButton();
|
||||
self.redraw();
|
||||
});
|
||||
}
|
||||
this.redraw();
|
||||
},
|
||||
redraw: function(updateExisting) {
|
||||
this._super();
|
||||
// TODO Depends on managed mime type
|
||||
if(node.is('img')) {
|
||||
var idOrUrl = node.data('id') || node.data('url') || node.attr('src');
|
||||
this.showFileView(idOrUrl).done(function(filefield) {
|
||||
filefield.updateFromNode(node);
|
||||
self.toggleCloseButton();
|
||||
self.redraw();
|
||||
});
|
||||
}
|
||||
this.redraw();
|
||||
},
|
||||
redraw: function(updateExisting) {
|
||||
this._super();
|
||||
|
||||
var node = this.getSelection(),
|
||||
hasItems = Boolean(this.find('.ss-htmleditorfield-file').length),
|
||||
editingSelected = node.is('img'),
|
||||
insertingURL = this.hasClass('insertingURL'),
|
||||
header = this.find('.header-edit');
|
||||
var node = this.getSelection(),
|
||||
hasItems = Boolean(this.find('.ss-htmleditorfield-file').length),
|
||||
editingSelected = node.is('img'),
|
||||
insertingURL = this.hasClass('insertingURL'),
|
||||
header = this.find('.header-edit');
|
||||
|
||||
// Only show second step if files are selected
|
||||
header[(hasItems) ? 'show' : 'hide']();
|
||||
// Only show second step if files are selected
|
||||
header[(hasItems) ? 'show' : 'hide']();
|
||||
|
||||
// Disable "insert" button if no files are selected
|
||||
this.closest('ui-dialog')
|
||||
.find('ui-dialog-buttonpane .media-insert')
|
||||
.button(hasItems ? 'enable' : 'disable')
|
||||
.toggleClass('ui-state-disabled', !hasItems);
|
||||
// Disable "insert" button if no files are selected
|
||||
this.closest('ui-dialog')
|
||||
.find('ui-dialog-buttonpane .media-insert')
|
||||
.button(hasItems ? 'enable' : 'disable')
|
||||
.toggleClass('ui-state-disabled', !hasItems);
|
||||
|
||||
// Hide file selection and step labels when editing an existing file
|
||||
this.find('.htmleditorfield-default-panel')[editingSelected || insertingURL ? 'hide' : 'show']();
|
||||
this.find('.htmleditorfield-web-panel')[editingSelected || !insertingURL ? 'hide' : 'show']();
|
||||
// Hide file selection and step labels when editing an existing file
|
||||
this.find('.htmleditorfield-default-panel')[editingSelected || insertingURL ? 'hide' : 'show']();
|
||||
this.find('.htmleditorfield-web-panel')[editingSelected || !insertingURL ? 'hide' : 'show']();
|
||||
|
||||
var mediaFormHeading = this.find('.htmleditorfield-mediaform-heading.insert');
|
||||
var mediaFormHeading = this.find('.htmleditorfield-mediaform-heading.insert');
|
||||
|
||||
if (editingSelected) {
|
||||
//When editing details of a file
|
||||
mediaFormHeading.hide();
|
||||
} else if (insertingURL) {
|
||||
//When inserting an image from a URL
|
||||
mediaFormHeading
|
||||
.show()
|
||||
.text(i18n._t("HtmlEditorField.INSERTURL"))
|
||||
.prepend('<button class="back-button font-icon-left-open no-text" title="' + i18n._t("HtmlEditorField.BACK") + '"></button>');
|
||||
if (editingSelected) {
|
||||
//When editing details of a file
|
||||
mediaFormHeading.hide();
|
||||
} else if (insertingURL) {
|
||||
//When inserting an image from a URL
|
||||
mediaFormHeading
|
||||
.show()
|
||||
.text(i18n._t("HtmlEditorField.INSERTURL"))
|
||||
.prepend('<button class="back-button font-icon-left-open no-text" title="' + i18n._t("HtmlEditorField.BACK") + '"></button>');
|
||||
|
||||
this.find('.htmleditorfield-web-panel input.remoteurl').focus();
|
||||
} else {
|
||||
//Default view when modal is opened
|
||||
mediaFormHeading
|
||||
.show()
|
||||
.text(i18n._t("HtmlEditorField.INSERTFROM"))
|
||||
.find('.back-button').remove();
|
||||
}
|
||||
this.find('.htmleditorfield-web-panel input.remoteurl').focus();
|
||||
} else {
|
||||
//Default view when modal is opened
|
||||
mediaFormHeading
|
||||
.show()
|
||||
.text(i18n._t("HtmlEditorField.INSERTFROM"))
|
||||
.find('.back-button').remove();
|
||||
}
|
||||
|
||||
// TODO Way too much knowledge on UploadField internals, use viewfile URL directly instead
|
||||
this.find('.htmleditorfield-mediaform-heading.update')[editingSelected ? 'show' : 'hide']();
|
||||
this.find('.ss-uploadfield-item-actions')[editingSelected ? 'hide' : 'show']();
|
||||
this.find('.ss-uploadfield-item-name')[editingSelected ? 'hide' : 'show']();
|
||||
this.find('.ss-uploadfield-item-preview')[editingSelected ? 'hide' : 'show']();
|
||||
this.find('.btn-toolbar .media-update')[editingSelected ? 'show' : 'hide']();
|
||||
this.find('.ss-uploadfield-item-editform').toggleEditForm(editingSelected);
|
||||
this.find('.htmleditorfield-from-cms .field.treedropdown').css('left', $('.htmleditorfield-mediaform-heading:visible').outerWidth());
|
||||
this.closest('.ui-dialog').addClass('ss-uploadfield-dropzone');
|
||||
this.closest('.ui-dialog')
|
||||
.find('.ui-dialog-buttonpane .media-insert .ui-button-text')
|
||||
.text([editingSelected ? i18n._t(
|
||||
'HtmlEditorField.UPDATE',
|
||||
'Update'
|
||||
) : i18n._t(
|
||||
'HtmlEditorField.INSERT',
|
||||
'Insert'
|
||||
)]);
|
||||
},
|
||||
resetFields: function() {
|
||||
this.find('.ss-htmleditorfield-file').remove(); // Remove any existing views
|
||||
this.find('.ss-gridfield-items .ui-selected').removeClass('ui-selected'); // Unselect all items
|
||||
this.find('li.ss-uploadfield-item').remove(); // Remove all selected items
|
||||
this.redraw();
|
||||
// TODO Way too much knowledge on UploadField internals, use viewfile URL directly instead
|
||||
this.find('.htmleditorfield-mediaform-heading.update')[editingSelected ? 'show' : 'hide']();
|
||||
this.find('.ss-uploadfield-item-actions')[editingSelected ? 'hide' : 'show']();
|
||||
this.find('.ss-uploadfield-item-name')[editingSelected ? 'hide' : 'show']();
|
||||
this.find('.ss-uploadfield-item-preview')[editingSelected ? 'hide' : 'show']();
|
||||
this.find('.btn-toolbar .media-update')[editingSelected ? 'show' : 'hide']();
|
||||
this.find('.ss-uploadfield-item-editform').toggleEditForm(editingSelected);
|
||||
this.find('.htmleditorfield-from-cms .field.treedropdown').css('left', $('.htmleditorfield-mediaform-heading:visible').outerWidth());
|
||||
this.closest('.ui-dialog').addClass('ss-uploadfield-dropzone');
|
||||
this.closest('.ui-dialog')
|
||||
.find('.ui-dialog-buttonpane .media-insert .ui-button-text')
|
||||
.text([editingSelected ? i18n._t(
|
||||
'HtmlEditorField.UPDATE',
|
||||
'Update'
|
||||
) : i18n._t(
|
||||
'HtmlEditorField.INSERT',
|
||||
'Insert'
|
||||
)]);
|
||||
},
|
||||
resetFields: function() {
|
||||
this.find('.ss-htmleditorfield-file').remove(); // Remove any existing views
|
||||
this.find('.ss-gridfield-items .ui-selected').removeClass('ui-selected'); // Unselect all items
|
||||
this.find('li.ss-uploadfield-item').remove(); // Remove all selected items
|
||||
this.redraw();
|
||||
|
||||
this._super();
|
||||
},
|
||||
getFileView: function(idOrUrl) {
|
||||
return this.find('.ss-htmleditorfield-file[data-id=' + idOrUrl + ']');
|
||||
},
|
||||
showFileView: function(idOrUrl) {
|
||||
var self = this, params = (Number(idOrUrl) == idOrUrl) ? {ID: idOrUrl} : {FileURL: idOrUrl};
|
||||
this._super();
|
||||
},
|
||||
getFileView: function(idOrUrl) {
|
||||
return this.find('.ss-htmleditorfield-file[data-id=' + idOrUrl + ']');
|
||||
},
|
||||
showFileView: function(idOrUrl) {
|
||||
var self = this, params = (Number(idOrUrl) == idOrUrl) ? {ID: idOrUrl} : {FileURL: idOrUrl};
|
||||
|
||||
var item = $('<div class="ss-htmleditorfield-file loading" />');
|
||||
this.find('.content-edit').prepend(item);
|
||||
var item = $('<div class="ss-htmleditorfield-file loading" />');
|
||||
this.find('.content-edit').prepend(item);
|
||||
|
||||
var dfr = $.Deferred();
|
||||
var dfr = $.Deferred();
|
||||
|
||||
$.ajax({
|
||||
url: $.path.addSearchParams(this.attr('action').replace(/MediaForm/, 'viewfile'), params),
|
||||
success: function(html, status, xhr) {
|
||||
var newItem = $(html).filter('.ss-htmleditorfield-file');
|
||||
item.replaceWith(newItem);
|
||||
self.redraw();
|
||||
dfr.resolve(newItem);
|
||||
},
|
||||
error: function() {
|
||||
item.remove();
|
||||
dfr.reject();
|
||||
}
|
||||
});
|
||||
$.ajax({
|
||||
url: $.path.addSearchParams(this.attr('action').replace(/MediaForm/, 'viewfile'), params),
|
||||
success: function(html, status, xhr) {
|
||||
var newItem = $(html).filter('.ss-htmleditorfield-file');
|
||||
item.replaceWith(newItem);
|
||||
self.redraw();
|
||||
dfr.resolve(newItem);
|
||||
},
|
||||
error: function() {
|
||||
item.remove();
|
||||
dfr.reject();
|
||||
}
|
||||
});
|
||||
|
||||
return dfr.promise();
|
||||
}
|
||||
});
|
||||
return dfr.promise();
|
||||
}
|
||||
});
|
||||
|
||||
//When 'Insert from URL' button is clicked
|
||||
$('form.htmleditorfield-mediaform div.ss-upload .upload-url').entwine({
|
||||
onclick: function () {
|
||||
var form = this.closest('form');
|
||||
//When 'Insert from URL' button is clicked
|
||||
$('form.htmleditorfield-mediaform div.ss-upload .upload-url').entwine({
|
||||
onclick: function () {
|
||||
var form = this.closest('form');
|
||||
|
||||
form.addClass('insertingURL');
|
||||
form.redraw();
|
||||
}
|
||||
});
|
||||
form.addClass('insertingURL');
|
||||
form.redraw();
|
||||
}
|
||||
});
|
||||
|
||||
//When back button is clicked while inserting URL
|
||||
$('form.htmleditorfield-mediaform .htmleditorfield-mediaform-heading .back-button').entwine({
|
||||
onclick: function() {
|
||||
var form = this.closest('form');
|
||||
//When back button is clicked while inserting URL
|
||||
$('form.htmleditorfield-mediaform .htmleditorfield-mediaform-heading .back-button').entwine({
|
||||
onclick: function() {
|
||||
var form = this.closest('form');
|
||||
|
||||
form.removeClass('insertingURL');
|
||||
form.redraw();
|
||||
}
|
||||
});
|
||||
form.removeClass('insertingURL');
|
||||
form.redraw();
|
||||
}
|
||||
});
|
||||
|
||||
$('form.htmleditorfield-mediaform .ss-gridfield-items').entwine({
|
||||
onselectableselected: function(e, ui) {
|
||||
$('form.htmleditorfield-mediaform .ss-gridfield-items').entwine({
|
||||
onselectableselected: function(e, ui) {
|
||||
var form = this.closest('form'), item = $(ui.selected);
|
||||
if(!item.is('.ss-gridfield-item')) return;
|
||||
form.closest('form').showFileView(item.data('id'));
|
||||
form.redraw();
|
||||
form.closest('form').showFileView(item.data('id'));
|
||||
form.redraw();
|
||||
|
||||
form.parent().trigger('scroll');
|
||||
},
|
||||
onselectableunselected: function(e, ui) {
|
||||
var form = this.closest('form'), item = $(ui.unselected);
|
||||
if(!item.is('.ss-gridfield-item')) return;
|
||||
form.getFileView(item.data('id')).remove();
|
||||
form.redraw();
|
||||
}
|
||||
});
|
||||
form.parent().trigger('scroll');
|
||||
},
|
||||
onselectableunselected: function(e, ui) {
|
||||
var form = this.closest('form'), item = $(ui.unselected);
|
||||
if(!item.is('.ss-gridfield-item')) return;
|
||||
form.getFileView(item.data('id')).remove();
|
||||
form.redraw();
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Show the second step after uploading an image
|
||||
*/
|
||||
$('form.htmleditorfield-form.htmleditorfield-mediaform div.ss-assetuploadfield').entwine({
|
||||
//the UploadField div.ss-uploadfield-editandorganize is hidden in CSS,
|
||||
// because we use the detail view for each individual file instead
|
||||
onfileuploadstop: function(e) {
|
||||
var form = this.closest('form');
|
||||
/**
|
||||
* Show the second step after uploading an image
|
||||
*/
|
||||
$('form.htmleditorfield-form.htmleditorfield-mediaform div.ss-assetuploadfield').entwine({
|
||||
//the UploadField div.ss-uploadfield-editandorganize is hidden in CSS,
|
||||
// because we use the detail view for each individual file instead
|
||||
onfileuploadstop: function(e) {
|
||||
var form = this.closest('form');
|
||||
|
||||
//update the editFields to show those Files that are newly uploaded
|
||||
var editFieldIDs = [];
|
||||
form.find('div.content-edit').find('div.ss-htmleditorfield-file').each(function(){
|
||||
//get the uploaded file ID when this event triggers, signaling the upload has compeleted successfully
|
||||
editFieldIDs.push($(this).data('id'));
|
||||
});
|
||||
// we only want this .ss-uploadfield-files - else we get all ss-uploadfield-files wich include the ones not related to #tinymce insertmedia
|
||||
var uploadedFiles = $('.ss-uploadfield-files', this).children('.ss-uploadfield-item');
|
||||
uploadedFiles.each(function(){
|
||||
var uploadedID = $(this).data('fileid');
|
||||
if (uploadedID && $.inArray(uploadedID, editFieldIDs) == -1) {
|
||||
//trigger the detail view for filling out details about the file we are about to insert into TinyMCE
|
||||
$(this).remove(); // Remove successfully added item from the queue
|
||||
form.showFileView(uploadedID);
|
||||
}
|
||||
});
|
||||
//update the editFields to show those Files that are newly uploaded
|
||||
var editFieldIDs = [];
|
||||
form.find('div.content-edit').find('div.ss-htmleditorfield-file').each(function(){
|
||||
//get the uploaded file ID when this event triggers, signaling the upload has compeleted successfully
|
||||
editFieldIDs.push($(this).data('id'));
|
||||
});
|
||||
// we only want this .ss-uploadfield-files - else we get all ss-uploadfield-files wich include the ones not related to #tinymce insertmedia
|
||||
var uploadedFiles = $('.ss-uploadfield-files', this).children('.ss-uploadfield-item');
|
||||
uploadedFiles.each(function(){
|
||||
var uploadedID = $(this).data('fileid');
|
||||
if (uploadedID && $.inArray(uploadedID, editFieldIDs) == -1) {
|
||||
//trigger the detail view for filling out details about the file we are about to insert into TinyMCE
|
||||
$(this).remove(); // Remove successfully added item from the queue
|
||||
form.showFileView(uploadedID);
|
||||
}
|
||||
});
|
||||
|
||||
form.parent().trigger('scroll');
|
||||
form.parent().trigger('scroll');
|
||||
|
||||
form.redraw();
|
||||
}
|
||||
form.redraw();
|
||||
}
|
||||
|
||||
});
|
||||
});
|
||||
|
||||
$('form.htmleditorfield-form.htmleditorfield-mediaform input.remoteurl').entwine({
|
||||
onadd: function() {
|
||||
this._super();
|
||||
this.validate();
|
||||
},
|
||||
$('form.htmleditorfield-form.htmleditorfield-mediaform input.remoteurl').entwine({
|
||||
onadd: function() {
|
||||
this._super();
|
||||
this.validate();
|
||||
},
|
||||
|
||||
onkeyup: function() {
|
||||
this.validate();
|
||||
},
|
||||
onkeyup: function() {
|
||||
this.validate();
|
||||
},
|
||||
|
||||
onchange: function() {
|
||||
this.validate();
|
||||
},
|
||||
onchange: function() {
|
||||
this.validate();
|
||||
},
|
||||
|
||||
getAddButton: function() {
|
||||
return this.closest('.CompositeField').find('button.add-url');
|
||||
},
|
||||
getAddButton: function() {
|
||||
return this.closest('.CompositeField').find('button.add-url');
|
||||
},
|
||||
|
||||
validate: function() {
|
||||
var val = this.val(), orig = val;
|
||||
validate: function() {
|
||||
var val = this.val(), orig = val;
|
||||
|
||||
val = $.trim(val);
|
||||
val = val.replace(/^https?:\/\//i, '');
|
||||
if (orig !== val) this.val(val);
|
||||
val = $.trim(val);
|
||||
val = val.replace(/^https?:\/\//i, '');
|
||||
if (orig !== val) this.val(val);
|
||||
|
||||
this.getAddButton().button(!!val ? 'enable' : 'disable');
|
||||
return !!val;
|
||||
}
|
||||
});
|
||||
this.getAddButton().button(!!val ? 'enable' : 'disable');
|
||||
return !!val;
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Show the second step after adding a URL
|
||||
*/
|
||||
$('form.htmleditorfield-form.htmleditorfield-mediaform .add-url').entwine({
|
||||
getURLField: function() {
|
||||
return this.closest('.CompositeField').find('input.remoteurl');
|
||||
},
|
||||
/**
|
||||
* Show the second step after adding a URL
|
||||
*/
|
||||
$('form.htmleditorfield-form.htmleditorfield-mediaform .add-url').entwine({
|
||||
getURLField: function() {
|
||||
return this.closest('.CompositeField').find('input.remoteurl');
|
||||
},
|
||||
|
||||
onclick: function(e) {
|
||||
var urlField = this.getURLField(), container = this.closest('.CompositeField'), form = this.closest('form');
|
||||
onclick: function(e) {
|
||||
var urlField = this.getURLField(), container = this.closest('.CompositeField'), form = this.closest('form');
|
||||
|
||||
if (urlField.validate()) {
|
||||
container.addClass('loading');
|
||||
form.showFileView('http://' + urlField.val()).done(function() {
|
||||
container.removeClass('loading');
|
||||
if (urlField.validate()) {
|
||||
container.addClass('loading');
|
||||
form.showFileView('http://' + urlField.val()).done(function() {
|
||||
container.removeClass('loading');
|
||||
|
||||
form.parent().trigger('scroll');
|
||||
});
|
||||
form.redraw();
|
||||
}
|
||||
form.parent().trigger('scroll');
|
||||
});
|
||||
form.redraw();
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
});
|
||||
return false;
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Represents a single selected file, together with a set of form fields to edit its properties.
|
||||
* Overload this based on the media type to determine how the HTML should be created.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file').entwine({
|
||||
/**
|
||||
* @return {Object} Map of HTML attributes which can be set on the created DOM node.
|
||||
*/
|
||||
getAttributes: function() {
|
||||
},
|
||||
/**
|
||||
* @return {Object} Map of additional properties which can be evaluated
|
||||
* by the specific media type.
|
||||
*/
|
||||
getExtraData: function() {
|
||||
},
|
||||
/**
|
||||
* @return {String} HTML suitable for insertion into the rich text editor
|
||||
*/
|
||||
getHTML: function() {
|
||||
// Assumes UploadField markup structure
|
||||
return $('<div>').append(
|
||||
$('<a/>').attr({href: this.data('url')}).text(this.find('.name').text())
|
||||
).html();
|
||||
},
|
||||
/**
|
||||
* Insert updated HTML content into the rich text editor
|
||||
*/
|
||||
insertHTML: function(ed) {
|
||||
// Insert content
|
||||
ed.replaceContent(this.getHTML());
|
||||
},
|
||||
/**
|
||||
* Updates the form values from an existing node in the editor.
|
||||
*
|
||||
* @param {DOMElement}
|
||||
*/
|
||||
updateFromNode: function(node) {
|
||||
},
|
||||
/**
|
||||
* Transforms values set on the dimensions form fields based on two constraints:
|
||||
* An aspect ration, and max width/height values. Writes back to the field properties as required.
|
||||
*
|
||||
* @param {String} The dimension to constrain the other value by, if any ("Width" or "Height")
|
||||
* @param {Int} Optional max width
|
||||
* @param {Int} Optional max height
|
||||
*/
|
||||
updateDimensions: function(constrainBy, maxW, maxH) {
|
||||
var widthEl = this.find(':input[name=Width]'),
|
||||
heightEl = this.find(':input[name=Height]'),
|
||||
w = widthEl.val(),
|
||||
h = heightEl.val(),
|
||||
aspect;
|
||||
/**
|
||||
* Represents a single selected file, together with a set of form fields to edit its properties.
|
||||
* Overload this based on the media type to determine how the HTML should be created.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file').entwine({
|
||||
/**
|
||||
* @return {Object} Map of HTML attributes which can be set on the created DOM node.
|
||||
*/
|
||||
getAttributes: function() {
|
||||
},
|
||||
/**
|
||||
* @return {Object} Map of additional properties which can be evaluated
|
||||
* by the specific media type.
|
||||
*/
|
||||
getExtraData: function() {
|
||||
},
|
||||
/**
|
||||
* @return {String} HTML suitable for insertion into the rich text editor
|
||||
*/
|
||||
getHTML: function() {
|
||||
// Assumes UploadField markup structure
|
||||
return $('<div>').append(
|
||||
$('<a/>').attr({href: this.data('url')}).text(this.find('.name').text())
|
||||
).html();
|
||||
},
|
||||
/**
|
||||
* Insert updated HTML content into the rich text editor
|
||||
*/
|
||||
insertHTML: function(ed) {
|
||||
// Insert content
|
||||
ed.replaceContent(this.getHTML());
|
||||
},
|
||||
/**
|
||||
* Updates the form values from an existing node in the editor.
|
||||
*
|
||||
* @param {DOMElement}
|
||||
*/
|
||||
updateFromNode: function(node) {
|
||||
},
|
||||
/**
|
||||
* Transforms values set on the dimensions form fields based on two constraints:
|
||||
* An aspect ration, and max width/height values. Writes back to the field properties as required.
|
||||
*
|
||||
* @param {String} The dimension to constrain the other value by, if any ("Width" or "Height")
|
||||
* @param {Int} Optional max width
|
||||
* @param {Int} Optional max height
|
||||
*/
|
||||
updateDimensions: function(constrainBy, maxW, maxH) {
|
||||
var widthEl = this.find(':input[name=Width]'),
|
||||
heightEl = this.find(':input[name=Height]'),
|
||||
w = widthEl.val(),
|
||||
h = heightEl.val(),
|
||||
aspect;
|
||||
|
||||
// Proportionate updating of heights, using the original values
|
||||
if(w && h) {
|
||||
if(constrainBy) {
|
||||
aspect = heightEl.getOrigVal() / widthEl.getOrigVal();
|
||||
// Uses floor() and ceil() to avoid both fields constantly lowering each other's values in rounding situations
|
||||
if(constrainBy == 'Width') {
|
||||
if(maxW && w > maxW) w = maxW;
|
||||
h = Math.floor(w * aspect);
|
||||
} else if(constrainBy == 'Height') {
|
||||
if(maxH && h > maxH) h = maxH;
|
||||
w = Math.ceil(h / aspect);
|
||||
}
|
||||
} else {
|
||||
if(maxW && w > maxW) w = maxW;
|
||||
if(maxH && h > maxH) h = maxH;
|
||||
}
|
||||
// Proportionate updating of heights, using the original values
|
||||
if(w && h) {
|
||||
if(constrainBy) {
|
||||
aspect = heightEl.getOrigVal() / widthEl.getOrigVal();
|
||||
// Uses floor() and ceil() to avoid both fields constantly lowering each other's values in rounding situations
|
||||
if(constrainBy == 'Width') {
|
||||
if(maxW && w > maxW) w = maxW;
|
||||
h = Math.floor(w * aspect);
|
||||
} else if(constrainBy == 'Height') {
|
||||
if(maxH && h > maxH) h = maxH;
|
||||
w = Math.ceil(h / aspect);
|
||||
}
|
||||
} else {
|
||||
if(maxW && w > maxW) w = maxW;
|
||||
if(maxH && h > maxH) h = maxH;
|
||||
}
|
||||
|
||||
widthEl.val(w);
|
||||
heightEl.val(h);
|
||||
}
|
||||
}
|
||||
});
|
||||
widthEl.val(w);
|
||||
heightEl.val(h);
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file.image').entwine({
|
||||
getAttributes: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'src' : this.find(':input[name=URL]').val(),
|
||||
'alt' : this.find(':input[name=AltText]').val(),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'title' : this.find(':input[name=Title]').val(),
|
||||
'class' : this.find(':input[name=CSSClass]').val(),
|
||||
'data-id' : this.find(':input[name=FileID]').val()
|
||||
};
|
||||
},
|
||||
getExtraData: function() {
|
||||
return {
|
||||
'CaptionText': this.find(':input[name=CaptionText]').val()
|
||||
};
|
||||
},
|
||||
getHTML: function() {
|
||||
/* NOP */
|
||||
},
|
||||
/**
|
||||
* Logic similar to TinyMCE 'advimage' plugin, insertAndClose() method.
|
||||
*/
|
||||
insertHTML: function(ed) {
|
||||
var form = this.closest('form');
|
||||
var node = form.getSelection();
|
||||
if (!ed) ed = form.getEditor();
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file.image').entwine({
|
||||
getAttributes: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'src' : this.find(':input[name=URL]').val(),
|
||||
'alt' : this.find(':input[name=AltText]').val(),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'title' : this.find(':input[name=Title]').val(),
|
||||
'class' : this.find(':input[name=CSSClass]').val(),
|
||||
'data-id' : this.find(':input[name=FileID]').val()
|
||||
};
|
||||
},
|
||||
getExtraData: function() {
|
||||
return {
|
||||
'CaptionText': this.find(':input[name=CaptionText]').val()
|
||||
};
|
||||
},
|
||||
getHTML: function() {
|
||||
/* NOP */
|
||||
},
|
||||
/**
|
||||
* Logic similar to TinyMCE 'advimage' plugin, insertAndClose() method.
|
||||
*/
|
||||
insertHTML: function(ed) {
|
||||
var form = this.closest('form');
|
||||
var node = form.getSelection();
|
||||
if (!ed) ed = form.getEditor();
|
||||
|
||||
// Get the attributes & extra data
|
||||
var attrs = this.getAttributes(), extraData = this.getExtraData();
|
||||
// Get the attributes & extra data
|
||||
var attrs = this.getAttributes(), extraData = this.getExtraData();
|
||||
|
||||
// Find the element we are replacing - either the img, it's wrapper parent, or nothing (if creating)
|
||||
var replacee = (node && node.is('img')) ? node : null;
|
||||
if (replacee && replacee.parent().is('.captionImage')) replacee = replacee.parent();
|
||||
// Find the element we are replacing - either the img, it's wrapper parent, or nothing (if creating)
|
||||
var replacee = (node && node.is('img')) ? node : null;
|
||||
if (replacee && replacee.parent().is('.captionImage')) replacee = replacee.parent();
|
||||
|
||||
// Find the img node - either the existing img or a new one, and update it
|
||||
var img = (node && node.is('img')) ? node : $('<img />');
|
||||
img.attr(attrs);
|
||||
// Find the img node - either the existing img or a new one, and update it
|
||||
var img = (node && node.is('img')) ? node : $('<img />');
|
||||
img.attr(attrs);
|
||||
|
||||
// Any existing figure or caption node
|
||||
var container = img.parent('.captionImage'), caption = container.find('.caption');
|
||||
// Any existing figure or caption node
|
||||
var container = img.parent('.captionImage'), caption = container.find('.caption');
|
||||
|
||||
// If we've got caption text, we need a wrapping div.captionImage and sibling p.caption
|
||||
if (extraData.CaptionText) {
|
||||
if (!container.length) {
|
||||
container = $('<div></div>');
|
||||
}
|
||||
// If we've got caption text, we need a wrapping div.captionImage and sibling p.caption
|
||||
if (extraData.CaptionText) {
|
||||
if (!container.length) {
|
||||
container = $('<div></div>');
|
||||
}
|
||||
|
||||
container.attr('class', 'captionImage '+attrs['class']).css('width', attrs.width);
|
||||
container.attr('class', 'captionImage '+attrs['class']).css('width', attrs.width);
|
||||
|
||||
if (!caption.length) {
|
||||
caption = $('<p class="caption"></p>').appendTo(container);
|
||||
}
|
||||
if (!caption.length) {
|
||||
caption = $('<p class="caption"></p>').appendTo(container);
|
||||
}
|
||||
|
||||
caption.attr('class', 'caption '+attrs['class']).text(extraData.CaptionText);
|
||||
}
|
||||
// Otherwise forget they exist
|
||||
else {
|
||||
container = caption = null;
|
||||
}
|
||||
caption.attr('class', 'caption '+attrs['class']).text(extraData.CaptionText);
|
||||
}
|
||||
// Otherwise forget they exist
|
||||
else {
|
||||
container = caption = null;
|
||||
}
|
||||
|
||||
// The element we are replacing the replacee with
|
||||
var replacer = container ? container : img;
|
||||
// The element we are replacing the replacee with
|
||||
var replacer = container ? container : img;
|
||||
|
||||
// If we're replacing something, and it's not with itself, do so
|
||||
if (replacee && replacee.not(replacer).length) {
|
||||
replacee.replaceWith(replacer);
|
||||
}
|
||||
// If we're replacing something, and it's not with itself, do so
|
||||
if (replacee && replacee.not(replacer).length) {
|
||||
replacee.replaceWith(replacer);
|
||||
}
|
||||
|
||||
// If we have a wrapper element, make sure the img is the first child - img might be the
|
||||
// replacee, and the wrapper the replacer, and we can't do this till after the replace has happened
|
||||
if (container) {
|
||||
container.prepend(img);
|
||||
}
|
||||
// If we have a wrapper element, make sure the img is the first child - img might be the
|
||||
// replacee, and the wrapper the replacer, and we can't do this till after the replace has happened
|
||||
if (container) {
|
||||
container.prepend(img);
|
||||
}
|
||||
|
||||
// If we don't have a replacee, then we need to insert the whole HTML
|
||||
if (!replacee) {
|
||||
// Otherwise insert the whole HTML content
|
||||
ed.repaint();
|
||||
ed.insertContent($('<div />').append(replacer).html(), {skip_undo : 1});
|
||||
}
|
||||
// If we don't have a replacee, then we need to insert the whole HTML
|
||||
if (!replacee) {
|
||||
// Otherwise insert the whole HTML content
|
||||
ed.repaint();
|
||||
ed.insertContent($('<div />').append(replacer).html(), {skip_undo : 1});
|
||||
}
|
||||
|
||||
ed.addUndo();
|
||||
ed.repaint();
|
||||
},
|
||||
updateFromNode: function(node) {
|
||||
this.find(':input[name=AltText]').val(node.attr('alt'));
|
||||
this.find(':input[name=Title]').val(node.attr('title'));
|
||||
this.find(':input[name=CSSClass]').val(node.attr('class'));
|
||||
this.find(':input[name=Width]').val(node.width());
|
||||
this.find(':input[name=Height]').val(node.height());
|
||||
this.find(':input[name=CaptionText]').val(node.siblings('.caption:first').text());
|
||||
this.find(':input[name=FileID]').val(node.data('id'));
|
||||
}
|
||||
});
|
||||
ed.addUndo();
|
||||
ed.repaint();
|
||||
},
|
||||
updateFromNode: function(node) {
|
||||
this.find(':input[name=AltText]').val(node.attr('alt'));
|
||||
this.find(':input[name=Title]').val(node.attr('title'));
|
||||
this.find(':input[name=CSSClass]').val(node.attr('class'));
|
||||
this.find(':input[name=Width]').val(node.width());
|
||||
this.find(':input[name=Height]').val(node.height());
|
||||
this.find(':input[name=CaptionText]').val(node.siblings('.caption:first').text());
|
||||
this.find(':input[name=FileID]').val(node.data('id'));
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
/**
|
||||
* Insert a flash object tag into the content.
|
||||
* Requires the 'media' plugin for serialization of tags into <img> placeholders.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file.flash').entwine({
|
||||
getAttributes: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'src' : this.find(':input[name=URL]').val(),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'data-fileid' : this.find(':input[name=FileID]').val()
|
||||
};
|
||||
},
|
||||
getHTML: function() {
|
||||
var attrs = this.getAttributes();
|
||||
/**
|
||||
* Insert a flash object tag into the content.
|
||||
* Requires the 'media' plugin for serialization of tags into <img> placeholders.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file.flash').entwine({
|
||||
getAttributes: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'src' : this.find(':input[name=URL]').val(),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'data-fileid' : this.find(':input[name=FileID]').val()
|
||||
};
|
||||
},
|
||||
getHTML: function() {
|
||||
var attrs = this.getAttributes();
|
||||
|
||||
// Emulate serialization from 'media' plugin
|
||||
var el = tinyMCE.activeEditor.plugins.media.dataToImg({
|
||||
'type': 'flash',
|
||||
'width': attrs.width,
|
||||
'height': attrs.height,
|
||||
'params': {'src': attrs.src},
|
||||
'video': {'sources': []}
|
||||
});
|
||||
// Emulate serialization from 'media' plugin
|
||||
var el = tinyMCE.activeEditor.plugins.media.dataToImg({
|
||||
'type': 'flash',
|
||||
'width': attrs.width,
|
||||
'height': attrs.height,
|
||||
'params': {'src': attrs.src},
|
||||
'video': {'sources': []}
|
||||
});
|
||||
|
||||
return $('<div />').append(el).html(); // Little hack to get outerHTML string
|
||||
},
|
||||
updateFromNode: function(node) {
|
||||
// TODO Not implemented
|
||||
}
|
||||
});
|
||||
return $('<div />').append(el).html(); // Little hack to get outerHTML string
|
||||
},
|
||||
updateFromNode: function(node) {
|
||||
// TODO Not implemented
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
/**
|
||||
* Insert an Embed object tag into the content.
|
||||
* Requires the 'media' plugin for serialization of tags into <img> placeholders.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file.embed').entwine({
|
||||
getAttributes: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'src' : this.find('.thumbnail-preview').attr('src'),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'class' : this.find(':input[name=CSSClass]').val(),
|
||||
'alt' : this.find(':input[name=AltText]').val(),
|
||||
'title' : this.find(':input[name=Title]').val(),
|
||||
'data-fileid' : this.find(':input[name=FileID]').val()
|
||||
};
|
||||
},
|
||||
getExtraData: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'CaptionText': this.find(':input[name=CaptionText]').val(),
|
||||
'Url': this.find(':input[name=URL]').val(),
|
||||
'thumbnail': this.find('.thumbnail-preview').attr('src'),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'cssclass': this.find(':input[name=CSSClass]').val()
|
||||
};
|
||||
},
|
||||
getHTML: function() {
|
||||
var el,
|
||||
attrs = this.getAttributes(),
|
||||
extraData = this.getExtraData(),
|
||||
// imgEl = $('<img id="_ss_tmp_img" />');
|
||||
imgEl = $('<img />').attr(attrs).addClass('ss-htmleditorfield-file embed');
|
||||
/**
|
||||
* Insert an Embed object tag into the content.
|
||||
* Requires the 'media' plugin for serialization of tags into <img> placeholders.
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file.embed').entwine({
|
||||
getAttributes: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'src' : this.find('.thumbnail-preview').attr('src'),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'class' : this.find(':input[name=CSSClass]').val(),
|
||||
'alt' : this.find(':input[name=AltText]').val(),
|
||||
'title' : this.find(':input[name=Title]').val(),
|
||||
'data-fileid' : this.find(':input[name=FileID]').val()
|
||||
};
|
||||
},
|
||||
getExtraData: function() {
|
||||
var width = this.find(':input[name=Width]').val(),
|
||||
height = this.find(':input[name=Height]').val();
|
||||
return {
|
||||
'CaptionText': this.find(':input[name=CaptionText]').val(),
|
||||
'Url': this.find(':input[name=URL]').val(),
|
||||
'thumbnail': this.find('.thumbnail-preview').attr('src'),
|
||||
'width' : width ? parseInt(width, 10) : null,
|
||||
'height' : height ? parseInt(height, 10) : null,
|
||||
'cssclass': this.find(':input[name=CSSClass]').val()
|
||||
};
|
||||
},
|
||||
getHTML: function() {
|
||||
var el,
|
||||
attrs = this.getAttributes(),
|
||||
extraData = this.getExtraData(),
|
||||
// imgEl = $('<img id="_ss_tmp_img" />');
|
||||
imgEl = $('<img />').attr(attrs).addClass('ss-htmleditorfield-file embed');
|
||||
|
||||
$.each(extraData, function (key, value) {
|
||||
imgEl.attr('data-' + key, value);
|
||||
});
|
||||
$.each(extraData, function (key, value) {
|
||||
imgEl.attr('data-' + key, value);
|
||||
});
|
||||
|
||||
if(extraData.CaptionText) {
|
||||
el = $('<div style="width: ' + attrs['width'] + 'px;" class="captionImage ' + attrs['class'] + '"><p class="caption">' + extraData.CaptionText + '</p></div>').prepend(imgEl);
|
||||
} else {
|
||||
el = imgEl;
|
||||
}
|
||||
return $('<div />').append(el).html(); // Little hack to get outerHTML string
|
||||
},
|
||||
updateFromNode: function(node) {
|
||||
this.find(':input[name=AltText]').val(node.attr('alt'));
|
||||
this.find(':input[name=Title]').val(node.attr('title'));
|
||||
this.find(':input[name=Width]').val(node.width());
|
||||
this.find(':input[name=Height]').val(node.height());
|
||||
this.find(':input[name=Title]').val(node.attr('title'));
|
||||
this.find(':input[name=CSSClass]').val(node.data('cssclass'));
|
||||
this.find(':input[name=FileID]').val(node.data('fileid'));
|
||||
}
|
||||
});
|
||||
if(extraData.CaptionText) {
|
||||
el = $('<div style="width: ' + attrs['width'] + 'px;" class="captionImage ' + attrs['class'] + '"><p class="caption">' + extraData.CaptionText + '</p></div>').prepend(imgEl);
|
||||
} else {
|
||||
el = imgEl;
|
||||
}
|
||||
return $('<div />').append(el).html(); // Little hack to get outerHTML string
|
||||
},
|
||||
updateFromNode: function(node) {
|
||||
this.find(':input[name=AltText]').val(node.attr('alt'));
|
||||
this.find(':input[name=Title]').val(node.attr('title'));
|
||||
this.find(':input[name=Width]').val(node.width());
|
||||
this.find(':input[name=Height]').val(node.height());
|
||||
this.find(':input[name=Title]').val(node.attr('title'));
|
||||
this.find(':input[name=CSSClass]').val(node.data('cssclass'));
|
||||
this.find(':input[name=FileID]').val(node.data('fileid'));
|
||||
}
|
||||
});
|
||||
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file .dimensions :input').entwine({
|
||||
OrigVal: null,
|
||||
onmatch: function () {
|
||||
this._super();
|
||||
$('form.htmleditorfield-mediaform .ss-htmleditorfield-file .dimensions :input').entwine({
|
||||
OrigVal: null,
|
||||
onmatch: function () {
|
||||
this._super();
|
||||
|
||||
this.setOrigVal(parseInt(this.val(), 10));
|
||||
},
|
||||
onunmatch: function() {
|
||||
this._super();
|
||||
},
|
||||
onfocusout: function(e) {
|
||||
this.closest('.ss-htmleditorfield-file').updateDimensions(this.attr('name'));
|
||||
}
|
||||
});
|
||||
this.setOrigVal(parseInt(this.val(), 10));
|
||||
},
|
||||
onunmatch: function() {
|
||||
this._super();
|
||||
},
|
||||
onfocusout: function(e) {
|
||||
this.closest('.ss-htmleditorfield-file').updateDimensions(this.attr('name'));
|
||||
}
|
||||
});
|
||||
|
||||
/**
|
||||
* Deselect item and remove the 'edit' view
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-uploadfield-item .ss-uploadfield-item-cancel').entwine({
|
||||
onclick: function(e) {
|
||||
var form = this.closest('form'), file = this.closest('ss-uploadfield-item');
|
||||
form.find('.ss-gridfield-item[data-id=' + file.data('id') + ']').removeClass('ui-selected');
|
||||
this.closest('.ss-uploadfield-item').remove();
|
||||
form.redraw();
|
||||
e.preventDefault();
|
||||
}
|
||||
});
|
||||
/**
|
||||
* Deselect item and remove the 'edit' view
|
||||
*/
|
||||
$('form.htmleditorfield-mediaform .ss-uploadfield-item .ss-uploadfield-item-cancel').entwine({
|
||||
onclick: function(e) {
|
||||
var form = this.closest('form'), file = this.closest('ss-uploadfield-item');
|
||||
form.find('.ss-gridfield-item[data-id=' + file.data('id') + ']').removeClass('ui-selected');
|
||||
this.closest('.ss-uploadfield-item').remove();
|
||||
form.redraw();
|
||||
e.preventDefault();
|
||||
}
|
||||
});
|
||||
|
||||
$('div.ss-assetuploadfield .ss-uploadfield-item-edit, div.ss-assetuploadfield .ss-uploadfield-item-name').entwine({
|
||||
getEditForm: function() {
|
||||
return this.closest('.ss-uploadfield-item').find('.ss-uploadfield-item-editform');
|
||||
},
|
||||
$('div.ss-assetuploadfield .ss-uploadfield-item-edit, div.ss-assetuploadfield .ss-uploadfield-item-name').entwine({
|
||||
getEditForm: function() {
|
||||
return this.closest('.ss-uploadfield-item').find('.ss-uploadfield-item-editform');
|
||||
},
|
||||
|
||||
fromEditForm: {
|
||||
onchange: function(e){
|
||||
var form = $(e.target);
|
||||
form.removeClass('edited'); //so edited class is only there once
|
||||
form.addClass('edited');
|
||||
}
|
||||
},
|
||||
fromEditForm: {
|
||||
onchange: function(e){
|
||||
var form = $(e.target);
|
||||
form.removeClass('edited'); //so edited class is only there once
|
||||
form.addClass('edited');
|
||||
}
|
||||
},
|
||||
|
||||
onclick: function(e) {
|
||||
var editForm = this.getEditForm();
|
||||
onclick: function(e) {
|
||||
var editForm = this.getEditForm();
|
||||
|
||||
// Make sure we're in an HtmlEditorField here, or fall-back to _super(). HtmlEditorField with
|
||||
// AssetUploadField doesn't use iframes, so needs its own toggleEditForm() logic
|
||||
if (this.closest('.ss-uploadfield-item').hasClass('ss-htmleditorfield-file')) {
|
||||
editForm.parent('ss-uploadfield-item').removeClass('ui-state-warning');
|
||||
// Make sure we're in an HtmlEditorField here, or fall-back to _super(). HtmlEditorField with
|
||||
// AssetUploadField doesn't use iframes, so needs its own toggleEditForm() logic
|
||||
if (this.closest('.ss-uploadfield-item').hasClass('ss-htmleditorfield-file')) {
|
||||
editForm.parent('ss-uploadfield-item').removeClass('ui-state-warning');
|
||||
|
||||
editForm.toggleEditForm();
|
||||
editForm.toggleEditForm();
|
||||
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
e.preventDefault(); // Avoid a form submit
|
||||
|
||||
return false; // Avoid duplication from button
|
||||
}
|
||||
return false; // Avoid duplication from button
|
||||
}
|
||||
|
||||
this._super(e);
|
||||
}
|
||||
});
|
||||
this._super(e);
|
||||
}
|
||||
});
|
||||
|
||||
$('div.ss-assetuploadfield .ss-uploadfield-item-editform').entwine({
|
||||
toggleEditForm: function(bool) {
|
||||
var itemInfo = this.prev('.ss-uploadfield-item-info'), status = itemInfo.find('.ss-uploadfield-item-status');
|
||||
var text="";
|
||||
$('div.ss-assetuploadfield .ss-uploadfield-item-editform').entwine({
|
||||
toggleEditForm: function(bool) {
|
||||
var itemInfo = this.prev('.ss-uploadfield-item-info'), status = itemInfo.find('.ss-uploadfield-item-status');
|
||||
var text="";
|
||||
|
||||
if(bool === true || (bool !== false && this.height() === 0)) {
|
||||
text = i18n._t('UploadField.Editing', "Editing ...");
|
||||
this.height('auto');
|
||||
itemInfo.find('.toggle-details-icon').addClass('opened');
|
||||
status.removeClass('ui-state-success-text').removeClass('ui-state-warning-text');
|
||||
} else {
|
||||
this.height(0);
|
||||
itemInfo.find('.toggle-details-icon').removeClass('opened');
|
||||
if(!this.hasClass('edited')){
|
||||
text = i18n._t('UploadField.NOCHANGES', 'No Changes');
|
||||
status.addClass('ui-state-success-text');
|
||||
}else{
|
||||
text = i18n._t('UploadField.CHANGESSAVED', 'Changes Made');
|
||||
this.removeClass('edited');
|
||||
status.addClass('ui-state-success-text');
|
||||
}
|
||||
if(bool === true || (bool !== false && this.height() === 0)) {
|
||||
text = i18n._t('UploadField.Editing', "Editing ...");
|
||||
this.height('auto');
|
||||
itemInfo.find('.toggle-details-icon').addClass('opened');
|
||||
status.removeClass('ui-state-success-text').removeClass('ui-state-warning-text');
|
||||
} else {
|
||||
this.height(0);
|
||||
itemInfo.find('.toggle-details-icon').removeClass('opened');
|
||||
if(!this.hasClass('edited')){
|
||||
text = i18n._t('UploadField.NOCHANGES', 'No Changes');
|
||||
status.addClass('ui-state-success-text');
|
||||
}else{
|
||||
text = i18n._t('UploadField.CHANGESSAVED', 'Changes Made');
|
||||
this.removeClass('edited');
|
||||
status.addClass('ui-state-success-text');
|
||||
}
|
||||
|
||||
}
|
||||
status.attr('title',text).text(text);
|
||||
}
|
||||
});
|
||||
}
|
||||
status.attr('title',text).text(text);
|
||||
}
|
||||
});
|
||||
|
||||
|
||||
$('form.htmleditorfield-mediaform .field[id$="ParentID_Holder"] .TreeDropdownField').entwine({
|
||||
onadd: function() {
|
||||
this._super();
|
||||
$('form.htmleditorfield-mediaform .field[id$="ParentID_Holder"] .TreeDropdownField').entwine({
|
||||
onadd: function() {
|
||||
this._super();
|
||||
|
||||
// TODO Custom event doesn't fire in IE if registered through object literal
|
||||
var self = this;
|
||||
this.bind('change', function() {
|
||||
var fileList = self.closest('form').find('.grid-field');
|
||||
fileList.setState('ParentID', self.getValue());
|
||||
fileList.reload();
|
||||
});
|
||||
}
|
||||
});
|
||||
// TODO Custom event doesn't fire in IE if registered through object literal
|
||||
var self = this;
|
||||
this.bind('change', function() {
|
||||
var fileList = self.closest('form').find('.grid-field');
|
||||
fileList.setState('ParentID', self.getValue());
|
||||
fileList.reload();
|
||||
});
|
||||
}
|
||||
});
|
||||
|
||||
});
|
||||
|
Loading…
x
Reference in New Issue
Block a user