pad.libre-service.eu-etherpad/src/static/js/domline.js

323 lines
8.2 KiB
JavaScript
Raw Normal View History

/**
* This code is mostly from the old Etherpad. Please help us to comment this code.
* This helps other people to understand this code better and helps them to improve it.
* TL;DR COMMENTS ON THIS FILE ARE HIGHLY APPRECIATED
*/
2011-03-26 14:10:41 +01:00
// THIS FILE IS ALSO AN APPJET MODULE: etherpad.collab.ace.domline
// %APPJET%: import("etherpad.admin.plugins");
/**
* 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.
*/
// requires: top
// requires: plugins
// requires: undefined
2012-03-07 02:27:03 +01:00
var Security = require('./security');
var hooks = require('./pluginfw/hooks');
2012-03-17 13:36:42 +01:00
var _ = require('./underscore');
var lineAttributeMarker = require('./linestylefilter').lineAttributeMarker;
var noop = function(){};
2012-03-17 13:36:42 +01:00
2011-03-26 14:10:41 +01:00
var domline = {};
2011-07-07 19:59:34 +02:00
domline.addToLineClass = function(lineClass, cls)
{
2011-03-26 14:10:41 +01:00
// an "empty span" at any point can be used to add classes to
// the line, using line:className. otherwise, we ignore
// the span.
2011-07-07 19:59:34 +02:00
cls.replace(/\S+/g, function(c)
{
if (c.indexOf("line:") == 0)
{
2011-03-26 14:10:41 +01:00
// add class to line
2011-07-07 19:59:34 +02:00
lineClass = (lineClass ? lineClass + ' ' : '') + c.substring(5);
2011-03-26 14:10:41 +01:00
}
});
return lineClass;
}
// if "document" is falsy we don't create a DOM node, just
// an object with innerHTML and className
2011-07-07 19:59:34 +02:00
domline.createDomLine = function(nonEmpty, doesWrap, optBrowser, optDocument)
{
var result = {
node: null,
appendSpan: noop,
prepareForAdd: noop,
notifyAdded: noop,
clearSpans: noop,
finishUpdate: noop,
2011-07-07 19:59:34 +02:00
lineMarker: 0
};
2011-03-26 14:10:41 +01:00
var document = optDocument;
2011-07-07 19:59:34 +02:00
if (document)
{
2011-03-26 14:10:41 +01:00
result.node = document.createElement("div");
}
2011-07-07 19:59:34 +02:00
else
{
result.node = {
innerHTML: '',
className: ''
};
2011-03-26 14:10:41 +01:00
}
var html = [];
2015-12-23 15:35:52 +01:00
var preHtml = '',
postHtml = '';
2011-03-26 14:10:41 +01:00
var curHTML = null;
2011-07-07 19:59:34 +02:00
function processSpaces(s)
{
2011-03-26 14:10:41 +01:00
return domline.processSpaces(s, doesWrap);
}
2012-03-17 13:36:42 +01:00
var perTextNodeProcess = (doesWrap ? _.identity : processSpaces);
var perHtmlLineProcess = (doesWrap ? processSpaces : _.identity);
2011-03-26 14:10:41 +01:00
var lineClass = 'ace-line';
2015-01-21 15:25:24 +01:00
2011-07-07 19:59:34 +02:00
result.appendSpan = function(txt, cls)
{
2015-01-21 15:25:24 +01:00
var processedMarker = false;
// Handle lineAttributeMarker, if present
if (cls.indexOf(lineAttributeMarker) >= 0)
2011-07-07 19:59:34 +02:00
{
2011-03-26 14:10:41 +01:00
var listType = /(?:^| )list:(\S+)/.exec(cls);
2012-01-15 18:20:20 +01:00
var start = /(?:^| )start:(\S+)/.exec(cls);
2014-03-05 00:14:15 +01:00
_.map(hooks.callAll("aceDomLinePreProcessLineAttributes", {
domline: domline,
cls: cls
}), function(modifier)
{
preHtml += modifier.preHtml;
postHtml += modifier.postHtml;
processedMarker |= modifier.processedMarker;
});
2011-07-07 19:59:34 +02:00
if (listType)
{
2011-03-26 14:10:41 +01:00
listType = listType[1];
2011-07-07 19:59:34 +02:00
if (listType)
{
2012-01-15 18:20:20 +01:00
if(listType.indexOf("number") < 0)
{
2014-03-05 22:44:32 +01:00
preHtml += '<ul class="list-' + Security.escapeHTMLAttribute(listType) + '"><li>';
postHtml = '</li></ul>' + postHtml;
2012-01-15 18:20:20 +01:00
}
else
{
if(start){ // is it a start of a list with more than one item in?
if(start[1] == 1){ // if its the first one at this level?
lineClass = lineClass + " " + "list-start-" + listType; // Add start class to DIV node
}
2014-03-05 22:44:32 +01:00
preHtml += '<ol start='+start[1]+' class="list-' + Security.escapeHTMLAttribute(listType) + '"><li>';
2014-03-05 00:14:15 +01:00
}else{
2014-03-05 22:44:32 +01:00
preHtml += '<ol class="list-' + Security.escapeHTMLAttribute(listType) + '"><li>'; // Handles pasted contents into existing lists
}
2014-03-05 22:44:32 +01:00
postHtml += '</li></ol>';
2012-01-15 18:20:20 +01:00
}
}
processedMarker = true;
}
_.map(hooks.callAll("aceDomLineProcessLineAttributes", {
domline: domline,
cls: cls
}), function(modifier)
{
preHtml += modifier.preHtml;
postHtml += modifier.postHtml;
processedMarker |= modifier.processedMarker;
});
if( processedMarker ){
2011-03-26 14:10:41 +01:00
result.lineMarker += txt.length;
return; // don't append any text
}
2011-03-26 14:10:41 +01:00
}
var href = null;
var simpleTags = null;
2011-07-07 19:59:34 +02:00
if (cls.indexOf('url') >= 0)
{
cls = cls.replace(/(^| )url:(\S+)/g, function(x0, space, url)
{
href = url;
return space + "url";
2011-03-26 14:10:41 +01:00
});
}
2011-07-07 19:59:34 +02:00
if (cls.indexOf('tag') >= 0)
{
cls = cls.replace(/(^| )tag:(\S+)/g, function(x0, space, tag)
{
if (!simpleTags) simpleTags = [];
simpleTags.push(tag.toLowerCase());
return space + tag;
2011-03-26 14:10:41 +01:00
});
}
var extraOpenTags = "";
var extraCloseTags = "";
2012-03-17 13:36:42 +01:00
_.map(hooks.callAll("aceCreateDomLine", {
2011-07-07 19:59:34 +02:00
domline: domline,
cls: cls
2012-03-17 13:36:42 +01:00
}), function(modifier)
2011-07-07 19:59:34 +02:00
{
2011-03-26 14:10:41 +01:00
cls = modifier.cls;
2011-07-07 19:59:34 +02:00
extraOpenTags = extraOpenTags + modifier.extraOpenTags;
extraCloseTags = modifier.extraCloseTags + extraCloseTags;
2011-03-26 14:10:41 +01:00
});
2011-07-07 19:59:34 +02:00
if ((!txt) && cls)
{
2011-03-26 14:10:41 +01:00
lineClass = domline.addToLineClass(lineClass, cls);
}
2011-07-07 19:59:34 +02:00
else if (txt)
{
if (href)
{
urn_schemes = new RegExp("^(about|geo|mailto|tel):");
if(!~href.indexOf("://") && !urn_schemes.test(href)) // if the url doesn't include a protocol prefix, assume http
{
href = "http://"+href;
}
extraOpenTags = extraOpenTags + '<a href="' + Security.escapeHTMLAttribute(href) + '">';
2011-07-07 19:59:34 +02:00
extraCloseTags = '</a>' + extraCloseTags;
2011-03-26 14:10:41 +01:00
}
2011-07-07 19:59:34 +02:00
if (simpleTags)
{
simpleTags.sort();
extraOpenTags = extraOpenTags + '<' + simpleTags.join('><') + '>';
simpleTags.reverse();
extraCloseTags = '</' + simpleTags.join('></') + '>' + extraCloseTags;
2011-03-26 14:10:41 +01:00
}
html.push('<span class="', Security.escapeHTMLAttribute(cls || ''), '">', extraOpenTags, perTextNodeProcess(Security.escapeHTML(txt)), extraCloseTags, '</span>');
2011-03-26 14:10:41 +01:00
}
};
2011-07-07 19:59:34 +02:00
result.clearSpans = function()
{
2011-03-26 14:10:41 +01:00
html = [];
lineClass = ''; // non-null to cause update
result.lineMarker = 0;
};
2011-07-07 19:59:34 +02:00
function writeHTML()
{
2011-03-26 14:10:41 +01:00
var newHTML = perHtmlLineProcess(html.join(''));
2011-07-07 19:59:34 +02:00
if (!newHTML)
{
if ((!document) || (!optBrowser))
{
2011-03-26 14:10:41 +01:00
newHTML += '&nbsp;';
}
2015-01-21 03:55:03 +01:00
else if (!optBrowser.msie)
2011-07-07 19:59:34 +02:00
{
2011-03-26 14:10:41 +01:00
newHTML += '<br/>';
}
}
2011-07-07 19:59:34 +02:00
if (nonEmpty)
{
newHTML = (preHtml || '') + newHTML + (postHtml || '');
2011-03-26 14:10:41 +01:00
}
html = preHtml = postHtml = ''; // free memory
2011-07-07 19:59:34 +02:00
if (newHTML !== curHTML)
{
2011-03-26 14:10:41 +01:00
curHTML = newHTML;
result.node.innerHTML = curHTML;
}
if (lineClass !== null) result.node.className = lineClass;
2014-03-05 00:14:15 +01:00
hooks.callAll("acePostWriteDomLineHTML", {
node: result.node
2014-03-05 00:36:16 +01:00
});
2011-03-26 14:10:41 +01:00
}
result.prepareForAdd = writeHTML;
result.finishUpdate = writeHTML;
2011-07-07 19:59:34 +02:00
result.getInnerHTML = function()
{
return curHTML || '';
};
2011-03-26 14:10:41 +01:00
return result;
};
2011-07-07 19:59:34 +02:00
domline.processSpaces = function(s, doesWrap)
{
if (s.indexOf("<") < 0 && !doesWrap)
{
2011-03-26 14:10:41 +01:00
// short-cut
return s.replace(/ /g, '&nbsp;');
}
var parts = [];
2011-07-07 19:59:34 +02:00
s.replace(/<[^>]*>?| |[^ <]+/g, function(m)
{
parts.push(m);
});
if (doesWrap)
{
2011-03-26 14:10:41 +01:00
var endOfLine = true;
var beforeSpace = false;
// last space in a run is normal, others are nbsp,
// end of line is nbsp
2011-07-07 19:59:34 +02:00
for (var i = parts.length - 1; i >= 0; i--)
{
2011-03-26 14:10:41 +01:00
var p = parts[i];
2011-07-07 19:59:34 +02:00
if (p == " ")
{
if (endOfLine || beforeSpace) parts[i] = '&nbsp;';
endOfLine = false;
beforeSpace = true;
2011-03-26 14:10:41 +01:00
}
2011-07-07 19:59:34 +02:00
else if (p.charAt(0) != "<")
{
endOfLine = false;
beforeSpace = false;
2011-03-26 14:10:41 +01:00
}
}
// beginning of line is nbsp
2011-07-07 19:59:34 +02:00
for (var i = 0; i < parts.length; i++)
{
2011-03-26 14:10:41 +01:00
var p = parts[i];
2011-07-07 19:59:34 +02:00
if (p == " ")
{
parts[i] = '&nbsp;';
break;
2011-03-26 14:10:41 +01:00
}
2011-07-07 19:59:34 +02:00
else if (p.charAt(0) != "<")
{
break;
2011-03-26 14:10:41 +01:00
}
}
}
2011-07-07 19:59:34 +02:00
else
{
for (var i = 0; i < parts.length; i++)
{
2011-03-26 14:10:41 +01:00
var p = parts[i];
2011-07-07 19:59:34 +02:00
if (p == " ")
{
parts[i] = '&nbsp;';
2011-03-26 14:10:41 +01:00
}
}
}
return parts.join('');
};
exports.domline = domline;