mirror of
https://github.com/classilla/tenfourfox.git
synced 2024-06-26 11:29:31 +00:00
980 lines
32 KiB
JavaScript
980 lines
32 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
var { utils: Cu, interfaces: Ci, classes: Cc } = Components;
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
|
|
XPCOMUtils.defineLazyModuleGetter(this, "BrowserUtils",
|
|
"resource://gre/modules/BrowserUtils.jsm");
|
|
XPCOMUtils.defineLazyModuleGetter(this, "DeferredTask",
|
|
"resource://gre/modules/DeferredTask.jsm");
|
|
|
|
const NS_XHTML = "http://www.w3.org/1999/xhtml";
|
|
const BUNDLE_URL = "chrome://global/locale/viewSource.properties";
|
|
|
|
// These are markers used to delimit the selection during processing. They
|
|
// are removed from the final rendering.
|
|
// We use noncharacter Unicode codepoints to minimize the risk of clashing
|
|
// with anything that might legitimately be present in the document.
|
|
// U+FDD0..FDEF <noncharacters>
|
|
const MARK_SELECTION_START = "\uFDD0";
|
|
const MARK_SELECTION_END = "\uFDEF";
|
|
|
|
var global = this;
|
|
|
|
/**
|
|
* ViewSourceContent should be loaded in the <xul:browser> of the
|
|
* view source window, and initialized as soon as it has loaded.
|
|
*/
|
|
var ViewSourceContent = {
|
|
/**
|
|
* We'll act as an nsISelectionListener as well so that we can send
|
|
* updates to the view source window's status bar.
|
|
*/
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsISelectionListener]),
|
|
|
|
/**
|
|
* These are the messages that ViewSourceContent is prepared to listen
|
|
* for. If you need ViewSourceContent to handle more messages, add them
|
|
* here.
|
|
*/
|
|
messages: [
|
|
"ViewSource:LoadSource",
|
|
"ViewSource:LoadSourceDeprecated",
|
|
"ViewSource:LoadSourceWithSelection",
|
|
"ViewSource:GoToLine",
|
|
"ViewSource:ToggleWrapping",
|
|
"ViewSource:ToggleSyntaxHighlighting",
|
|
"ViewSource:SetCharacterSet",
|
|
],
|
|
|
|
/**
|
|
* When showing selection source, chrome will construct a page fragment to
|
|
* show, and then instruct content to draw a selection after load. This is
|
|
* set true when there is a pending request to draw selection.
|
|
*/
|
|
needsDrawSelection: false,
|
|
|
|
/**
|
|
* ViewSourceContent is attached as an nsISelectionListener on pageshow,
|
|
* and removed on pagehide. When the initial about:blank is transitioned
|
|
* away from, a pagehide is fired without us having attached ourselves
|
|
* first. We use this boolean to keep track of whether or not we're
|
|
* attached, so we don't attempt to remove our listener when it's not
|
|
* yet there (which throws).
|
|
*/
|
|
selectionListenerAttached: false,
|
|
|
|
get isViewSource() {
|
|
let uri = content.document.documentURI;
|
|
return uri.startsWith("view-source:") ||
|
|
(uri.startsWith("data:") && uri.includes("MathML"));
|
|
},
|
|
|
|
get isAboutBlank() {
|
|
let uri = content.document.documentURI;
|
|
return uri == "about:blank";
|
|
},
|
|
|
|
/**
|
|
* This should be called as soon as this frame script has loaded.
|
|
*/
|
|
init() {
|
|
this.messages.forEach((msgName) => {
|
|
addMessageListener(msgName, this);
|
|
});
|
|
|
|
addEventListener("pagehide", this, true);
|
|
addEventListener("pageshow", this, true);
|
|
addEventListener("click", this);
|
|
addEventListener("unload", this);
|
|
Services.els.addSystemEventListener(global, "contextmenu", this, false);
|
|
},
|
|
|
|
/**
|
|
* This should be called when the frame script is being unloaded,
|
|
* and the browser is tearing down.
|
|
*/
|
|
uninit() {
|
|
this.messages.forEach((msgName) => {
|
|
removeMessageListener(msgName, this);
|
|
});
|
|
|
|
removeEventListener("pagehide", this, true);
|
|
removeEventListener("pageshow", this, true);
|
|
removeEventListener("click", this);
|
|
removeEventListener("unload", this);
|
|
|
|
Services.els.removeSystemEventListener(global, "contextmenu", this, false);
|
|
|
|
// Cancel any pending toolbar updates.
|
|
if (this.updateStatusTask) {
|
|
this.updateStatusTask.disarm();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Anything added to the messages array will get handled here, and should
|
|
* get dispatched to a specific function for the message name.
|
|
*/
|
|
receiveMessage(msg) {
|
|
if (!this.isViewSource && !this.isAboutBlank) {
|
|
return;
|
|
}
|
|
let data = msg.data;
|
|
let objects = msg.objects;
|
|
switch(msg.name) {
|
|
case "ViewSource:LoadSource":
|
|
this.viewSource(data.URL, data.outerWindowID, data.lineNumber,
|
|
data.shouldWrap);
|
|
break;
|
|
case "ViewSource:LoadSourceDeprecated":
|
|
this.viewSourceDeprecated(data.URL, objects.pageDescriptor, data.lineNumber,
|
|
data.forcedCharSet);
|
|
break;
|
|
case "ViewSource:LoadSourceWithSelection":
|
|
this.viewSourceWithSelection(data.URL, data.drawSelection, data.baseURI);
|
|
break;
|
|
case "ViewSource:GoToLine":
|
|
this.goToLine(data.lineNumber);
|
|
break;
|
|
case "ViewSource:ToggleWrapping":
|
|
this.toggleWrapping();
|
|
break;
|
|
case "ViewSource:ToggleSyntaxHighlighting":
|
|
this.toggleSyntaxHighlighting();
|
|
break;
|
|
case "ViewSource:SetCharacterSet":
|
|
this.setCharacterSet(data.charset, data.doPageLoad);
|
|
break;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Any events should get handled here, and should get dispatched to
|
|
* a specific function for the event type.
|
|
*/
|
|
handleEvent(event) {
|
|
if (!this.isViewSource) {
|
|
return;
|
|
}
|
|
switch(event.type) {
|
|
case "pagehide":
|
|
this.onPageHide(event);
|
|
break;
|
|
case "pageshow":
|
|
this.onPageShow(event);
|
|
break;
|
|
case "click":
|
|
this.onClick(event);
|
|
break;
|
|
case "unload":
|
|
this.uninit();
|
|
break;
|
|
case "contextmenu":
|
|
this.onContextMenu(event);
|
|
break;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* A getter for the view source string bundle.
|
|
*/
|
|
get bundle() {
|
|
delete this.bundle;
|
|
this.bundle = Services.strings.createBundle(BUNDLE_URL);
|
|
return this.bundle;
|
|
},
|
|
|
|
/**
|
|
* A shortcut to the nsISelectionController for the content.
|
|
*/
|
|
get selectionController() {
|
|
return docShell.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
.getInterface(Ci.nsISelectionDisplay)
|
|
.QueryInterface(Ci.nsISelectionController);
|
|
},
|
|
|
|
/**
|
|
* A shortcut to the nsIWebBrowserFind for the content.
|
|
*/
|
|
get webBrowserFind() {
|
|
return docShell.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
.getInterface(Ci.nsIWebBrowserFind);
|
|
},
|
|
|
|
/**
|
|
* Called when the parent sends a message to view some source code.
|
|
*
|
|
* @param URL (required)
|
|
* The URL string of the source to be shown.
|
|
* @param outerWindowID (optional)
|
|
* The outerWindowID of the content window that has hosted
|
|
* the document, in case we want to retrieve it from the network
|
|
* cache.
|
|
* @param lineNumber (optional)
|
|
* The line number to focus as soon as the source has finished
|
|
* loading.
|
|
*/
|
|
viewSource(URL, outerWindowID, lineNumber) {
|
|
let pageDescriptor, forcedCharSet;
|
|
|
|
if (outerWindowID) {
|
|
let contentWindow = Services.wm.getOuterWindowWithId(outerWindowID);
|
|
let requestor = contentWindow.QueryInterface(Ci.nsIInterfaceRequestor);
|
|
|
|
try {
|
|
let otherWebNav = requestor.getInterface(Ci.nsIWebNavigation);
|
|
pageDescriptor = otherWebNav.QueryInterface(Ci.nsIWebPageDescriptor)
|
|
.currentDescriptor;
|
|
} catch(e) {
|
|
// We couldn't get the page descriptor, so we'll probably end up re-retrieving
|
|
// this document off of the network.
|
|
}
|
|
|
|
let utils = requestor.getInterface(Ci.nsIDOMWindowUtils);
|
|
let doc = contentWindow.document;
|
|
let forcedCharSet = utils.docCharsetIsForced ? doc.characterSet
|
|
: null;
|
|
}
|
|
|
|
this.loadSource(URL, pageDescriptor, lineNumber, forcedCharSet);
|
|
},
|
|
|
|
/**
|
|
* Called when the parent is using the deprecated API for viewSource.xul.
|
|
* This function will throw if it's called on a remote browser.
|
|
*
|
|
* @param URL (required)
|
|
* The URL string of the source to be shown.
|
|
* @param pageDescriptor (optional)
|
|
* The currentDescriptor off of an nsIWebPageDescriptor, in the
|
|
* event that the caller wants to try to load the source out of
|
|
* the network cache.
|
|
* @param lineNumber (optional)
|
|
* The line number to focus as soon as the source has finished
|
|
* loading.
|
|
* @param forcedCharSet (optional)
|
|
* The document character set to use instead of the default one.
|
|
*/
|
|
viewSourceDeprecated(URL, pageDescriptor, lineNumber, forcedCharSet) {
|
|
// This should not be called if this frame script is running
|
|
// in a content process!
|
|
if (Services.appinfo.processType != Services.appinfo.PROCESS_TYPE_DEFAULT) {
|
|
throw new Error("ViewSource deprecated API should not be used with " +
|
|
"remote browsers.");
|
|
}
|
|
|
|
this.loadSource(URL, pageDescriptor, lineNumber, forcedCharSet);
|
|
},
|
|
|
|
/**
|
|
* Common utility function used by both the current and deprecated APIs
|
|
* for loading source.
|
|
*
|
|
* @param URL (required)
|
|
* The URL string of the source to be shown.
|
|
* @param pageDescriptor (optional)
|
|
* The currentDescriptor off of an nsIWebPageDescriptor, in the
|
|
* event that the caller wants to try to load the source out of
|
|
* the network cache.
|
|
* @param lineNumber (optional)
|
|
* The line number to focus as soon as the source has finished
|
|
* loading.
|
|
* @param forcedCharSet (optional)
|
|
* The document character set to use instead of the default one.
|
|
*/
|
|
loadSource(URL, pageDescriptor, lineNumber, forcedCharSet) {
|
|
const viewSrcURL = "view-source:" + URL;
|
|
let loadFromURL = false;
|
|
|
|
if (forcedCharSet) {
|
|
docShell.charset = forcedCharSet;
|
|
}
|
|
|
|
if (lineNumber && lineNumber > 0) {
|
|
let doneLoading = (event) => {
|
|
// Ignore possible initial load of about:blank
|
|
if (this.isAboutBlank ||
|
|
!content.document.body) {
|
|
return;
|
|
}
|
|
this.goToLine(lineNumber);
|
|
removeEventListener("pageshow", doneLoading);
|
|
};
|
|
|
|
addEventListener("pageshow", doneLoading);
|
|
}
|
|
|
|
if (!pageDescriptor) {
|
|
this.loadSourceFromURL(viewSrcURL);
|
|
return;
|
|
}
|
|
|
|
try {
|
|
let pageLoader = docShell.QueryInterface(Ci.nsIWebPageDescriptor);
|
|
pageLoader.loadPage(pageDescriptor,
|
|
Ci.nsIWebPageDescriptor.DISPLAY_AS_SOURCE);
|
|
} catch(e) {
|
|
// We were not able to load the source from the network cache.
|
|
this.loadSourceFromURL(viewSrcURL);
|
|
return;
|
|
}
|
|
|
|
let shEntrySource = pageDescriptor.QueryInterface(Ci.nsISHEntry);
|
|
let shEntry = Cc["@mozilla.org/browser/session-history-entry;1"]
|
|
.createInstance(Ci.nsISHEntry);
|
|
shEntry.setURI(BrowserUtils.makeURI(viewSrcURL, null, null));
|
|
shEntry.setTitle(viewSrcURL);
|
|
shEntry.loadType = Ci.nsIDocShellLoadInfo.loadHistory;
|
|
shEntry.cacheKey = shEntrySource.cacheKey;
|
|
docShell.QueryInterface(Ci.nsIWebNavigation)
|
|
.sessionHistory
|
|
.QueryInterface(Ci.nsISHistoryInternal)
|
|
.addEntry(shEntry, true);
|
|
},
|
|
|
|
/**
|
|
* Load some URL in the browser.
|
|
*
|
|
* @param URL
|
|
* The URL string to load.
|
|
*/
|
|
loadSourceFromURL(URL) {
|
|
let loadFlags = Ci.nsIWebNavigation.LOAD_FLAGS_NONE;
|
|
let webNav = docShell.QueryInterface(Ci.nsIWebNavigation);
|
|
webNav.loadURI(URL, loadFlags, null, null, null);
|
|
},
|
|
|
|
/**
|
|
* This handler is for click events from:
|
|
* * error page content, which can show up if the user attempts to view the
|
|
* source of an attack page.
|
|
* * in-page context menu actions
|
|
*/
|
|
onClick(event) {
|
|
let target = event.originalTarget;
|
|
// Check for content menu actions
|
|
if (target.id) {
|
|
this.contextMenuItems.forEach(itemSpec => {
|
|
if (itemSpec.id !== target.id) {
|
|
return;
|
|
}
|
|
itemSpec.handler.call(this, event);
|
|
event.stopPropagation();
|
|
});
|
|
}
|
|
|
|
// Don't trust synthetic events
|
|
if (!event.isTrusted || event.target.localName != "button")
|
|
return;
|
|
|
|
let errorDoc = target.ownerDocument;
|
|
|
|
if (/^about:blocked/.test(errorDoc.documentURI)) {
|
|
// The event came from a button on a malware/phishing block page
|
|
|
|
if (target == errorDoc.getElementById("getMeOutButton")) {
|
|
// Instead of loading some safe page, just close the window
|
|
sendAsyncMessage("ViewSource:Close");
|
|
} else if (target == errorDoc.getElementById("reportButton")) {
|
|
// This is the "Why is this site blocked" button. We redirect
|
|
// to the generic page describing phishing/malware protection.
|
|
let URL = Services.urlFormatter.formatURLPref("app.support.baseURL");
|
|
sendAsyncMessage("ViewSource:OpenURL", { URL })
|
|
} else if (target == errorDoc.getElementById("ignoreWarningButton")) {
|
|
// Allow users to override and continue through to the site
|
|
docShell.QueryInterface(Ci.nsIWebNavigation)
|
|
.loadURIWithOptions(content.location.href,
|
|
Ci.nsIWebNavigation.LOAD_FLAGS_BYPASS_CLASSIFIER,
|
|
null, Ci.nsIHttpChannel.REFERRER_POLICY_DEFAULT,
|
|
null, null, null);
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Handler for the pageshow event.
|
|
*
|
|
* @param event
|
|
* The pageshow event being handled.
|
|
*/
|
|
onPageShow(event) {
|
|
let selection = content.getSelection();
|
|
if (selection) {
|
|
selection.QueryInterface(Ci.nsISelectionPrivate)
|
|
.addSelectionListener(this);
|
|
this.selectionListenerAttached = true;
|
|
}
|
|
content.focus();
|
|
|
|
// If we need to draw the selection, wait until an actual view source page
|
|
// has loaded, instead of about:blank.
|
|
if (this.needsDrawSelection &&
|
|
content.document.documentURI.startsWith("view-source:")) {
|
|
this.needsDrawSelection = false;
|
|
this.drawSelection();
|
|
}
|
|
|
|
if (content.document.body) {
|
|
this.injectContextMenu();
|
|
}
|
|
|
|
sendAsyncMessage("ViewSource:SourceLoaded");
|
|
},
|
|
|
|
/**
|
|
* Handler for the pagehide event.
|
|
*
|
|
* @param event
|
|
* The pagehide event being handled.
|
|
*/
|
|
onPageHide(event) {
|
|
// The initial about:blank will fire pagehide before we
|
|
// ever set a selectionListener, so we have a boolean around
|
|
// to keep track of when the listener is attached.
|
|
if (this.selectionListenerAttached) {
|
|
content.getSelection()
|
|
.QueryInterface(Ci.nsISelectionPrivate)
|
|
.removeSelectionListener(this);
|
|
this.selectionListenerAttached = false;
|
|
}
|
|
sendAsyncMessage("ViewSource:SourceUnloaded");
|
|
},
|
|
|
|
onContextMenu(event) {
|
|
let addonInfo = {};
|
|
let subject = {
|
|
event: event,
|
|
addonInfo: addonInfo,
|
|
};
|
|
|
|
subject.wrappedJSObject = subject;
|
|
Services.obs.notifyObservers(subject, "content-contextmenu", null);
|
|
|
|
let node = event.target;
|
|
|
|
let result = {
|
|
isEmail: false,
|
|
isLink: false,
|
|
href: "",
|
|
// We have to pass these in the event that we're running in
|
|
// a remote browser, so that ViewSourceChrome knows where to
|
|
// open the context menu.
|
|
screenX: event.screenX,
|
|
screenY: event.screenY,
|
|
};
|
|
|
|
if (node && node.localName == "a") {
|
|
result.isLink = node.href.startsWith("view-source:");
|
|
result.isEmail = node.href.startsWith("mailto:");
|
|
result.href = node.href.substring(node.href.indexOf(":") + 1);
|
|
}
|
|
|
|
sendSyncMessage("ViewSource:ContextMenuOpening", result);
|
|
},
|
|
|
|
/**
|
|
* Attempts to go to a particular line in the source code being
|
|
* shown. If it succeeds in finding the line, it will fire a
|
|
* "ViewSource:GoToLine:Success" message, passing up an object
|
|
* with the lineNumber we just went to. If it cannot find the line,
|
|
* it will fire a "ViewSource:GoToLine:Failed" message.
|
|
*
|
|
* @param lineNumber
|
|
* The line number to attempt to go to.
|
|
*/
|
|
goToLine(lineNumber) {
|
|
let body = content.document.body;
|
|
|
|
// The source document is made up of a number of pre elements with
|
|
// id attributes in the format <pre id="line123">, meaning that
|
|
// the first line in the pre element is number 123.
|
|
// Do binary search to find the pre element containing the line.
|
|
// However, in the plain text case, we have only one pre without an
|
|
// attribute, so assume it begins on line 1.
|
|
let pre;
|
|
for (let lbound = 0, ubound = body.childNodes.length; ; ) {
|
|
let middle = (lbound + ubound) >> 1;
|
|
pre = body.childNodes[middle];
|
|
|
|
let firstLine = pre.id ? parseInt(pre.id.substring(4)) : 1;
|
|
|
|
if (lbound == ubound - 1) {
|
|
break;
|
|
}
|
|
|
|
if (lineNumber >= firstLine) {
|
|
lbound = middle;
|
|
} else {
|
|
ubound = middle;
|
|
}
|
|
}
|
|
|
|
let result = {};
|
|
let found = this.findLocation(pre, lineNumber, null, -1, false, result);
|
|
|
|
if (!found) {
|
|
sendAsyncMessage("ViewSource:GoToLine:Failed");
|
|
return;
|
|
}
|
|
|
|
let selection = content.getSelection();
|
|
selection.removeAllRanges();
|
|
|
|
// In our case, the range's startOffset is after "\n" on the previous line.
|
|
// Tune the selection at the beginning of the next line and do some tweaking
|
|
// to position the focusNode and the caret at the beginning of the line.
|
|
selection.QueryInterface(Ci.nsISelectionPrivate)
|
|
.interlinePosition = true;
|
|
|
|
selection.addRange(result.range);
|
|
|
|
if (!selection.isCollapsed) {
|
|
selection.collapseToEnd();
|
|
|
|
let offset = result.range.startOffset;
|
|
let node = result.range.startContainer;
|
|
if (offset < node.data.length) {
|
|
// The same text node spans across the "\n", just focus where we were.
|
|
selection.extend(node, offset);
|
|
}
|
|
else {
|
|
// There is another tag just after the "\n", hook there. We need
|
|
// to focus a safe point because there are edgy cases such as
|
|
// <span>...\n</span><span>...</span> vs.
|
|
// <span>...\n<span>...</span></span><span>...</span>
|
|
node = node.nextSibling ? node.nextSibling : node.parentNode.nextSibling;
|
|
selection.extend(node, 0);
|
|
}
|
|
}
|
|
|
|
let selCon = this.selectionController;
|
|
selCon.setDisplaySelection(Ci.nsISelectionController.SELECTION_ON);
|
|
selCon.setCaretVisibilityDuringSelection(true);
|
|
|
|
// Scroll the beginning of the line into view.
|
|
selCon.scrollSelectionIntoView(
|
|
Ci.nsISelectionController.SELECTION_NORMAL,
|
|
Ci.nsISelectionController.SELECTION_FOCUS_REGION,
|
|
true);
|
|
|
|
sendAsyncMessage("ViewSource:GoToLine:Success", { lineNumber });
|
|
},
|
|
|
|
|
|
/**
|
|
* Some old code from the original view source implementation. Original
|
|
* documentation follows:
|
|
*
|
|
* "Loops through the text lines in the pre element. The arguments are either
|
|
* (pre, line) or (node, offset, interlinePosition). result is an out
|
|
* argument. If (pre, line) are specified (and node == null), result.range is
|
|
* a range spanning the specified line. If the (node, offset,
|
|
* interlinePosition) are specified, result.line and result.col are the line
|
|
* and column number of the specified offset in the specified node relative to
|
|
* the whole file."
|
|
*/
|
|
findLocation(pre, lineNumber, node, offset, interlinePosition, result) {
|
|
if (node && !pre) {
|
|
// Look upwards to find the current pre element.
|
|
for (pre = node;
|
|
pre.nodeName != "PRE";
|
|
pre = pre.parentNode);
|
|
}
|
|
|
|
// The source document is made up of a number of pre elements with
|
|
// id attributes in the format <pre id="line123">, meaning that
|
|
// the first line in the pre element is number 123.
|
|
// However, in the plain text case, there is only one <pre> without an id,
|
|
// so assume line 1.
|
|
let curLine = pre.id ? parseInt(pre.id.substring(4)) : 1;
|
|
|
|
// Walk through each of the text nodes and count newlines.
|
|
let treewalker = content.document
|
|
.createTreeWalker(pre, Ci.nsIDOMNodeFilter.SHOW_TEXT, null);
|
|
|
|
// The column number of the first character in the current text node.
|
|
let firstCol = 1;
|
|
|
|
let found = false;
|
|
for (let textNode = treewalker.firstChild();
|
|
textNode && !found;
|
|
textNode = treewalker.nextNode()) {
|
|
|
|
// \r is not a valid character in the DOM, so we only check for \n.
|
|
let lineArray = textNode.data.split(/\n/);
|
|
let lastLineInNode = curLine + lineArray.length - 1;
|
|
|
|
// Check if we can skip the text node without further inspection.
|
|
if (node ? (textNode != node) : (lastLineInNode < lineNumber)) {
|
|
if (lineArray.length > 1) {
|
|
firstCol = 1;
|
|
}
|
|
firstCol += lineArray[lineArray.length - 1].length;
|
|
curLine = lastLineInNode;
|
|
continue;
|
|
}
|
|
|
|
// curPos is the offset within the current text node of the first
|
|
// character in the current line.
|
|
for (var i = 0, curPos = 0;
|
|
i < lineArray.length;
|
|
curPos += lineArray[i++].length + 1) {
|
|
|
|
if (i > 0) {
|
|
curLine++;
|
|
}
|
|
|
|
if (node) {
|
|
if (offset >= curPos && offset <= curPos + lineArray[i].length) {
|
|
// If we are right after the \n of a line and interlinePosition is
|
|
// false, the caret looks as if it were at the end of the previous
|
|
// line, so we display that line and column instead.
|
|
|
|
if (i > 0 && offset == curPos && !interlinePosition) {
|
|
result.line = curLine - 1;
|
|
var prevPos = curPos - lineArray[i - 1].length;
|
|
result.col = (i == 1 ? firstCol : 1) + offset - prevPos;
|
|
} else {
|
|
result.line = curLine;
|
|
result.col = (i == 0 ? firstCol : 1) + offset - curPos;
|
|
}
|
|
found = true;
|
|
|
|
break;
|
|
}
|
|
|
|
} else {
|
|
if (curLine == lineNumber && !("range" in result)) {
|
|
result.range = content.document.createRange();
|
|
result.range.setStart(textNode, curPos);
|
|
|
|
// This will always be overridden later, except when we look for
|
|
// the very last line in the file (this is the only line that does
|
|
// not end with \n).
|
|
result.range.setEndAfter(pre.lastChild);
|
|
|
|
} else if (curLine == lineNumber + 1) {
|
|
result.range.setEnd(textNode, curPos - 1);
|
|
found = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return found || ("range" in result);
|
|
},
|
|
|
|
/**
|
|
* Toggles the "wrap" class on the document body, which sets whether
|
|
* or not long lines are wrapped. Notifies parent to update the pref.
|
|
*/
|
|
toggleWrapping() {
|
|
let body = content.document.body;
|
|
let state = body.classList.toggle("wrap");
|
|
sendAsyncMessage("ViewSource:StoreWrapping", { state });
|
|
},
|
|
|
|
/**
|
|
* Toggles the "highlight" class on the document body, which sets whether
|
|
* or not syntax highlighting is displayed. Notifies parent to update the
|
|
* pref.
|
|
*/
|
|
toggleSyntaxHighlighting() {
|
|
let body = content.document.body;
|
|
let state = body.classList.toggle("highlight");
|
|
sendAsyncMessage("ViewSource:StoreSyntaxHighlighting", { state });
|
|
},
|
|
|
|
/**
|
|
* Called when the parent has changed the character set to view the
|
|
* source with.
|
|
*
|
|
* @param charset
|
|
* The character set to use.
|
|
* @param doPageLoad
|
|
* Whether or not we should reload the page ourselves with the
|
|
* nsIWebPageDescriptor. Part of a workaround for bug 136322.
|
|
*/
|
|
setCharacterSet(charset, doPageLoad) {
|
|
docShell.charset = charset;
|
|
if (doPageLoad) {
|
|
this.reload();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Reloads the content.
|
|
*/
|
|
reload() {
|
|
let pageLoader = docShell.QueryInterface(Ci.nsIWebPageDescriptor);
|
|
try {
|
|
pageLoader.loadPage(pageLoader.currentDescriptor,
|
|
Ci.nsIWebPageDescriptor.DISPLAY_NORMAL);
|
|
} catch(e) {
|
|
let webNav = docShell.QueryInterface(Ci.nsIWebNavigation);
|
|
webNav.reload(Ci.nsIWebNavigation.LOAD_FLAGS_NONE);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* A reference to a DeferredTask that is armed every time the
|
|
* selection changes.
|
|
*/
|
|
updateStatusTask: null,
|
|
|
|
/**
|
|
* Called once the DeferredTask fires. Sends a message up to the
|
|
* parent to update the status bar text.
|
|
*/
|
|
updateStatus() {
|
|
let selection = content.getSelection();
|
|
|
|
if (!selection.focusNode) {
|
|
sendAsyncMessage("ViewSource:UpdateStatus", { label: "" });
|
|
return;
|
|
}
|
|
if (selection.focusNode.nodeType != Ci.nsIDOMNode.TEXT_NODE) {
|
|
return;
|
|
}
|
|
|
|
let selCon = this.selectionController;
|
|
selCon.setDisplaySelection(Ci.nsISelectionController.SELECTION_ON);
|
|
selCon.setCaretVisibilityDuringSelection(true);
|
|
|
|
let interlinePosition = selection.QueryInterface(Ci.nsISelectionPrivate)
|
|
.interlinePosition;
|
|
|
|
let result = {};
|
|
this.findLocation(null, -1,
|
|
selection.focusNode, selection.focusOffset, interlinePosition, result);
|
|
|
|
let label = this.bundle.formatStringFromName("statusBarLineCol",
|
|
[result.line, result.col], 2);
|
|
sendAsyncMessage("ViewSource:UpdateStatus", { label });
|
|
},
|
|
|
|
/**
|
|
* Loads a view source selection showing the given view-source url and
|
|
* highlight the selection.
|
|
*
|
|
* @param uri view-source uri to show
|
|
* @param drawSelection true to highlight the selection
|
|
* @param baseURI base URI of the original document
|
|
*/
|
|
viewSourceWithSelection(uri, drawSelection, baseURI)
|
|
{
|
|
this.needsDrawSelection = drawSelection;
|
|
|
|
// all our content is held by the data:URI and URIs are internally stored as utf-8 (see nsIURI.idl)
|
|
let loadFlags = Ci.nsIWebNavigation.LOAD_FLAGS_NONE;
|
|
let referrerPolicy = Ci.nsIHttpChannel.REFERRER_POLICY_DEFAULT;
|
|
let webNav = docShell.QueryInterface(Ci.nsIWebNavigation);
|
|
webNav.loadURIWithOptions(uri, loadFlags,
|
|
null, referrerPolicy, // referrer
|
|
null, null, // postData, headers
|
|
Services.io.newURI(baseURI, null, null));
|
|
},
|
|
|
|
/**
|
|
* nsISelectionListener
|
|
*/
|
|
|
|
/**
|
|
* Gets called every time the selection is changed. Coalesces frequent
|
|
* changes, and calls updateStatus after 100ms of no selection change
|
|
* activity.
|
|
*/
|
|
notifySelectionChanged(doc, sel, reason) {
|
|
if (!this.updateStatusTask) {
|
|
this.updateStatusTask = new DeferredTask(() => {
|
|
this.updateStatus();
|
|
}, 100);
|
|
}
|
|
|
|
this.updateStatusTask.arm();
|
|
},
|
|
|
|
/**
|
|
* Using special markers left in the serialized source, this helper makes the
|
|
* underlying markup of the selected fragment to automatically appear as
|
|
* selected on the inflated view-source DOM.
|
|
*/
|
|
drawSelection() {
|
|
content.document.title =
|
|
this.bundle.GetStringFromName("viewSelectionSourceTitle");
|
|
|
|
// find the special selection markers that we added earlier, and
|
|
// draw the selection between the two...
|
|
var findService = null;
|
|
try {
|
|
// get the find service which stores the global find state
|
|
findService = Cc["@mozilla.org/find/find_service;1"]
|
|
.getService(Ci.nsIFindService);
|
|
} catch(e) { }
|
|
if (!findService)
|
|
return;
|
|
|
|
// cache the current global find state
|
|
var matchCase = findService.matchCase;
|
|
var entireWord = findService.entireWord;
|
|
var wrapFind = findService.wrapFind;
|
|
var findBackwards = findService.findBackwards;
|
|
var searchString = findService.searchString;
|
|
var replaceString = findService.replaceString;
|
|
|
|
// setup our find instance
|
|
var findInst = this.webBrowserFind;
|
|
findInst.matchCase = true;
|
|
findInst.entireWord = false;
|
|
findInst.wrapFind = true;
|
|
findInst.findBackwards = false;
|
|
|
|
// ...lookup the start mark
|
|
findInst.searchString = MARK_SELECTION_START;
|
|
var startLength = MARK_SELECTION_START.length;
|
|
findInst.findNext();
|
|
|
|
var selection = content.getSelection();
|
|
if (!selection.rangeCount)
|
|
return;
|
|
|
|
var range = selection.getRangeAt(0);
|
|
|
|
var startContainer = range.startContainer;
|
|
var startOffset = range.startOffset;
|
|
|
|
// ...lookup the end mark
|
|
findInst.searchString = MARK_SELECTION_END;
|
|
var endLength = MARK_SELECTION_END.length;
|
|
findInst.findNext();
|
|
|
|
var endContainer = selection.anchorNode;
|
|
var endOffset = selection.anchorOffset;
|
|
|
|
// reset the selection that find has left
|
|
selection.removeAllRanges();
|
|
|
|
// delete the special markers now...
|
|
endContainer.deleteData(endOffset, endLength);
|
|
startContainer.deleteData(startOffset, startLength);
|
|
if (startContainer == endContainer)
|
|
endOffset -= startLength; // has shrunk if on same text node...
|
|
range.setEnd(endContainer, endOffset);
|
|
|
|
// show the selection and scroll it into view
|
|
selection.addRange(range);
|
|
// the default behavior of the selection is to scroll at the end of
|
|
// the selection, whereas in this situation, it is more user-friendly
|
|
// to scroll at the beginning. So we override the default behavior here
|
|
try {
|
|
this.selectionController.scrollSelectionIntoView(
|
|
Ci.nsISelectionController.SELECTION_NORMAL,
|
|
Ci.nsISelectionController.SELECTION_ANCHOR_REGION,
|
|
true);
|
|
}
|
|
catch(e) { }
|
|
|
|
// restore the current find state
|
|
findService.matchCase = matchCase;
|
|
findService.entireWord = entireWord;
|
|
findService.wrapFind = wrapFind;
|
|
findService.findBackwards = findBackwards;
|
|
findService.searchString = searchString;
|
|
findService.replaceString = replaceString;
|
|
|
|
findInst.matchCase = matchCase;
|
|
findInst.entireWord = entireWord;
|
|
findInst.wrapFind = wrapFind;
|
|
findInst.findBackwards = findBackwards;
|
|
findInst.searchString = searchString;
|
|
},
|
|
|
|
/**
|
|
* In-page context menu items that are injected after page load.
|
|
*/
|
|
contextMenuItems: [
|
|
{
|
|
id: "goToLine",
|
|
accesskey: true,
|
|
handler() {
|
|
sendAsyncMessage("ViewSource:PromptAndGoToLine");
|
|
}
|
|
},
|
|
{
|
|
id: "wrapLongLines",
|
|
get checked() {
|
|
return Services.prefs.getBoolPref("view_source.wrap_long_lines");
|
|
},
|
|
handler() {
|
|
this.toggleWrapping();
|
|
}
|
|
},
|
|
{
|
|
id: "highlightSyntax",
|
|
get checked() {
|
|
return Services.prefs.getBoolPref("view_source.syntax_highlight");
|
|
},
|
|
handler() {
|
|
this.toggleSyntaxHighlighting();
|
|
}
|
|
},
|
|
],
|
|
|
|
/**
|
|
* Add context menu items for view source specific actions.
|
|
*/
|
|
injectContextMenu() {
|
|
let doc = content.document;
|
|
|
|
let menu = doc.createElementNS(NS_XHTML, "menu");
|
|
menu.setAttribute("type", "context");
|
|
menu.setAttribute("id", "actions");
|
|
doc.body.appendChild(menu);
|
|
doc.body.setAttribute("contextmenu", "actions");
|
|
|
|
this.contextMenuItems.forEach(itemSpec => {
|
|
let item = doc.createElementNS(NS_XHTML, "menuitem");
|
|
item.setAttribute("id", itemSpec.id);
|
|
let labelName = `context_${itemSpec.id}_label`;
|
|
let label = this.bundle.GetStringFromName(labelName);
|
|
item.setAttribute("label", label);
|
|
if ("checked" in itemSpec) {
|
|
item.setAttribute("type", "checkbox");
|
|
}
|
|
if (itemSpec.accesskey) {
|
|
let accesskeyName = `context_${itemSpec.id}_accesskey`;
|
|
item.setAttribute("accesskey",
|
|
this.bundle.GetStringFromName(accesskeyName))
|
|
}
|
|
menu.appendChild(item);
|
|
});
|
|
|
|
this.updateContextMenu();
|
|
},
|
|
|
|
/**
|
|
* Update state of checkbox-style context menu items.
|
|
*/
|
|
updateContextMenu() {
|
|
let doc = content.document;
|
|
this.contextMenuItems.forEach(itemSpec => {
|
|
if (!("checked" in itemSpec)) {
|
|
return;
|
|
}
|
|
let item = doc.getElementById(itemSpec.id);
|
|
if (itemSpec.checked) {
|
|
item.setAttribute("checked", true);
|
|
} else {
|
|
item.removeAttribute("checked");
|
|
}
|
|
});
|
|
},
|
|
};
|
|
ViewSourceContent.init();
|