Skip to content

Commit

Permalink
Return Promise<void> from Turbo.visit (#650)
Browse files Browse the repository at this point in the history
When consumer applications navigate through the `Turbo.visit`, return a
`Promise<void>` that will resolve when the visit is complete.

If a visit fails or is cancelled, the `Promise` will be rejected.

To achieve this behavior, extract the `ResolvingFunctions<T>` type
from the `Renderer` class into the shared `src/core/types.ts` module.
Following the pattern established by `Renderer` instances, add a
`promise: Promise` property to the `Visit` class, and use that `Promise`
to assign a `resolvingFunctions: ResolvingFunctions<void>` property to
make the underlying `resolve()` and `reject()` calls available to the
`Visit`.

When a `Visit` completes successfully, call
`this.resolvingFunctions.resolve()`. When it fails or is canceled from
the outside, call `this.resolvingFunctions.reject()`.
  • Loading branch information
seanpdoyle authored Jul 28, 2022
1 parent 82937c6 commit aeeaae8
Show file tree
Hide file tree
Showing 10 changed files with 86 additions and 16 deletions.
10 changes: 8 additions & 2 deletions src/core/drive/navigator.ts
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@ import { PageSnapshot } from "./page_snapshot"

export type NavigatorDelegate = VisitDelegate & {
allowsVisitingLocationWithAction(location: URL, action?: Action): boolean
visitProposedToLocation(location: URL, options: Partial<VisitOptions>): void
visitProposedToLocation(location: URL, options: Partial<VisitOptions>): Promise<void>
notifyApplicationAfterVisitingSamePageLocation(oldURL: URL, newURL: URL): void
}

Expand All @@ -26,10 +26,14 @@ export class Navigator {
proposeVisit(location: URL, options: Partial<VisitOptions> = {}) {
if (this.delegate.allowsVisitingLocationWithAction(location, options.action)) {
if (locationIsVisitable(location, this.view.snapshot.rootLocation)) {
this.delegate.visitProposedToLocation(location, options)
return this.delegate.visitProposedToLocation(location, options)
} else {
window.location.href = location.toString()

return Promise.resolve()
}
} else {
return Promise.reject()
}
}

Expand All @@ -41,6 +45,8 @@ export class Navigator {
...options,
})
this.currentVisit.start()

return this.currentVisit.promise
}

submitForm(form: HTMLFormElement, submitter?: HTMLElement) {
Expand Down
12 changes: 11 additions & 1 deletion src/core/drive/visit.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import { History } from "./history"
import { getAnchor } from "../url"
import { Snapshot } from "../snapshot"
import { PageSnapshot } from "./page_snapshot"
import { Action } from "../types"
import { Action, ResolvingFunctions } from "../types"
import { getHistoryMethodForAction, uuid } from "../../util"
import { PageView } from "./page_view"

Expand Down Expand Up @@ -81,6 +81,9 @@ export class Visit implements FetchRequestDelegate {
readonly visitCachedSnapshot: (snapshot: Snapshot) => void
readonly willRender: boolean
readonly updateHistory: boolean
readonly promise: Promise<void>

private resolvingFunctions!: ResolvingFunctions<void>

followedRedirect = false
frame?: number
Expand All @@ -105,6 +108,7 @@ export class Visit implements FetchRequestDelegate {
this.delegate = delegate
this.location = location
this.restorationIdentifier = restorationIdentifier || uuid()
this.promise = new Promise((resolve, reject) => (this.resolvingFunctions = { resolve, reject }))

const {
action,
Expand Down Expand Up @@ -169,6 +173,8 @@ export class Visit implements FetchRequestDelegate {
}
this.cancelRender()
this.state = VisitState.canceled

this.resolvingFunctions.reject()
}
}

Expand All @@ -182,13 +188,17 @@ export class Visit implements FetchRequestDelegate {
this.adapter.visitCompleted(this)
this.delegate.visitCompleted(this)
}

this.resolvingFunctions.resolve()
}
}

fail() {
if (this.state == VisitState.started) {
this.state = VisitState.failed
this.adapter.visitFailed(this)

this.resolvingFunctions.reject()
}
}

