214 lines
7.6 KiB
JavaScript
214 lines
7.6 KiB
JavaScript
|
'use strict';
|
||
|
|
||
|
var Mixin = require('../../utils/mixin'),
|
||
|
Tokenizer = require('../../tokenizer'),
|
||
|
LocationInfoTokenizerMixin = require('./tokenizer_mixin'),
|
||
|
PositionTrackingPreprocessorMixin = require('../position_tracking/preprocessor_mixin'),
|
||
|
LocationInfoOpenElementStackMixin = require('./open_element_stack_mixin'),
|
||
|
HTML = require('../../common/html'),
|
||
|
inherits = require('util').inherits;
|
||
|
|
||
|
|
||
|
//Aliases
|
||
|
var $ = HTML.TAG_NAMES;
|
||
|
|
||
|
var LocationInfoParserMixin = module.exports = function (parser) {
|
||
|
Mixin.call(this, parser);
|
||
|
|
||
|
this.parser = parser;
|
||
|
this.posTracker = null;
|
||
|
this.lastStartTagToken = null;
|
||
|
this.lastFosterParentingLocation = null;
|
||
|
this.currentToken = null;
|
||
|
};
|
||
|
|
||
|
inherits(LocationInfoParserMixin, Mixin);
|
||
|
|
||
|
|
||
|
LocationInfoParserMixin.prototype._setStartLocation = function (element) {
|
||
|
if (this.lastStartTagToken) {
|
||
|
element.__location = Object.create(this.lastStartTagToken.location);
|
||
|
element.__location.startTag = this.lastStartTagToken.location;
|
||
|
}
|
||
|
else
|
||
|
element.__location = null;
|
||
|
};
|
||
|
|
||
|
LocationInfoParserMixin.prototype._setEndLocation = function (element, closingToken) {
|
||
|
var loc = element.__location;
|
||
|
|
||
|
if (loc) {
|
||
|
if (closingToken.location) {
|
||
|
var ctLoc = closingToken.location,
|
||
|
tn = this.parser.treeAdapter.getTagName(element);
|
||
|
|
||
|
// NOTE: For cases like <p> <p> </p> - First 'p' closes without a closing
|
||
|
// tag and for cases like <td> <p> </td> - 'p' closes without a closing tag.
|
||
|
var isClosingEndTag = closingToken.type === Tokenizer.END_TAG_TOKEN && tn === closingToken.tagName;
|
||
|
|
||
|
if (isClosingEndTag) {
|
||
|
loc.endTag = Object.create(ctLoc);
|
||
|
loc.endOffset = ctLoc.endOffset;
|
||
|
}
|
||
|
|
||
|
else
|
||
|
loc.endOffset = ctLoc.startOffset;
|
||
|
}
|
||
|
|
||
|
else if (closingToken.type === Tokenizer.EOF_TOKEN)
|
||
|
loc.endOffset = this.posTracker.offset;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
LocationInfoParserMixin.prototype._getOverriddenMethods = function (mxn, orig) {
|
||
|
return {
|
||
|
_bootstrap: function (document, fragmentContext) {
|
||
|
orig._bootstrap.call(this, document, fragmentContext);
|
||
|
|
||
|
mxn.lastStartTagToken = null;
|
||
|
mxn.lastFosterParentingLocation = null;
|
||
|
mxn.currentToken = null;
|
||
|
mxn.posTracker = new PositionTrackingPreprocessorMixin(this.tokenizer.preprocessor);
|
||
|
|
||
|
new LocationInfoTokenizerMixin(this.tokenizer);
|
||
|
|
||
|
new LocationInfoOpenElementStackMixin(this.openElements, {
|
||
|
onItemPop: function (element) {
|
||
|
mxn._setEndLocation(element, mxn.currentToken);
|
||
|
}
|
||
|
});
|
||
|
},
|
||
|
|
||
|
_runParsingLoop: function (scriptHandler) {
|
||
|
orig._runParsingLoop.call(this, scriptHandler);
|
||
|
|
||
|
// NOTE: generate location info for elements
|
||
|
// that remains on open element stack
|
||
|
for (var i = this.openElements.stackTop; i >= 0; i--)
|
||
|
mxn._setEndLocation(this.openElements.items[i], mxn.currentToken);
|
||
|
},
|
||
|
|
||
|
|
||
|
//Token processing
|
||
|
_processTokenInForeignContent: function (token) {
|
||
|
mxn.currentToken = token;
|
||
|
orig._processTokenInForeignContent.call(this, token);
|
||
|
},
|
||
|
|
||
|
_processToken: function (token) {
|
||
|
mxn.currentToken = token;
|
||
|
orig._processToken.call(this, token);
|
||
|
|
||
|
//NOTE: <body> and <html> are never popped from the stack, so we need to updated
|
||
|
//their end location explicitly.
|
||
|
var requireExplicitUpdate = token.type === Tokenizer.END_TAG_TOKEN &&
|
||
|
(token.tagName === $.HTML ||
|
||
|
token.tagName === $.BODY && this.openElements.hasInScope($.BODY));
|
||
|
|
||
|
if (requireExplicitUpdate) {
|
||
|
for (var i = this.openElements.stackTop; i >= 0; i--) {
|
||
|
var element = this.openElements.items[i];
|
||
|
|
||
|
if (this.treeAdapter.getTagName(element) === token.tagName) {
|
||
|
mxn._setEndLocation(element, token);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
|
||
|
//Doctype
|
||
|
_setDocumentType: function (token) {
|
||
|
orig._setDocumentType.call(this, token);
|
||
|
|
||
|
var documentChildren = this.treeAdapter.getChildNodes(this.document),
|
||
|
cnLength = documentChildren.length;
|
||
|
|
||
|
for (var i = 0; i < cnLength; i++) {
|
||
|
var node = documentChildren[i];
|
||
|
|
||
|
if (this.treeAdapter.isDocumentTypeNode(node)) {
|
||
|
node.__location = token.location;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
},
|
||
|
|
||
|
|
||
|
//Elements
|
||
|
_attachElementToTree: function (element) {
|
||
|
//NOTE: _attachElementToTree is called from _appendElement, _insertElement and _insertTemplate methods.
|
||
|
//So we will use token location stored in this methods for the element.
|
||
|
mxn._setStartLocation(element);
|
||
|
mxn.lastStartTagToken = null;
|
||
|
orig._attachElementToTree.call(this, element);
|
||
|
},
|
||
|
|
||
|
_appendElement: function (token, namespaceURI) {
|
||
|
mxn.lastStartTagToken = token;
|
||
|
orig._appendElement.call(this, token, namespaceURI);
|
||
|
},
|
||
|
|
||
|
_insertElement: function (token, namespaceURI) {
|
||
|
mxn.lastStartTagToken = token;
|
||
|
orig._insertElement.call(this, token, namespaceURI);
|
||
|
},
|
||
|
|
||
|
_insertTemplate: function (token) {
|
||
|
mxn.lastStartTagToken = token;
|
||
|
orig._insertTemplate.call(this, token);
|
||
|
|
||
|
var tmplContent = this.treeAdapter.getTemplateContent(this.openElements.current);
|
||
|
|
||
|
tmplContent.__location = null;
|
||
|
},
|
||
|
|
||
|
_insertFakeRootElement: function () {
|
||
|
orig._insertFakeRootElement.call(this);
|
||
|
this.openElements.current.__location = null;
|
||
|
},
|
||
|
|
||
|
//Comments
|
||
|
_appendCommentNode: function (token, parent) {
|
||
|
orig._appendCommentNode.call(this, token, parent);
|
||
|
|
||
|
var children = this.treeAdapter.getChildNodes(parent),
|
||
|
commentNode = children[children.length - 1];
|
||
|
|
||
|
commentNode.__location = token.location;
|
||
|
},
|
||
|
|
||
|
//Text
|
||
|
_findFosterParentingLocation: function () {
|
||
|
//NOTE: store last foster parenting location, so we will be able to find inserted text
|
||
|
//in case of foster parenting
|
||
|
mxn.lastFosterParentingLocation = orig._findFosterParentingLocation.call(this);
|
||
|
|
||
|
return mxn.lastFosterParentingLocation;
|
||
|
},
|
||
|
|
||
|
_insertCharacters: function (token) {
|
||
|
orig._insertCharacters.call(this, token);
|
||
|
|
||
|
var hasFosterParent = this._shouldFosterParentOnInsertion(),
|
||
|
parent = hasFosterParent && mxn.lastFosterParentingLocation.parent ||
|
||
|
this.openElements.currentTmplContent ||
|
||
|
this.openElements.current,
|
||
|
siblings = this.treeAdapter.getChildNodes(parent),
|
||
|
textNodeIdx = hasFosterParent && mxn.lastFosterParentingLocation.beforeElement ?
|
||
|
siblings.indexOf(mxn.lastFosterParentingLocation.beforeElement) - 1 :
|
||
|
siblings.length - 1,
|
||
|
textNode = siblings[textNodeIdx];
|
||
|
|
||
|
//NOTE: if we have location assigned by another token, then just update end position
|
||
|
if (textNode.__location)
|
||
|
textNode.__location.endOffset = token.location.endOffset;
|
||
|
|
||
|
else
|
||
|
textNode.__location = token.location;
|
||
|
}
|
||
|
};
|
||
|
};
|
||
|
|