From 40e143e3f2f0943ecd8bc3eb1ce4e5e8ef703e55 Mon Sep 17 00:00:00 2001 From: Evan Morikawa Date: Tue, 1 Dec 2015 18:49:40 -0800 Subject: [PATCH] test(contenteditable): add test harness --- .../contenteditable/list-manager-spec.coffee | 6 +- spec/jasmine-helper.coffee | 5 +- spec_integration/app-boot-spec.es6 | 2 +- .../contenteditable-integration-spec.es6 | 74 +++++++++++++++++++ .../contenteditable-test-harness.es6 | 58 +++++++++++++++ spec_integration/integrated-unit-spec.es6 | 2 +- spec_integration/integration-helper.es6 | 61 +++++++++++---- src/browser/application.coffee | 2 +- 8 files changed, 190 insertions(+), 20 deletions(-) create mode 100644 spec_integration/contenteditable-integration-spec.es6 create mode 100644 spec_integration/contenteditable-test-harness.es6 diff --git a/spec/components/contenteditable/list-manager-spec.coffee b/spec/components/contenteditable/list-manager-spec.coffee index e558ff758..ff6924d56 100644 --- a/spec/components/contenteditable/list-manager-spec.coffee +++ b/spec/components/contenteditable/list-manager-spec.coffee @@ -2,7 +2,7 @@ ContenteditableTestHarness = require './contenteditable-test-harness' return unless NylasEnv.inIntegrationSpecMode() -fdescribe "ListManager", -> +xdescribe "ListManager", -> beforeEach -> # console.log "--> Before each" @ce = new ContenteditableTestHarness @@ -22,7 +22,7 @@ fdescribe "ListManager", -> @ce.expectSelection (dom) -> node: dom.querySelectorAll("li")[0] - ffit "Undoes ordered list creation with backspace", -> waitsForPromise => + it "Undoes ordered list creation with backspace", -> waitsForPromise => @ce.keys(['1', '.', 'Space', 'Back space']).then => @ce.expectHTML "1. " @ce.expectSelection (dom) -> @@ -35,7 +35,7 @@ fdescribe "ListManager", -> @ce.expectSelection (dom) -> node: dom.querySelectorAll("li")[0] - xit "Undoes unordered list creation with backspace", -> + it "Undoes unordered list creation with backspace", -> aitsForPromise => @ce.keys(['*', 'Space', 'Back space']).then => @ce.expectHTML "* " diff --git a/spec/jasmine-helper.coffee b/spec/jasmine-helper.coffee index db7271423..e8c4cec28 100644 --- a/spec/jasmine-helper.coffee +++ b/spec/jasmine-helper.coffee @@ -20,7 +20,10 @@ module.exports.runSpecSuite = (specSuite, logFile, logErrors=true) -> else remote.process.stdout.write(str) - if NylasEnv.getLoadSettings().exitWhenDone + if NylasEnv.getLoadSettings().showSpecsInWindow + N1SpecReporter = require './n1-spec-reporter' + reporter = new N1SpecReporter() + else if NylasEnv.getLoadSettings().exitWhenDone reporter = new TerminalReporter color: true print: (str) -> diff --git a/spec_integration/app-boot-spec.es6 b/spec_integration/app-boot-spec.es6 index 67e5dd6b3..66e2d6fd3 100644 --- a/spec_integration/app-boot-spec.es6 +++ b/spec_integration/app-boot-spec.es6 @@ -3,7 +3,7 @@ import {N1Launcher} from './integration-helper' describe('Nylas Prod Bootup Tests', function() { beforeAll((done)=>{ // Boot in dev mode with no arguments - this.app = new N1Launcher([]); + this.app = new N1Launcher(["--dev"]); this.app.mainWindowReady().finally(done); }); diff --git a/spec_integration/contenteditable-integration-spec.es6 b/spec_integration/contenteditable-integration-spec.es6 new file mode 100644 index 000000000..47c7e4157 --- /dev/null +++ b/spec_integration/contenteditable-integration-spec.es6 @@ -0,0 +1,74 @@ +import Promise from 'bluebird' +import {N1Launcher} from './integration-helper' +import ContenteditableTestHarness from './contenteditable-test-harness.es6' + +fdescribe('Contenteditable Integration Spec', function() { + beforeAll((done)=>{ + console.log("----------- BEFORE ALL"); + // Boot in dev mode with no arguments + this.app = new N1Launcher(["--dev"]); + this.app.popoutComposerWindowReady().finally(done); + }); + + beforeEach((done) => { + console.log("----------- BEFORE EACH"); + this.ce = new ContenteditableTestHarness(this.app.client, expect) + this.ce.init().finally(done); + }); + + afterAll((done)=> { + if (this.app && this.app.isRunning()) { + this.app.stop().then(done); + } else { + done() + } + }); + + fit("Creates ordered lists", (done)=> { + console.log("RUNNING KEYS"); + this.app.client.keys(["1", ".", "Space"]).then(()=>{ + console.log("DONE FIRING KEYS"); + e1 = this.ce.expectHTML("
  1. WOOO
