Skip to content

Commit

Permalink
adjust after reverting hotwired#412
Browse files Browse the repository at this point in the history
  • Loading branch information
seanpdoyle committed Oct 6, 2022
1 parent c212ae7 commit fa20fcd
Show file tree
Hide file tree
Showing 5 changed files with 41 additions and 44 deletions.
20 changes: 5 additions & 15 deletions src/core/frames/frame_controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,7 +26,7 @@ import { FrameView } from "./frame_view"
import { LinkInterceptor, LinkInterceptorDelegate } from "./link_interceptor"
import { FormLinkClickObserver, FormLinkClickObserverDelegate } from "../../observers/form_link_click_observer"
import { FrameRenderer } from "./frame_renderer"
import { TurboClickEvent, session } from "../index"
import { session } from "../index"
import { isAction, Action } from "../types"
import { VisitOptions } from "../drive/visit"
import { TurboBeforeFrameRenderEvent } from "../session"
Expand Down Expand Up @@ -214,12 +214,12 @@ export class FrameController

// Link interceptor delegate

shouldInterceptLinkClick(element: Element, url: string, originalEvent: MouseEvent) {
return this.shouldInterceptNavigation(element) && this.frameAllowsVisitingLocation(element, url, originalEvent)
shouldInterceptLinkClick(element: Element, _location: string, _event: MouseEvent) {
return this.shouldInterceptNavigation(element)
}

linkClickIntercepted(element: Element, url: string) {
this.navigateFrame(element, url)
linkClickIntercepted(element: Element, location: string) {
this.navigateFrame(element, location)
}

// Form submit observer delegate
Expand Down Expand Up @@ -555,16 +555,6 @@ export class FrameController
return expandURL(root)
}

private frameAllowsVisitingLocation(target: Element, url: string, originalEvent: MouseEvent): boolean {
const event = dispatch<TurboClickEvent>("turbo:click", {
target,
detail: { url, originalEvent },
cancelable: true,
})

return !event.defaultPrevented
}

private isIgnoringChangesTo(attributeName: FrameElementObservedAttribute): boolean {
return this.ignoredAttributes.has(attributeName)
}
Expand Down
39 changes: 19 additions & 20 deletions src/core/frames/frame_redirector.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,15 +2,15 @@ import { FormSubmitObserver, FormSubmitObserverDelegate } from "../../observers/
import { FrameElement } from "../../elements/frame_element"
import { LinkInterceptor, LinkInterceptorDelegate } from "./link_interceptor"
import { expandURL, getAction, locationIsVisitable } from "../url"
import { TurboClickEvent } from "../session"
import { dispatch } from "../../util"

import { Session } from "../session"
export class FrameRedirector implements LinkInterceptorDelegate, FormSubmitObserverDelegate {
readonly session: Session
readonly element: Element
readonly linkInterceptor: LinkInterceptor
readonly formSubmitObserver: FormSubmitObserver

constructor(element: Element) {
constructor(session: Session, element: Element) {
this.session = session
this.element = element
this.linkInterceptor = new LinkInterceptor(this, element)
this.formSubmitObserver = new FormSubmitObserver(this, element)
Expand All @@ -26,14 +26,14 @@ export class FrameRedirector implements LinkInterceptorDelegate, FormSubmitObser
this.formSubmitObserver.stop()
}

shouldInterceptLinkClick(element: Element, url: string, originalEvent: MouseEvent) {
return this.shouldRedirect(element) && this.frameAllowsVisitingLocation(element, url, originalEvent)
shouldInterceptLinkClick(element: Element, _location: string, _event: MouseEvent) {
return this.shouldRedirect(element)
}

linkClickIntercepted(element: Element, url: string, originalEvent: MouseEvent) {
linkClickIntercepted(element: Element, url: string, event: MouseEvent) {
const frame = this.findFrameElement(element)
if (frame) {
frame.delegate.linkClickIntercepted(element, url, originalEvent)
frame.delegate.linkClickIntercepted(element, url, event)
}
}

Expand All @@ -52,16 +52,6 @@ export class FrameRedirector implements LinkInterceptorDelegate, FormSubmitObser
}
}

private frameAllowsVisitingLocation(target: Element, url: string, originalEvent: MouseEvent): boolean {
const event = dispatch<TurboClickEvent>("turbo:click", {
target,
detail: { url, originalEvent },
cancelable: true,
})

return !event.defaultPrevented
}

private shouldSubmit(form: HTMLFormElement, submitter?: HTMLElement) {
const action = getAction(form, submitter)
const meta = this.element.ownerDocument.querySelector<HTMLMetaElement>(`meta[name="turbo-root"]`)
Expand All @@ -71,8 +61,17 @@ export class FrameRedirector implements LinkInterceptorDelegate, FormSubmitObser
}

private shouldRedirect(element: Element, submitter?: HTMLElement) {
const frame = this.findFrameElement(element, submitter)
return frame ? frame != element.closest("turbo-frame") : false
const isNavigatable =
element instanceof HTMLFormElement
? this.session.submissionIsNavigatable(element, submitter)
: this.session.elementIsNavigatable(element)

if (isNavigatable) {
const frame = this.findFrameElement(element, submitter)
return frame ? frame != element.closest("turbo-frame") : false
} else {
return false
}
}

private findFrameElement(element: Element, submitter?: HTMLElement) {
Expand Down
2 changes: 1 addition & 1 deletion src/core/session.ts
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@ export class Session
readonly scrollObserver = new ScrollObserver(this)
readonly streamObserver = new StreamObserver(this)
readonly formLinkClickObserver = new FormLinkClickObserver(this, document.documentElement)
readonly frameRedirector = new FrameRedirector(document.documentElement)
readonly frameRedirector = new FrameRedirector(this, document.documentElement)
readonly streamMessageRenderer = new StreamMessageRenderer()

drive = true
Expand Down
12 changes: 4 additions & 8 deletions src/tests/functional/frame_tests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -2,9 +2,9 @@ import { Page, test } from "@playwright/test"
import { assert, Assertion } from "chai"
import {
attributeForSelector,
cancelNextEvent,
hasSelector,
innerHTMLForSelector,
listenForEventOnTarget,
nextAttributeMutationNamed,
noNextAttributeMutationNamed,
nextBeat,
Expand Down Expand Up @@ -442,6 +442,7 @@ test("test navigating a frame from an outer form fires events", async ({ page })
})

test("test navigating a frame from an outer link fires events", async ({ page }) => {
await listenForEventOnTarget(page, "outside-frame-form", "turbo:click")
await page.click("#outside-frame-form")

await nextEventOnTarget(page, "outside-frame-form", "turbo:click")
Expand All @@ -456,14 +457,8 @@ test("test navigating a frame from an outer link fires events", async ({ page })
assert.equal(otherEvents.length, 0, "no more events")
})

test("test canceling a turbo:cilck event falls back to built-in browser navigation", async ({ page }) => {
await cancelNextEvent(page, "turbo:click")
await Promise.all([page.waitForNavigation(), page.click("#link-frame")])

assert.equal(pathname(page.url()), "/src/tests/fixtures/frames/frame.html")
})

test("test navigating a frame from an inner link fires events", async ({ page }) => {
await listenForEventOnTarget(page, "link-frame", "turbo:click")
await page.click("#link-frame")

await nextEventOnTarget(page, "link-frame", "turbo:click")
Expand All @@ -479,6 +474,7 @@ test("test navigating a frame from an inner link fires events", async ({ page })
})

test("test navigating a frame targeting _top from an outer link fires events", async ({ page }) => {
await listenForEventOnTarget(page, "outside-navigate-top-link", "turbo:click")
await page.click("#outside-navigate-top-link")

await nextEventOnTarget(page, "outside-navigate-top-link", "turbo:click")
Expand Down
12 changes: 12 additions & 0 deletions src/tests/helpers/page.ts
Original file line number Diff line number Diff line change
Expand Up @@ -100,6 +100,18 @@ export async function nextEventOnTarget(page: Page, elementId: string, eventName
return record[1]
}

export async function listenForEventOnTarget(page: Page, elementId: string, eventName: string): Promise<void> {
return page.locator("#" + elementId).evaluate((element, eventName) => {
const eventLogs = (window as any).eventLogs

element.addEventListener(eventName, ({ target, type }) => {
if (target instanceof Element) {
eventLogs.push([type, {}, target.id])
}
})
}, eventName)
}

export async function nextAttributeMutationNamed(
page: Page,
elementId: string,
Expand Down

0 comments on commit fa20fcd

Please sign in to comment.