diff --git a/src/core/frames/frame_controller.ts b/src/core/frames/frame_controller.ts index b3bb376fb..e276a8fe2 100644 --- a/src/core/frames/frame_controller.ts +++ b/src/core/frames/frame_controller.ts @@ -32,6 +32,8 @@ import { VisitOptions } from "../drive/visit" import { TurboBeforeFrameRenderEvent, TurboFetchRequestErrorEvent } from "../session" import { StreamMessage } from "../streams/stream_message" +export type TurboFrameMissingEvent = CustomEvent<{ fetchResponse: FetchResponse }> + export class FrameController implements AppearanceObserverDelegate, @@ -147,23 +149,29 @@ export class FrameController const html = await fetchResponse.responseHTML if (html) { const { body } = parseHTMLDocument(html) - const snapshot = new Snapshot(await this.extractForeignFrameElement(body)) - const renderer = new FrameRenderer( - this, - this.view.snapshot, - snapshot, - FrameRenderer.renderElement, - false, - false - ) - if (this.view.renderPromise) await this.view.renderPromise - this.changeHistory() - - await this.view.render(renderer) - this.complete = true - session.frameRendered(fetchResponse, this.element) - session.frameLoaded(this.element) - this.fetchResponseLoaded(fetchResponse) + const newFrameElement = await this.extractForeignFrameElement(body) + + if (newFrameElement) { + const snapshot = new Snapshot(newFrameElement) + const renderer = new FrameRenderer( + this, + this.view.snapshot, + snapshot, + FrameRenderer.renderElement, + false, + false + ) + if (this.view.renderPromise) await this.view.renderPromise + this.changeHistory() + + await this.view.render(renderer) + this.complete = true + session.frameRendered(fetchResponse, this.element) + session.frameLoaded(this.element) + this.fetchResponseLoaded(fetchResponse) + } else if (this.sessionWillHandleMissingFrame(fetchResponse)) { + await session.frameMissing(this.element, fetchResponse) + } } } catch (error) { console.error(error) @@ -390,12 +398,24 @@ export class FrameController } } + private sessionWillHandleMissingFrame(fetchResponse: FetchResponse) { + this.element.setAttribute("complete", "") + + const event = dispatch("turbo:frame-missing", { + target: this.element, + detail: { fetchResponse }, + cancelable: true, + }) + + return !event.defaultPrevented + } + private findFrameElement(element: Element, submitter?: HTMLElement) { const id = getAttribute("data-turbo-frame", submitter, element) || this.element.getAttribute("target") return getFrameElementById(id) ?? this.element } - async extractForeignFrameElement(container: ParentNode): Promise { + async extractForeignFrameElement(container: ParentNode): Promise { let element const id = CSS.escape(this.id) @@ -410,13 +430,12 @@ export class FrameController await element.loaded return await this.extractForeignFrameElement(element) } - - console.error(`Response has no matching element`) } catch (error) { console.error(error) + return new FrameElement() } - return new FrameElement() + return null } private formActionIsVisitable(form: HTMLFormElement, submitter?: HTMLElement) { diff --git a/src/core/index.ts b/src/core/index.ts index 54b033821..75cbb2f90 100644 --- a/src/core/index.ts +++ b/src/core/index.ts @@ -28,6 +28,7 @@ export { } from "./session" export { TurboSubmitStartEvent, TurboSubmitEndEvent } from "./drive/form_submission" +export { TurboFrameMissingEvent } from "./frames/frame_controller" export { TurboBeforeFetchRequestEvent, TurboBeforeFetchResponseEvent } from "../http/fetch_request" export { TurboBeforeStreamRenderEvent } from "../elements/stream_element" diff --git a/src/core/session.ts b/src/core/session.ts index a1a36fb1a..cbc01286c 100644 --- a/src/core/session.ts +++ b/src/core/session.ts @@ -309,6 +309,11 @@ export class Session this.notifyApplicationAfterFrameRender(fetchResponse, frame) } + frameMissing(frame: FrameElement, fetchResponse: FetchResponse): Promise { + console.warn(`Completing full-page visit as matching frame for #${frame.id} was missing from the response`) + return this.visit(fetchResponse.location) + } + // Application events applicationAllowsFollowingLinkToLocation(link: Element, location: URL, ev: MouseEvent) { diff --git a/src/tests/fixtures/test.js b/src/tests/fixtures/test.js index b22b09248..e8043a048 100644 --- a/src/tests/fixtures/test.js +++ b/src/tests/fixtures/test.js @@ -51,5 +51,6 @@ "turbo:fetch-request-error", "turbo:frame-load", "turbo:frame-render", + "turbo:frame-missing", "turbo:reload" ]) diff --git a/src/tests/functional/frame_tests.ts b/src/tests/functional/frame_tests.ts index e5e160433..565b892b6 100644 --- a/src/tests/functional/frame_tests.ts +++ b/src/tests/functional/frame_tests.ts @@ -113,10 +113,59 @@ test("test following a link driving a frame toggles the [aria-busy=true] attribu ) }) -test("test following a link to a page without a matching frame results in an empty frame", async ({ page }) => { +test("test following a link to a page without a matching frame dispatches a turbo:frame-missing event", async ({ + page, +}) => { await page.click("#missing a") - await nextBeat() - assert.notOk(await innerHTMLForSelector(page, "#missing")) + await noNextEventOnTarget(page, "missing", "turbo:frame-render") + await noNextEventOnTarget(page, "missing", "turbo:frame-load") + const { fetchResponse } = await nextEventOnTarget(page, "missing", "turbo:frame-missing") + await nextEventNamed(page, "turbo:load") + + assert.ok(fetchResponse, "dispatchs turbo:frame-missing with event.detail.fetchResponse") + assert.equal(pathname(page.url()), "/src/tests/fixtures/frames/frame.html", "navigates the page") + + await page.goBack() + await nextEventNamed(page, "turbo:load") + + assert.equal(pathname(page.url()), "/src/tests/fixtures/frames.html") + assert.ok(await innerHTMLForSelector(page, "#missing")) +}) + +test("test following a link to a page without a matching frame dispatches a turbo:frame-missing event that can be cancelled", async ({ + page, +}) => { + await page.locator("#missing").evaluate((frame) => { + frame.addEventListener( + "turbo:frame-missing", + (event) => { + event.preventDefault() + + if (event.target instanceof HTMLElement) { + event.target.textContent = "Overridden" + } + }, + { once: true } + ) + }) + await page.click("#missing a") + await nextEventOnTarget(page, "missing", "turbo:frame-missing") + + assert.equal(await page.textContent("#missing"), "Overridden") +}) + +test("test following a link to a page with a matching frame does not dispatch a turbo:frame-missing event", async ({ + page, +}) => { + await page.click("#link-frame") + await noNextEventNamed(page, "turbo:frame-missing") + await nextEventOnTarget(page, "frame", "turbo:frame-load") + + const src = await attributeForSelector(page, "#frame", "src") + assert( + src?.includes("/src/tests/fixtures/frames/frame.html"), + "navigates frame without dispatching turbo:frame-missing" + ) }) test("test following a link within a frame with a target set navigates the target frame", async ({ page }) => {