Bug 968727 - Cleans up all markupview tests for consistency and to reduce file sizes. r=bgrins, a=test-only
authorPatrick Brosset <pbrosset@mozilla.com>
Tue, 25 Mar 2014 09:55:05 +0100
changeset 192927 bbdc0136061edb72bb94f131886b65cbf01f9737
parent 192926 76622a1dc44e73f8317397ba4589ef75e2b5f72c
child 192928 329ad4441fd9d2174f9138a5ee8e796e20af3770
push id474
push userasasaki@mozilla.com
push dateMon, 02 Jun 2014 21:01:02 +0000
treeherdermozilla-release@967f4cf1b31c [default view] [failures only]
perfherder[talos] [build metrics] [platform microbench] (compared to previous push)
reviewersbgrins, test-only
bugs968727
milestone30.0a2
Bug 968727 - Cleans up all markupview tests for consistency and to reduce file sizes. r=bgrins, a=test-only
browser/devtools/markupview/test/browser.ini
browser/devtools/markupview/test/browser_bug896181_css_mixed_completion_new_attribute.js
browser/devtools/markupview/test/browser_inspector_markup_765105_tooltip.js
browser/devtools/markupview/test/browser_inspector_markup_950732.js
browser/devtools/markupview/test/browser_inspector_markup_962647_search.js
browser/devtools/markupview/test/browser_inspector_markup_964014_copy_image_data.js
browser/devtools/markupview/test/browser_inspector_markup_968316_highlight_node_after_mouseleave_mousemove.js
browser/devtools/markupview/test/browser_inspector_markup_968316_highlit_node_on_hover_then_select.js
browser/devtools/markupview/test/browser_inspector_markup_add_attributes.js
browser/devtools/markupview/test/browser_inspector_markup_edit.js
browser/devtools/markupview/test/browser_inspector_markup_edit_2.js
browser/devtools/markupview/test/browser_inspector_markup_edit_3.js
browser/devtools/markupview/test/browser_inspector_markup_edit_4.js
browser/devtools/markupview/test/browser_inspector_markup_edit_outerhtml.js
browser/devtools/markupview/test/browser_inspector_markup_edit_outerhtml2.js
browser/devtools/markupview/test/browser_inspector_markup_mutation.js
browser/devtools/markupview/test/browser_inspector_markup_mutation_flashing.js
browser/devtools/markupview/test/browser_inspector_markup_navigation.js
browser/devtools/markupview/test/browser_inspector_markup_subset.js
browser/devtools/markupview/test/head.js
browser/devtools/styleinspector/rule-view.js
--- a/browser/devtools/markupview/test/browser.ini
+++ b/browser/devtools/markupview/test/browser.ini
@@ -7,18 +7,16 @@ support-files =
   browser_inspector_markup_navigation.html
   browser_inspector_markup_subset.html
   browser_inspector_markup_765105_tooltip.png
   browser_inspector_markup_950732.html
   browser_inspector_markup_962647_search.html
   head.js
 
 [browser_bug896181_css_mixed_completion_new_attribute.js]
-# Bug 916763 - too many intermittent failures
-skip-if = true
 [browser_inspector_markup_edit.js]
 [browser_inspector_markup_edit_2.js]
 [browser_inspector_markup_edit_3.js]
 [browser_inspector_markup_edit_4.js]
 [browser_inspector_markup_add_attributes.js]
 [browser_inspector_markup_edit_outerhtml.js]
 [browser_inspector_markup_edit_outerhtml2.js]
 [browser_inspector_markup_mutation.js]
--- a/browser/devtools/markupview/test/browser_bug896181_css_mixed_completion_new_attribute.js
+++ b/browser/devtools/markupview/test/browser_bug896181_css_mixed_completion_new_attribute.js
@@ -1,176 +1,142 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
    http://creativecommons.org/publicdomain/zero/1.0/ */
+
 // Test CSS state is correctly determined and the corresponding suggestions are
 // displayed. i.e. CSS property suggestions are shown when cursor is like:
 // ```style="di|"``` where | is the cursor; And CSS value suggestion is
 // displayed when the cursor is like: ```style="display:n|"``` properly. No
 // suggestions should ever appear when the attribute is not a style attribute.
 // The correctness and cycling of the suggestions is covered in the ruleview
 // tests.
