Skip to content

Commit

Permalink
Allow Turbo Streams w/ GET via data-turbo-stream (#612)
Browse files Browse the repository at this point in the history
Turbo Streams are normally supported only for [non-GET requests][0].
However there are cases where Turbo Streams responses to GET requests
are useful.

This commit adds the ability to use Turbo Streams with specific GET
requests by setting `data-turbo-stream="true"` on a form or link.

[0]: #52
  • Loading branch information
kevinmcconnell authored Jul 14, 2022
1 parent 98cdc40 commit bdad71e
Show file tree
Hide file tree
Showing 6 changed files with 66 additions and 9 deletions.
11 changes: 10 additions & 1 deletion src/core/drive/form_submission.ts
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { FetchRequest, FetchMethod, fetchMethodFromString, FetchRequestHeaders } from "../../http/fetch_request"
import { FetchResponse } from "../../http/fetch_response"
import { expandURL } from "../url"
import { dispatch } from "../../util"
import { attributeTrue, dispatch } from "../../util"
import { StreamMessage } from "../streams/stream_message"

export interface FormSubmissionDelegate {
Expand Down Expand Up @@ -153,6 +153,9 @@ export class FormSubmission {
if (token) {
headers["X-CSRF-Token"] = token
}
}

if (this.requestAcceptsTurboStreamResponse(request)) {
headers["Accept"] = [StreamMessage.contentType, headers["Accept"]].join(", ")
}
}
Expand Down Expand Up @@ -204,9 +207,15 @@ export class FormSubmission {
this.delegate.formSubmissionFinished(this)
}

// Private

requestMustRedirect(request: FetchRequest) {
return !request.isIdempotent && this.mustRedirect
}

requestAcceptsTurboStreamResponse(request: FetchRequest) {
return !request.isIdempotent || attributeTrue(this.formElement, "data-turbo-stream")
}
}