") + e2 = this.ce.expectSelection((dom) => { + return {node: dom.querySelectorAll("li")[0]} + }) + return Promise.all(e1,e2) + }).catch((err)=>{ console.log("XXXX ERROR"); console.log(err); }).finally(done) + }); + + it("has main window visible", (done)=> { + this.app.client.isWindowVisible() + .then((result)=>{ expect(result).toBe(true) }) + .finally(done) + }); + + it("has main window focused", (done)=> { + this.app.client.isWindowFocused() + .then((result)=>{ expect(result).toBe(true) }) + .finally(done) + }); + + it("isn't minimized", (done)=> { + this.app.client.isWindowMinimized() + .then((result)=>{ expect(result).toBe(false) }) + .finally(done) + }); + + it("doesn't have the dev tools open", (done)=> { + this.app.client.isWindowDevToolsOpened() + .then((result)=>{ expect(result).toBe(false) }) + .finally(done) + }); + + it("has width", (done)=> { + this.app.client.getWindowWidth() + .then((result)=>{ expect(result).toBeGreaterThan(0) }) + .finally(done) + }); + + it("has height", (done)=> { + this.app.client.getWindowHeight() + .then((result)=>{ expect(result).toBeGreaterThan(0) }) + .finally(done) + }); +}); diff --git a/spec_integration/contenteditable-test-harness.es6 b/spec_integration/contenteditable-test-harness.es6 new file mode 100644 index 000000000..3766b1203 --- /dev/null +++ b/spec_integration/contenteditable-test-harness.es6 @@ -0,0 +1,58 @@ +import Promise from 'bluebird' + +class ContenteditableTestHarness { + + constructor(client, expect) { + this.expect = expect + this.client = client; + } + + init() { + console.log("INIT TEST HARNESS"); + return this.client.execute(() => { + ce = document.querySelector(".contenteditable") + ce.innerHTML = "" + ce.focus() + }).then(({value})=>{ + console.log(value); + }) + } + + expectHTML(expectedHTML) { + console.log("EXPECTING HTML"); + console.log(expectedHTML); + return this.client.execute((expect, arg2) => { + console.log(expect); + console.log(arg2); + ce = document.querySelector(".contenteditable") + expect(ce.innerHTML).toBe(expectedHTML) + return ce.innerHTML + }, this.expect, "FOOO").then(({value})=>{ + console.log("GOT HTML VALUE"); + console.log(value); + }).catch((err)=>{ + console.log("XXXXXXXXXX GOT ERROR") + console.log(err); + }) + } + + expectSelection(callback) { + return this.client.execute(() => { + ce = document.querySelector(".contenteditable") + expectSel = callback(ce) + + anchorNode = expectSel.anchorNode || expectSel.node || "No anchorNode found" + focusNode = expectSel.focusNode || expectSel.node || "No focusNode found" + anchorOffset = expectSel.anchorOffset || expectSel.offset || 0 + focusOffset = expectSel.focusOffset || expectSel.offset || 0 + + selection = document.getSelection() + + this.expect(selection.anchorNode).toBe(anchorNode) + this.expect(selection.focusNode).toBe(focusNode) + this.expect(selection.anchorOffset).toBe(anchorOffset) + this.expect(selection.focusOffset).toBe(focusOffset) + }) + } +} +module.exports = ContenteditableTestHarness diff --git a/spec_integration/integrated-unit-spec.es6 b/spec_integration/integrated-unit-spec.es6 index 0b8bc45b4..1c2ad5300 100644 --- a/spec_integration/integrated-unit-spec.es6 +++ b/spec_integration/integrated-unit-spec.es6 @@ -2,7 +2,7 @@ import {N1Launcher} from './integration-helper' // Some unit tests, such as the Contenteditable specs need to be run with // Spectron availble in the environment. -fdescribe('Integrated Unit Tests', function() { +describe('Integrated Unit Tests', function() { beforeAll((done)=>{ // Boot in dev mode with no arguments this.app = new N1Launcher(["--test=window"]); diff --git a/spec_integration/integration-helper.es6 b/spec_integration/integration-helper.es6 index d9c346b5d..cc7f945a6 100644 --- a/spec_integration/integration-helper.es6 +++ b/spec_integration/integration-helper.es6 @@ -11,14 +11,48 @@ class N1Launcher extends Application { } mainWindowReady() { + return this.windowReady(N1Launcher.mainWindowMatcher) + } + + popoutComposerWindowReady() { + return this.windowReady(N1Launcher.mainWindowMatcher).then(() => { + return this.client.execute(()=>{ + require('nylas-exports').Actions.composeNewBlankDraft(); + }) + }).then(()=>{ + return N1Launcher.waitUntilMatchingWindowLoaded(this.client, N1Launcher.composerWindowMatcher).then((windowId)=>{ + return this.client.window(windowId) + }) + }) + } + + windowReady(matcher) { // Wrap in a Bluebird promise so we have `.finally on the return` return Promise.resolve(this.start().then(()=>{ - return N1Launcher.waitUntilMainWindowLoaded(this.client).then((mainWindowId)=>{ - return this.client.window(mainWindowId) + return N1Launcher.waitUntilMatchingWindowLoaded(this.client, matcher).then((windowId)=>{ + return this.client.window(windowId) }) })); } + static mainWindowMatcher(client) { + return client.isExisting(".main-window-loaded").then((exists)=>{ + if (exists) {return true} else {return false} + }) + } + + static composerWindowMatcher(client) { + return client.execute(()=>{ + return NylasEnv.getLoadSettings().windowType; + }).then(({value})=>{ + if(value === "composer") { + return client.isWindowVisible() + } else { + return false + } + }) + } + static defaultNylasArgs() { return ["--enable-logging", `--resource-path=${jasmine.NYLAS_ROOT_PATH}`] } @@ -47,22 +81,22 @@ class N1Launcher extends Application { // // Returns a promise that resolves with the main window's ID once it's // loaded. - static waitUntilMainWindowLoaded(client, lastCheck=0) { - var CHECK_EVERY = 1000 + static waitUntilMatchingWindowLoaded(client, matcher, lastCheck=0) { + var CHECK_EVERY = 500 return new Promise((resolve, reject) => { client.windowHandles().then(({value}) => { return Promise.mapSeries(value, (windowId)=>{ - return N1Launcher.switchAndCheckForMain(client, windowId) + return N1Launcher.switchAndCheckForMatch(client, windowId, matcher) }) - }).then((mainChecks)=>{ - for (mainWindowId of mainChecks) { - if (mainWindowId) {return resolve(mainWindowId)} + }).then((windowIdChecks)=>{ + for (windowId of windowIdChecks) { + if (windowId) {return resolve(windowId)} } var now = Date.now(); var delay = Math.max(CHECK_EVERY - (now - lastCheck), 0) setTimeout(()=>{ - N1Launcher.waitUntilMainWindowLoaded(client, now).then(resolve) + N1Launcher.waitUntilMatchingWindowLoaded(client, matcher, now).then(resolve) }, delay) }).catch((err) => { console.error(err); @@ -71,11 +105,12 @@ class N1Launcher extends Application { } // Returns false or the window ID of the main window - static switchAndCheckForMain(client, windowId) { + // The `matcher` resolves to a boolean. + static switchAndCheckForMatch(client, windowId, matcher) { return client.window(windowId).then(()=>{ - return client.isExisting(".main-window-loaded").then((exists)=>{ - if (exists) {return windowId} else {return false} - }) + return matcher(client).then((isMatch) => { + if (isMatch) {return windowId} else {return false} + }); }) } } diff --git a/src/browser/application.coffee b/src/browser/application.coffee index e9ff529c4..6a41ada7f 100644 --- a/src/browser/application.coffee +++ b/src/browser/application.coffee @@ -493,4 +493,4 @@ class Application isSpec = true devMode = true safeMode ?= false - new NylasWindow({bootstrapScript, resourcePath, exitWhenDone, isSpec, devMode, specDirectory, specFilePattern, logFile, safeMode}) + new NylasWindow({bootstrapScript, resourcePath, exitWhenDone, isSpec, devMode, specDirectory, specFilePattern, logFile, safeMode, showSpecsInWindow})