-function test() {
-  let inspector;
-  let {
-    getInplaceEditorForSpan: inplaceEditor
-  } = devtools.require("devtools/shared/inplace-editor");
-
-  waitForExplicitFinish();
-
-  // Will hold the doc we're viewing
-  let doc;
 
-  // Holds the MarkupTool object we're testing.
-  let markup;
-  let editor;
-  let state;
-  // format :
-  //  [
-  //    what key to press,
-  //    expected input box value after keypress,
-  //    expected input.selectionStart,
-  //    expected input.selectionEnd,
-  //    is popup expected to be open ?
-  //  ]
-  let testData = [
-    ['s', 's', 1, 1, false],
-    ['t', 'st', 2, 2, false],
-    ['y', 'sty', 3, 3, false],
-    ['l', 'styl', 4, 4, false],
-    ['e', 'style', 5, 5, false],
-    ['=', 'style=', 6, 6, false],
-    ['"', 'style="', 7, 7, false],
-    ['d', 'style="direction', 8, 16, true],
-    ['VK_DOWN', 'style="display', 8, 14, true],
-    ['VK_TAB', 'style="display', 14, 14, true],
-    ['VK_TAB', 'style="dominant-baseline', 24, 24, true],
-    ['VK_TAB', 'style="direction', 16, 16, true],
-    ['click_1', 'style="display', 14, 14, false],
-    [':', 'style="display:', 15, 15, false],
-    ['n', 'style="display:none', 16, 19, false],
-    ['VK_BACK_SPACE', 'style="display:n', 16, 16, false],
-    ['VK_BACK_SPACE', 'style="display:', 15, 15, false],
-    [' ', 'style="display: ', 16, 16, false],
-    [' ', 'style="display:  ', 17, 17, false],
-    ['i', 'style="display:  inherit', 18, 24, true],
-    ['VK_RIGHT', 'style="display:  inherit', 24, 24, false],
-    [';', 'style="display:  inherit;', 25, 25, false],
-    [' ', 'style="display:  inherit; ', 26, 26, false],
-    [' ', 'style="display:  inherit;  ', 27, 27, false],
-    ['VK_LEFT', 'style="display:  inherit;  ', 26, 26, false],
-    ['c', 'style="display:  inherit; caption-side ', 27, 38, true],
-    ['o', 'style="display:  inherit; color ', 28, 31, true],
-    ['VK_RIGHT', 'style="display:  inherit; color ', 31, 31, false],
-    [' ', 'style="display:  inherit; color  ', 32, 32, false],
-    ['c', 'style="display:  inherit; color c ', 33, 33, false],
-    ['VK_BACK_SPACE', 'style="display:  inherit; color  ', 32, 32, false],
-    [':', 'style="display:  inherit; color : ', 33, 33, false],
-    ['c', 'style="display:  inherit; color :cadetblue ', 34, 42, true],
-    ['VK_DOWN', 'style="display:  inherit; color :chartreuse ', 34, 43, true],
-    ['VK_RIGHT', 'style="display:  inherit; color :chartreuse ', 43, 43, false],
-    [' ', 'style="display:  inherit; color :chartreuse  ', 44, 44, false],
-    ['!', 'style="display:  inherit; color :chartreuse !important; ', 45, 55, false],
-    ['VK_RIGHT', 'style="display:  inherit; color :chartreuse !important; ', 55, 55, false],
-    ['VK_RETURN', 'style="display:  inherit; color :chartreuse !important;"', -1, -1, false]
-  ];
-  function startTests() {
-    markup = inspector.markup;
-    markup.expandAll().then(() => {
-      let node = getContainerForRawNode(markup, doc.querySelector("#node14")).editor;
-      let attr = node.newAttr;
-      attr.focus();
-      EventUtils.sendKey("return", inspector.panelWin);
-      editor = inplaceEditor(attr);
-      checkStateAndMoveOn(0);
-    });
+const TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_edit.html";
+// test data format :
+//  [
+//    what key to press,
+//    expected input box value after keypress,
+//    expected input.selectionStart,
+//    expected input.selectionEnd,
+//    is popup expected to be open ?
+//  ]
+const TEST_DATA = [
+  ['s', 's', 1, 1, false],
+  ['t', 'st', 2, 2, false],
+  ['y', 'sty', 3, 3, false],
+  ['l', 'styl', 4, 4, false],
+  ['e', 'style', 5, 5, false],
+  ['=', 'style=', 6, 6, false],
+  ['"', 'style="', 7, 7, false],
+  ['d', 'style="direction', 8, 16, true],
+  ['VK_DOWN', 'style="display', 8, 14, true],
+  ['VK_TAB', 'style="display', 14, 14, true],
+  ['VK_TAB', 'style="dominant-baseline', 24, 24, true],
+  ['VK_TAB', 'style="direction', 16, 16, true],
+  ['click_1', 'style="display', 14, 14, false],
+  [':', 'style="display:', 15, 15, false],
+  ['n', 'style="display:none', 16, 19, false],
+  ['VK_BACK_SPACE', 'style="display:n', 16, 16, false],
+  ['VK_BACK_SPACE', 'style="display:', 15, 15, false],
+  [' ', 'style="display: ', 16, 16, false],
+  [' ', 'style="display:  ', 17, 17, false],
+  ['i', 'style="display:  inherit', 18, 24, true],
+  ['VK_RIGHT', 'style="display:  inherit', 24, 24, false],
+  [';', 'style="display:  inherit;', 25, 25, false],
+  [' ', 'style="display:  inherit; ', 26, 26, false],
+  [' ', 'style="display:  inherit;  ', 27, 27, false],
+  ['VK_LEFT', 'style="display:  inherit;  ', 26, 26, false],
+  ['c', 'style="display:  inherit; caption-side ', 27, 38, true],
+  ['o', 'style="display:  inherit; color ', 28, 31, true],
+  ['VK_RIGHT', 'style="display:  inherit; color ', 31, 31, false],
+  [' ', 'style="display:  inherit; color  ', 32, 32, false],
+  ['c', 'style="display:  inherit; color c ', 33, 33, false],
+  ['VK_BACK_SPACE', 'style="display:  inherit; color  ', 32, 32, false],
+  [':', 'style="display:  inherit; color : ', 33, 33, false],
+  ['c', 'style="display:  inherit; color :cadetblue ', 34, 42, true],
+  ['VK_DOWN', 'style="display:  inherit; color :chartreuse ', 34, 43, true],
+  ['VK_RIGHT', 'style="display:  inherit; color :chartreuse ', 43, 43, false],
+  [' ', 'style="display:  inherit; color :chartreuse  ', 44, 44, false],
+  ['!', 'style="display:  inherit; color :chartreuse !important; ', 45, 55, false],
+  ['VK_RIGHT', 'style="display:  inherit; color :chartreuse !important; ', 55, 55, false],
+  ['VK_RETURN', 'style="display:  inherit; color :chartreuse !important;"', -1, -1, false]
+];
+
+let test = asyncTest(function*() {
+  info("Opening the inspector on the test URL");
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
+
+  yield inspector.markup.expandAll();
+
+  let node = getContainerForRawNode("#node14", inspector).editor;
+  let attr = node.newAttr;
+  attr.focus();
+  EventUtils.sendKey("return", inspector.panelWin);
+  let editor = inplaceEditor(attr);
+
+  for (let i = 0; i < TEST_DATA.length; i ++) {
+    yield enterData(i, editor, inspector);
+    checkData(i, editor, inspector);
   }
 
-  function checkStateAndMoveOn(index) {
-    if (index == testData.length) {
-      finishUp();
-      return;
-    }
+  while (inspector.markup.undo.canUndo()) {
+    yield undoChange(inspector);
+  }
 
-    let [key] = testData[index];
-    state = index;
+  yield inspector.once("inspector-updated");
+});
+
+function enterData(index, editor, inspector) {
+  let [key] = TEST_DATA[index];
+  info("Entering test data " + index + ": " + key + ", expecting: [" + TEST_DATA[index].slice(1) + "]");
 
-    info("pressing key " + key + " to get result: [" + testData[index].slice(1) +
-         "] for state " + state);
-    if (/(down|left|right|back_space|return)/ig.test(key)) {
-      info("added event listener for down|left|right|back_space|return keys");
-      editor.input.addEventListener("keypress", function onKeypress() {
-        if (editor.input) {
-          editor.input.removeEventListener("keypress", onKeypress);
-        }
-        info("inside event listener");
-        checkState();
-      })
-    }
-    else if (/click_[0-9]/.test(key)) {
-      editor.once("after-suggest", checkState);
-      let index = +key.split("_")[1];
-      editor.popup._list.childNodes[index].click();
-      editor.input.blur();
-      return;
-    }
-    else {
-      editor.once("after-suggest", checkState);
-    }
-    EventUtils.synthesizeKey(key, {}, inspector.panelWin);
+  let def = promise.defer();
+
+  if (/click_[0-9]/.test(key)) {
+    let nb = +key.split("_")[1];
+    info("Clicking on item " + nb + " in the list");
+    editor.once("after-suggest", () => {
+      executeSoon(def.resolve);
+    });
+    editor.popup._list.childNodes[nb].click();
+    editor.input.blur();
+    return def.promise;
   }
 
-  function checkState() {
-    executeSoon(() => {
-      info("After keypress for state " + state);
-      let [key, completion, selStart, selEnd, popupOpen] = testData[state];
-      if (selEnd != -1) {
-        is(editor.input.value, completion,
-           "Correct value is autocompleted for state " + state);
-        is(editor.input.selectionStart, selStart,
-           "Selection is starting at the right location for state " + state);
-        is(editor.input.selectionEnd, selEnd,
-           "Selection is ending at the right location for state " + state);
-        if (popupOpen) {
-          ok(editor.popup.isOpen, "Popup is open for state " + state);
-        }
-        else {
-          ok(editor.popup._panel.state != "open" &&
-             editor.popup._panel.state != "showing",
-             "Popup is closed for state " + state);
-        }
+  if (/(down|left|right|back_space|return)/ig.test(key)) {
+    info("Adding event listener for down|left|right|back_space|return keys");
+    editor.input.addEventListener("keypress", function onKeypress() {
+      if (editor.input) {
+        editor.input.removeEventListener("keypress", onKeypress);
       }
-      else {
-        let editor = getContainerForRawNode(markup, doc.querySelector("#node14")).editor;
-        let attr = editor.attrs["style"].querySelector(".editable");
-        is(attr.textContent, completion,
-           "Correct value is persisted after pressing Enter for state " + state);
-      }
-      checkStateAndMoveOn(state + 1);
+      executeSoon(def.resolve);
+    });
+  } else {
+    editor.once("after-suggest", () => {
+      executeSoon(def.resolve);
     });
   }
+  EventUtils.synthesizeKey(key, {}, inspector.panelWin);
 
-  // Create the helper tab for parsing...
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(setupTest, content);
-  }, true);
-  content.location = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_edit.html";
+  return def.promise;
+}
+
+function checkData(index, editor, inspector) {
+  let [key, completion, selStart, selEnd, popupOpen] = TEST_DATA[index];
+  info("Test data " + index + " entered. Checking state.");
 
-  function setupTest() {
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      startTests();
-    });
-  }
-
-  function finishUp() {
-    while (markup.undo.canUndo()) {
-      markup.undo.undo();
+  if (selEnd != -1) {
+    is(editor.input.value, completion, "Completed value is correct");
+    is(editor.input.selectionStart, selStart, "Selection start position is correct");
+    is(editor.input.selectionEnd, selEnd, "Selection end position is correct");
+    if (popupOpen) {
+      ok(editor.popup.isOpen, "Popup is open");
+    } else {
+      ok(editor.popup._panel.state != "open" && editor.popup._panel.state != "showing",
+        "Popup is closed");
     }
-    doc = inspector = null;
-    gBrowser.removeCurrentTab();
-    finish();
+  } else {
+    let editor = getContainerForRawNode("#node14", inspector).editor;
+    let attr = editor.attrs["style"].querySelector(".editable");
+    is(attr.textContent, completion, "Correct value is persisted after pressing Enter");
   }
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_765105_tooltip.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_765105_tooltip.js
@@ -1,125 +1,85 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
-let {PanelFactory} = devtools.require("devtools/shared/widgets/Tooltip");
-
-let contentDoc;
-let inspector;
-let markup;
+// Test that image preview tooltips are shown on img and canvas tags in the
+// markup-view and that the tooltip actually contains an image and shows the
+// right dimension label
 
 const PAGE_CONTENT = [
   '<img class="local" src="chrome://branding/content/about-logo.png" />',
   '<img class="data" src="" />',
-  '<img class="remote" src="http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_765105_tooltip.png" />',
+  '<img class="remote" src="' + TEST_URL_ROOT + 'browser_inspector_markup_765105_tooltip.png" />',
   '<canvas class="canvas" width="600" height="600"></canvas>'
 ].join("\n");
 
 const TEST_NODES = [
   {selector: "img.local", size: "192 x 192"},
   {selector: "img.data", size: "64 x 64"},
   {selector: "img.remote", size: "22 x 23"},
   {selector: ".canvas", size: "600 x 600"}
 ];
 
-function test() {
-  waitForExplicitFinish();
+let test = asyncTest(function*() {
+  yield addTab("data:text/html,markup view tooltip test");
+  createPage();
 
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function(evt) {
-    gBrowser.selectedBrowser.removeEventListener(evt.type, arguments.callee, true);
-    contentDoc = content.document;
-    waitForFocus(createDocument, content);
-  }, true);
+  let {inspector} = yield openInspector();
 
-  content.location = "data:text/html,markup view tooltip test";
-}
+  info("Selecting the first <img> tag");
+  yield selectNode("img", inspector);
 
-function createDocument() {
-  contentDoc.body.innerHTML = PAGE_CONTENT;
+  for (let testNode of TEST_NODES) {
+    let target = getImageTooltipTarget(testNode, inspector);
+    yield assertTooltipShownOn(target, inspector);
+    checkImageTooltip(testNode, inspector);
+  }
+});
 
-  var target = TargetFactory.forTab(gBrowser.selectedTab);
-  gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-    inspector = toolbox.getCurrentPanel();
-    markup = inspector.markup;
-    startTests();
-  });
-}
+function createPage() {
+  info("Fill the page with the test content");
+  content.document.body.innerHTML = PAGE_CONTENT;
 
-function startTests() {
-  // Draw something in the canvas :)
+  info("Fill the canvas");
   let doc = content.document;
   let context = doc.querySelector(".canvas").getContext("2d");
 
   context.beginPath();
   context.moveTo(300, 0);
   context.lineTo(600, 600);
   context.lineTo(0, 600);
   context.closePath();
   context.fillStyle = "#ffc821";
   context.fill();
-
-  // Actually start testing
-  inspector.selection.setNode(contentDoc.querySelector("img"));
-  inspector.once("inspector-updated", () => {
-    testImageTooltip(0);
-  });
 }
 
-function endTests() {
-  contentDoc = inspector = markup = null;
-  gBrowser.removeCurrentTab();
-  finish();
-}
-
-function testImageTooltip(index) {
-  if (index === TEST_NODES.length) {
-    return endTests();
-  }
-
-  let node = contentDoc.querySelector(TEST_NODES[index].selector);
-  ok(node, "We have the [" + TEST_NODES[index].selector + "] image node to test for tooltip");
+function getImageTooltipTarget({selector}, inspector) {
+  let node = getNode(selector);
   let isImg = node.tagName.toLowerCase() === "img";
 
-  let container = getContainerForRawNode(markup, node);
-
-  let target = container.editor.tag;
-  if (isImg) {
-    target = container.editor.getAttributeElement("src");
-  }
+  let container = getContainerForRawNode(node, inspector);
 
-  assertTooltipShownOn(target).then(() => {
-    let images = markup.tooltip.panel.getElementsByTagName("image");
-    is(images.length, 1,
-      "Tooltip for [" + TEST_NODES[index].selector + "] contains an image");
-
-    let label = markup.tooltip.panel.querySelector(".devtools-tooltip-caption");
-    is(label.textContent, TEST_NODES[index].size,
-      "Tooltip label for [" + TEST_NODES[index].selector + "] displays the right image size")
-
-    testImageTooltip(index + 1);
-  });
+   let target = container.editor.tag;
+   if (isImg) {
+     target = container.editor.getAttributeElement("src");
+   }
+  return target;
 }
 
-function compareImageData(img, imgData) {
-  let canvas = content.document.createElement("canvas");
-  canvas.width = img.naturalWidth;
-  canvas.height = img.naturalHeight;
-  let ctx = canvas.getContext("2d");
-  let data = "";
-  try {
-    ctx.drawImage(img, 0, 0);
-    data = canvas.toDataURL("image/png");
-  } catch (e) {}
-
-  is(data, imgData, "Tooltip image has the right content");
-}
-
-function assertTooltipShownOn(element, cb) {
+function assertTooltipShownOn(element, {markup}) {
   return Task.spawn(function*() {
     info("Is the element a valid hover target");
-
     let isValid = yield markup.tooltip.isValidHoverTarget(element);
     ok(isValid, "The element is a valid hover target for the image tooltip");
   });
 }
+
+function checkImageTooltip({selector, size}, {markup}) {
+  let images = markup.tooltip.panel.getElementsByTagName("image");
+  is(images.length, 1, "Tooltip for [" + selector + "] contains an image");
+
+  let label = markup.tooltip.panel.querySelector(".devtools-tooltip-caption");
+  is(label.textContent, size, "Tooltip label for [" + selector + "] displays the right image size");
+
+  markup.tooltip.hide();
+}
--- a/browser/devtools/markupview/test/browser_inspector_markup_950732.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_950732.js
@@ -1,105 +1,44 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
-/**
- * Tests that the markup view loads only as many nodes as specified
- * by the devtools.markup.pagesize preference.
- */
+// Tests that the markup view loads only as many nodes as specified
+// by the devtools.markup.pagesize preference and that pressing the "show all nodes"
+// actually shows the nodes
 
-let {devtools} = Cu.import("resource://gre/modules/devtools/Loader.jsm", {});
-let promise = devtools.require("sdk/core/promise");
-let {Task} = Cu.import("resource://gre/modules/Task.jsm", {});
+const TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_950732.html";
 
 // Make sure nodes are hidden when there are more than 5 in a row
-registerCleanupFunction(function() {
-  Services.prefs.clearUserPref("devtools.markup.pagesize");
-});
 Services.prefs.setIntPref("devtools.markup.pagesize", 5);
 
-function test() {
-  waitForExplicitFinish();
-
-  let doc;
-  let inspector;
-  let markup;
-
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(runTests, content);
-  }, true);
-  content.location = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_950732.html";
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
 
-  function runTests() {
-    Task.spawn(function() {
-      yield openMarkupView();
-      yield selectUL();
-      yield reloadPage();
-      yield showAllNodes();
-
-      assertAllNodesAreVisible();
-      finishUp();
-    }).then(null, Cu.reportError);
-  }
+  info("Selecting the UL node");
+  yield clickContainer("ul", inspector);
+  info("Reloading the page with the UL node selected will expand its children");
+  yield reloadPage(inspector);
+  yield inspector.markup._waitForChildren();
 
-  function openMarkupView() {
-    let deferred = promise.defer();
+  info("Click on the 'show all nodes' button in the UL's list of children");
+  yield showAllNodes(inspector);
 
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    let toolbox = gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      markup = inspector.markup;
-      inspector.once("inspector-updated", deferred.resolve);
-    });
-
-    return deferred.promise;
-  }
-
-  function selectUL() {
-    let deferred = promise.defer();
+  assertAllNodesAreVisible(inspector);
+});
 
-    let container = getContainerForRawNode(markup, doc.querySelector("ul"));
-    let win = container.elt.ownerDocument.defaultView;
-
-    EventUtils.sendMouseEvent({type: "mousedown"}, container.elt, win);
-    inspector.once("inspector-updated", deferred.resolve);
-
-    return deferred.promise;
-  }
-
-  function reloadPage() {
-    let deferred = promise.defer();
-
-    inspector.once("new-root", () => {
-      doc = content.document;
-      markup = inspector.markup;
-      markup._waitForChildren().then(deferred.resolve);
-    });
-    content.location.reload();
-
-    return deferred.promise;
-  }
+function showAllNodes(inspector) {
+  let container = getContainerForRawNode("ul", inspector);
+  let button = container.elt.querySelector("button");
+  ok(button, "All nodes button is here");
+  let win = button.ownerDocument.defaultView;
 
-  function showAllNodes() {
-    let container = getContainerForRawNode(markup, doc.querySelector("ul"));
-    let button = container.elt.querySelector("button");
-    let win = button.ownerDocument.defaultView;
-
-    EventUtils.sendMouseEvent({type: "click"}, button, win);
-    return markup._waitForChildren();
-  }
+  EventUtils.sendMouseEvent({type: "click"}, button, win);
+  return inspector.markup._waitForChildren();
+}
 
-  function assertAllNodesAreVisible() {
-    let ul = doc.querySelector("ul");
-    let container = getContainerForRawNode(markup, ul);
-    ok(!container.elt.querySelector("button"), "All nodes button isn't here");
-    is(container.children.childNodes.length, ul.children.length);
-  }
-
-  function finishUp() {
-    doc = inspector = markup = null;
-    gBrowser.removeCurrentTab();
-    finish();
-  }
+function assertAllNodesAreVisible(inspector) {
+  let ul = getNode("ul");
+  let container = getContainerForRawNode(ul, inspector);
+  ok(!container.elt.querySelector("button"), "All nodes button isn't here anymore");
+  is(container.children.childNodes.length, ul.children.length);
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_962647_search.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_962647_search.js
@@ -1,50 +1,39 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
 // Test that searching for nodes using the selector-search input expands and
 // selects the right nodes in the markup-view, even when those nodes are deeply
 // nested (and therefore not attached yet when the markup-view is initialized).
 
-const TEST_URL = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_962647_search.html";
-
-function test() {
-  waitForExplicitFinish();
+const TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_962647_search.html";
 
-  let p = content.document.querySelector("p");
-  Task.spawn(function() {
-    info("loading the test page");
-    yield addTab(TEST_URL);
+let test = asyncTest(function*() {
+  let {inspector, toolbox} = yield addTab(TEST_URL).then(openInspector);
 
-    info("opening the inspector");
-    let {inspector, toolbox} = yield openInspector();
-
-    ok(!getContainerForRawNode(inspector.markup, getNode("em")),
-      "The <em> tag isn't present yet in the markup-view");
+  ok(!getContainerForRawNode("em", inspector),
+    "The <em> tag isn't present yet in the markup-view");
 
-    // Searching for the innermost element first makes sure that the inspector
-    // back-end is able to attach the resulting node to the tree it knows at the
-    // moment. When the inspector is started, the <body> is the default selected
-    // node, and only the parents up to the ROOT are known, and its direct children
-    info("searching for the innermost child: <em>");
-    let updated = inspector.once("inspector-updated");
-    searchUsingSelectorSearch("em", inspector);
-    yield updated;
+  // Searching for the innermost element first makes sure that the inspector
+  // back-end is able to attach the resulting node to the tree it knows at the
+  // moment. When the inspector is started, the <body> is the default selected
+  // node, and only the parents up to the ROOT are known, and its direct children
+  info("searching for the innermost child: <em>");
+  let updated = inspector.once("inspector-updated");
+  searchUsingSelectorSearch("em", inspector);
+  yield updated;
 
-    ok(getContainerForRawNode(inspector.markup, getNode("em")),
-      "The <em> tag is now imported in the markup-view");
-    is(inspector.selection.node, getNode("em"),
-      "The <em> tag is the currently selected node");
+  ok(getContainerForRawNode("em", inspector),
+    "The <em> tag is now imported in the markup-view");
+  is(inspector.selection.node, getNode("em"),
+    "The <em> tag is the currently selected node");
 
-    info("searching for other nodes too");
-    for (let node of ["span", "li", "ul"]) {
-      let updated = inspector.once("inspector-updated");
-      searchUsingSelectorSearch(node, inspector);
-      yield updated;
-      is(inspector.selection.node, getNode(node),
-        "The <" + node + "> tag is the currently selected node");
-    }
-
-    gBrowser.removeCurrentTab();
-  }).then(null, ok.bind(null, false)).then(finish);
-}
+  info("searching for other nodes too");
+  for (let node of ["span", "li", "ul"]) {
+    let updated = inspector.once("inspector-updated");
+    searchUsingSelectorSearch(node, inspector);
+    yield updated;
+    is(inspector.selection.node, getNode(node),
+      "The <" + node + "> tag is the currently selected node");
+  }
+});
--- a/browser/devtools/markupview/test/browser_inspector_markup_964014_copy_image_data.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_964014_copy_image_data.js
@@ -1,131 +1,109 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
 // Test that image nodes have the "copy data-uri" contextual menu item enabled
 // and that clicking it puts the image data into the clipboard
 
-let doc;
-let inspector;
-let markup;
-
 const PAGE_CONTENT = [
   '<div></div>',
   '<img class="data" src="" />',
   '<canvas class="canvas" width="600" height="600"></canvas>'
 ].join("\n");
 
-function test() {
-  waitForExplicitFinish();
+let test = asyncTest(function*() {
+  yield addTab("data:text/html,markup view copy image as data-uri");
+  createDocument();
+  let doc = content.document;
+
+  let {inspector} = yield openInspector();
+
+  yield selectNode("div", inspector);
+  yield assertCopyImageDataNotAvailable(inspector);
 
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload(evt) {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(createDocument, content);
-  }, true);
+  yield selectNode("img", inspector);
+  yield assertCopyImageDataAvailable(inspector);
+  yield triggerCopyImageUrlAndWaitForClipboard(doc.querySelector("img").src, inspector);
 
-  content.location = "data:text/html,markup view copy image as data-uri";
-}
+  yield selectNode("canvas", inspector);
+  yield assertCopyImageDataAvailable(inspector);
+  let canvas = doc.querySelector(".canvas");
+  yield triggerCopyImageUrlAndWaitForClipboard(canvas.toDataURL(), inspector);
+
+  // Check again that the menu isn't available on the DIV (to make sure our
+  // menu updating mechanism works)
+  yield selectNode("div", inspector);
+  yield assertCopyImageDataNotAvailable(inspector);
+});
 
 function createDocument() {
+  let doc = content.document;
+
   doc.body.innerHTML = PAGE_CONTENT;
   let context = doc.querySelector(".canvas").getContext("2d");
   context.beginPath();
   context.moveTo(300, 0);
   context.lineTo(600, 600);
   context.lineTo(0, 600);
   context.closePath();
   context.fillStyle = "#ffc821";
   context.fill();
-
-  openInspector().then(startTests);
 }
 
-function startTests({inspector: aInspector, toolbox: aToolbox}) {
-  inspector = aInspector;
-  markup = inspector.markup;
-
-  Task.spawn(function() {
-    yield selectNode("div", inspector);
-    yield assertCopyImageDataNotAvailable();
-
-    yield selectNode("img", inspector);
-    yield assertCopyImageDataAvailable();
-    yield triggerCopyImageUrlAndWaitForClipboard(doc.querySelector("img").src);
-
-    yield selectNode("canvas", inspector);
-    yield assertCopyImageDataAvailable();
-    let canvas = doc.querySelector(".canvas");
-    yield triggerCopyImageUrlAndWaitForClipboard(canvas.toDataURL());
-
-    // Check again that the menu isn't available on the DIV (to make sure our
-    // menu updating mechanism works)
-    yield selectNode("div", inspector);
-    yield assertCopyImageDataNotAvailable();
-  }).then(null, ok.bind(null, false)).then(endTests);
-}
-
-function endTests() {
-  doc = inspector = markup = null;
-  gBrowser.removeCurrentTab();
-  finish();
-}
-
-function assertCopyImageDataNotAvailable() {
-  return openNodeMenu().then(menu => {
+function assertCopyImageDataNotAvailable(inspector) {
+  return openNodeMenu(inspector).then(menu => {
     let item = menu.getElementsByAttribute("id", "node-menu-copyimagedatauri")[0];
     ok(item, "The menu item was found in the contextual menu");
     is(item.getAttribute("disabled"), "true", "The menu item is disabled");
-  }).then(closeNodeMenu);
+  }).then(() => closeNodeMenu(inspector));
 }
 
-function assertCopyImageDataAvailable() {
-  return openNodeMenu().then(menu => {
+function assertCopyImageDataAvailable(inspector) {
+  return openNodeMenu(inspector).then(menu => {
     let item = menu.getElementsByAttribute("id", "node-menu-copyimagedatauri")[0];
     ok(item, "The menu item was found in the contextual menu");
     is(item.getAttribute("disabled"), "", "The menu item is enabled");
-  }).then(closeNodeMenu);
+  }).then(() => closeNodeMenu(inspector));
 }
 
-function openNodeMenu() {
-  let deferred = promise.defer();
+function openNodeMenu(inspector) {
+  let def = promise.defer();
 
   inspector.nodemenu.addEventListener("popupshown", function onOpen() {
     inspector.nodemenu.removeEventListener("popupshown", onOpen, false);
-    deferred.resolve(inspector.nodemenu);
+    def.resolve(inspector.nodemenu);
   }, false);
   inspector.nodemenu.hidden = false;
   inspector.nodemenu.openPopup();
 
-  return deferred.promise;
+  return def.promise;
 }
 
-function closeNodeMenu() {
-  let deferred = promise.defer();
+function closeNodeMenu(inspector) {
+  let def = promise.defer();
 
   inspector.nodemenu.addEventListener("popuphidden", function onClose() {
     inspector.nodemenu.removeEventListener("popuphidden", onClose, false);
-    deferred.resolve(inspector.nodemenu);
+    def.resolve(inspector.nodemenu);
   }, false);
   inspector.nodemenu.hidden = true;
   inspector.nodemenu.hidePopup();
 
-  return deferred.promise;
+  return def.promise;
 }
 
-function triggerCopyImageUrlAndWaitForClipboard(expected) {
-  let deferred = promise.defer();
+function triggerCopyImageUrlAndWaitForClipboard(expected, inspector) {
+  let def = promise.defer();
 
   SimpleTest.waitForClipboard(expected, () => {
-    markup.getContainer(inspector.selection.nodeFront).copyImageDataUri();
+    inspector.markup.getContainer(inspector.selection.nodeFront).copyImageDataUri();
   }, () => {
     ok(true, "The clipboard contains the expected value " + expected.substring(0, 50) + "...");
-    deferred.resolve();
+    def.resolve();
   }, () => {
     ok(false, "The clipboard doesn't contain the expected value " + expected.substring(0, 50) + "...");
-    deferred.resolve();
+    def.resolve();
   });
 
-  return deferred.promise;
+  return def.promise;
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_968316_highlight_node_after_mouseleave_mousemove.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_968316_highlight_node_after_mouseleave_mousemove.js
@@ -1,47 +1,26 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
 // Test that when after an element is selected and highlighted on hover, if the
 // mouse leaves the markup-view and comes back again on the same element, that
 // the highlighter is shown again on the node
 
-function test() {
-  waitForExplicitFinish();
-
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload(evt) {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    waitForFocus(startTests, content);
-  }, true);
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab("data:text/html,<p>Select me!</p>").then(openInspector);
 
-  content.location = "data:text/html;charset=utf-8,<p>Select me!</p>";
-}
-
-function startTests(aInspector, aToolbox) {
-  let p = content.document.querySelector("p");
-  Task.spawn(function() {
-    info("opening the inspector tool");
-    let {inspector, toolbox} = yield openInspector();
+  info("hover over the <p> line in the markup-view so that it's the currently hovered node");
+  yield hoverContainer("p", inspector);
 
-    info("hover over the <p> line in the markup-view so that it's the currently hovered node");
-    yield hoverContainer(p, inspector);
-
-    info("select the <p> markup-container line by clicking");
-    yield clickContainer(p, inspector);
-    ok(isHighlighterVisible(), "the highlighter is shown");
-
-    info("mouse-leave the markup-view");
-    yield mouseLeaveMarkupView(inspector);
-    ok(!isHighlighterVisible(), "the highlighter is hidden after mouseleave");
+  info("select the <p> markup-container line by clicking");
+  yield clickContainer("p", inspector);
+  ok(isHighlighterVisible(), "the highlighter is shown");
 
-    info("hover over the <p> line again, which is still selected");
-    yield hoverContainer(p, inspector);
-    ok(isHighlighterVisible(), "the highlighter is visible again");
-  }).then(null, ok.bind(null, false)).then(endTests);
-}
+  info("mouse-leave the markup-view");
+  yield mouseLeaveMarkupView(inspector);
+  ok(!isHighlighterVisible(), "the highlighter is hidden after mouseleave");
 
-function endTests() {
-  gBrowser.removeCurrentTab();
-  finish();
-}
+  info("hover over the <p> line again, which is still selected");
+  yield hoverContainer("p", inspector);
+  ok(isHighlighterVisible(), "the highlighter is visible again");
+});
--- a/browser/devtools/markupview/test/browser_inspector_markup_968316_highlit_node_on_hover_then_select.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_968316_highlit_node_on_hover_then_select.js
@@ -1,54 +1,36 @@
 /* vim: set ts=2 et sw=2 tw=80: */
 /* Any copyright is dedicated to the Public Domain.
  http://creativecommons.org/publicdomain/zero/1.0/ */
 
 // Test that when first hovering over a node and immediately after selecting it
 // by clicking on it leaves the highlighter visible for as long as the mouse is
 // over the node
 
-function test() {
-  waitForExplicitFinish();
-
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload(evt) {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    waitForFocus(startTests, content);
-  }, true);
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab("data:text/html,<p>It's going to be legen....</p>").then(openInspector);
+  let p = getNode("p");
 
-  content.location = "data:text/html;charset=utf-8,<p>It's going to be legen....</p>";
-}
-
-function startTests(aInspector, aToolbox) {
-  let p = content.document.querySelector("p");
-  Task.spawn(function() {
-    info("opening the inspector tool");
-    let {inspector, toolbox} = yield openInspector();
+  info("hovering over the <p> line in the markup-view");
+  yield hoverContainer(p, inspector);
+  ok(isHighlighterVisible(), "the highlighter is still visible");
 
-    info("hovering over the <p> line in the markup-view");
-    yield hoverContainer(p, inspector);
-    ok(isHighlighterVisible(), "the highlighter is still visible");
-
-    info("selecting the <p> line by clicking in the markup-view");
-    yield clickContainer(p, inspector);
+  info("selecting the <p> line by clicking in the markup-view");
+  yield clickContainer(p, inspector);
 
-    p.textContent = "wait for it ....";
-    info("wait and see if the highlighter stays visible even after the node was selected");
-    yield waitForTheBrieflyShowBoxModelTimeout();
+  p.textContent = "wait for it ....";
+  info("wait and see if the highlighter stays visible even after the node was selected");
+  yield waitForTheBrieflyShowBoxModelTimeout();
 
-    p.textContent = "dary!!!!";
-    ok(isHighlighterVisible(), "the highlighter is still visible");
-  }).then(null, ok.bind(null, false)).then(endTests);
-}
-
-function endTests() {
-  gBrowser.removeCurrentTab();
-  finish();
-}
+  let updated = inspector.once("inspector-updated");
+  p.textContent = "dary!!!!";
+  ok(isHighlighterVisible(), "the highlighter is still visible");
+  yield updated;
+});
 
 function waitForTheBrieflyShowBoxModelTimeout() {
   let deferred = promise.defer();
   // Note that the current timeout is 1 sec and is neither configurable nor
   // exported anywhere we can access, so hard-coding the timeout
   content.setTimeout(deferred.resolve, 1500);
   return deferred.promise;
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_add_attributes.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_add_attributes.js
@@ -8,17 +8,16 @@ http://creativecommons.org/publicdomain/
  * works as expected. Also checks that the changes are properly undoable and
  * redoable. For each step in the test, we:
  * - Create a new DIV
  * - Make the change, check that the change was made as we expect
  * - Undo the change, check that the node is back in its original state
  * - Redo the change, check that the node change was made again correctly.
  */
 
-waitForExplicitFinish();
 requestLongerTimeout(2);
 
 let TEST_URL = "data:text/html,<div>markup-view attributes addition test</div>";
 let TEST_DATA = [{
   desc: "Add an attribute value without closing \"",
   enteredText: 'style="display: block;',
   expectedAttributes: {
     style: "display: block;"
@@ -129,43 +128,38 @@ let TEST_DATA = [{
   desc: "Add event handlers",
   enteredText: "onclick=\"javascript: throw new Error('wont fire');\" onload=\"alert('here');\"",
   expectedAttributes: {
     onclick: "javascript: throw new Error('wont fire');",
     onload: "alert('here');"
   }
 }];
 
-function test() {
-  Task.spawn(function() {
-    info("Opening the inspector on the test URL");
-    let args = yield addTab(TEST_URL).then(openInspector);
-    let inspector = args.inspector;
-    let markup = inspector.markup;
+let test = asyncTest(function*() {
+  info("Opening the inspector on the test URL");
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
+  let markup = inspector.markup;
 
-    info("Selecting the test node");
-    let div = getNode("div");
-    yield selectNode(div, inspector);
-    let editor = getContainerForRawNode(markup, div).editor;
+  info("Selecting the test node");
+  let div = getNode("div");
+  yield selectNode(div, inspector);
+  let editor = getContainerForRawNode(div, inspector).editor;
 
-    for (let test of TEST_DATA) {
-      info("Starting test: " + test.desc);
+  for (let test of TEST_DATA) {
+    info("Starting test: " + test.desc);
 
-      info("Enter the new attribute(s) test: " + test.enteredText);
-      let nodeMutated = inspector.once("markupmutation");
-      setEditableFieldValue(editor.newAttr, test.enteredText, inspector);
-      yield nodeMutated;
+    info("Enter the new attribute(s) test: " + test.enteredText);
+    let nodeMutated = inspector.once("markupmutation");
+    setEditableFieldValue(editor.newAttr, test.enteredText, inspector);
+    yield nodeMutated;
 
-      info("Assert that the attribute(s) has/have been applied correctly");
-      assertAttributes(div, test.expectedAttributes);
+    info("Assert that the attribute(s) has/have been applied correctly");
+    assertAttributes(div, test.expectedAttributes);
 
-      info("Undo the change");
-      yield undoChange(inspector);
+    info("Undo the change");
+    yield undoChange(inspector);
 
-      info("Assert that the attribute(s) has/have been removed correctly");
-      assertAttributes(div, {});
-    }
+    info("Assert that the attribute(s) has/have been removed correctly");
+    assertAttributes(div, {});
+  }
 
-    yield inspector.once("inspector-updated");
-
-    gBrowser.removeCurrentTab();
-  }).then(null, ok.bind(null, false)).then(finish);
-}
+  yield inspector.once("inspector-updated");
+});
--- a/browser/devtools/markupview/test/browser_inspector_markup_edit.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_edit.js
@@ -1,441 +1,409 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
-/**
- * Tests that various editors work as expected. Also checks
- * that the various changes are properly undoable and redoable.
- * For each step in the test, we:
- * - Run the setup for that test (if any)
- * - Check that the node we're editing is as we expect
- * - Make the change, check that the change was made as we expect
- * - Undo the change, check that the node is back in its original state
- * - Redo the change, check that the node change was made again correctly.
- *
- * This test mostly tries to verify that the editor makes changes to the
- * underlying DOM, not that the UI updates - UI updates are based on
- * underlying DOM changes, and the mutation tests should cover those cases.
- */
+// Tests that various editors work as expected. Also checks
+// that the various changes are properly undoable and redoable.
+// For each step in the test, we:
+// - Check that the node we're editing is as we expect
+// - Make the change, check that the change was made as we expect
+// - Undo the change, check that the node is back in its original state
+// - Redo the change, check that the node change was made again correctly.
+// This test mostly tries to verify that the editor makes changes to the
+// underlying DOM, not that the UI updates - UI updates are based on
+// underlying DOM changes, and the mutation tests should cover those cases.
 
-waitForExplicitFinish();
-
-let doc, inspector, markup;
-
-let TEST_URL = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_edit.html";
+let TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_edit.html";
 let LONG_ATTRIBUTE = "ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ";
 let LONG_ATTRIBUTE_COLLAPSED = "ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEF\u2026UVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ-ABCDEFGHIJKLMNOPQRSTUVWXYZ";
 let DATA_URL_INLINE_STYLE='color: red; background: url("");';
 let DATA_URL_INLINE_STYLE_COLLAPSED='color: red; background: url("\u2026NDDeNGe4Ug9C9zwz3gVLMDA/A6P9/AFGGFyjOXZtQAAAAAElFTkSuQmCC");';
 let DATA_URL_ATTRIBUTE = "";
 let DATA_URL_ATTRIBUTE_COLLAPSED = "\u20269/AFGGFyjOXZtQAAAAAElFTkSuQmCC";
 
 let TEST_DATA = [
   {
     desc: "Change an attribute",
-    before: function() {
-      assertAttributes(doc.querySelector("#node1"), {
+    before: function(inspector) {
+      assertAttributes("#node1", {
         id: "node1",
         class: "node1"
       });
     },
-    execute: function(after) {
+    execute: function(after, inspector) {
       inspector.once("markupmutation", after);
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node1")).editor;
+      let editor = getContainerForRawNode("#node1", inspector).editor;
       let attr = editor.attrs["class"].querySelector(".editable");
       setEditableFieldValue(attr, 'class="changednode1"', inspector);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node1"), {
+    after: function(inspector) {
+      assertAttributes("#node1", {
         id: "node1",
         class: "changednode1"
       });
     }
   },
   {
     desc: 'Try changing an attribute to a quote (") - this should result ' +
           'in it being set to an empty string',
-    before: function() {
-      assertAttributes(doc.querySelector("#node22"), {
+    before: function(inspector) {
+      assertAttributes("#node22", {
         id: "node22",
         class: "unchanged"
       });
     },
-    execute: function(after) {
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node22")).editor;
+    execute: function(after, inspector) {
+      let editor = getContainerForRawNode("#node22", inspector).editor;
       let attr = editor.attrs["class"].querySelector(".editable");
       setEditableFieldValue(attr, 'class="""', inspector);
       inspector.once("markupmutation", after);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node22"), {
+    after: function(inspector) {
+      assertAttributes("#node22", {
         id: "node22",
         class: ""
       });
     }
   },
   {
     desc: "Remove an attribute",
-    before: function() {
-      assertAttributes(doc.querySelector("#node4"), {
+    before: function(inspector) {
+      assertAttributes("#node4", {
         id: "node4",
         class: "node4"
       });
     },
-    execute: function(after) {
+    execute: function(after, inspector) {
       inspector.once("markupmutation", after);
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node4")).editor;
+      let editor = getContainerForRawNode("#node4", inspector).editor;
       let attr = editor.attrs["class"].querySelector(".editable");
       setEditableFieldValue(attr, '', inspector);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node4"), {
+    after: function(inspector) {
+      assertAttributes("#node4", {
         id: "node4",
       });
     }
   },
   {
     desc: "Add an attribute by clicking the empty space after a node",
-    before: function() {
-      assertAttributes(doc.querySelector("#node14"), {
+    before: function(inspector) {
+      assertAttributes("#node14", {
         id: "node14",
       });
     },
-    execute: function(after) {
+    execute: function(after, inspector) {
       inspector.once("markupmutation", after);
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node14")).editor;
+      let editor = getContainerForRawNode("#node14", inspector).editor;
       let attr = editor.newAttr;
       setEditableFieldValue(attr, 'class="newclass" style="color:green"', inspector);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node14"), {
+    after: function(inspector) {
+      assertAttributes("#node14", {
         id: "node14",
         class: "newclass",
         style: "color:green"
       });
     }
   },
   {
     desc: 'Try add an attribute containing a quote (") attribute by ' +
           'clicking the empty space after a node - this should result ' +
           'in it being set to an empty string',
-    before: function() {
-      assertAttributes(doc.querySelector("#node23"), {
+    before: function(inspector) {
+      assertAttributes("#node23", {
         id: "node23",
       });
     },
-    execute: function(after) {
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node23")).editor;
+    execute: function(after, inspector) {
+      let editor = getContainerForRawNode("#node23", inspector).editor;
       let attr = editor.newAttr;
       setEditableFieldValue(attr, 'class="newclass" style="""', inspector);
       inspector.once("markupmutation", after);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node23"), {
+    after: function(inspector) {
+      assertAttributes("#node23", {
         id: "node23",
         class: "newclass",
         style: ""
       });
     }
   },
   {
     desc: "Try add attributes by adding to an existing attribute's entry",
-    before: function() {
-      assertAttributes(doc.querySelector("#node24"), {
+    before: function(inspector) {
+      assertAttributes("#node24", {
         id: "node24",
       });
     },
-    execute: function(after) {
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node24")).editor;
+    execute: function(after, inspector) {
+      let editor = getContainerForRawNode("#node24", inspector).editor;
       let attr = editor.attrs["id"].querySelector(".editable");
       setEditableFieldValue(attr, attr.textContent + ' class="""', inspector);
       inspector.once("markupmutation", after);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node24"), {
+    after: function(inspector) {
+      assertAttributes("#node24", {
         id: "node24",
         class: ""
       });
     }
   },
   {
     desc: "Try to add long attribute to make sure it is collapsed in attribute editor.",
-    before: function() {
-      assertAttributes(doc.querySelector("#node24"), {
+    before: function(inspector) {
+      assertAttributes("#node24", {
         id: "node24",
         class: ""
       });
     },
-    execute: function(after) {
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node24")).editor;
+    execute: function(after, inspector) {
+      let editor = getContainerForRawNode("#node24", inspector).editor;
       let attr = editor.newAttr;
       setEditableFieldValue(attr, 'data-long="'+LONG_ATTRIBUTE+'"', inspector);
       inspector.once("markupmutation", after);
     },
-    after: function() {
-
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node24")).editor;
+    after: function(inspector) {
+      let editor = getContainerForRawNode("#node24", inspector).editor;
       let visibleAttrText = editor.attrs["data-long"].querySelector(".attr-value").textContent;
       is (visibleAttrText, LONG_ATTRIBUTE_COLLAPSED)
 
-      assertAttributes(doc.querySelector("#node24"), {
+      assertAttributes("#node24", {
         id: "node24",
         class: "",
         'data-long':LONG_ATTRIBUTE
       });
     }
   },
   {
     desc: "Try to modify the collapsed long attribute, making sure it expands.",
-    before: function() {
-      assertAttributes(doc.querySelector("#node24"), {
+    before: function(inspector) {
+      assertAttributes("#node24", {
         id: "node24",
         class: "",
         'data-long': LONG_ATTRIBUTE
       });
     },
-    execute: function(after) {
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node24")).editor;
+    execute: function(after, inspector) {
+      let editor = getContainerForRawNode("#node24", inspector).editor;
       let attr = editor.attrs["data-long"].querySelector(".editable");
 
       // Check to make sure it has expanded after focus
       attr.focus();
       EventUtils.sendKey("return", inspector.panelWin);
       let input = inplaceEditor(attr).input;
       is (input.value, 'data-long="'+LONG_ATTRIBUTE+'"');
       EventUtils.sendKey("escape", inspector.panelWin);
 
       setEditableFieldValue(attr, input.value  + ' data-short="ABC"', inspector);
       inspector.once("markupmutation", after);
     },
-    after: function() {
-
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node24")).editor;
+    after: function(inspector) {
+      let editor = getContainerForRawNode("#node24", inspector).editor;
       let visibleAttrText = editor.attrs["data-long"].querySelector(".attr-value").textContent;
       is (visibleAttrText, LONG_ATTRIBUTE_COLLAPSED)
 
-      assertAttributes(doc.querySelector("#node24"), {
+      assertAttributes("#node24", {
         id: "node24",
         class: "",
         'data-long': LONG_ATTRIBUTE,
         "data-short": "ABC"
       });
     }
   },
   {
     desc: "Try to add long data URL to make sure it is collapsed in attribute editor.",
-    before: function() {
-      assertAttributes(doc.querySelector("#node-data-url"), {
+    before: function(inspector) {
+      assertAttributes("#node-data-url", {
         id: "node-data-url"
       });
     },
-    execute: function(after) {
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node-data-url")).editor;
+    execute: function(after, inspector) {
+      let editor = getContainerForRawNode("#node-data-url", inspector).editor;
       let attr = editor.newAttr;
       setEditableFieldValue(attr, 'src="'+DATA_URL_ATTRIBUTE+'"', inspector);
       inspector.once("markupmutation", after);
     },
-    after: function() {
-
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node-data-url")).editor;
+    after: function(inspector) {
+      let editor = getContainerForRawNode("#node-data-url", inspector).editor;
       let visibleAttrText = editor.attrs["src"].querySelector(".attr-value").textContent;
       is (visibleAttrText, DATA_URL_ATTRIBUTE_COLLAPSED);
 
-      let node = doc.querySelector("#node-data-url");
+      let node = getNode("#node-data-url");
       is (node.width, 16, "Image width has been set after data url src.");
       is (node.height, 16, "Image height has been set after data url src.");
 
       assertAttributes(node, {
         id: "node-data-url",
         "src": DATA_URL_ATTRIBUTE
       });
     }
   },
   {
     desc: "Try to add long data URL to make sure it is collapsed in attribute editor.",
-    before: function() {
-      assertAttributes(doc.querySelector("#node-data-url-style"), {
+    before: function(inspector) {
+      assertAttributes("#node-data-url-style", {
         id: "node-data-url-style"
       });
     },
-    execute: function(after) {
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node-data-url-style")).editor;
+    execute: function(after, inspector) {
+      let editor = getContainerForRawNode("#node-data-url-style", inspector).editor;
       let attr = editor.newAttr;
       setEditableFieldValue(attr, "style='"+DATA_URL_INLINE_STYLE+"'", inspector);
       inspector.once("markupmutation", after);
     },
-    after: function() {
-
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node-data-url-style")).editor;
+    after: function(inspector) {
+      let editor = getContainerForRawNode("#node-data-url-style", inspector).editor;
       let visibleAttrText = editor.attrs["style"].querySelector(".attr-value").textContent;
       is (visibleAttrText, DATA_URL_INLINE_STYLE_COLLAPSED)
 
-      assertAttributes(doc.querySelector("#node-data-url-style"), {
+      assertAttributes("#node-data-url-style", {
         id: "node-data-url-style",
         'style': DATA_URL_INLINE_STYLE
       });
     }
   },
   {
     desc: "Edit text",
-    before: function() {
-      let node = doc.querySelector('.node6').firstChild;
+    before: function(inspector) {
+      let node = getNode('.node6').firstChild;
       is(node.nodeValue, "line6", "Text should be unchanged");
     },
-    execute: function(after) {
+    execute: function(after, inspector) {
       inspector.once("markupmutation", after);
-      let node = doc.querySelector('.node6').firstChild;
-      let editor = getContainerForRawNode(markup, node).editor;
+      let node = getNode('.node6').firstChild;
+      let editor = getContainerForRawNode(node, inspector).editor;
       let field = editor.elt.querySelector("pre");
       setEditableFieldValue(field, "New text", inspector);
     },
-    after: function() {
-      let node = doc.querySelector('.node6').firstChild;
+    after: function(inspector) {
+      let node = getNode('.node6').firstChild;
       is(node.nodeValue, "New text", "Text should be changed.");
     },
   },
   {
     desc: "Add an attribute value containing < > &uuml; \" & '",
-    before: function() {
-      assertAttributes(doc.querySelector("#node25"), {
+    before: function(inspector) {
+      assertAttributes("#node25", {
         id: "node25",
       });
     },
-    execute: function(after) {
+    execute: function(after, inspector) {
       inspector.once("markupmutation", after);
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node25")).editor;
+      let editor = getContainerForRawNode("#node25", inspector).editor;
       let attr = editor.newAttr;
       setEditableFieldValue(attr, 'src="somefile.html?param1=<a>&param2=&uuml;&param3=\'&quot;\'"', inspector);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node25"), {
+    after: function(inspector) {
+      assertAttributes("#node25", {
         id: "node25",
         src: "somefile.html?param1=<a>&param2=\xfc&param3='\"'"
       });
     }
   },
   {
     desc: "Modify inline style containing \"",
-    before: function() {
-      assertAttributes(doc.querySelector("#node26"), {
+    before: function(inspector) {
+      assertAttributes("#node26", {
         id: "node26",
         style: 'background-image: url("moz-page-thumb://thumbnail?url=http%3A%2F%2Fwww.mozilla.org%2F");'
       });
     },
-    execute: function(after) {
+    execute: function(after, inspector) {
       inspector.once("markupmutation", after);
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node26")).editor;
+      let editor = getContainerForRawNode("#node26", inspector).editor;
       let attr = editor.attrs["style"].querySelector(".editable");
 
-
       attr.focus();
       EventUtils.sendKey("return", inspector.panelWin);
 
       let input = inplaceEditor(attr).input;
       let value = input.value;
 
       is (value,
         "style='background-image: url(\"moz-page-thumb://thumbnail?url=http%3A%2F%2Fwww.mozilla.org%2F\");'",
         "Value contains actual double quotes"
       );
 
       value = value.replace(/mozilla\.org/, "mozilla.com");
       input.value = value;
 
       EventUtils.sendKey("return", inspector.panelWin);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node26"), {
+    after: function(inspector) {
+      assertAttributes("#node26", {
         id: "node26",
         style: 'background-image: url("moz-page-thumb://thumbnail?url=http%3A%2F%2Fwww.mozilla.com%2F");'
       });
     }
   },
   {
     desc: "Modify inline style containing \" and \'",
-    before: function() {
-      assertAttributes(doc.querySelector("#node27"), {
+    before: function(inspector) {
+      assertAttributes("#node27", {
         id: "node27",
         class: 'Double " and single \''
       });
     },
-    execute: function(after) {
+    execute: function(after, inspector) {
       inspector.once("markupmutation", after);
-      let editor = getContainerForRawNode(markup, doc.querySelector("#node27")).editor;
+      let editor = getContainerForRawNode("#node27", inspector).editor;
       let attr = editor.attrs["class"].querySelector(".editable");
 
       attr.focus();
       EventUtils.sendKey("return", inspector.panelWin);
 
       let input = inplaceEditor(attr).input;
       let value = input.value;
 
       is (value, "class=\"Double &quot; and single '\"", "Value contains &quot;");
 
       value = value.replace(/Double/, "&quot;").replace(/single/, "'");
       input.value = value;
 
       EventUtils.sendKey("return", inspector.panelWin);
     },
-    after: function() {
-      assertAttributes(doc.querySelector("#node27"), {
+    after: function(inspector) {
+      assertAttributes("#node27", {
         id: "node27",
         class: '" " and \' \''
       });
     }
   }
 ];
 
-function test() {
-  addTab(TEST_URL).then(openInspector).then(args => {
-    inspector = args.inspector;
-    doc = content.document;
-    markup = inspector.markup;
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
 
-    markup.expandAll().then(() => {
-      // Iterate through the items in TEST_DATA
-      let cursor = 0;
+  info("Expanding all nodes in the markup-view");
+  yield inspector.markup.expandAll();
 
-      function nextEditTest() {
-        executeSoon(function() {
-          if (cursor >= TEST_DATA.length) {
-            return finishUp();
-          }
+  for (let step of TEST_DATA) {
+    yield executeStep(step, inspector);
+  }
+  yield inspector.once("inspector-updated");
+});
 
-          let step = TEST_DATA[cursor++];
-          info("Start test for: " + step.desc);
-          if (step.setup) {
-            step.setup();
-          }
-          step.before();
-          step.execute(function() {
-            step.after();
+function executeStep(step, inspector) {
+  let def = promise.defer();
 
-            undoChange(inspector).then(() => {
-              step.before();
+  info("Start test for: " + step.desc);
+  step.before(inspector);
+  step.execute(function() {
+    step.after(inspector);
 
-              redoChange(inspector).then(() => {
-                step.after();
-                info("End test for: " + step.desc);
-                nextEditTest();
-              });
-            });
-          });
-        });
-      }
+    undoChange(inspector).then(() => {
+      step.before(inspector);
 
-      nextEditTest();
+      redoChange(inspector).then(() => {
+        step.after(inspector);
+        info("End test for: " + step.desc);
+        def.resolve();
+      });
     });
-  });
-}
+  }, inspector);
 
-function finishUp() {
-  while (markup.undo.canUndo()) {
-    markup.undo.undo();
-  }
-  inspector.once("inspector-updated", () => {
-    doc = inspector = markup = null;
-    gBrowser.removeCurrentTab();
-    finish();
-  });
+  return def.promise;
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_edit_2.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_edit_2.js
@@ -1,51 +1,46 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
 // Tests that an existing attribute can be modified
 
-waitForExplicitFinish();
-
 const TEST_URL = "data:text/html,<div id='test-div'>Test modifying my ID attribute</div>";
 
-function test() {
-  Task.spawn(function() {
-    info("Opening the inspector on the test page");
-    let {toolbox, inspector} = yield addTab(TEST_URL).then(openInspector);
+let test = asyncTest(function*() {
+  info("Opening the inspector on the test page");
+  let {toolbox, inspector} = yield addTab(TEST_URL).then(openInspector);
 
-    info("Selecting the test node");
-    let node = content.document.getElementById("test-div");
-    yield selectNode(node, inspector);
+  info("Selecting the test node");
+  let node = content.document.getElementById("test-div");
+  yield selectNode(node, inspector);
 
-    info("Verify attributes, only ID should be there for now");
-    assertAttributes(node, {
-      id: "test-div"
-    });
+  info("Verify attributes, only ID should be there for now");
+  assertAttributes(node, {
+    id: "test-div"
+  });
 
-    info("Focus the ID attribute and change its content");
-    let editor = getContainerForRawNode(inspector.markup, node).editor;
-    let attr = editor.attrs["id"].querySelector(".editable");
-    let mutated = inspector.once("markupmutation");
-    setEditableFieldValue(attr,
-      attr.textContent + ' class="newclass" style="color:green"', inspector);
-    yield mutated;
+  info("Focus the ID attribute and change its content");
+  let editor = getContainerForRawNode(node, inspector).editor;
+  let attr = editor.attrs["id"].querySelector(".editable");
+  let mutated = inspector.once("markupmutation");
+  setEditableFieldValue(attr,
+    attr.textContent + ' class="newclass" style="color:green"', inspector);
+  yield mutated;
 
-    info("Verify attributes, should have ID, class and style");
-    assertAttributes(node, {
-      id: "test-div",
-      class: "newclass",
-      style: "color:green"
-    });
+  info("Verify attributes, should have ID, class and style");
+  assertAttributes(node, {
+    id: "test-div",
+    class: "newclass",
+    style: "color:green"
+  });
 
-    info("Trying to undo the change");
-    yield undoChange(inspector);
-    assertAttributes(node, {
-      id: "test-div"
-    });
+  info("Trying to undo the change");
+  yield undoChange(inspector);
+  assertAttributes(node, {
+    id: "test-div"
+  });
 
-    yield inspector.once("inspector-updated");
-
-    gBrowser.removeCurrentTab();
-  }).then(null, ok.bind(null, false)).then(finish);
-}
+  yield inspector.once("inspector-updated");
+});
--- a/browser/devtools/markupview/test/browser_inspector_markup_edit_3.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_edit_3.js
@@ -1,45 +1,39 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
 // Tests that a node's tagname can be edited in the markup-view
 
-waitForExplicitFinish();
-
 const TEST_URL = "data:text/html,<div id='retag-me'><div id='retag-me-2'></div></div>";
 
-function test() {
-  Task.spawn(function() {
-    info("Opening the inspector on the test page");
-    let {toolbox, inspector} = yield addTab(TEST_URL).then(openInspector);
+let test = asyncTest(function*() {
+  let {toolbox, inspector} = yield addTab(TEST_URL).then(openInspector);
 
-    yield inspector.markup.expandAll();
+  yield inspector.markup.expandAll();
 
-    info("Selecting the test node");
-    let node = content.document.getElementById("retag-me");
-    let child = content.document.querySelector("#retag-me-2");
-    yield selectNode(node, inspector);
+  info("Selecting the test node");
+  let node = content.document.getElementById("retag-me");
+  let child = content.document.querySelector("#retag-me-2");
+  yield selectNode(node, inspector);
 
-    let container = getContainerForRawNode(inspector.markup, node);
-    is(node.tagName, "DIV", "We've got #retag-me element, it's a DIV");
-    ok(container.expanded, "It is expanded");
-    is(child.parentNode, node, "Child #retag-me-2 is inside #retag-me");
+  let container = getContainerForRawNode(node, inspector);
+  is(node.tagName, "DIV", "We've got #retag-me element, it's a DIV");
+  ok(container.expanded, "It is expanded");
+  is(child.parentNode, node, "Child #retag-me-2 is inside #retag-me");
 
-    info("Changing the tagname");
-    let mutated = inspector.once("markupmutation");
-    let tagEditor = container.editor.tag;
-    setEditableFieldValue(tagEditor, "p", inspector);
-    yield mutated;
+  info("Changing the tagname");
+  let mutated = inspector.once("markupmutation");
+  let tagEditor = container.editor.tag;
+  setEditableFieldValue(tagEditor, "p", inspector);
+  yield mutated;
 
-    info("Checking that the tagname change was done");
-    let node = content.document.getElementById("retag-me");
-    let container = getContainerForRawNode(inspector.markup, node);
-    is(node.tagName, "P", "We've got #retag-me, it should now be a P");
-    ok(container.expanded, "It is still expanded");
-    ok(container.selected, "It is still selected");
-    is(child.parentNode, node, "Child #retag-me-2 is still inside #retag-me");
-
-    gBrowser.removeCurrentTab();
-  }).then(null, ok.bind(null, false)).then(finish);
-}
+  info("Checking that the tagname change was done");
+  let node = content.document.getElementById("retag-me");
+  let container = getContainerForRawNode(node, inspector);
+  is(node.tagName, "P", "We've got #retag-me, it should now be a P");
+  ok(container.expanded, "It is still expanded");
+  ok(container.selected, "It is still selected");
+  is(child.parentNode, node, "Child #retag-me-2 is still inside #retag-me");
+});
--- a/browser/devtools/markupview/test/browser_inspector_markup_edit_4.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_edit_4.js
@@ -1,36 +1,30 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 "use strict";
 
 // Tests that a node can be deleted from the markup-view with the delete key
 
-waitForExplicitFinish();
-
 const TEST_URL = "data:text/html,<div id='delete-me'></div>";
 
-function test() {
-  Task.spawn(function() {
-    info("Opening the inspector on the test page");
-    let {toolbox, inspector} = yield addTab(TEST_URL).then(openInspector);
+let test = asyncTest(function*() {
+  let {toolbox, inspector} = yield addTab(TEST_URL).then(openInspector);
 
-    info("Selecting the test node by clicking on it to make sure it receives focus");
-    let node = content.document.getElementById("delete-me");
-    yield clickContainer(node, inspector);
+  info("Selecting the test node by clicking on it to make sure it receives focus");
+  let node = content.document.getElementById("delete-me");
+  yield clickContainer(node, inspector);
 
-    info("Deleting the element with the keyboard");
-    let mutated = inspector.once("markupmutation");
-    EventUtils.sendKey("delete", inspector.panelWin);
-    yield mutated;
-
-    info("Checking that it's gone, baby gone!");
-    ok(!content.document.getElementById("delete-me"), "The test node does not exist");
+  info("Deleting the element with the keyboard");
+  let mutated = inspector.once("markupmutation");
+  EventUtils.sendKey("delete", inspector.panelWin);
+  yield mutated;
 
-    yield undoChange(inspector);
-    ok(content.document.getElementById("delete-me"), "The test node is back!");
+  info("Checking that it's gone, baby gone!");
+  ok(!content.document.getElementById("delete-me"), "The test node does not exist");
 
-    yield inspector.once("inspector-updated");
+  yield undoChange(inspector);
+  ok(content.document.getElementById("delete-me"), "The test node is back!");
 
-    gBrowser.removeCurrentTab();
-  }).then(null, ok.bind(null, false)).then(finish);
-}
+  yield inspector.once("inspector-updated");
+});
--- a/browser/devtools/markupview/test/browser_inspector_markup_edit_outerhtml.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_edit_outerhtml.js
@@ -1,232 +1,199 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
-
-
-function test() {
-  let inspector;
-  let doc;
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
-  waitForExplicitFinish();
-
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(setupTest, content);
-  }, true);
+// Test outerHTML edition via the markup-view
 
-  let outerHTMLs = [
-    {
-      selector: "#one",
-      oldHTML: '<div id="one">First <em>Div</em></div>',
-      newHTML: '<div id="one">First Div</div>',
-      validate: function(pageNode, selectedNode) {
-        is (pageNode.textContent, "First Div", "New div has expected text content");
-        ok (!doc.querySelector("#one em"), "No em remaining")
-      }
-    },
-    {
-      selector: "#removedChildren",
-      oldHTML: '<div id="removedChildren">removedChild <i>Italic <b>Bold <u>Underline</u></b></i> Normal</div>',
-      newHTML: '<div id="removedChildren">removedChild</div>'
-    },
-    {
-      selector: "#addedChildren",
-      oldHTML: '<div id="addedChildren">addedChildren</div>',
-      newHTML: '<div id="addedChildren">addedChildren <i>Italic <b>Bold <u>Underline</u></b></i> Normal</div>'
-    },
-    {
-      selector: "#addedAttribute",
-      oldHTML: '<div id="addedAttribute">addedAttribute</div>',
-      newHTML: '<div id="addedAttribute" class="important" disabled checked>addedAttribute</div>',
-      validate: function(pageNode, selectedNode) {
-        is (pageNode, selectedNode, "Original element is selected");
-        is (pageNode.outerHTML, '<div id="addedAttribute" class="important" disabled="" checked="">addedAttribute</div>',
-              "Attributes have been added");
-      }
-    },
-    {
-      selector: "#changedTag",
-      oldHTML: '<div id="changedTag">changedTag</div>',
-      newHTML: '<p id="changedTag" class="important">changedTag</p>'
-    },
-    {
-      selector: "#badMarkup1",
-      oldHTML: '<div id="badMarkup1">badMarkup1</div>',
-      newHTML: '<div id="badMarkup1">badMarkup1</div> hanging</div>',
-      validate: function(pageNode, selectedNode) {
-        is (pageNode, selectedNode, "Original element is selected");
-
-        let textNode = pageNode.nextSibling;
+const TEST_DATA = [
+  {
+    selector: "#one",
+    oldHTML: '<div id="one">First <em>Div</em></div>',
+    newHTML: '<div id="one">First Div</div>',
+    validate: function(pageNode, selectedNode) {
+      is(pageNode.textContent, "First Div", "New div has expected text content");
+      ok(!getNode("#one em"), "No em remaining")
+    }
+  },
+  {
+    selector: "#removedChildren",
+    oldHTML: '<div id="removedChildren">removedChild <i>Italic <b>Bold <u>Underline</u></b></i> Normal</div>',
+    newHTML: '<div id="removedChildren">removedChild</div>'
+  },
+  {
+    selector: "#addedChildren",
+    oldHTML: '<div id="addedChildren">addedChildren</div>',
+    newHTML: '<div id="addedChildren">addedChildren <i>Italic <b>Bold <u>Underline</u></b></i> Normal</div>'
+  },
+  {
+    selector: "#addedAttribute",
+    oldHTML: '<div id="addedAttribute">addedAttribute</div>',
+    newHTML: '<div id="addedAttribute" class="important" disabled checked>addedAttribute</div>',
+    validate: function(pageNode, selectedNode) {
+      is(pageNode, selectedNode, "Original element is selected");
+      is(pageNode.outerHTML, '<div id="addedAttribute" class="important" disabled="" checked="">addedAttribute</div>',
+            "Attributes have been added");
+    }
+  },
+  {
+    selector: "#changedTag",
+    oldHTML: '<div id="changedTag">changedTag</div>',
+    newHTML: '<p id="changedTag" class="important">changedTag</p>'
+  },
+  {
+    selector: "#badMarkup1",
+    oldHTML: '<div id="badMarkup1">badMarkup1</div>',
+    newHTML: '<div id="badMarkup1">badMarkup1</div> hanging</div>',
+    validate: function(pageNode, selectedNode) {
+      is(pageNode, selectedNode, "Original element is selected");
 
-        is (textNode.nodeName, "#text", "Sibling is a text element");
-        is (textNode.data, " hanging", "New text node has expected text content");
-      }
-    },
-    {
-      selector: "#badMarkup2",
-      oldHTML: '<div id="badMarkup2">badMarkup2</div>',
-      newHTML: '<div id="badMarkup2">badMarkup2</div> hanging<div></div></div></div></body>',
-      validate: function(pageNode, selectedNode) {
-        is (pageNode, selectedNode, "Original element is selected");
-
-        let textNode = pageNode.nextSibling;
+      let textNode = pageNode.nextSibling;
 
-        is (textNode.nodeName, "#text", "Sibling is a text element");
-        is (textNode.data, " hanging", "New text node has expected text content");
-      }
-    },
-    {
-      selector: "#badMarkup3",
-      oldHTML: '<div id="badMarkup3">badMarkup3</div>',
-      newHTML: '<div id="badMarkup3">badMarkup3 <em>Emphasized <strong> and strong</div>',
-      validate: function(pageNode, selectedNode) {
-        is (pageNode, selectedNode, "Original element is selected");
+      is(textNode.nodeName, "#text", "Sibling is a text element");
+      is(textNode.data, " hanging", "New text node has expected text content");
+    }
+  },
+  {
+    selector: "#badMarkup2",
+    oldHTML: '<div id="badMarkup2">badMarkup2</div>',
+    newHTML: '<div id="badMarkup2">badMarkup2</div> hanging<div></div></div></div></body>',
+    validate: function(pageNode, selectedNode) {
+      is(pageNode, selectedNode, "Original element is selected");
 
-        let em = doc.querySelector("#badMarkup3 em");
-        let strong = doc.querySelector("#badMarkup3 strong");
+      let textNode = pageNode.nextSibling;
 
-        is (em.textContent, "Emphasized  and strong", "<em> was auto created");
-        is (strong.textContent, " and strong", "<strong> was auto created");
-      }
-    },
-    {
-      selector: "#badMarkup4",
-      oldHTML: '<div id="badMarkup4">badMarkup4</div>',
-      newHTML: '<div id="badMarkup4">badMarkup4</p>',
-      validate: function(pageNode, selectedNode) {
-        is (pageNode, selectedNode, "Original element is selected");
+      is(textNode.nodeName, "#text", "Sibling is a text element");
+      is(textNode.data, " hanging", "New text node has expected text content");
+    }
+  },
+  {
+    selector: "#badMarkup3",
+    oldHTML: '<div id="badMarkup3">badMarkup3</div>',
+    newHTML: '<div id="badMarkup3">badMarkup3 <em>Emphasized <strong> and strong</div>',
+    validate: function(pageNode, selectedNode) {
+      is(pageNode, selectedNode, "Original element is selected");
+
+      let em = getNode("#badMarkup3 em");
+      let strong = getNode("#badMarkup3 strong");
 
-        let div = doc.querySelector("#badMarkup4");
-        let p = doc.querySelector("#badMarkup4 p");
+      is(em.textContent, "Emphasized  and strong", "<em> was auto created");
+      is(strong.textContent, " and strong", "<strong> was auto created");
+    }
+  },
+  {
+    selector: "#badMarkup4",
+    oldHTML: '<div id="badMarkup4">badMarkup4</div>',
+    newHTML: '<div id="badMarkup4">badMarkup4</p>',
+    validate: function(pageNode, selectedNode) {
+      is(pageNode, selectedNode, "Original element is selected");
 
-        is (div.textContent, "badMarkup4", "textContent is correct");
-        is (div.tagName, "DIV", "did not change to <p> tag");
-        is (p.textContent, "", "The <p> tag has no children");
-        is (p.tagName, "P", "Created an empty <p> tag");
-      }
-    },
-    {
-      selector: "#badMarkup5",
-      oldHTML: '<p id="badMarkup5">badMarkup5</p>',
-      newHTML: '<p id="badMarkup5">badMarkup5 <div>with a nested div</div></p>',
-      validate: function(pageNode, selectedNode) {
-        is (pageNode, selectedNode, "Original element is selected");
+      let div = getNode("#badMarkup4");
+      let p = getNode("#badMarkup4 p");
 
-        let p = doc.querySelector("#badMarkup5");
-        let nodiv = doc.querySelector("#badMarkup5 div");
-        let div = doc.querySelector("#badMarkup5 ~ div");
+      is(div.textContent, "badMarkup4", "textContent is correct");
+      is(div.tagName, "DIV", "did not change to <p> tag");
+      is(p.textContent, "", "The <p> tag has no children");
+      is(p.tagName, "P", "Created an empty <p> tag");
+    }
+  },
+  {
+    selector: "#badMarkup5",
+    oldHTML: '<p id="badMarkup5">badMarkup5</p>',
+    newHTML: '<p id="badMarkup5">badMarkup5 <div>with a nested div</div></p>',
+    validate: function(pageNode, selectedNode) {
+      is(pageNode, selectedNode, "Original element is selected");
 
-        ok (!nodiv, "The invalid markup got created as a sibling");
-        is (p.textContent, "badMarkup5 ", "The <p> tag does not take in the <div> content");
-        is (p.tagName, "P", "Did not change to a <div> tag");
-        is (div.textContent, "with a nested div", "textContent is correct");
-        is (div.tagName, "DIV", "Did not change to <p> tag");
-      }
-    },
-    {
-      selector: "#siblings",
-      oldHTML: '<div id="siblings">siblings</div>',
-      newHTML: '<div id="siblings-before-sibling">before sibling</div>' +
-               '<div id="siblings">siblings (updated)</div>' +
-               '<div id="siblings-after-sibling">after sibling</div>',
-      validate: function(pageNode, selectedNode) {
-        let beforeSiblingNode = doc.querySelector("#siblings-before-sibling");
-        let afterSiblingNode = doc.querySelector("#siblings-after-sibling");
+      let p = getNode("#badMarkup5");
+      let nodiv = getNode("#badMarkup5 div");
+      let div = getNode("#badMarkup5 ~ div");
 
-        is (beforeSiblingNode, selectedNode, "Sibling has been selected");
-        is (pageNode.textContent, "siblings (updated)", "New div has expected text content");
-        is (beforeSiblingNode.textContent, "before sibling", "Sibling has been inserted");
-        is (afterSiblingNode.textContent, "after sibling", "Sibling has been inserted");
-      }
+      ok(!nodiv, "The invalid markup got created as a sibling");
+      is(p.textContent, "badMarkup5 ", "The <p> tag does not take in the <div> content");
+      is(p.tagName, "P", "Did not change to a <div> tag");
+      is(div.textContent, "with a nested div", "textContent is correct");
+      is(div.tagName, "DIV", "Did not change to <p> tag");
     }
-  ];
-  content.location = "data:text/html," +
-    "<!DOCTYPE html>" +
-    "<head><meta charset='utf-8' /></head>" +
-    "<body>" +
-    [outer.oldHTML for (outer of outerHTMLs) ].join("\n") +
-    "</body>" +
-    "</html>";
+  },
+  {
+    selector: "#siblings",
+    oldHTML: '<div id="siblings">siblings</div>',
+    newHTML: '<div id="siblings-before-sibling">before sibling</div>' +
+             '<div id="siblings">siblings (updated)</div>' +
+             '<div id="siblings-after-sibling">after sibling</div>',
+    validate: function(pageNode, selectedNode) {
+      let beforeSiblingNode = getNode("#siblings-before-sibling");
+      let afterSiblingNode = getNode("#siblings-after-sibling");
 
-  function setupTest() {
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      inspector.once("inspector-updated", startTests);
-    });
+      is(beforeSiblingNode, selectedNode, "Sibling has been selected");
+      is(pageNode.textContent, "siblings (updated)", "New div has expected text content");
+      is(beforeSiblingNode.textContent, "before sibling", "Sibling has been inserted");
+      is(afterSiblingNode.textContent, "after sibling", "Sibling has been inserted");
+    }
   }
+];
 
-  function startTests() {
-    inspector.markup._frame.focus();
-    nextStep(0);
-  }
+const TEST_URL = "data:text/html," +
+  "<!DOCTYPE html>" +
+  "<head><meta charset='utf-8' /></head>" +
+  "<body>" +
+  [outer.oldHTML for (outer of TEST_DATA)].join("\n") +
+  "</body>" +
+  "</html>";
 
-  function nextStep(cursor) {
-    if (cursor >= outerHTMLs.length) {
-      finishUp();
-      return;
-    }
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
 
-    let currentTestData = outerHTMLs[cursor];
-    let selector = currentTestData.selector;
-    let oldHTML = currentTestData.oldHTML;
-    let newHTML = currentTestData.newHTML;
-    let rawNode = doc.querySelector(selector);
+  inspector.markup._frame.focus();
 
-    inspector.selection.once("new-node", () => {
-
-      let oldNodeFront = inspector.selection.nodeFront;
+  for (let step of TEST_DATA) {
+    yield selectNode(step.selector, inspector);
+    yield executeStep(step, inspector);
+  }
+});
 
-      // markupmutation fires once the outerHTML is set, with a target
-      // as the parent node and a type of "childList".
-      inspector.once("markupmutation", (e, aMutations) => {
+function executeStep(step, inspector) {
+  let rawNode = getNode(step.selector);
+  let oldNodeFront = inspector.selection.nodeFront;
 
-        // Check to make the sure the correct mutation has fired, and that the
-        // parent is selected (this will be reset to the child once the mutation is complete.
-        let node = inspector.selection.node;
-        let nodeFront = inspector.selection.nodeFront;
-        let mutation = aMutations[0];
-        let isFromOuterHTML = mutation.removed.some((n) => {
-          return n === oldNodeFront;
-        });
+  // markupmutation fires once the outerHTML is set, with a target
+  // as the parent node and a type of "childList".
+  return editHTML(step, inspector).then(mutations => {
+    // Check to make the sure the correct mutation has fired, and that the
+    // parent is selected (this will be reset to the child once the mutation is complete.
+    let node = inspector.selection.node;
+    let nodeFront = inspector.selection.nodeFront;
+    let mutation = mutations[0];
+    let isFromOuterHTML = mutation.removed.some(n => n === oldNodeFront);
 
-        ok (isFromOuterHTML, "The node is in the 'removed' list of the mutation");
-        is (mutation.type, "childList", "Mutation is a childList after updating outerHTML");
-        is (mutation.target, nodeFront, "Parent node is selected immediately after setting outerHTML");
-
-        // Wait for node to be reselected after outerHTML has been set
-        inspector.selection.once("new-node", () => {
+    ok(isFromOuterHTML, "The node is in the 'removed' list of the mutation");
+    is(mutation.type, "childList", "Mutation is a childList after updating outerHTML");
+    is(mutation.target, nodeFront, "Parent node is selected immediately after setting outerHTML");
 
-          // Typically selectedNode will === pageNode, but if a new element has been injected in front
-          // of it, this will not be the case.  If this happens.
-          let selectedNode = inspector.selection.node;
-          let nodeFront = inspector.selection.nodeFront;
-          let pageNode = doc.querySelector(selector);
+    // Wait for node to be reselected after outerHTML has been set
+    return inspector.selection.once("new-node").then(() => {
+      // Typically selectedNode will === pageNode, but if a new element has been injected in front
+      // of it, this will not be the case.  If this happens.
+      let selectedNode = inspector.selection.node;
+      let nodeFront = inspector.selection.nodeFront;
+      let pageNode = getNode(step.selector);
 
-          if (currentTestData.validate) {
-            currentTestData.validate(pageNode, selectedNode);
-          } else {
-            is (pageNode, selectedNode, "Original node (grabbed by selector) is selected");
-            is (pageNode.outerHTML, newHTML, "Outer HTML has been updated");
-          }
-
-          nextStep(cursor + 1);
-        });
+      if (step.validate) {
+        step.validate(pageNode, selectedNode);
+      } else {
+        is(pageNode, selectedNode, "Original node (grabbed by selector) is selected");
+        is(pageNode.outerHTML, step.newHTML, "Outer HTML has been updated");
+      }
 
-      });
-
-      is (inspector.selection.node, rawNode, "Selection is on the correct node");
-      inspector.markup.updateNodeOuterHTML(inspector.selection.nodeFront, newHTML, oldHTML);
+      // Wait for the inspector to be fully updated to avoid causing errors by
+      // abruptly closing hanging requests when the test ends
+      return inspector.once("inspector-updated");
     });
+  });
+}
 
-    inspector.selection.setNode(rawNode);
-  }
-
-
-  function finishUp() {
-    doc = inspector = null;
-    gBrowser.removeCurrentTab();
-    finish();
-  }
+function editHTML({oldHTML, newHTML}, inspector) {
+  // markupmutation fires once the outerHTML is set, with a target
+  // as the parent node and a type of "childList".
+  let mutated = inspector.once("markupmutation");
+  inspector.markup.updateNodeOuterHTML(inspector.selection.nodeFront, newHTML, oldHTML);
+  return mutated;
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_edit_outerhtml2.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_edit_outerhtml2.js
@@ -1,170 +1,178 @@
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
-function test() {
-  let inspector;
-  let doc;
-  let selector = "#keyboard";
-  let oldHTML = '<div id="keyboard"></div>';
-  let newHTML = '<div id="keyboard">Edited</div>';
-
-  waitForExplicitFinish();
+// Test that outerHTML editing keybindings work as expected and that *special*
+// elements like <html>, <body> and <head> can be edited correctly.
 
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(setupTest, content);
-  }, true);
+const TEST_URL = "data:text/html," +
+  "<!DOCTYPE html>" +
+  "<head><meta charset='utf-8' /></head>" +
+  "<body>" +
+  "<div id=\"keyboard\"></div>" +
+  "</body>" +
+  "</html>";
+const SELECTOR = "#keyboard";
+const OLD_HTML = '<div id="keyboard"></div>';
+const NEW_HTML = '<div id="keyboard">Edited</div>';
 
-  content.location = "data:text/html," +
-    "<!DOCTYPE html>" +
-    "<head><meta charset='utf-8' /></head>" +
-    "<body>" +
-    "<div id=\"keyboard\"></div>" +
-    "</body>" +
-    "</html>";
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
 
-  function setupTest() {
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      inspector.once("inspector-updated", startTests);
-    });
-  }
+  inspector.markup._frame.focus();
 
-  function startTests() {
-    inspector.markup._frame.focus();
-    testEscapeCancels();
-  }
+  info("Checking that pressing escape cancels edits");
+  yield testEscapeCancels(inspector);
 
-  function testEscapeCancels() {
-    info("Checking to make sure that pressing escape cancels edits");
-    let rawNode = doc.querySelector(selector);
+  info("Checking that pressing F2 commits edits");
+  yield testF2Commits(inspector);
 
-    inspector.selection.once("new-node", () => {
-
-      inspector.markup.htmlEditor.on("popupshown", function onPopupShown() {
-        inspector.markup.htmlEditor.off("popupshown", onPopupShown);
+  info("Checking that editing the <body> element works like other nodes");
+  yield testBody(inspector);
 
-        ok (inspector.markup.htmlEditor._visible, "HTML Editor is visible");
-        is (rawNode.outerHTML, oldHTML, "The node is starting with old HTML.");
-
-        inspector.markup.htmlEditor.on("popuphidden", function onPopupHidden() {
-          inspector.markup.htmlEditor.off("popuphidden", onPopupHidden);
-          ok (!inspector.markup.htmlEditor._visible, "HTML Editor is not visible");
+  info("Checking that editing the <head> element works like other nodes");
+  yield testHead(inspector);
 
-          let rawNode = doc.querySelector(selector);
-          is (rawNode.outerHTML, oldHTML, "Escape cancels edits");
-          testF2Commits();
-        });
-
-        inspector.markup.htmlEditor.editor.setText(newHTML);
-
-        EventUtils.sendKey("ESCAPE", inspector.markup.htmlEditor.doc.defaultView);
-      });
+  info("Checking that editing the <html> element works like other nodes");
+  yield testDocumentElement(inspector);
 
-      EventUtils.sendKey("F2", inspector.markup._frame.contentWindow);
-    });
+  info("Checking (again) that editing the <html> element works like other nodes");
+  yield testDocumentElement2(inspector);
+});
 
-    inspector.selection.setNode(rawNode);
-  }
+function testEscapeCancels(inspector) {
+  let def = promise.defer();
+  let node = getNode(SELECTOR);
 
-  function testF2Commits() {
-    info("Checking to make sure that pressing F2 commits edits");
-    let rawNode = doc.querySelector(selector);
-
+  selectNode(node, inspector).then(() => {
     inspector.markup.htmlEditor.on("popupshown", function onPopupShown() {
       inspector.markup.htmlEditor.off("popupshown", onPopupShown);
 
-      ok (inspector.markup.htmlEditor._visible, "HTML Editor is visible");
-      is (rawNode.outerHTML, oldHTML, "The node is starting with old HTML.");
+      ok(inspector.markup.htmlEditor._visible, "HTML Editor is visible");
+      is(node.outerHTML, OLD_HTML, "The node is starting with old HTML.");
 
-      inspector.once("markupmutation", (e, aMutations) => {
-        ok (!inspector.markup.htmlEditor._visible, "HTML Editor is not visible");
+      inspector.markup.htmlEditor.on("popuphidden", function onPopupHidden() {
+        inspector.markup.htmlEditor.off("popuphidden", onPopupHidden);
+        ok(!inspector.markup.htmlEditor._visible, "HTML Editor is not visible");
 
-        let rawNode = doc.querySelector(selector);
-        is (rawNode.outerHTML, newHTML, "F2 commits edits - the node has new HTML.");
-        testBody();
+        let node = getNode(SELECTOR);
+        is(node.outerHTML, OLD_HTML, "Escape cancels edits");
+        def.resolve();
       });
 
-      inspector.markup.htmlEditor.editor.setText(newHTML);
-      EventUtils.sendKey("F2", inspector.markup._frame.contentWindow);
+      inspector.markup.htmlEditor.editor.setText(NEW_HTML);
+
+      EventUtils.sendKey("ESCAPE", inspector.markup.htmlEditor.doc.defaultView);
+    });
+
+    EventUtils.sendKey("F2", inspector.markup._frame.contentWindow);
+  });
+
+  return def.promise;
+}
+
+function testF2Commits(inspector) {
+  let def = promise.defer();
+  let node = getNode(SELECTOR);
+
+  inspector.markup.htmlEditor.on("popupshown", function onPopupShown() {
+    inspector.markup.htmlEditor.off("popupshown", onPopupShown);
+
+    ok(inspector.markup.htmlEditor._visible, "HTML Editor is visible");
+    is(node.outerHTML, OLD_HTML, "The node is starting with old HTML.");
+
+    inspector.once("markupmutation", (e, aMutations) => {
+      ok(!inspector.markup.htmlEditor._visible, "HTML Editor is not visible");
+
+      let node = getNode(SELECTOR);
+      is(node.outerHTML, NEW_HTML, "F2 commits edits - the node has new HTML.");
+      def.resolve();
     });
 
-    inspector.markup._frame.contentDocument.documentElement.focus();
+    inspector.markup.htmlEditor.editor.setText(NEW_HTML);
     EventUtils.sendKey("F2", inspector.markup._frame.contentWindow);
-  }
+  });
+
+  inspector.markup._frame.contentDocument.documentElement.focus();
+  EventUtils.sendKey("F2", inspector.markup._frame.contentWindow);
+
+  return def.promise;
+}
+
+function testBody(inspector) {
+  let body = getNode("body");
+  let bodyHTML = '<body id="updated"><p></p></body>';
+  let bodyFront = inspector.markup.walker.frontForRawNode(body);
+  let doc = content.document;
+
+  let mutated = inspector.once("markupmutation");
+  inspector.markup.updateNodeOuterHTML(bodyFront, bodyHTML, body.outerHTML);
 
-  function testBody() {
-    info("Checking to make sure that editing the <body> element works like other nodes");
-    let body = doc.querySelector("body");
-    let bodyHTML = '<body id="updated"><p></p></body>';
-    let bodyFront = inspector.markup.walker.frontForRawNode(body);
-    inspector.once("markupmutation", (e, aMutations) => {
-      is (doc.querySelector("body").outerHTML, bodyHTML, "<body> HTML has been updated");
-      is (doc.querySelectorAll("head").length, 1, "no extra <head>s have been added");
-      testHead();
-    });
-    inspector.markup.updateNodeOuterHTML(bodyFront, bodyHTML, body.outerHTML);
-  }
+  return mutated.then(mutations => {
+    is(getNode("body").outerHTML, bodyHTML, "<body> HTML has been updated");
+    is(doc.querySelectorAll("head").length, 1, "no extra <head>s have been added");
+    return inspector.once("inspector-updated");
+  });
+}
 
-  function testHead() {
-    info("Checking to make sure that editing the <head> element works like other nodes");
-    let head = doc.querySelector("head");
-    let headHTML = '<head id="updated"><title>New Title</title><script>window.foo="bar";</script></head>';
-    let headFront = inspector.markup.walker.frontForRawNode(head);
-    inspector.once("markupmutation", (e, aMutations) => {
-      is (doc.title, "New Title", "New title has been added");
-      is (doc.defaultView.foo, undefined, "Script has not been executed");
-      is (doc.querySelector("head").outerHTML, headHTML, "<head> HTML has been updated");
-      is (doc.querySelectorAll("body").length, 1, "no extra <body>s have been added");
-      testDocumentElement();
-    });
-    inspector.markup.updateNodeOuterHTML(headFront, headHTML, head.outerHTML);
-  }
+function testHead(inspector) {
+  let head = getNode("head");
+  let headHTML = '<head id="updated"><title>New Title</title><script>window.foo="bar";</script></head>';
+  let headFront = inspector.markup.walker.frontForRawNode(head);
+  let doc = content.document;
+
+  let mutated = inspector.once("markupmutation");
+  inspector.markup.updateNodeOuterHTML(headFront, headHTML, head.outerHTML);
+
+  return mutated.then(mutations => {
+    is(doc.title, "New Title", "New title has been added");
+    is(doc.defaultView.foo, undefined, "Script has not been executed");
+    is(doc.querySelector("head").outerHTML, headHTML, "<head> HTML has been updated");
+    is(doc.querySelectorAll("body").length, 1, "no extra <body>s have been added");
+    return inspector.once("inspector-updated");
+  });
+}
 
-  function testDocumentElement() {
-    info("Checking to make sure that editing the <html> element works like other nodes");
-    let docElement = doc.documentElement;
-    let docElementHTML = '<html id="updated" foo="bar"><head><title>Updated from document element</title><script>window.foo="bar";</script></head><body><p>Hello</p></body></html>';
-    let docElementFront = inspector.markup.walker.frontForRawNode(docElement);
-    inspector.once("markupmutation", (e, aMutations) => {
-      is (doc.title, "Updated from document element", "New title has been added");
-      is (doc.defaultView.foo, undefined, "Script has not been executed");
-      is (doc.documentElement.id, "updated", "<html> ID has been updated");
-      is (doc.documentElement.className, "", "<html> class has been updated");
-      is (doc.documentElement.getAttribute("foo"), "bar", "<html> attribute has been updated");
-      is (doc.documentElement.outerHTML, docElementHTML, "<html> HTML has been updated");
-      is (doc.querySelectorAll("head").length, 1, "no extra <head>s have been added");
-      is (doc.querySelectorAll("body").length, 1, "no extra <body>s have been added");
-      is (doc.body.textContent, "Hello", "document.body.textContent has been updated");
-      testDocumentElement2();
-    });
-    inspector.markup.updateNodeOuterHTML(docElementFront, docElementHTML, docElement.outerHTML);
-  }
+function testDocumentElement(inspector) {
+  let doc = content.document;
+  let docElement = doc.documentElement;
+  let docElementHTML = '<html id="updated" foo="bar"><head><title>Updated from document element</title><script>window.foo="bar";</script></head><body><p>Hello</p></body></html>';
+  let docElementFront = inspector.markup.walker.frontForRawNode(docElement);
+
+  let mutated = inspector.once("markupmutation");
+  inspector.markup.updateNodeOuterHTML(docElementFront, docElementHTML, docElement.outerHTML);
+
+  return mutated.then(mutations => {
+    is(doc.title, "Updated from document element", "New title has been added");
+    is(doc.defaultView.foo, undefined, "Script has not been executed");
+    is(doc.documentElement.id, "updated", "<html> ID has been updated");
+    is(doc.documentElement.className, "", "<html> class has been updated");
+    is(doc.documentElement.getAttribute("foo"), "bar", "<html> attribute has been updated");
+    is(doc.documentElement.outerHTML, docElementHTML, "<html> HTML has been updated");
+    is(doc.querySelectorAll("head").length, 1, "no extra <head>s have been added");
+    is(doc.querySelectorAll("body").length, 1, "no extra <body>s have been added");
+    is(doc.body.textContent, "Hello", "document.body.textContent has been updated");
+  });
+}
 
-  function testDocumentElement2() {
-    info("Checking to make sure (again) that editing the <html> element works like other nodes");
-    let docElement = doc.documentElement;
-    let docElementHTML = '<html class="updated" id="somethingelse"><head><title>Updated again from document element</title><script>window.foo="bar";</script></head><body><p>Hello again</p></body></html>';
-    let docElementFront = inspector.markup.walker.frontForRawNode(docElement);
-    inspector.once("markupmutation", (e, aMutations) => {
-      is (doc.title, "Updated again from document element", "New title has been added");
-      is (doc.defaultView.foo, undefined, "Script has not been executed");
-      is (doc.documentElement.id, "somethingelse", "<html> ID has been updated");
-      is (doc.documentElement.className, "updated", "<html> class has been updated");
-      is (doc.documentElement.getAttribute("foo"), null, "<html> attribute has been removed");
-      is (doc.documentElement.outerHTML, docElementHTML, "<html> HTML has been updated");
-      is (doc.querySelectorAll("head").length, 1, "no extra <head>s have been added");
-      is (doc.querySelectorAll("body").length, 1, "no extra <body>s have been added");
-      is (doc.body.textContent, "Hello again", "document.body.textContent has been updated");
-      finishUp();
-    });
-    inspector.markup.updateNodeOuterHTML(docElementFront, docElementHTML, docElement.outerHTML);
-  }
+function testDocumentElement2(inspector) {
+  let doc = content.document;
+  let docElement = doc.documentElement;
+  let docElementHTML = '<html class="updated" id="somethingelse"><head><title>Updated again from document element</title><script>window.foo="bar";</script></head><body><p>Hello again</p></body></html>';
+  let docElementFront = inspector.markup.walker.frontForRawNode(docElement);
+
+  let mutated = inspector.once("markupmutation");
+  inspector.markup.updateNodeOuterHTML(docElementFront, docElementHTML, docElement.outerHTML);
 
-  function finishUp() {
-    doc = inspector = null;
-    gBrowser.removeCurrentTab();
-    finish();
-  }
+  return mutated.then(mutations => {
+    is(doc.title, "Updated again from document element", "New title has been added");
+    is(doc.defaultView.foo, undefined, "Script has not been executed");
+    is(doc.documentElement.id, "somethingelse", "<html> ID has been updated");
+    is(doc.documentElement.className, "updated", "<html> class has been updated");
+    is(doc.documentElement.getAttribute("foo"), null, "<html> attribute has been removed");
+    is(doc.documentElement.outerHTML, docElementHTML, "<html> HTML has been updated");
+    is(doc.querySelectorAll("head").length, 1, "no extra <head>s have been added");
+    is(doc.querySelectorAll("body").length, 1, "no extra <body>s have been added");
+    is(doc.body.textContent, "Hello again", "document.body.textContent has been updated");
+  });
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_mutation.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_mutation.js
@@ -1,190 +1,154 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
 /**
- * Tests that various mutations to the dom update the markup tool correctly.
- * The test for comparing the markup tool to the real dom is a bit weird:
- * - Select the text in the markup tool
+ * Tests that various mutations to the dom update the markup view correctly.
+ * The test for comparing the markup view to the real dom is a bit weird:
+ * - Select the text in the markup view
  * - Parse that as innerHTML in a document we've created for the purpose.
  * - Remove extraneous whitespace in that tree
  * - Compare it to the real dom with isEqualNode.
  */
 
-function fail(err) {
-  ok(false, err)
-}
-
-function test() {
-  waitForExplicitFinish();
-
-  // Will hold the doc we're viewing
-  let contentTab;
-  let doc;
-
-  // Holds the MarkupTool object we're testing.
-  let markup;
-
-  // Holds the document we use to help re-parse the markup tool's output.
-  let parseTab;
-  let parseDoc;
-
-  let inspector;
-
-  // Strip whitespace from a node and its children.
-  function stripWhitespace(node)
+const TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_mutation.html";
+// All the mutation types we want to test.
+const TEST_DATA = [
   {
-    node.normalize();
-    let iter = node.ownerDocument.createNodeIterator(node, NodeFilter.SHOW_TEXT + NodeFilter.SHOW_COMMENT,
-      null);
-
-    while ((node = iter.nextNode())) {
-      node.nodeValue = node.nodeValue.replace(/\s+/g, '');
-      if (node.nodeType == Node.TEXT_NODE &&
-        !/[^\s]/.exec(node.nodeValue)) {
-        node.parentNode.removeChild(node);
-      }
+    desc: "Adding an attribute",
+    test: () => {
+      let node1 = getNode("#node1");
+      node1.setAttribute("newattr", "newattrval");
     }
-  }
-
-  // Verify that the markup in the tool is the same as the markup in the document.
-  function checkMarkup()
+  },
+  {
+    desc: "Removing an attribute",
+    test: () => {
+      let node1 = getNode("#node1");
+      node1.removeAttribute("newattr");
+    }
+  },
   {
-    return markup.expandAll().then(checkMarkup2);
-  }
-
-  function checkMarkup2()
+    desc: "Updating the text-content",
+    test: () => {
+      let node1 = getNode("#node1");
+      node1.textContent = "newtext";
+    }
+  },
   {
-    let contentNode = doc.querySelector("body");
-    let panelNode = getContainerForRawNode(markup, contentNode).elt;
-    let parseNode = parseDoc.querySelector("body");
-
-    // Grab the text from the markup panel...
-    let sel = panelNode.ownerDocument.defaultView.getSelection();
-    sel.selectAllChildren(panelNode);
-
-    // Parse it
-    parseNode.outerHTML = sel;
-    parseNode = parseDoc.querySelector("body");
-
-    // Pull whitespace out of text and comment nodes, there will
-    // be minor unimportant differences.
-    stripWhitespace(parseNode);
-
-    ok(contentNode.isEqualNode(parseNode), "Markup panel should match document.");
-  }
-
-  // All the mutation types we want to test.
-  let mutations = [
-    // Add an attribute
-    function() {
-      let node1 = doc.querySelector("#node1");
-      node1.setAttribute("newattr", "newattrval");
-    },
-    function() {
-      let node1 = doc.querySelector("#node1");
-      node1.removeAttribute("newattr");
-    },
-    function() {
-      let node1 = doc.querySelector("#node1");
-      node1.textContent = "newtext";
-    },
-    function() {
-      let node2 = doc.querySelector("#node2");
+    desc: "Updating the innerHTML",
+    test: () => {
+      let node2 = getNode("#node2");
       node2.innerHTML = "<div><span>foo</span></div>";
-    },
-
-    function() {
-      let node4 = doc.querySelector("#node4");
+    }
+  },
+  {
+    desc: "Removing child nodes",
+    test: () => {
+      let node4 = getNode("#node4");
       while (node4.firstChild) {
         node4.removeChild(node4.firstChild);
       }
-    },
-    function() {
-      // Move a child to a new parent.
-      let node17 = doc.querySelector("#node17");
-      let node1 = doc.querySelector("#node2");
+    }
+  },
+  {
+    desc: "Appending a child to a different parent",
+    test: () => {
+      let node17 = getNode("#node17");
+      let node1 = getNode("#node2");
       node1.appendChild(node17);
-    },
+    }
+  },
+  {
+    desc: "Swapping a parent and child element, putting them in the same tree",
+    // body
+    //  node1
+    //  node18
+    //    node19
+    //      node20
+    //        node21
+    // will become:
+    // body
+    //   node1
+    //     node20
+    //      node21
+    //      node18
+    //        node19
+    test: () => {
+      let node18 = getNode("#node18");
+      let node20 = getNode("#node20");
 
-    function() {
-      // Swap a parent and child element, putting them in the same tree.
-      // body
-      //  node1
-      //  node18
-      //    node19
-      //      node20
-      //        node21
-      // will become:
-      // body
-      //   node1
-      //     node20
-      //      node21
-      //      node18
-      //        node19
-      let node18 = doc.querySelector("#node18");
-      let node20 = doc.querySelector("#node20");
-
-      let node1 = doc.querySelector("#node1");
+      let node1 = getNode("#node1");
 
       node1.appendChild(node20);
       node20.appendChild(node18);
-    },
-  ];
+    }
+  }
+];
 
-  // Create the helper tab for parsing...
-  parseTab = gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    parseDoc = content.document;
+let test = asyncTest(function*() {
+  info("Creating the helper tab for parsing");
+  let parseTab = yield addTab("data:text/html,<html></html>");
+  let parseDoc = content.document;
 
-    // Then create the actual dom we're inspecting...
-    contentTab = gBrowser.selectedTab = gBrowser.addTab();
-    gBrowser.selectedBrowser.addEventListener("load", function onload2() {
-      gBrowser.selectedBrowser.removeEventListener("load", onload2, true);
-      doc = content.document;
-      // Strip whitespace from the doc for easier comparison.
-      stripWhitespace(doc.documentElement);
-      waitForFocus(setupTest, content);
-    }, true);
-    content.location = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_mutation.html";
-  }, true);
+  info("Creating the test tab");
+  let contentTab = yield addTab(TEST_URL);
+  let doc = content.document;
+  // Strip whitespace in the document for easier comparison
+  stripWhitespace(doc.documentElement);
 
-  content.location = "data:text/html,<html></html>";
+  let {inspector} = yield openInspector();
+  let markup = inspector.markup;
+
+  info("Expanding all markup-view nodes");
+  yield markup.expandAll();
+
+  for (let step of TEST_DATA) {
+    info("Starting test: " + step.desc);
 
-  function setupTest() {
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      startTests();
-    });
-  }
+    info("Executing the test markup mutation, listening for inspector-updated before moving on");
+    let updated = inspector.once("inspector-updated");
+    step.test();
+    yield updated;
 
-  function startTests() {
-    markup = inspector.markup;
-    checkMarkup().then(() => {
-      nextStep(0);
-    }).then(null, fail);
-  }
+    info("Expanding all markup-view nodes to make sure new nodes are imported");
+    yield markup.expandAll();
 
-  function nextStep(cursor) {
-    if (cursor >= mutations.length) {
-      finishUp();
-      return;
+    info("Comparing the markup-view markup with the content document");
+    compareMarkup(parseDoc, inspector);
+  }
+});
+
+function stripWhitespace(node) {
+  node.normalize();
+  let iter = node.ownerDocument.createNodeIterator(node,
+    NodeFilter.SHOW_TEXT + NodeFilter.SHOW_COMMENT, null);
+
+  while ((node = iter.nextNode())) {
+    node.nodeValue = node.nodeValue.replace(/\s+/g, '');
+    if (node.nodeType == Node.TEXT_NODE &&
+      !/[^\s]/.exec(node.nodeValue)) {
+      node.parentNode.removeChild(node);
     }
-    mutations[cursor]();
-    inspector.once("markupmutation", function() {
-      executeSoon(function() {
-        checkMarkup().then(() => {
-          nextStep(cursor + 1);
-        }).then(null, fail);
-      });
-    });
-  }
-
-  function finishUp() {
-    doc = inspector = null;
-    gBrowser.removeTab(contentTab);
-    gBrowser.removeTab(parseTab);
-    finish();
   }
 }
+
+function compareMarkup(parseDoc, inspector) {
+  // Grab the text from the markup panel...
+  let markupContainerEl = getContainerForRawNode("body", inspector).elt;
+  let sel = markupContainerEl.ownerDocument.defaultView.getSelection();
+  sel.selectAllChildren(markupContainerEl);
+
+  // Parse it
+  let parseNode = parseDoc.querySelector("body");
+  parseNode.outerHTML = sel;
+  parseNode = parseDoc.querySelector("body");
+
+  // Pull whitespace out of text and comment nodes, there will
+  // be minor unimportant differences.
+  stripWhitespace(parseNode);
+
+  // console.log(contentNode.innerHTML, parseNode.innerHTML);
+  ok(getNode("body").isEqualNode(parseNode),
+    "Markup panel matches what's in the content document.");
+}
--- a/browser/devtools/markupview/test/browser_inspector_markup_mutation_flashing.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_mutation_flashing.js
@@ -1,130 +1,89 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
-
-function test() {
-  waitForExplicitFinish();
-
-  // Will hold the doc we're viewing
-  let contentTab;
-  let doc;
-  let listElement;
-
-  // Holds the MarkupTool object we're testing.
-  let markup;
-  let inspector;
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
-  // Then create the actual dom we're inspecting...
-  contentTab = gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(setupTest, content);
-  }, true);
-  content.location = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_mutation_flashing.html";
-
-  function setupTest() {
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      startTests();
-    });
-  }
-
-  function startTests() {
-    markup = inspector.markup;
-
-    // Get the content UL element
-    listElement = doc.querySelector(".list");
+// Test that markup-containers in the markup-view do flash when their
+// corresponding DOM nodes mutate
 
-    // Making sure children are expanded
-    inspector.selection.setNode(listElement.lastElementChild);
-    inspector.once("inspector-updated", () => {
-      // testData contains a list of mutations to test
-      // Each array item is an object with:
-      // - mutate: a function that should make changes to the content DOM
-      // - assert: a function that should test flashing background
-      let testData = [{
-        // Adding a new node should flash the new node
-        mutate: () => {
-          let newLi = doc.createElement("LI");
-          newLi.textContent = "new list item";
-          listElement.appendChild(newLi);
-        },
-        assert: () => {
-          assertNodeFlashing(listElement.lastElementChild);
-        }
-      }, {
-        // Removing a node should flash its parent
-        mutate: () => {
-          listElement.removeChild(listElement.lastElementChild);
-        },
-        assert: () => {
-          assertNodeFlashing(listElement);
-        }
-      }, {
-        // Re-appending an existing node should only flash this node
-        mutate: () => {
-          listElement.appendChild(listElement.firstElementChild);
-        },
-        assert: () => {
-          assertNodeFlashing(listElement.lastElementChild);
-        }
-      }, {
-        // Adding an attribute should flash the node
-        mutate: () => {
-          listElement.setAttribute("name-" + Date.now(), "value-" + Date.now());
-        },
-        assert: () => {
-          assertNodeFlashing(listElement);
-        }
-      }, {
-        // Editing an attribute should flash the node
-        mutate: () => {
-          listElement.setAttribute("class", "list value-" + Date.now());
-        },
-        assert: () => {
-          assertNodeFlashing(listElement);
-        }
-      }, {
-        // Removing an attribute should flash the node
-        mutate: () => {
-          listElement.removeAttribute("class");
-        },
-        assert: () => {
-          assertNodeFlashing(listElement);
-        }
-      }];
-      testMutation(testData, 0);
-    });
+const TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_mutation_flashing.html";
+// The test data contains a list of mutations to test.
+// Each item is an object:
+// - desc: a description of the test step, for better logging
+// - mutate: a function that should make changes to the content DOM
+// - shouldFlash: a function that returns the element that should be the one flashing
+const TEST_DATA = [{
+  desc: "Adding a new node should flash the new node",
+  mutate: (doc, rootNode) => {
+    let newLi = doc.createElement("LI");
+    newLi.textContent = "new list item";
+    rootNode.appendChild(newLi);
+  },
+  shouldFlash: rootNode => rootNode.lastElementChild
+}, {
+  desc: "Removing a node should flash its parent",
+  mutate: (doc, rootNode) => {
+    rootNode.removeChild(rootNode.lastElementChild);
+  },
+  shouldFlash: rootNode => rootNode
+}, {
+  desc: "Re-appending an existing node should only flash this node",
+  mutate: (doc, rootNode) => {
+    rootNode.appendChild(rootNode.firstElementChild);
+  },
+  shouldFlash: rootNode => rootNode.lastElementChild
+}, {
+  desc: "Adding an attribute should flash the node",
+  mutate: (doc, rootNode) => {
+    rootNode.setAttribute("name-" + Date.now(), "value-" + Date.now());
+  },
+  shouldFlash: rootNode => rootNode
+}, {
+  desc: "Editing an attribute should flash the node",
+  mutate: (doc, rootNode) => {
+    rootNode.setAttribute("class", "list value-" + Date.now());
+  },
+  shouldFlash: rootNode => rootNode
+}, {
+  desc: "Removing an attribute should flash the node",
+  mutate: (doc, rootNode) => {
+    rootNode.removeAttribute("class");
+  },
+  shouldFlash: rootNode => rootNode
+}];
+
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
+
+  info("Getting the <ul.list> root node to test mutations on");
+  let rootNode = getNode(".list");
+
+  info("Selecting the last element of the root node before starting");
+  yield selectNode(rootNode.lastElementChild, inspector);
+
+  for (let {mutate, shouldFlash, desc} of TEST_DATA) {
+    info("Starting test: " + desc);
+
+    info("Mutating the DOM and listening for markupmutation event");
+    let mutated = inspector.once("markupmutation");
+    let updated = inspector.once("inspector-updated");
+    mutate(content.document, rootNode);
+    yield mutated;
+
+    info("Asserting that the correct markup-container is flashing");
+    assertNodeFlashing(shouldFlash(rootNode), inspector);
+
+    // Making sure the inspector has finished updating before moving on
+    yield updated;
   }
+});
 
-  function testMutation(testData, cursor) {
-    if (cursor < testData.length) {
-      let {mutate, assert} = testData[cursor];
-      mutate();
-      inspector.once("markupmutation", () => {
-        assert();
-        testMutation(testData, cursor + 1);
-      });
-    } else {
-      endTests();
-    }
-  }
+function assertNodeFlashing(node, inspector) {
+  let container = getContainerForRawNode(node, inspector);
 
-  function endTests() {
-    gBrowser.removeTab(contentTab);
-    doc = inspector = contentTab = markup = listElement = null;
-    finish();
-  }
-
-  function assertNodeFlashing(rawNode) {
-    let container = getContainerForRawNode(markup, rawNode);
-
-    if(!container) {
-      ok(false, "Node not found");
-    } else {
-      ok(container.tagState.classList.contains("theme-bg-contrast"),
-        "Node is flashing");
-    }
+  if (!container) {
+    ok(false, "Node not found");
+  } else {
+    ok(container.tagState.classList.contains("theme-bg-contrast"),
+      "Node is flashing");
   }
 }
--- a/browser/devtools/markupview/test/browser_inspector_markup_navigation.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_navigation.js
@@ -1,150 +1,134 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
-
-
-function test() {
-  let inspector;
-
-  waitForExplicitFinish();
-
-  let doc;
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
-  let keySequences = [
-    ["pageup", "*doctype*"],
-    ["down", "html"],
-    ["down", "head"],
-    ["down", "body"],
-    ["down", "node0"],
-    ["right", "node0"],
-    ["down", "node1"],
-    ["down", "node2"],
-    ["down", "node3"],
-    ["down", "*comment*"],
-    ["down", "node4"],
-    ["right", "node4"],
-    ["down", "*text*"],
-    ["down", "node5"],
-    ["down", "node6"],
-    ["down", "*comment*"],
-    ["down" , "node7"],
-    ["right", "node7"],
-    ["down", "*text*"],
-    ["down", "node8"],
-    ["left", "node7"],
-    ["left", "node7"],
-    ["right", "node7"],
-    ["right", "*text*"],
-    ["down", "node8"],
-    ["right", "node8"],
-    ["left", "node8"],
-    ["down", "node9"],
-    ["down", "node10"],
-    ["down", "node11"],
-    ["down", "node12"],
-    ["right", "node12"],
-    ["down", "*text*"],
-    ["down", "node13"],
-    ["down", "node14"],
-    ["down", "node15"],
-    ["down", "node15"],
-    ["down", "node15"],
-    ["up", "node14"],
-    ["up", "node13"],
-    ["up", "*text*"],
-    ["up", "node12"],
-    ["left", "node12"],
-    ["down", "node14"],
-    ["home", "*doctype*"],
-    ["pagedown", "*text*"],
-    ["down", "node5"],
-    ["down", "node6"],
-    ["down", "*comment*"],
-    ["down", "node7"],
-    ["left", "node7"],
-    ["down", "node9"],
-    ["down", "node10"],
-    ["pageup", "node2"],
-    ["pageup", "*doctype*"],
-    ["down", "html"],
-    ["left", "html"],
-    ["down", "html"]
-  ];
+// Test that the markup-view nodes can be navigated to with the keyboard
 
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(setupTest, content);
-  }, true);
-
-  content.location = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_navigation.html";
-
-  function setupTest() {
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      inspector.once("inspector-updated", startNavigation);
-    });
-  }
-
-  function startNavigation() {
-    nextStep(0);
-  }
-
-  function nextStep(cursor) {
-    if (cursor >= keySequences.length) {
-      finishUp();
-      return;
-    }
-
-    let key = keySequences[cursor][0];
-    let className = keySequences[cursor][1];
-    inspector.markup._frame.focus();
+const TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_navigation.html";
+const TEST_DATA = [
+  ["pageup", "*doctype*"],
+  ["down", "html"],
+  ["down", "head"],
+  ["down", "body"],
+  ["down", "node0"],
+  ["right", "node0"],
+  ["down", "node1"],
+  ["down", "node2"],
+  ["down", "node3"],
+  ["down", "*comment*"],
+  ["down", "node4"],
+  ["right", "node4"],
+  ["down", "*text*"],
+  ["down", "node5"],
+  ["down", "node6"],
+  ["down", "*comment*"],
+  ["down" , "node7"],
+  ["right", "node7"],
+  ["down", "*text*"],
+  ["down", "node8"],
+  ["left", "node7"],
+  ["left", "node7"],
+  ["right", "node7"],
+  ["right", "*text*"],
+  ["down", "node8"],
+  ["right", "node8"],
+  ["left", "node8"],
+  ["down", "node9"],
+  ["down", "node10"],
+  ["down", "node11"],
+  ["down", "node12"],
+  ["right", "node12"],
+  ["down", "*text*"],
+  ["down", "node13"],
+  ["down", "node14"],
+  ["down", "node15"],
+  ["down", "node15"],
+  ["down", "node15"],
+  ["up", "node14"],
+  ["up", "node13"],
+  ["up", "*text*"],
+  ["up", "node12"],
+  ["left", "node12"],
+  ["down", "node14"],
+  ["home", "*doctype*"],
+  ["pagedown", "*text*"],
+  ["down", "node5"],
+  ["down", "node6"],
+  ["down", "*comment*"],
+  ["down", "node7"],
+  ["left", "node7"],
+  ["down", "node9"],
+  ["down", "node10"],
+  ["pageup", "node2"],
+  ["pageup", "*doctype*"],
+  ["down", "html"],
+  ["left", "html"],
+  ["down", "html"]
+];
 
-    switch(key) {
-      case "right":
-        EventUtils.synthesizeKey("VK_RIGHT", {});
-        break;
-      case "down":
-        EventUtils.synthesizeKey("VK_DOWN", {});
-        break;
-      case "left":
-        EventUtils.synthesizeKey("VK_LEFT", {});
-        break;
-      case "up":
-        EventUtils.synthesizeKey("VK_UP", {});
-        break;
-      case "pageup":
-        EventUtils.synthesizeKey("VK_PAGE_UP", {});
-        break;
-      case "pagedown":
-        EventUtils.synthesizeKey("VK_PAGE_DOWN", {});
-        break;
-      case "home":
-        EventUtils.synthesizeKey("VK_HOME", {});
-        break;
-    }
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
+
+  info("Making sure the markup-view frame is focused");
+  inspector.markup._frame.focus();
+
+  info("Starting to iterate through the test data");
+  for (let [key, className] of TEST_DATA) {
+    info("Testing step: " + key + " to navigate to " + className);
+    pressKey(key);
+
+    info("Making sure markup-view children get updated");
+    yield waitForChildrenUpdated(inspector);
+
+    info("Checking the right node is selected");
+    checkSelectedNode(key, className, inspector);
+  }
+});
 
-    inspector.markup._waitForChildren().then(() => executeSoon(() => {
-      let node = inspector.selection.node;
-
-      if (className == "*comment*") {
-        is(node.nodeType, Node.COMMENT_NODE, "[" + cursor + "] should be a comment after moving " + key);
-      } else if (className == "*text*") {
-        is(node.nodeType, Node.TEXT_NODE, "[" + cursor + "] should be text after moving " + key);
-      } else if (className == "*doctype*") {
-        is(node.nodeType, Node.DOCUMENT_TYPE_NODE, "[" + cursor + "] should be doctype after moving " + key);
-      } else {
-        is(node.className, className, "[" + cursor + "] right node selected: " + className + " after moving " + key);
-      }
-
-      nextStep(cursor + 1);
-    }));
-  }
-
-  function finishUp() {
-    doc = inspector = null;
-    gBrowser.removeCurrentTab();
-    finish();
+function pressKey(key) {
+  switch(key) {
+    case "right":
+      EventUtils.synthesizeKey("VK_RIGHT", {});
+      break;
+    case "down":
+      EventUtils.synthesizeKey("VK_DOWN", {});
+      break;
+    case "left":
+      EventUtils.synthesizeKey("VK_LEFT", {});
+      break;
+    case "up":
+      EventUtils.synthesizeKey("VK_UP", {});
+      break;
+    case "pageup":
+      EventUtils.synthesizeKey("VK_PAGE_UP", {});
+      break;
+    case "pagedown":
+      EventUtils.synthesizeKey("VK_PAGE_DOWN", {});
+      break;
+    case "home":
+      EventUtils.synthesizeKey("VK_HOME", {});
+      break;
   }
 }
+
+function waitForChildrenUpdated(inspector) {
+  let def = promise.defer();
+  inspector.markup._waitForChildren().then(() => {
+    executeSoon(def.resolve);
+  });
+  return def.promise;
+}
+
+function checkSelectedNode(key, className, inspector) {
+  let node = inspector.selection.node;
+
+  if (className == "*comment*") {
+    is(node.nodeType, Node.COMMENT_NODE, "Found a comment after pressing " + key);
+  } else if (className == "*text*") {
+    is(node.nodeType, Node.TEXT_NODE, "Found text after pressing " + key);
+  } else if (className == "*doctype*") {
+    is(node.nodeType, Node.DOCUMENT_TYPE_NODE, "Found the doctype after pressing " + key);
+  } else {
+    is(node.className, className, "Found node: " + className + " after pressing " + key);
+  }
+}
--- a/browser/devtools/markupview/test/browser_inspector_markup_subset.js
+++ b/browser/devtools/markupview/test/browser_inspector_markup_subset.js
@@ -1,150 +1,84 @@
-/* Any copyright", " is dedicated to the Public Domain.
-http://creativecommons.org/publicdomain/zero/1.0/ */
+/* vim: set ts=2 et sw=2 tw=80: */
+/* Any copyright is dedicated to the Public Domain.
+ http://creativecommons.org/publicdomain/zero/1.0/ */
 
-/**
- * Tests that the markup view loads only as many nodes as specified
- * by the devtools.markup.pagesize preference.
- */
+// Tests that the markup view loads only as many nodes as specified by the
+// devtools.markup.pagesize preference.
 
-registerCleanupFunction(function() {
-  Services.prefs.clearUserPref("devtools.markup.pagesize");
-});
 Services.prefs.setIntPref("devtools.markup.pagesize", 5);
 
-
-function test() {
-  waitForExplicitFinish();
-
-  // Will hold the doc we're viewing
-  let doc;
-
-  let inspector;
-
-  // Holds the MarkupTool object we're testing.
-  let markup;
+const TEST_URL = TEST_URL_ROOT + "browser_inspector_markup_subset.html";
+const TEST_DATA = [{
+  desc: "Select the last item",
+  selector: "#z",
+  expected: "*more*vwxyz"
+}, {
+  desc: "Select the first item",
+  selector: "#a",
+  expected: "abcde*more*"
+}, {
+  desc: "Select the last item",
+  selector: "#z",
+  expected: "*more*vwxyz"
+}, {
+  desc: "Select an already-visible item",
+  selector: "#v",
+  // Because "v" was already visible, we shouldn't have loaded
+  // a different page.
+  expected: "*more*vwxyz"
+}, {
+  desc: "Verify childrenDirty reloads the page",
+  selector: "#w",
+  forceReload: true,
+  // But now that we don't already have a loaded page, selecting
+  // w should center around w.
+  expected: "*more*uvwxy*more*"
+}];
 
-  function assertChildren(expected)
-  {
-    let container = getContainerForRawNode(markup, doc.querySelector("body"));
-    let found = [];
-    for (let child of container.children.children) {
-      if (child.classList.contains("more-nodes")) {
-        found += "*more*";
-      } else {
-        found += child.container.node.getAttribute("id");
-      }
+let test = asyncTest(function*() {
+  let {inspector} = yield addTab(TEST_URL).then(openInspector);
+
+  info("Start iterating through the test data");
+  for (let step of TEST_DATA) {
+    info("Start test: " + step.desc);
+
+    if (step.forceReload) {
+      forceReload(inspector);
     }
-    is(found, expected, "Got the expected children.");
-  }
+    info("Selecting the node that corresponds to " + step.selector);
+    yield selectNode(step.selector, inspector);
 
-  function forceReload()
-  {
-    let container = getContainerForRawNode(markup, doc.querySelector("body"));
-    container.childrenDirty = true;
+    info("Checking that the right nodes are shwon");
+    assertChildren(step.expected, inspector);
   }
 
-  let selections = [
-    {
-      desc: "Select the last item",
-      selector: "#z",
-      before: function() {},
-      after: function() {
-        assertChildren("*more*vwxyz");
-      }
-    },
-    {
-      desc: "Select the first item",
-      selector: "#a",
-      before: function() {
-      },
-      after: function() {
-        assertChildren("abcde*more*");
-      }
-    },
-    {
-      desc: "Select the last item",
-      selector: "#z",
-      before: function() {},
-      after: function() {
-        assertChildren("*more*vwxyz");
-      }
-    },
-    {
-      desc: "Select an already-visible item",
-      selector: "#v",
-      before: function() {},
-      after: function() {
-        // Because "v" was already visible, we shouldn't have loaded
-        // a different page.
-        assertChildren("*more*vwxyz");
-      },
-    },
-    {
-      desc: "Verify childrenDirty reloads the page",
-      selector: "#w",
-      before: function() {
-        forceReload();
-      },
-      after: function() {
-        // But now that we don't already have a loaded page, selecting
-        // w should center around w.
-        assertChildren("*more*uvwxy*more*");
-      },
-    },
-  ];
+  info("Checking that clicking the more button loads everything");
+  clickShowMoreNodes(inspector);
+  yield inspector.markup._waitForChildren();
+  assertChildren("abcdefghijklmnopqrstuvwxyz", inspector);
+});
 
-  // Create the helper tab for parsing...
-  gBrowser.selectedTab = gBrowser.addTab();
-  gBrowser.selectedBrowser.addEventListener("load", function onload() {
-    gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    doc = content.document;
-    waitForFocus(setupTest, content);
-  }, true);
-  content.location = "http://mochi.test:8888/browser/browser/devtools/markupview/test/browser_inspector_markup_subset.html";
-
-  function setupTest() {
-    var target = TargetFactory.forTab(gBrowser.selectedTab);
-    let toolbox = gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-      inspector = toolbox.getCurrentPanel();
-      markup = inspector.markup;
-      inspector.once("inspector-updated", runNextSelection);
-    });
+function assertChildren(expected, inspector) {
+  let container = getContainerForRawNode("body", inspector);
+  let found = "";
+  for (let child of container.children.children) {
+    if (child.classList.contains("more-nodes")) {
+      found += "*more*";
+    } else {
+      found += child.container.node.getAttribute("id");
+    }
   }
-
-  function runNextSelection() {
-    let selection = selections.shift();
-    if (!selection) {
-      clickMore();
-      return;
-    }
+  is(found, expected, "Got the expected children.");
+}
 
-    info(selection.desc);
-    selection.before();
-    inspector.once("inspector-updated", function() {
-      selection.after();
-      runNextSelection();
-    });
-    inspector.selection.setNode(doc.querySelector(selection.selector));
-  }
+function forceReload(inspector) {
+  let container = getContainerForRawNode("body", inspector);
+  container.childrenDirty = true;
+}
 
-  function clickMore() {
-    info("Check that clicking more loads the whole thing.");
-    // Make sure that clicking the "more" button loads all the nodes.
-    let container = getContainerForRawNode(markup, doc.querySelector("body"));
-    let button = container.elt.querySelector("button");
-    let win = button.ownerDocument.defaultView;
-
-    EventUtils.sendMouseEvent({type: "click"}, button, win);
-
-    markup._waitForChildren().then(() => {
-      assertChildren("abcdefghijklmnopqrstuvwxyz");
-      finishUp();
-    });
-  }
-
-  function finishUp() {
-    doc = inspector = markup = null;
-    gBrowser.removeCurrentTab();
-    finish();
-  }
+function clickShowMoreNodes(inspector) {
+  let container = getContainerForRawNode("body", inspector);
+  let button = container.elt.querySelector("button");
+  let win = button.ownerDocument.defaultView;
+  EventUtils.sendMouseEvent({type: "click"}, button, win);
 }
--- a/browser/devtools/markupview/test/head.js
+++ b/browser/devtools/markupview/test/head.js
@@ -1,151 +1,188 @@
 /* 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/. */
 
 const Cu = Components.utils;
-
 let {devtools} = Cu.import("resource://gre/modules/devtools/Loader.jsm", {});
 let TargetFactory = devtools.TargetFactory;
 let {console} = Cu.import("resource://gre/modules/devtools/Console.jsm", {});
 let promise = devtools.require("sdk/core/promise");
 let {getInplaceEditorForSpan: inplaceEditor} = devtools.require("devtools/shared/inplace-editor");
 
+// All test are asynchronous
+waitForExplicitFinish();
+
 //Services.prefs.setBoolPref("devtools.dump.emit", true);
 
+// Set the testing flag on gDevTools and reset it when the test ends
 gDevTools.testing = true;
-SimpleTest.registerCleanupFunction(() => {
-  gDevTools.testing = false;
-});
+registerCleanupFunction(() => gDevTools.testing = false);
 
 // Clear preferences that may be set during the course of tests.
-function clearUserPrefs() {
+registerCleanupFunction(() => {
   Services.prefs.clearUserPref("devtools.inspector.htmlPanelOpen");
   Services.prefs.clearUserPref("devtools.inspector.sidebarOpen");
   Services.prefs.clearUserPref("devtools.inspector.activeSidebar");
   Services.prefs.clearUserPref("devtools.dump.emit");
-}
+  Services.prefs.clearUserPref("devtools.markup.pagesize");
+});
 
-registerCleanupFunction(clearUserPrefs);
+// Auto close the toolbox and close the test tabs when the test ends
+registerCleanupFunction(() => {
+  try {
+    let target = TargetFactory.forTab(gBrowser.selectedTab);
+    gDevTools.closeToolbox(target);
+  } catch (ex) {
+    dump(ex);
+  }
+  while (gBrowser.tabs.length > 1) {
+    gBrowser.removeCurrentTab();
+  }
+});
+
+const TEST_URL_ROOT = "http://mochi.test:8888/browser/browser/devtools/markupview/test/";
+
+/**
+ * Define an async test based on a generator function
+ */
+function asyncTest(generator) {
+  return () => Task.spawn(generator).then(null, ok.bind(null, false)).then(finish);
+}
 
 /**
  * Add a new test tab in the browser and load the given url.
  * @param {String} url The url to be loaded in the new tab
- * @return a promise that resolves when the url is loaded
+ * @return a promise that resolves to the tab object when the url is loaded
  */
 function addTab(url) {
+  info("Adding a new tab with URL: '" + url + "'");
   let def = promise.defer();
 
-  gBrowser.selectedTab = gBrowser.addTab();
+  let tab = gBrowser.selectedTab = gBrowser.addTab();
   gBrowser.selectedBrowser.addEventListener("load", function onload() {
     gBrowser.selectedBrowser.removeEventListener("load", onload, true);
-    info("URL " + url + " loading complete into new test tab");
-    waitForFocus(def.resolve, content);
+    info("URL '" + url + "' loading complete");
+    waitForFocus(() => {
+      def.resolve(tab);
+    }, content);
   }, true);
   content.location = url;
 
   return def.promise;
 }
 
 /**
+ * Reload the current page
+ * @return a promise that resolves when the inspector has emitted the event
+ * new-root
+ */
+function reloadPage(inspector) {
+  info("Reloading the page");
+  let newRoot = inspector.once("new-root");
+  content.location.reload();
+  return newRoot;
+}
+
+/**
  * Open the toolbox, with the inspector tool visible.
  * @return a promise that resolves when the inspector is ready
  */
 function openInspector() {
+  info("Opening the inspector panel");
   let def = promise.defer();
 
   let target = TargetFactory.forTab(gBrowser.selectedTab);
   gDevTools.showToolbox(target, "inspector").then(function(toolbox) {
-    info("Toolbox open");
+    info("The toolbox is open");
     let inspector = toolbox.getCurrentPanel();
     inspector.once("inspector-updated", () => {
-      info("Inspector panel active and ready");
+      info("The inspector panel is active and ready");
       def.resolve({toolbox: toolbox, inspector: inspector});
     });
   }).then(null, console.error);
 
   return def.promise;
 }
 
 /**
- * Get the MarkupContainer object instance that corresponds to the given
- * HTML node
- * @param {MarkupView} markupView The instance of MarkupView currently loaded into the inspector panel
- * @param {DOMNode} rawNode The DOM node for which the container is required
- * @return {MarkupContainer}
- */
-function getContainerForRawNode(markupView, rawNode) {
-  let front = markupView.walker.frontForRawNode(rawNode);
-  let container = markupView.getContainer(front);
-  return container;
-}
-
-/**
  * Simple DOM node accesor function that takes either a node or a string css
  * selector as argument and returns the corresponding node
  * @param {String|DOMNode} nodeOrSelector
  * @return {DOMNode}
  */
 function getNode(nodeOrSelector) {
-  let node = nodeOrSelector;
-
-  if (typeof nodeOrSelector === "string") {
-    node = content.document.querySelector(nodeOrSelector);
-    ok(node, "A node was found for selector " + nodeOrSelector);
-  }
-
-  return node;
+  info("Getting the node for '" + nodeOrSelector + "'");
+  return typeof nodeOrSelector === "string" ?
+    content.document.querySelector(nodeOrSelector) :
+    nodeOrSelector;
 }
 
 /**
  * Set the inspector's current selection to a node or to the first match of the
  * given css selector
  * @param {String|DOMNode} nodeOrSelector
  * @param {InspectorPanel} inspector The instance of InspectorPanel currently loaded in the toolbox
  * @param {String} reason Defaults to "test" which instructs the inspector not to highlight the node upon selection
  * @return a promise that resolves when the inspector is updated with the new
  * node
  */
 function selectNode(nodeOrSelector, inspector, reason="test") {
-  info("Selecting the node " + nodeOrSelector);
+  info("Selecting the node for '" + nodeOrSelector + "'");
   let node = getNode(nodeOrSelector);
   let updated = inspector.once("inspector-updated");
   inspector.selection.setNode(node, reason);
   return updated;
 }
 
 /**
+ * Get the MarkupContainer object instance that corresponds to the given
+ * HTML node
+ * @param {DOMNode|String} nodeOrSelector The DOM node for which the
+ * container is required
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * loaded in the toolbox
+ * @return {MarkupContainer}
+ */
+function getContainerForRawNode(nodeOrSelector, {markup}) {
+  let front = markup.walker.frontForRawNode(getNode(nodeOrSelector));
+  let container = markup.getContainer(front);
+  info("Markup-container object for " + nodeOrSelector + " " + container);
+  return container;
+}
+
+/**
  * Simulate a mouse-over on the markup-container (a line in the markup-view)
  * that corresponds to the node or selector passed.
  * @param {String|DOMNode} nodeOrSelector
  * @param {InspectorPanel} inspector The instance of InspectorPanel currently loaded in the toolbox
  * @return a promise that resolves when the container is hovered and the higlighter
  * is shown on the corresponding node
  */
 function hoverContainer(nodeOrSelector, inspector) {
   info("Hovering over the markup-container for node " + nodeOrSelector);
   let highlit = inspector.toolbox.once("node-highlight");
-  let container = getContainerForRawNode(inspector.markup, getNode(nodeOrSelector));
+  let container = getContainerForRawNode(getNode(nodeOrSelector), inspector);
   EventUtils.synthesizeMouseAtCenter(container.tagLine, {type: "mousemove"},
     inspector.markup.doc.defaultView);
   return highlit;
 }
 
 /**
  * Simulate a click on the markup-container (a line in the markup-view)
  * that corresponds to the node or selector passed.
  * @param {String|DOMNode} nodeOrSelector
  * @param {InspectorPanel} inspector The instance of InspectorPanel currently loaded in the toolbox
  * @return a promise that resolves when the node has been selected.
  */
 function clickContainer(nodeOrSelector, inspector) {
   info("Clicking on the markup-container for node " + nodeOrSelector);
   let updated = inspector.once("inspector-updated");
-  let container = getContainerForRawNode(inspector.markup, getNode(nodeOrSelector));
+  let container = getContainerForRawNode(getNode(nodeOrSelector), inspector);
   EventUtils.synthesizeMouseAtCenter(container.tagLine, {type: "mousedown"},
     inspector.markup.doc.defaultView);
   EventUtils.synthesizeMouseAtCenter(container.tagLine, {type: "mouseup"},
     inspector.markup.doc.defaultView);
   return updated;
 }
 
 /**
@@ -174,52 +211,57 @@ function mouseLeaveMarkupView(inspector)
     inspector.toolbox.doc.defaultView);
   executeSoon(def.resolve);
 
   return def.promise;
 }
 
 /**
  * Focus a given editable element, enter edit mode, set value, and commit
- * @param {DOMNode} field The element that gets editable after receiving focus and <ENTER> keypress
+ * @param {DOMNode} field The element that gets editable after receiving focus
+ * and <ENTER> keypress
  * @param {String} value The string value to be set into the edited field
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently loaded in the toolbox
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * loaded in the toolbox
  */
 function setEditableFieldValue(field, value, inspector) {
   field.focus();
   EventUtils.sendKey("return", inspector.panelWin);
   let input = inplaceEditor(field).input;
   ok(input, "Found editable field for setting value: " + value);
   input.value = value;
   EventUtils.sendKey("return", inspector.panelWin);
 }
 
 /**
  * Checks that a node has the given attributes
  *
- * @param {HTMLNode} element The node to check.
+ * @param {DOMNode|String} nodeOrSelector The node or node selector to check.
  * @param {Object} attrs An object containing the attributes to check.
  *        e.g. {id: "id1", class: "someclass"}
  *
  * Note that node.getAttribute() returns attribute values provided by the HTML
  * parser. The parser only provides unescaped entities so &amp; will return &.
  */
-function assertAttributes(element, attrs) {
-  is(element.attributes.length, Object.keys(attrs).length,
+function assertAttributes(nodeOrSelector, attrs) {
+  let node = getNode(nodeOrSelector);
+
+  is(node.attributes.length, Object.keys(attrs).length,
     "Node has the correct number of attributes.");
   for (let attr in attrs) {
-    is(element.getAttribute(attr), attrs[attr],
+    is(node.getAttribute(attr), attrs[attr],
       "Node has the correct " + attr + " attribute.");
   }
 }
 
 /**
  * Undo the last markup-view action and wait for the corresponding mutation to
  * occur
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently loaded in the toolbox
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * loaded in the toolbox
  * @return a promise that resolves when the markup-mutation has been treated or
  * rejects if no undo action is possible
  */
 function undoChange(inspector) {
   let canUndo = inspector.markup.undo.canUndo();
   ok(canUndo, "The last change in the markup-view can be undone");
   if (!canUndo) {
     return promise.reject();
@@ -228,17 +270,18 @@ function undoChange(inspector) {
   let mutated = inspector.once("markupmutation");
   inspector.markup.undo.undo();
   return mutated;
 }
 
 /**
  * Redo the last markup-view action and wait for the corresponding mutation to
  * occur
- * @param {InspectorPanel} inspector The instance of InspectorPanel currently loaded in the toolbox
+ * @param {InspectorPanel} inspector The instance of InspectorPanel currently
+ * loaded in the toolbox
  * @return a promise that resolves when the markup-mutation has been treated or
  * rejects if no redo action is possible
  */
 function redoChange(inspector) {
   let canRedo = inspector.markup.undo.canRedo();
   ok(canRedo, "The last change in the markup-view can be redone");
   if (!canRedo) {
     return promise.reject();
--- a/browser/devtools/styleinspector/rule-view.js
+++ b/browser/devtools/styleinspector/rule-view.js
@@ -1379,19 +1379,23 @@ CssRuleView.prototype = {
     this._viewedElement = aElement;
     if (!this._viewedElement) {
       this._showEmpty();
       return promise.resolve(undefined);
     }
 
     this._elementStyle = new ElementStyle(aElement, this.store, this.pageStyle);
     return this._populate().then(() => {
-      this._elementStyle.onChanged = () => {
-        this._changed();
-      };
+      // A new node may already be selected, in which this._elementStyle will
+      // be null.
+      if (this._elementStyle) {
+        this._elementStyle.onChanged = () => {
+          this._changed();
+        };
+      }
     }).then(null, console.error);
   },
 
   /**
    * Update the rules for the currently highlighted element.
    */
   nodeChanged: function() {
     // Ignore refreshes during editing or when no element is selected.