Expand Down
4 changes: 2 additions & 2 deletions src/core/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -64,8 +64,8 @@ export function registerAdapter(adapter: Adapter) {
* @param options.snapshotHTML Cached snapshot to render
* @param options.response Response of the specified location
*/
export function visit(location: Locatable, options?: Partial<VisitOptions>) {
session.visit(location, options)
export function visit(location: Locatable, options?: Partial<VisitOptions>): Promise<void> {
return session.visit(location, options)
}

/**
Expand Down
2 changes: 1 addition & 1 deletion src/core/native/adapter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { FormSubmission } from "../drive/form_submission"
import { ReloadReason } from "./browser_adapter"

export interface Adapter {
visitProposedToLocation(location: URL, options?: Partial<VisitOptions>): void
visitProposedToLocation(location: URL, options?: Partial<VisitOptions>): Promise<void>
visitStarted(visit: Visit): void
visitCompleted(visit: Visit): void
visitFailed(visit: Visit): void
Expand Down
2 changes: 1 addition & 1 deletion src/core/native/browser_adapter.ts
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ export class BrowserAdapter implements Adapter {
}

visitProposedToLocation(location: URL, options?: Partial<VisitOptions>) {
this.navigator.startVisit(location, options?.restorationIdentifier || uuid(), options)
return this.navigator.startVisit(location, options?.restorationIdentifier || uuid(), options)
}

visitStarted(visit: Visit) {
Expand Down
6 changes: 1 addition & 5 deletions src/core/renderer.ts
Original file line number Diff line number Diff line change
@@ -1,13 +1,9 @@
import { ResolvingFunctions } from "./types"
import { Bardo, BardoDelegate } from "./bardo"
import { Snapshot } from "./snapshot"
import { ReloadReason } from "./native/browser_adapter"
import { getMetaContent } from "../util"

type ResolvingFunctions<T = unknown> = {
resolve(value: T | PromiseLike<T>): void
reject(reason?: any): void
}

export type Render<E> = (newElement: E, currentElement: E) => void

export abstract class Renderer<E extends Element, S extends Snapshot<E> = Snapshot<E>> implements BardoDelegate {
Expand Down
6 changes: 3 additions & 3 deletions src/core/session.ts
Original file line number Diff line number Diff line change
Expand Up @@ -106,8 +106,8 @@ export class Session
this.adapter = adapter
}

visit(location: Locatable, options: Partial<VisitOptions> = {}) {
this.navigator.proposeVisit(expandURL(location), options)
visit(location: Locatable, options: Partial<VisitOptions> = {}): Promise<void> {
return this.navigator.proposeVisit(expandURL(location), options)
}

connectStreamSource(source: StreamSource) {
Expand Down Expand Up @@ -194,7 +194,7 @@ export class Session

visitProposedToLocation(location: URL, options: Partial<VisitOptions>) {
extendURLWithDeprecatedProperties(location)
this.adapter.visitProposedToLocation(location, options)
return this.adapter.visitProposedToLocation(location, options)
}

visitStarted(visit: Visit) {
Expand Down
5 changes: 5 additions & 0 deletions src/core/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,3 +18,8 @@ export type StreamSource = {
options?: boolean | EventListenerOptions
): void
}

export type ResolvingFunctions<T = unknown> = {
resolve(value: T | PromiseLike<T>): void
reject(reason?: any): void
}
51 changes: 51 additions & 0 deletions src/tests/functional/visit_tests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,24 @@ test("test programmatically visiting a same-origin location", async ({ page }) =
assert.ok(timing)
})

test("test programmatically Turbo.visit-ing a same-origin location", async ({ page }) => {
const urlBeforeVisit = page.url()
await page.evaluate(async () => await window.Turbo.visit("/src/tests/fixtures/one.html"))

const urlAfterVisit = page.url()
assert.notEqual(urlBeforeVisit, urlAfterVisit)
assert.equal(await visitAction(page), "advance")

const { url: urlFromBeforeVisitEvent } = await nextEventNamed(page, "turbo:before-visit")
assert.equal(urlFromBeforeVisitEvent, urlAfterVisit)

const { url: urlFromVisitEvent } = await nextEventNamed(page, "turbo:visit")
assert.equal(urlFromVisitEvent, urlAfterVisit)

const { timing } = await nextEventNamed(page, "turbo:load")
assert.ok(timing)
})

test("skip programmatically visiting a cross-origin location falls back to window.location", async ({ page }) => {
const urlBeforeVisit = page.url()
await visitLocation(page, "about:blank")
Expand All @@ -37,6 +55,17 @@ test("skip programmatically visiting a cross-origin location falls back to windo
assert.equal(await visitAction(page), "load")
})

test("skip programmatically Turbo.visit-ing a cross-origin location falls back to window.location", async ({
page,
}) => {
const urlBeforeVisit = page.url()
await page.evaluate(async () => await window.Turbo.visit("about:blank"))

const urlAfterVisit = page.url()
assert.notEqual(urlBeforeVisit, urlAfterVisit)
assert.equal(await visitAction(page), "load")
})

test("test visiting a location served with a non-HTML content type", async ({ page }) => {
const urlBeforeVisit = page.url()
await visitLocation(page, "/src/tests/fixtures/svg.svg")
Expand Down Expand Up @@ -66,6 +95,28 @@ test("test canceling a before-visit event prevents navigation", async ({ page })
assert.equal(urlAfterVisit, urlBeforeVisit)
})

test("test canceling a before-visit event prevents a Turbo.visit-initiated navigation", async ({ page }) => {
await cancelNextVisit(page)
const urlBeforeVisit = page.url()

assert.notOk<boolean>(
await willChangeBody(page, async () => {
await page.evaluate(async () => {
try {
return await window.Turbo.visit("/src/tests/fixtures/one.html")
} catch {
const title = document.querySelector("h1")
if (title) title.innerHTML = "Visit canceled"
}
})
})
)

const urlAfterVisit = page.url()
assert.equal(urlAfterVisit, urlBeforeVisit)
assert.equal(await page.textContent("h1"), "Visit canceled")
})

test("test navigation by history is not cancelable", async ({ page }) => {
await page.click("#same-origin-link")
await nextEventNamed(page, "turbo:load")
Expand Down
4 changes: 3 additions & 1 deletion src/tests/unit/deprecated_adapter_support_test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -34,8 +34,10 @@ export class DeprecatedAdapterSupportTest extends DOMTestCase implements Adapter

// Adapter interface

visitProposedToLocation(location: URL, _options?: Partial<VisitOptions>): void {
visitProposedToLocation(location: URL, _options?: Partial<VisitOptions>): Promise<void> {
this.locations.push(location)

return Promise.resolve()
}

visitStarted(visit: Visit): void {
Expand Down

0 comments on commit aeeaae8

Please sign in to comment.