function buildFormData(formElement: HTMLFormElement, submitter?: HTMLElement): FormData {
Expand Down
4 changes: 2 additions & 2 deletions src/core/frames/frame_controller.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@ import {
import { FetchMethod, FetchRequest, FetchRequestDelegate, FetchRequestHeaders } from "../../http/fetch_request"
import { FetchResponse } from "../../http/fetch_response"
import { AppearanceObserver, AppearanceObserverDelegate } from "../../observers/appearance_observer"
import { clearBusyState, getAttribute, parseHTMLDocument, markAsBusy } from "../../util"
import { clearBusyState, getAttribute, parseHTMLDocument, markAsBusy, attributeTrue } from "../../util"
import { FormSubmission, FormSubmissionDelegate } from "../drive/form_submission"
import { Snapshot } from "../snapshot"
import { ViewDelegate } from "../view"
Expand Down Expand Up @@ -149,7 +149,7 @@ export class FrameController
// Link interceptor delegate

shouldInterceptLinkClick(element: Element, _url: string) {
if (element.hasAttribute("data-turbo-method")) {
if (element.hasAttribute("data-turbo-method") || attributeTrue(element, "data-turbo-stream")) {
return false
} else {
return this.shouldInterceptNavigation(element)
Expand Down
16 changes: 10 additions & 6 deletions src/core/session.ts
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ import { ScrollObserver } from "../observers/scroll_observer"
import { StreamMessage } from "./streams/stream_message"
import { StreamObserver } from "../observers/stream_observer"
import { Action, Position, StreamSource, isAction } from "./types"
import { clearBusyState, dispatch, markAsBusy } from "../util"
import { attributeTrue, clearBusyState, dispatch, markAsBusy } from "../util"
import { PageView, PageViewDelegate } from "./drive/page_view"
import { Visit, VisitOptions } from "./drive/visit"
import { PageSnapshot } from "./drive/page_snapshot"
Expand Down Expand Up @@ -165,16 +165,20 @@ export class Session

convertLinkWithMethodClickToFormSubmission(link: Element) {
const linkMethod = link.getAttribute("data-turbo-method")
const useTurboStream = attributeTrue(link, "data-turbo-stream")

if (linkMethod) {
if (linkMethod || useTurboStream) {
const form = document.createElement("form")
form.setAttribute("method", linkMethod)
form.setAttribute("method", linkMethod || "get")
form.action = link.getAttribute("href") || "undefined"
form.hidden = true

if (link.hasAttribute("data-turbo-confirm")) {
form.setAttribute("data-turbo-confirm", link.getAttribute("data-turbo-confirm")!)
}
const attributes = ["data-turbo-confirm", "data-turbo-stream"]
attributes.forEach((attribute) => {
if (link.hasAttribute(attribute)) {
form.setAttribute(attribute, link.getAttribute(attribute)!)
}
})

const frame = this.getTargetFrameForLink(link)
if (frame) {
Expand Down
8 changes: 8 additions & 0 deletions src/tests/fixtures/form.html
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,11 @@ <h1>Form</h1>
<input type="hidden" name="greeting" value="Hello from a redirect">
<input id="standard-get-form-submit" type="submit" value="form[method=get]">
</form>
<form action="/__turbo/redirect" method="get" data-turbo-stream="true" class="redirect">
<input type="hidden" name="path" value="/src/tests/fixtures/form.html">
<input type="hidden" name="greeting" value="Hello from a redirect">
<input id="standard-get-form-with-stream-opt-in-submit" type="submit" value="form[method=get]">
</form>
<hr>
<form>
<button id="form-action-none-q-a" name="q" value="a">Submit ?q=a to form:not([action])</button>
Expand Down Expand Up @@ -254,6 +259,8 @@ <h2>Frame: Form</h2>
<a href="/src/tests/fixtures/frames/hello.html" data-turbo-method="get" data-turbo-frame="_top" id="link-method-inside-frame-target-top">Break-out of frame with method link inside frame</a><br />
<a href="/src/tests/fixtures/frames/hello.html" data-turbo-method="get" data-turbo-frame="hello" id="link-method-inside-frame-with-target">Method link inside frame targeting another frame</a><br />
<a href="/__turbo/messages?content=Link!&type=stream" data-turbo-method="post" id="stream-link-method-inside-frame">Stream link inside frame</a>
<a href="/__turbo/messages?content=Link!&type=stream" data-turbo-method="get" data-turbo-stream="true" id="stream-link-get-method-inside-frame">Stream link GET inside frame</a>
<a href="/__turbo/messages?content=Link!&type=stream" data-turbo-stream="true" id="stream-link-inside-frame">Stream link (no method) inside frame</a>
<a href="/__turbo/messages?content=Link!&type=stream" data-turbo-method="post" data-turbo-confirm="Are you sure?" id="link-method-inside-frame-with-confirmation"data-turbo-confirm="Are you sure?">Stream link inside frame with confirmation</a>
<form>
<a href="/src/tests/fixtures/frames/frame.html" data-turbo-method="get" id="method-link-within-form-inside-frame">Method link within form inside frame</a><br />
Expand Down Expand Up @@ -283,6 +290,7 @@ <h2>Frame: Form</h2>
</turbo-frame>
<a href="/src/tests/fixtures/frames/hello.html" data-turbo-method="get" id="link-method-outside-frame">Method link outside frame</a><br />
<a href="/__turbo/messages?content=Link!&type=stream" data-turbo-method="post" id="stream-link-method-outside-frame">Stream link outside frame</a>
<a href="/__turbo/messages?content=Link!&type=stream" data-turbo-stream="true" id="stream-link-outside-frame">Stream link (no method) outside frame</a>
<form>
<a href="/src/tests/fixtures/frames/hello.html" data-turbo-method="get" id="link-method-within-form-outside-frame">Method link within form outside frame</a><br />
<a href="/__turbo/messages?content=Link!&type=stream" data-turbo-method="post" id="stream-link-method-within-form-outside-frame">Stream link within form outside frame</a>
Expand Down
32 changes: 32 additions & 0 deletions src/tests/functional/form_submission_tests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -150,6 +150,14 @@ export class FormSubmissionTests extends TurboDriveTestCase {
this.assert.equal(await this.getSearchParam("greeting"), "Hello from a form")
}

async "test standard GET form submission with data-turbo-stream"() {
await this.clickSelector("#standard-get-form-with-stream-opt-in-submit")

const { fetchOptions } = await this.nextEventNamed("turbo:before-fetch-request")

this.assert.ok(fetchOptions.headers["Accept"].includes("text/vnd.turbo-stream.html"))
}

async "test standard GET form submission events"() {
await this.clickSelector("#standard-get-form-submit")

Expand Down Expand Up @@ -800,6 +808,30 @@ export class FormSubmissionTests extends TurboDriveTestCase {
this.assert.equal(await message.getVisibleText(), "Link!")
}

async "test stream link GET method form submission inside frame"() {
await this.clickSelector("#stream-link-get-method-inside-frame")

const { fetchOptions } = await this.nextEventNamed("turbo:before-fetch-request")

this.assert.ok(fetchOptions.headers["Accept"].includes("text/vnd.turbo-stream.html"))
}

async "test stream link inside frame"() {
await this.clickSelector("#stream-link-inside-frame")

const { fetchOptions } = await this.nextEventNamed("turbo:before-fetch-request")

this.assert.ok(fetchOptions.headers["Accept"].includes("text/vnd.turbo-stream.html"))
}

async "test stream link outside frame"() {
await this.clickSelector("#stream-link-outside-frame")

const { fetchOptions } = await this.nextEventNamed("turbo:before-fetch-request")

this.assert.ok(fetchOptions.headers["Accept"].includes("text/vnd.turbo-stream.html"))
}

async "test link method form submission within form inside frame"() {
await this.clickSelector("#stream-link-method-within-form-inside-frame")
await this.nextBeat
Expand Down
4 changes: 4 additions & 0 deletions src/util.ts
Original file line number Diff line number Diff line change
Expand Up @@ -92,3 +92,7 @@ export function clearBusyState(...elements: Element[]) {
element.removeAttribute("aria-busy")
}
}

export function attributeTrue(element: Element, attributeName: string) {
return element.getAttribute(attributeName) === "true"
}

0 comments on commit bdad71e

Please sign in to comment.