ENHANCEMENT Simplified "insert link" interface and logic by reverting to TinyMCE's standard behaviour: Only allow linking on an existing selection, don't allow modifying the link content in the "insert" interface. Now that we're presenting this interface as a dialog rather than persistent sidebar its no longer necessary to have "unselected link creation+insertion".

This commit is contained in:
Ingo Schommer 2012-01-04 23:31:40 +01:00
parent 6bafc76118
commit 98852677d5
2 changed files with 13 additions and 74 deletions

View File

@ -265,7 +265,6 @@ class HtmlEditorField_Toolbar extends RequestHandler {
new EmailField('email', _t('HtmlEditorField.EMAIL', 'Email address')), new EmailField('email', _t('HtmlEditorField.EMAIL', 'Email address')),
new TreeDropdownField('file', _t('HtmlEditorField.FILE', 'File'), 'File', 'Filename', 'Title', true), new TreeDropdownField('file', _t('HtmlEditorField.FILE', 'File'), 'File', 'Filename', 'Title', true),
new TextField('Anchor', _t('HtmlEditorField.ANCHORVALUE', 'Anchor')), new TextField('Anchor', _t('HtmlEditorField.ANCHORVALUE', 'Anchor')),
new TextField('LinkText', _t('HtmlEditorField.LINKTEXT', 'Link text')),
new TextField('Description', _t('HtmlEditorField.LINKDESCR', 'Link description')), new TextField('Description', _t('HtmlEditorField.LINKDESCR', 'Link description')),
new CheckboxField('TargetBlank', _t('HtmlEditorField.LINKOPENNEWWIN', 'Open link in a new window?')), new CheckboxField('TargetBlank', _t('HtmlEditorField.LINKOPENNEWWIN', 'Open link in a new window?')),
new HiddenField('Locale', null, $this->controller->Locale) new HiddenField('Locale', null, $this->controller->Locale)

View File

@ -86,6 +86,9 @@
$.entwine('ss', function($) { $.entwine('ss', function($) {
$('form.htmleditorfield-form').entwine({ $('form.htmleditorfield-form').entwine({
Bookmark: null,
onmatch: function() { onmatch: function() {
// Move title from headline to (jQuery compatible) title attribute // Move title from headline to (jQuery compatible) title attribute
var titleEl = this.find(':header:first'); var titleEl = this.find(':header:first');
@ -103,10 +106,12 @@
}, },
close: function() { close: function() {
this.dialog('close'); this.dialog('close');
this.setBookmark(null);
}, },
open: function() { open: function() {
this.dialog('open'); this.dialog('open');
this.redraw(); this.redraw();
this.setBookmark(this.getEditor().selection.getBookmark());
}, },
getEditor: function() { getEditor: function() {
return tinyMCE.activeEditor; return tinyMCE.activeEditor;
@ -115,8 +120,6 @@
$('form.htmleditorfield-linkform').entwine({ $('form.htmleditorfield-linkform').entwine({
OriginalSelection: null,
onmatch: function() { onmatch: function() {
this._super(); this._super();
@ -131,7 +134,7 @@
this.respondToNodeChange(); this.respondToNodeChange();
this.dialog('open'); this.dialog('open');
this.redraw(); this.redraw();
this.setOriginalSelection(null); this.setBookmark(this.getEditor().selection.getBookmark());
}, },
close: function() { close: function() {
@ -176,7 +179,7 @@
}, },
insertLink: function() { insertLink: function() {
var href, target = null, anchor = this.find(':input[name=Anchor]').val(); var href, target = null, anchor = this.find(':input[name=Anchor]').val(), ed = this.getEditor();
// Determine target // Determine target
if(this.find(':input[name=TargetBlank]').is(':checked')) target = '_blank'; if(this.find(':input[name=TargetBlank]').is(':checked')) target = '_blank';
@ -210,66 +213,16 @@
break; break;
} }
if(this.getOriginalSelection()) {
tinyMCE.activeEditor.selection.setRng(this.getOriginalSelection());
}
var linkText = this.find(':input[name=LinkText]').val();
var attributes = { var attributes = {
href : href, href : href,
target : target, target : target,
title : this.find(':input[name=Description]').val(), title : this.find(':input[name=Description]').val()
innerHTML : linkText ? linkText : "Your Link"
}; };
// Add the new link // Add the new link
this._insertLink(attributes); ed.selection.moveToBookmark(this.getBookmark());
}, ed.execCommand("mceInsertLink", false, attributes);
/**
* Insert a link into the given editor.
* Replaces mceInsertLink in that innerHTML can also be set
*/
_insertLink: function(attributes) {
var ed = this.getEditor(), v = attributes,
s = ed.selection, e = ed.dom.getParent(s.getNode(), 'A');
if(tinymce.is(attributes, 'string')) attributes = {href : attributes};
function set(e) {
tinymce.each(attributes, function(v, k) {
if(k == 'innerHTML') e.innerHTML = v;
else ed.dom.setAttrib(e, k, v);
});
try {
s.select(e);
this.updateSelection();
} catch(er) {}
};
function replace() {
tinymce.each(ed.dom.select('a'), function(e) {
if (e.href == 'javascript:mctmp(0);') set(e);
});
}
if(attributes.innerHTML && !ed.selection.getContent()) {
if(tinymce.isIE) var rng = ed.selection.getRng();
e = ed.getDoc().createElement('a');
e.href = 'javascript:mctmp(0);';
s.setNode(e);
if(tinymce.isIE) tinyMCE.activeEditor.selection.setRng(rng);
replace();
}
if (!e) {
ed.execCommand('CreateLink', false, 'javascript:mctmp(0);');
replace();
} else {
if (attributes.href) set(e);
else ed.dom.remove(e, 1);
}
this.trigger('onafterinsert', attributes); this.trigger('onafterinsert', attributes);
this.respondToNodeChange(); this.respondToNodeChange();
@ -280,11 +233,6 @@
this.close(); this.close();
}, },
updateSelection: function() {
var ed = this.getEditor();
if(ed.selection.getRng()) this.setOriginalSelection(ed.selection.getRng());
},
addAnchorSelector: function() { addAnchorSelector: function() {
// Avoid adding twice // Avoid adding twice
if(this.find(':input[name=AnchorSelector]').length) return; if(this.find(':input[name=AnchorSelector]').length) return;
@ -323,7 +271,7 @@
var selector = this.find(':input[name=AnchorSelector]'), anchors = new Array(); var selector = this.find(':input[name=AnchorSelector]'), anchors = new Array();
// name attribute is defined as CDATA, should accept all characters and entities // 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 // http://www.w3.org/TR/1999/REC-html401-19991224/struct/links.html#h-12.2
var raw = tinyMCE.activeEditor.getContent().match(/name="([^"]+?)"|name='([^']+?)'/gim); var raw = this.getEditor().getContent().match(/name="([^"]+?)"|name='([^']+?)'/gim);
if (raw && raw.length) { if (raw && raw.length) {
for(var i = 0; i < raw.length; i++) { for(var i = 0; i < raw.length; i++) {
anchors.push(raw[i].substr(6).replace(/"$/, '')); anchors.push(raw[i].substr(6).replace(/"$/, ''));
@ -402,8 +350,6 @@
href = RegExp.$1; href = RegExp.$1;
} }
var linkText = ed.selection.getContent({format : 'html'}).replace(/<\/?a[^>]*>/ig,'');
// Get rid of TinyMCE's temporary URLs // Get rid of TinyMCE's temporary URLs
if(href.match(/^javascript:\s*mctmp/)) href = ''; if(href.match(/^javascript:\s*mctmp/)) href = '';
@ -411,30 +357,26 @@
return { return {
LinkType: 'email', LinkType: 'email',
email: RegExp.$1, email: RegExp.$1,
LinkText: linkText,
Description: title Description: title
} }
} else if(href.match(/^(assets\/.*)$/)) { } else if(href.match(/^(assets\/.*)$/)) {
return { return {
LinkType: 'file', LinkType: 'file',
file: RegExp.$1, file: RegExp.$1,
LinkText: linkText,
Description: title Description: title
} }
} else if(href.match(/^#(.*)$/)) { } else if(href.match(/^#(.*)$/)) {
return { return {
LinkType: 'anchor', LinkType: 'anchor',
Anchor: RegExp.$1, Anchor: RegExp.$1,
LinkText: linkText,
Description: title, Description: title,
TargetBlank: target ? true : false TargetBlank: target ? true : false
} }
} else if(href.match(/^\[sitetree_link id=([0-9]+)\]?(#.*)?$/)) { } else if(href.match(/^\[sitetree_link\s*(?:%20)?id=([0-9]+)\]?(#.*)?$/)) {
return { return {
LinkType: 'internal', LinkType: 'internal',
internal: RegExp.$1, internal: RegExp.$1,
Anchor: RegExp.$2 ? RegExp.$2.substr(1) : '', Anchor: RegExp.$2 ? RegExp.$2.substr(1) : '',
LinkText: linkText,
Description: title, Description: title,
TargetBlank: target ? true : false TargetBlank: target ? true : false
} }
@ -442,14 +384,12 @@
return { return {
LinkType: 'external', LinkType: 'external',
external: href, external: href,
LinkText: linkText,
Description: title, Description: title,
TargetBlank: target ? true : false TargetBlank: target ? true : false
} }
} else { } else {
return { return {
LinkType: 'internal', LinkType: 'internal'
LinkText: linkText
} }
} }
} }