Skip to content

Latest commit

 

History

History
219 lines (160 loc) · 8.03 KB

API.md

File metadata and controls

219 lines (160 loc) · 8.03 KB

16.0.3 API Reference

[new ]Tallahassee(origin[, options])

Create new instance of Tallahasse.

  • origin: Optional http request origin, defaults to process.env.PORT as port
    • origin: fully fledged origin with protocol, host, and port
    • port: port number to local http server
    • Express App
    • requestListener: function passed to http.createServer
  • options: optional options
    • headers: default headers for local traffic, e.g.:
      • host: host when navigating to origin. Set cookies will use host to set explicit domain if missing
      • x-forwarded-host: overrides, or acts as, host
      • x-forwarded-proto: origin protocol
    • matchMedia: Window matchMedia RegExp pattern, if no match then default behaviour is executed
    • console: pass console instance directly to Window context

navigateTo(route[, headers, expectedStatusCode])

Navigate to route.

  • route: path to app route
  • headers: optional object with headers
  • expectedStatusCode: optional expected status code, defaults to 200

Returns promise with browser context.

  • $: Cheerio context
  • document
  • focus: Focus this browser
  • focusIframe: set focus to Iframe
  • navigateTo: navigate to route with preserved cookie
  • runScripts: run scripts found in script tags
  • setElementsToScroll: set elements to scroll
  • scrollToBottomOfElement: scroll to bottom of element
  • scrollToTopOfElement: scroll to top of element
  • stickElementToTop: stick element to top
  • unstickElementFromTop: unstick elements from top
  • window
  • response: response object from node-fetch

load([markup])

Load markup and return browser context.

  • markup: optional string with html

Returns promise with browser context.

browser.navigateTo(route[, headers, expectedStatusCode])

Returns promise with new browser context with preserved cookies. Takes the same arguments as navigateTo

browser.runScripts([scopeElement])

Runs script in all script tags.

  • scopeElement: run descendant script in context element, defaults to document.documentElement.

Scroll

Test you abundant sticky logic.

browser.scrollToTopOfElement()

"use strict";

const Browser = require("@expressen/tallahassee");
const Script = require("@bonniernews/wichita");
const { IntersectionObserver: fakeIntersectionObserver } = require("@expressen/tallahassee/lib");
const { app } = require("../app/app.js");

describe("Window scroller", () => {
  describe("use with IntersectionObserver", () => {
    it("listens to window scroll", async () => {
      const browser = await new Browser(app).navigateTo("/");
      browser.window.IntersectionObserver = fakeIntersectionObserver(browser);

      browser.setElementsToScroll((document) => {
        return document.getElementsByTagName("img");
      });

      const [ lazyLoaded ] = browser.document.getElementsByClassName("lazy-load");
      lazyLoaded._setBoundingClientRect({ top: 300 });

      await new Script("../app/assets/scripts/main.js").run(browser.window);

      browser.scrollToTopOfElement(lazyLoaded);

      expect(lazyLoaded.classList.contains("lazy-load")).to.be.false;
    });
  });
});

browser.scrollToBottomOfElement()

Scrolls so that element bottom matches window.innerHeight.

browser.stickElementToTop()

Set element sticky by fixating element to top.

browser.unstickElementFromTop()

Resets element top to wherever it was before sticked.

IntersectionObserver

"use strict";

const Browser = require("@expressen/tallahassee");
const Script = require("@bonniernews/wichita");
const { IntersectionObserver: fakeIntersectionObserver } = require("@expressen/tallahassee/lib");
const { app } = require("../app/app.js");

describe("IntersectionObserver", () => {
  it("observes elements", async () => {
    const browser = await new Browser(app).navigateTo("/", { cookie: "_ga=1" });
    const intersectionObserver = browser.window.IntersectionObserver = fakeIntersectionObserver(browser);

    await new Script("../app/assets/scripts/main.js").run(browser.window);

    expect(intersectionObserver._getObserved()).to.have.length(1);
  });

  it("acts on window scroll", async () => {
    const browser = await new Browser(app).navigateTo("/", { cookie: "_ga=1" });
    browser.window.IntersectionObserver = fakeIntersectionObserver(browser);

    await new Script("../app/assets/scripts/main.js").run(browser.window);

    browser.setElementsToScroll((document) => {
      return document.getElementsByClassName("lazy-load");
    });

    browser.scrollToTopOfElement(browser.document.getElementsByClassName("lazy-load")[0]);

    expect(browser.document.getElementsByClassName("lazy-load").length).to.equal(0);

    expect(browser.document.getElementsByTagName("img")[1].src).to.be.ok;
  });
});

iframe scope

Switch scopes between iframe window and main window.

"use strict";

const Browser = require("@expressen/tallahassee");
const nock = require("nock");
const { app } = require("../app/app.js");

describe("Iframe", () => {
  it("iframe from same host scopes window and document and sets frameElement and inherits cookie", async () => {
    const browser = await new Browser(app).navigateTo("/", { cookie: "_ga=2" });
    const element = browser.document.createElement("iframe");

    element.id = "friendly-frame";
    element.src = "/friendly/";
    browser.document.body.appendChild(element);

    const iframe = browser.document.getElementById("friendly-frame");
    const iframeScope = await browser.focusIframe(iframe);

    expect(iframeScope.window === browser.window, "scoped window").to.be.false;
    expect(iframeScope.window.top === browser.window, "window.top").to.be.true;
    expect(iframeScope.document === browser.document, "scoped document").to.be.false;
    expect(iframeScope.document.cookie, "scoped document cookie").to.equal("_ga=2");
    expect(iframeScope.window.frameElement === iframe, "window.frameElement property").to.be.true;
  });

  it("iframe from other host scopes window and document", async () => {
    nock("http://example.com")
      .get("/framed-content")
      .replyWithFile(200, "./app/assets/public/index.html", { "Content-Type": "text/html" });

    const browser = await new Browser(app).navigateTo("/", { cookie: "_ga=2" });
    const element = browser.document.createElement("iframe");
    element.id = "iframe";
    element.src = "//example.com/framed-content";
    browser.document.body.appendChild(element);
    const iframe = browser.document.getElementById("iframe");
    const iframeScope = await browser.focusIframe(iframe);
    expect(iframeScope.window === browser.window, "scoped window").to.be.false;
    expect(iframeScope.window.top, "window.top").to.be.ok;
    expect(() => iframeScope.window.top.location.pathname).to.throw("Blocked a frame with origin \"http://example.com\" from accessing a cross-origin frame.");
    expect(iframeScope.document === browser.document, "scoped document").to.be.false;
    expect(iframeScope.document.cookie, "scoped document cookie").to.equal("");
    expect(iframeScope.window.frameElement, "window.frameElement property").to.be.undefined;
  });
});