summaryrefslogtreecommitdiffstats
path: root/devtools/client/shared/view-source.js
diff options
context:
space:
mode:
Diffstat (limited to 'devtools/client/shared/view-source.js')
-rw-r--r--devtools/client/shared/view-source.js245
1 files changed, 245 insertions, 0 deletions
diff --git a/devtools/client/shared/view-source.js b/devtools/client/shared/view-source.js
new file mode 100644
index 0000000000..bbbe921fb3
--- /dev/null
+++ b/devtools/client/shared/view-source.js
@@ -0,0 +1,245 @@
+/* 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/. */
+
+"use strict";
+
+/**
+ * Tries to open a Stylesheet file in the Style Editor. If the file is not
+ * found, it is opened in source view instead.
+ * Returns a promise resolving to a boolean indicating whether or not
+ * the source was able to be displayed in the StyleEditor, as the built-in
+ * Firefox View Source is the fallback.
+ *
+ * @param {Toolbox} toolbox
+ * @param {string|Object} stylesheetFrontOrGeneratedURL
+ * @param {number} generatedLine
+ * @param {number} generatedColumn
+ *
+ * @return {Promise<boolean>}
+ */
+exports.viewSourceInStyleEditor = async function(
+ toolbox,
+ stylesheetFrontOrGeneratedURL,
+ generatedLine,
+ generatedColumn
+) {
+ const originalPanelId = toolbox.currentToolId;
+
+ try {
+ const panel = await toolbox.selectTool("styleeditor", "view-source", {
+ // This will be only used in case the styleeditor wasn't loaded yet, to make the
+ // initialization faster in case we already have a stylesheet resource. We still
+ // need the rest of this function to handle subsequent calls and sourcemapped stylesheets.
+ stylesheetToSelect: {
+ stylesheet: stylesheetFrontOrGeneratedURL,
+ line: generatedLine,
+ column: generatedColumn,
+ },
+ });
+
+ let stylesheetFront;
+ if (typeof stylesheetFrontOrGeneratedURL === "string") {
+ stylesheetFront = panel.getStylesheetFrontForGeneratedURL(
+ stylesheetFrontOrGeneratedURL
+ );
+ } else {
+ stylesheetFront = stylesheetFrontOrGeneratedURL;
+ }
+
+ const originalLocation = stylesheetFront
+ ? await getOriginalLocation(
+ toolbox,
+ stylesheetFront.resourceId,
+ generatedLine,
+ generatedColumn
+ )
+ : null;
+
+ if (originalLocation) {
+ await panel.selectOriginalSheet(
+ originalLocation.sourceId,
+ originalLocation.line,
+ originalLocation.column
+ );
+ return true;
+ }
+
+ if (stylesheetFront) {
+ await panel.selectStyleSheet(
+ stylesheetFront,
+ generatedLine,
+ generatedColumn
+ );
+ return true;
+ }
+ } catch (e) {
+ console.error("Failed to view source in style editor", e);
+ }
+
+ // If we weren't able to select the stylesheet in the style editor, display it in a
+ // view-source tab
+ exports.viewSource(
+ toolbox,
+ typeof stylesheetFrontOrGeneratedURL === "string"
+ ? stylesheetFrontOrGeneratedURL
+ : stylesheetFrontOrGeneratedURL.href ||
+ stylesheetFrontOrGeneratedURL.nodeHref,
+ generatedLine
+ );
+
+ // As we might have moved to the styleeditor, switch back to the original panel
+ await toolbox.selectTool(originalPanelId);
+
+ return false;
+};
+
+/**
+ * Tries to open a JavaScript file in the Debugger. If the file is not found,
+ * it is opened in source view instead. Either the source URL or source actor ID
+ * can be specified. If both are specified, the source actor ID is used.
+ *
+ * Returns a promise resolving to a boolean indicating whether or not
+ * the source was able to be displayed in the Debugger, as the built-in Firefox
+ * View Source is the fallback.
+ *
+ * @param {Toolbox} toolbox
+ * @param {string} sourceURL
+ * @param {number} sourceLine
+ * @param {number} sourceColumn
+ * @param {string} sourceID
+ * @param {(string|object)} [reason=unknown]
+ *
+ * @return {Promise<boolean>}
+ */
+exports.viewSourceInDebugger = async function(
+ toolbox,
+ generatedURL,
+ generatedLine,
+ generatedColumn,
+ sourceActorId,
+ reason = "unknown"
+) {
+ const location = await getViewSourceInDebuggerLocation(
+ toolbox,
+ generatedURL,
+ generatedLine,
+ generatedColumn,
+ sourceActorId
+ );
+
+ if (location) {
+ const { id, line, column } = location;
+
+ const dbg = await toolbox.selectTool("jsdebugger", reason);
+ try {
+ await dbg.selectSource(id, sourceActorId, line, column);
+ return true;
+ } catch (err) {
+ console.error("Failed to view source in debugger", err);
+ }
+ }
+
+ exports.viewSource(toolbox, generatedURL, generatedLine);
+ return false;
+};
+
+async function getViewSourceInDebuggerLocation(
+ toolbox,
+ generatedURL,
+ generatedLine,
+ generatedColumn,
+ sourceActorId
+) {
+ const dbg = await toolbox.loadTool("jsdebugger");
+
+ const generatedSource = sourceActorId
+ ? dbg.getSourceByActorId(sourceActorId)
+ : dbg.getSourceByURL(generatedURL);
+ if (
+ !generatedSource ||
+ // Note: We're not entirely sure when this can happen, so we may want
+ // to revisit that at some point.
+ dbg.getSourceActorsForSource(generatedSource.id).length === 0
+ ) {
+ return null;
+ }
+
+ const generatedLocation = {
+ id: generatedSource.id,
+ line: generatedLine,
+ column: generatedColumn,
+ };
+
+ const originalLocation = await getOriginalLocation(
+ toolbox,
+ generatedLocation.id,
+ generatedLocation.line,
+ generatedLocation.column
+ );
+
+ if (!originalLocation) {
+ return generatedLocation;
+ }
+
+ const originalSource = dbg.getLocationSource(originalLocation);
+
+ if (!originalSource) {
+ return generatedLocation;
+ }
+
+ return {
+ id: originalSource.id,
+ line: originalLocation.line,
+ column: originalLocation.column,
+ };
+}
+
+async function getOriginalLocation(
+ toolbox,
+ generatedID,
+ generatedLine,
+ generatedColumn
+) {
+ // If there is no line number, then there's no chance that we'll get back
+ // a useful original location.
+ if (typeof generatedLine !== "number") {
+ return null;
+ }
+
+ let originalLocation = null;
+ try {
+ originalLocation = await toolbox.sourceMapLoader.getOriginalLocation({
+ sourceId: generatedID,
+ line: generatedLine,
+ column: generatedColumn,
+ });
+ if (originalLocation && originalLocation.sourceId === generatedID) {
+ originalLocation = null;
+ }
+ } catch (err) {
+ console.error(
+ "Failed to resolve sourcemapped location for the given source location",
+ { generatedID, generatedLine, generatedColumn },
+ err
+ );
+ }
+ return originalLocation;
+}
+
+/**
+ * Open a link in Firefox's View Source.
+ *
+ * @param {Toolbox} toolbox
+ * @param {string} sourceURL
+ * @param {number} sourceLine
+ *
+ * @return {Promise}
+ */
+exports.viewSource = async function(toolbox, sourceURL, sourceLine) {
+ const utils = toolbox.gViewSourceUtils;
+ utils.viewSource({
+ URL: sourceURL,
+ lineNumber: sourceLine || -1,
+ });
+};