mirror of
https://gerrit.wikimedia.org/r/mediawiki/extensions/WikiEditor
synced 2024-11-24 00:06:49 +00:00
Merge "WikiEditor: Remove unmaintained beta iframe mode"
This commit is contained in:
commit
1c94813742
|
@ -1,4 +0,0 @@
|
|||
# upstream lib from Google
|
||||
modules/contentCollector.js
|
||||
# messy
|
||||
modules/jquery.wikiEditor.iframe.js
|
|
@ -62,11 +62,6 @@ $wikiEditorTpl = array(
|
|||
);
|
||||
|
||||
$wgResourceModules += array(
|
||||
/* Third-party modules */
|
||||
|
||||
'contentCollector' => $wikiEditorTpl + array(
|
||||
'scripts' => 'contentCollector.js',
|
||||
),
|
||||
|
||||
/* WikiEditor jQuery plugin Resources */
|
||||
|
||||
|
@ -83,13 +78,6 @@ $wgResourceModules += array(
|
|||
'wikieditor-loading',
|
||||
),
|
||||
),
|
||||
'jquery.wikiEditor.iframe' => $wikiEditorTpl + array(
|
||||
'scripts' => 'jquery.wikiEditor.iframe.js',
|
||||
'dependencies' => array(
|
||||
'jquery.wikiEditor',
|
||||
'contentCollector',
|
||||
),
|
||||
),
|
||||
'jquery.wikiEditor.dialogs' => $wikiEditorTpl + array(
|
||||
'scripts' => 'jquery.wikiEditor.dialogs.js',
|
||||
'styles' => 'jquery.wikiEditor.dialogs.css',
|
||||
|
|
|
@ -1,439 +0,0 @@
|
|||
// THIS FILE HAS BEEN MODIFIED for use with the mediawiki wikiEditor
|
||||
// It no longer requires etherpad.collab.ace.easysync2.Changeset
|
||||
// THIS FILE WAS ORIGINALLY AN APPJET MODULE: etherpad.collab.ace.contentcollector
|
||||
|
||||
/**
|
||||
* Copyright 2009 Google Inc.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License"); you may not
|
||||
* use this file except in compliance with the License. You may obtain a copy of
|
||||
* the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS-IS" BASIS, WITHOUT
|
||||
* WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||
* License for the specific language governing permissions and limitations under
|
||||
* the License.
|
||||
*/
|
||||
|
||||
var _MAX_LIST_LEVEL = 8;
|
||||
|
||||
function sanitizeUnicode(s) {
|
||||
return s.replace(/[\uffff\ufffe\ufeff\ufdd0-\ufdef\ud800-\udfff]/g, '?');
|
||||
}
|
||||
|
||||
function makeContentCollector( browser, domInterface ) {
|
||||
browser = browser || {};
|
||||
|
||||
var dom = domInterface || {
|
||||
isNodeText : function(n) {
|
||||
return (n.nodeType == 3);
|
||||
},
|
||||
nodeTagName : function(n) {
|
||||
return n.tagName;
|
||||
},
|
||||
nodeValue : function(n) {
|
||||
try {
|
||||
return n.nodeValue;
|
||||
} catch ( err ) {
|
||||
return '';
|
||||
}
|
||||
},
|
||||
nodeName : function(n) {
|
||||
return n.nodeName;
|
||||
},
|
||||
nodeNumChildren : function(n) {
|
||||
return n.childNodes.length;
|
||||
},
|
||||
nodeChild : function(n, i) {
|
||||
return n.childNodes.item(i);
|
||||
},
|
||||
nodeProp : function(n, p) {
|
||||
return n[p];
|
||||
},
|
||||
nodeAttr : function(n, a) {
|
||||
return n.getAttribute(a);
|
||||
},
|
||||
optNodeInnerHTML : function(n) {
|
||||
return n.innerHTML;
|
||||
}
|
||||
};
|
||||
|
||||
var _blockElems = {
|
||||
"div" : 1,
|
||||
"p" : 1,
|
||||
"pre" : 1,
|
||||
"li" : 1
|
||||
};
|
||||
function isBlockElement(n) {
|
||||
return !!_blockElems[(dom.nodeTagName(n) || "").toLowerCase()];
|
||||
}
|
||||
function textify(str) {
|
||||
return sanitizeUnicode(str.replace(/[\n\r ]/g, ' ').replace(/\xa0/g,
|
||||
' ').replace(/\t/g, ' '));
|
||||
}
|
||||
function getAssoc(node, name) {
|
||||
return dom.nodeProp(node, "_magicdom_" + name);
|
||||
}
|
||||
|
||||
var lines = (function() {
|
||||
var textArray = [];
|
||||
var self = {
|
||||
length : function() {
|
||||
return textArray.length;
|
||||
},
|
||||
atColumnZero : function() {
|
||||
return textArray[textArray.length - 1] === "";
|
||||
},
|
||||
startNew : function() {
|
||||
textArray.push("");
|
||||
self.flush(true);
|
||||
},
|
||||
textOfLine : function(i) {
|
||||
return textArray[i];
|
||||
},
|
||||
appendText : function(txt, attrString) {
|
||||
textArray[textArray.length - 1] += txt;
|
||||
// dmesg(txt+" / "+attrString);
|
||||
},
|
||||
textLines : function() {
|
||||
return textArray.slice();
|
||||
},
|
||||
// call flush only when you're done
|
||||
flush : function(withNewline) {
|
||||
|
||||
}
|
||||
};
|
||||
self.startNew();
|
||||
return self;
|
||||
}());
|
||||
var cc = {};
|
||||
function _ensureColumnZero(state) {
|
||||
if (!lines.atColumnZero()) {
|
||||
_startNewLine(state);
|
||||
}
|
||||
}
|
||||
var selection, startPoint, endPoint;
|
||||
var selStart = [ -1, -1 ], selEnd = [ -1, -1 ];
|
||||
var blockElems = {
|
||||
"div" : 1,
|
||||
"p" : 1,
|
||||
"pre" : 1
|
||||
};
|
||||
function _isEmpty(node, state) {
|
||||
// consider clean blank lines pasted in IE to be empty
|
||||
if (dom.nodeNumChildren(node) == 0)
|
||||
return true;
|
||||
if (dom.nodeNumChildren(node) == 1 && getAssoc(node, "shouldBeEmpty")
|
||||
&& dom.optNodeInnerHTML(node) == " "
|
||||
&& !getAssoc(node, "unpasted")) {
|
||||
if (state) {
|
||||
var child = dom.nodeChild(node, 0);
|
||||
_reachPoint(child, 0, state);
|
||||
_reachPoint(child, 1, state);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
return false;
|
||||
}
|
||||
function _pointHere(charsAfter, state) {
|
||||
var ln = lines.length() - 1;
|
||||
var chr = lines.textOfLine(ln).length;
|
||||
if (chr == 0 && state.listType && state.listType != 'none') {
|
||||
chr += 1; // listMarker
|
||||
}
|
||||
chr += charsAfter;
|
||||
return [ ln, chr ];
|
||||
}
|
||||
function _reachBlockPoint(nd, idx, state) {
|
||||
if (!dom.isNodeText(nd))
|
||||
_reachPoint(nd, idx, state);
|
||||
}
|
||||
function _reachPoint(nd, idx, state) {
|
||||
if (startPoint && nd == startPoint.node && startPoint.index == idx) {
|
||||
selStart = _pointHere(0, state);
|
||||
}
|
||||
if (endPoint && nd == endPoint.node && endPoint.index == idx) {
|
||||
selEnd = _pointHere(0, state);
|
||||
}
|
||||
}
|
||||
function _incrementFlag(state, flagName) {
|
||||
state.flags[flagName] = (state.flags[flagName] || 0) + 1;
|
||||
}
|
||||
function _decrementFlag(state, flagName) {
|
||||
state.flags[flagName]--;
|
||||
}
|
||||
function _enterList(state, listType) {
|
||||
var oldListType = state.listType;
|
||||
state.listLevel = (state.listLevel || 0) + 1;
|
||||
if (listType != 'none') {
|
||||
state.listNesting = (state.listNesting || 0) + 1;
|
||||
}
|
||||
state.listType = listType;
|
||||
return oldListType;
|
||||
}
|
||||
function _exitList(state, oldListType) {
|
||||
state.listLevel--;
|
||||
if (state.listType != 'none') {
|
||||
state.listNesting--;
|
||||
}
|
||||
state.listType = oldListType;
|
||||
}
|
||||
function _produceListMarker(state) {
|
||||
|
||||
}
|
||||
function _startNewLine(state) {
|
||||
if (state) {
|
||||
var atBeginningOfLine = lines.textOfLine(lines.length() - 1).length == 0;
|
||||
if (atBeginningOfLine && state.listType && state.listType != 'none') {
|
||||
_produceListMarker(state);
|
||||
}
|
||||
}
|
||||
lines.startNew();
|
||||
}
|
||||
cc.notifySelection = function(sel) {
|
||||
if (sel) {
|
||||
selection = sel;
|
||||
startPoint = selection.startPoint;
|
||||
endPoint = selection.endPoint;
|
||||
}
|
||||
};
|
||||
cc.collectContent = function(node, state) {
|
||||
if (!state) {
|
||||
state = {
|
||||
flags : {/* name -> nesting counter */}
|
||||
};
|
||||
}
|
||||
var isBlock = isBlockElement(node);
|
||||
var isEmpty = _isEmpty(node, state);
|
||||
if (isBlock)
|
||||
_ensureColumnZero(state);
|
||||
var startLine = lines.length() - 1;
|
||||
_reachBlockPoint(node, 0, state);
|
||||
if (dom.isNodeText(node)) {
|
||||
var txt = dom.nodeValue(node);
|
||||
var rest = '';
|
||||
var x = 0; // offset into original text
|
||||
if (txt.length == 0) {
|
||||
if (startPoint && node == startPoint.node) {
|
||||
selStart = _pointHere(0, state);
|
||||
}
|
||||
if (endPoint && node == endPoint.node) {
|
||||
selEnd = _pointHere(0, state);
|
||||
}
|
||||
}
|
||||
while (txt.length > 0) {
|
||||
var consumed = 0;
|
||||
if (!browser.firefox || state.flags.preMode) {
|
||||
var firstLine = txt.split('\n', 1)[0];
|
||||
consumed = firstLine.length + 1;
|
||||
rest = txt.substring(consumed);
|
||||
txt = firstLine;
|
||||
} else { /* will only run this loop body once */
|
||||
}
|
||||
if (startPoint && node == startPoint.node
|
||||
&& startPoint.index - x <= txt.length) {
|
||||
selStart = _pointHere(startPoint.index - x, state);
|
||||
}
|
||||
if (endPoint && node == endPoint.node
|
||||
&& endPoint.index - x <= txt.length) {
|
||||
selEnd = _pointHere(endPoint.index - x, state);
|
||||
}
|
||||
var txt2 = txt;
|
||||
if ((!state.flags.preMode) && /^[\r\n]*$/.exec(txt)) {
|
||||
// prevents textnodes containing just "\n" from being
|
||||
// significant
|
||||
// in safari when pasting text, now that we convert them to
|
||||
// spaces instead of removing them, because in other cases
|
||||
// removing "\n" from pasted HTML will collapse words
|
||||
// together.
|
||||
txt2 = "";
|
||||
}
|
||||
var atBeginningOfLine = lines.textOfLine(lines.length() - 1).length == 0;
|
||||
if (atBeginningOfLine) {
|
||||
// newlines in the source mustn't become spaces at beginning
|
||||
// of line box
|
||||
txt2 = txt2.replace(/^\n*/, '');
|
||||
}
|
||||
if (atBeginningOfLine && state.listType
|
||||
&& state.listType != 'none') {
|
||||
_produceListMarker(state);
|
||||
}
|
||||
lines.appendText(textify(txt2));
|
||||
|
||||
x += consumed;
|
||||
txt = rest;
|
||||
if (txt.length > 0) {
|
||||
_startNewLine(state);
|
||||
}
|
||||
}
|
||||
|
||||
} else {
|
||||
var cls = dom.nodeProp(node, "className");
|
||||
var tname = (dom.nodeTagName(node) || "").toLowerCase();
|
||||
if (tname == "br") {
|
||||
_startNewLine(state);
|
||||
} else if (tname == "script" || tname == "style") {
|
||||
// ignore
|
||||
} else if (!isEmpty) {
|
||||
var styl = dom.nodeAttr(node, "style");
|
||||
|
||||
var isPre = (tname == "pre");
|
||||
if ((!isPre) && browser.safari) {
|
||||
isPre = (styl && /\bwhite-space:\s*pre\b/i.exec(styl));
|
||||
}
|
||||
if (isPre)
|
||||
_incrementFlag(state, 'preMode');
|
||||
var oldListTypeOrNull = null;
|
||||
|
||||
var nc = dom.nodeNumChildren(node);
|
||||
for ( var i = 0; i < nc; i++) {
|
||||
var c = dom.nodeChild(node, i);
|
||||
//very specific IE case where it inserts <span lang="en"> which we want to ginore.
|
||||
//to reproduce copy content from wordpad andpaste into the middle of a line in IE
|
||||
if ( browser.msie && cls.indexOf('wikiEditor') >= 0 && dom.nodeName(c) == 'SPAN' && dom.nodeAttr(c, 'lang') == "" ) {
|
||||
continue;
|
||||
}
|
||||
cc.collectContent(c, state);
|
||||
}
|
||||
|
||||
if (isPre)
|
||||
_decrementFlag(state, 'preMode');
|
||||
|
||||
if (oldListTypeOrNull) {
|
||||
_exitList(state, oldListTypeOrNull);
|
||||
}
|
||||
}
|
||||
}
|
||||
if (!browser.msie) {
|
||||
_reachBlockPoint(node, 1, state);
|
||||
}
|
||||
if (isBlock) {
|
||||
if (lines.length() - 1 == startLine) {
|
||||
_startNewLine(state);
|
||||
} else {
|
||||
_ensureColumnZero(state);
|
||||
}
|
||||
}
|
||||
|
||||
if (browser.msie) {
|
||||
// in IE, a point immediately after a DIV appears on the next line
|
||||
//_reachBlockPoint(node, 1, state);
|
||||
}
|
||||
};
|
||||
// can pass a falsy value for end of doc
|
||||
cc.notifyNextNode = function(node) {
|
||||
// an "empty block" won't end a line; this addresses an issue in IE with
|
||||
// typing into a blank line at the end of the document. typed text
|
||||
// goes into the body, and the empty line div still looks clean.
|
||||
// it is incorporated as dirty by the rule that a dirty region has
|
||||
// to end a line.
|
||||
if ((!node) || (isBlockElement(node) && !_isEmpty(node))) {
|
||||
_ensureColumnZero(null);
|
||||
}
|
||||
};
|
||||
// each returns [line, char] or [-1,-1]
|
||||
var getSelectionStart = function() {
|
||||
return selStart;
|
||||
};
|
||||
var getSelectionEnd = function() {
|
||||
return selEnd;
|
||||
};
|
||||
|
||||
// returns array of strings for lines found, last entry will be "" if
|
||||
// last line is complete (i.e. if a following span should be on a new line).
|
||||
// can be called at any point
|
||||
cc.getLines = function() {
|
||||
return lines.textLines();
|
||||
};
|
||||
|
||||
// cc.applyHints = function(hints) {
|
||||
// if (hints.pastedLines) {
|
||||
//
|
||||
// }
|
||||
// }
|
||||
|
||||
cc.finish = function() {
|
||||
lines.flush();
|
||||
var lineStrings = cc.getLines();
|
||||
|
||||
if ( lineStrings.length > 0 && !lineStrings[lineStrings.length - 1] ) {
|
||||
lineStrings.length--;
|
||||
}
|
||||
|
||||
var ss = getSelectionStart();
|
||||
var se = getSelectionEnd();
|
||||
|
||||
function fixLongLines() {
|
||||
// design mode does not deal with with really long lines!
|
||||
var lineLimit = 2000; // chars
|
||||
var buffer = 10; // chars allowed over before wrapping
|
||||
var linesWrapped = 0;
|
||||
var numLinesAfter = 0;
|
||||
for ( var i = lineStrings.length - 1; i >= 0; i--) {
|
||||
var oldString = lineStrings[i];
|
||||
if (oldString.length > lineLimit + buffer) {
|
||||
var newStrings = [];
|
||||
while (oldString.length > lineLimit) {
|
||||
// var semiloc = oldString.lastIndexOf(';',
|
||||
// lineLimit-1);
|
||||
// var lengthToTake = (semiloc >= 0 ? (semiloc+1) :
|
||||
// lineLimit);
|
||||
lengthToTake = lineLimit;
|
||||
newStrings.push(oldString.substring(0, lengthToTake));
|
||||
oldString = oldString.substring(lengthToTake);
|
||||
|
||||
}
|
||||
if (oldString.length > 0) {
|
||||
newStrings.push(oldString);
|
||||
}
|
||||
function fixLineNumber(lineChar) {
|
||||
if (lineChar[0] < 0)
|
||||
return;
|
||||
var n = lineChar[0];
|
||||
var c = lineChar[1];
|
||||
if (n > i) {
|
||||
n += (newStrings.length - 1);
|
||||
} else if (n == i) {
|
||||
var a = 0;
|
||||
while (c > newStrings[a].length) {
|
||||
c -= newStrings[a].length;
|
||||
a++;
|
||||
}
|
||||
n += a;
|
||||
}
|
||||
lineChar[0] = n;
|
||||
lineChar[1] = c;
|
||||
}
|
||||
fixLineNumber(ss);
|
||||
fixLineNumber(se);
|
||||
linesWrapped++;
|
||||
numLinesAfter += newStrings.length;
|
||||
|
||||
newStrings.unshift(i, 1);
|
||||
lineStrings.splice.apply(lineStrings, newStrings);
|
||||
|
||||
}
|
||||
}
|
||||
return {
|
||||
linesWrapped : linesWrapped,
|
||||
numLinesAfter : numLinesAfter
|
||||
};
|
||||
}
|
||||
var wrapData = fixLongLines();
|
||||
|
||||
return {
|
||||
selStart : ss,
|
||||
selEnd : se,
|
||||
linesWrapped : wrapData.linesWrapped,
|
||||
numLinesAfter : wrapData.numLinesAfter,
|
||||
lines : lineStrings
|
||||
};
|
||||
};
|
||||
|
||||
return cc;
|
||||
}
|
||||
|
||||
|
|
@ -1298,10 +1298,7 @@ $.wikiEditor.modules.dialogs.config = {
|
|||
} );
|
||||
$textarea.textSelection( 'scrollToCaretPosition' );
|
||||
$( this ).data( 'offset', end );
|
||||
var context = $( this ).data( 'context' );
|
||||
var textbox = typeof context.$iframe !== 'undefined' ?
|
||||
context.$iframe[0].contentWindow : $textarea[0];
|
||||
textbox.focus();
|
||||
$textarea[0].focus();
|
||||
}
|
||||
});
|
||||
},
|
||||
|
@ -1352,8 +1349,7 @@ $.wikiEditor.modules.dialogs.config = {
|
|||
var dialog = $(this).closest( '.ui-dialog' );
|
||||
var that = this;
|
||||
var context = $(this).data( 'context' );
|
||||
var textbox = typeof context.$iframe !== 'undefined' ?
|
||||
context.$iframe[0].contentWindow.document : context.$textarea;
|
||||
var textbox = context.$textarea;
|
||||
|
||||
$( textbox )
|
||||
.bind( 'keypress.srdialog', function ( e ) {
|
||||
|
@ -1370,8 +1366,7 @@ $.wikiEditor.modules.dialogs.config = {
|
|||
},
|
||||
close: function () {
|
||||
var context = $(this).data( 'context' );
|
||||
var textbox = typeof context.$iframe !== 'undefined' ?
|
||||
context.$iframe[0].contentWindow.document : context.$textarea;
|
||||
var textbox = context.$textarea;
|
||||
$( textbox ).unbind( 'keypress.srdialog' );
|
||||
$(this).closest( '.ui-dialog' ).data( 'dialogaction', false );
|
||||
}
|
||||
|
|
|
@ -77,7 +77,7 @@ $.wikiEditor.modules.dialogs = {
|
|||
create: function ( context, config ) {
|
||||
var mod, module, filtered, i, $existingDialog;
|
||||
|
||||
// Defer building of modules, but do check whether they need the iframe rightaway
|
||||
// Defer building of modules, unless they require immediate creation
|
||||
for ( mod in config ) {
|
||||
module = config[mod];
|
||||
// Only create the dialog if it's supported, isn't filtered and doesn't exist yet
|
||||
|
@ -99,10 +99,6 @@ $.wikiEditor.modules.dialogs = {
|
|||
$existingDialog = $( '#' + module.id );
|
||||
if ( !filtered && $.wikiEditor.isSupported( module ) && $existingDialog.length === 0 ) {
|
||||
$.wikiEditor.modules.dialogs.modules[mod] = module;
|
||||
// If this dialog requires the iframe, set it up
|
||||
if ( typeof context.$iframe === 'undefined' && $.wikiEditor.isRequired( module, 'iframe' ) ) {
|
||||
context.fn.setupIframe();
|
||||
}
|
||||
context.$textarea.trigger( 'wikiEditor-dialogs-setup-' + mod );
|
||||
// If this dialog requires immediate creation, create it now
|
||||
if ( typeof module.immediateCreate !== 'undefined' && module.immediateCreate ) {
|
||||
|
|
File diff suppressed because it is too large
Load diff
|
@ -48,7 +48,7 @@ $.wikiEditor = {
|
|||
msie: [['>=', 7]],
|
||||
// Layout issues in FF < 2
|
||||
firefox: [['>=', 2]],
|
||||
// Text selection bugs galore - this may be a different situation with the new iframe-based solution
|
||||
// Text selection bugs galore
|
||||
opera: [['>=', 9.6]],
|
||||
// jQuery minimums
|
||||
safari: [['>=', 3]],
|
||||
|
@ -64,7 +64,7 @@ $.wikiEditor = {
|
|||
msie: [['>=', 8]],
|
||||
// Layout issues in FF < 2
|
||||
firefox: [['>=', 2]],
|
||||
// Text selection bugs galore - this may be a different situation with the new iframe-based solution
|
||||
// Text selection bugs galore
|
||||
opera: [['>=', 9.6]],
|
||||
// jQuery minimums
|
||||
safari: [['>=', 3]],
|
||||
|
@ -239,24 +239,8 @@ if ( !context || typeof context === 'undefined' ) {
|
|||
'data': {},
|
||||
// Unique numeric ID of this instance used both for looking up and differentiating instances of wikiEditor
|
||||
'instance': $.wikiEditor.instances.push( $(this) ) - 1,
|
||||
// Array mapping elements in the textarea to character offsets
|
||||
'offsets': null,
|
||||
// Cache for context.fn.htmlToText()
|
||||
'htmlToTextMap': {},
|
||||
// The previous HTML of the iframe, stored to detect whether something really changed.
|
||||
'oldHTML': null,
|
||||
// Same for delayedChange()
|
||||
'oldDelayedHTML': null,
|
||||
// The previous selection of the iframe, stored to detect whether the selection has changed
|
||||
'oldDelayedSel': null,
|
||||
// Saved selection state for old IE (<=10)
|
||||
'savedSelection': null,
|
||||
// Stack of states in { html: [string] } form
|
||||
'history': [],
|
||||
// Current history state position - this is number of steps backwards, so it's always -1 or less
|
||||
'historyPosition': -1,
|
||||
/// The previous historyPosition, stored to detect if change events were due to an undo or redo action
|
||||
'oldDelayedHistoryPosition': -1,
|
||||
// List of extensions active on this context
|
||||
'extensions': []
|
||||
};
|
||||
|
|
|
@ -257,9 +257,6 @@ fn: {
|
|||
'encapsulateSelection',
|
||||
$.extend( {}, action.options, parts, { 'replace': replace } )
|
||||
);
|
||||
if ( context.$iframe !== undefined ) {
|
||||
context.$iframe[0].contentWindow.focus();
|
||||
}
|
||||
break;
|
||||
case 'callback':
|
||||
if ( typeof action.execute === 'function' ) {
|
||||
|
|
Loading…
Reference in a new issue