diff --git a/src/core/frames/frame_controller.js b/src/core/frames/frame_controller.js index 07764fc86..a7183006a 100644 --- a/src/core/frames/frame_controller.js +++ b/src/core/frames/frame_controller.js @@ -1,18 +1,12 @@ import { FrameElement, FrameLoadingStyle } from "../../elements/frame_element" -import { FetchMethod, FetchRequest } from "../../http/fetch_request" import { FetchResponse } from "../../http/fetch_response" import { AppearanceObserver } from "../../observers/appearance_observer" import { - clearBusyState, dispatch, getAttribute, - parseHTMLDocument, - markAsBusy, uuid, - getHistoryMethodForAction, - getVisitAction + getHistoryMethodForAction } from "../../util" -import { FormSubmission } from "../drive/form_submission" import { Snapshot } from "../snapshot" import { getAction, expandURL, urlsAreEqual, locationIsVisitable } from "../url" import { FormSubmitObserver } from "../../observers/form_submit_observer" @@ -21,18 +15,15 @@ import { LinkInterceptor } from "./link_interceptor" import { FormLinkClickObserver } from "../../observers/form_link_click_observer" import { FrameRenderer } from "./frame_renderer" import { session } from "../index" -import { StreamMessage } from "../streams/stream_message" import { PageSnapshot } from "../drive/page_snapshot" import { TurboFrameMissingError } from "../errors" +import { FrameVisit } from "./frame_visit" export class FrameController { - fetchResponseLoaded = (_fetchResponse) => Promise.resolve() - #currentFetchRequest = null - #resolveVisitPromise = () => {} #connected = false #hasBeenLoaded = false #ignoredAttributes = new Set() - action = null + #frameVisit = null constructor(element) { this.element = element @@ -70,6 +61,11 @@ export class FrameController { } } + visit(options) { + const frameVisit = new FrameVisit(this, this.element, options) + return frameVisit.start() + } + disabledChanged() { if (this.loadingStyle == FrameLoadingStyle.eager) { this.#loadSourceURL() @@ -107,39 +103,30 @@ export class FrameController { async #loadSourceURL() { if (this.enabled && this.isActive && !this.complete && this.sourceURL) { - this.element.loaded = this.#visit(expandURL(this.sourceURL)) - this.appearanceObserver.stop() - await this.element.loaded - this.#hasBeenLoaded = true + await this.visit({ url: this.sourceURL }) } } - async loadResponse(fetchResponse) { + async loadResponse(fetchResponse, frameVisit) { if (fetchResponse.redirected || (fetchResponse.succeeded && fetchResponse.isHTML)) { this.sourceURL = fetchResponse.response.url } - try { - const html = await fetchResponse.responseHTML - if (html) { - const document = parseHTMLDocument(html) - const pageSnapshot = PageSnapshot.fromDocument(document) - - if (pageSnapshot.isVisitable) { - await this.#loadFrameResponse(fetchResponse, document) - } else { - await this.#handleUnvisitableFrameResponse(fetchResponse) - } + const html = await fetchResponse.responseHTML + if (html) { + const pageSnapshot = PageSnapshot.fromHTMLString(html) + + if (pageSnapshot.isVisitable) { + await this.#loadFrameResponse(fetchResponse, pageSnapshot, frameVisit) + } else { + await this.#handleUnvisitableFrameResponse(fetchResponse) } - } finally { - this.fetchResponseLoaded = () => Promise.resolve() } } // Appearance observer delegate elementAppearedInViewport(element) { - this.proposeVisitIfNavigatedWithAction(element, getVisitAction(element)) this.#loadSourceURL() } @@ -171,81 +158,48 @@ export class FrameController { } formSubmitted(element, submitter) { - if (this.formSubmission) { - this.formSubmission.stop() - } - - this.formSubmission = new FormSubmission(this, element, submitter) - const { fetchRequest } = this.formSubmission - this.prepareRequest(fetchRequest) - this.formSubmission.start() - } - - // Fetch request delegate - - prepareRequest(request) { - request.headers["Turbo-Frame"] = this.id - - if (this.currentNavigationElement?.hasAttribute("data-turbo-stream")) { - request.acceptResponseType(StreamMessage.contentType) - } - } - - requestStarted(_request) { - markAsBusy(this.element) - } - - requestPreventedHandlingResponse(_request, _response) { - this.#resolveVisitPromise() - } - - async requestSucceededWithResponse(request, response) { - await this.loadResponse(response) - this.#resolveVisitPromise() - } - - async requestFailedWithResponse(request, response) { - await this.loadResponse(response) - this.#resolveVisitPromise() + const frame = this.#findFrameElement(element, submitter) + frame.delegate.visit(FrameVisit.optionsForSubmit(element, submitter)) } - requestErrored(request, error) { - console.error(error) - this.#resolveVisitPromise() - } + // Frame visit delegate - requestFinished(_request) { - clearBusyState(this.element) + shouldVisitFrame(_frameVisit) { + return this.enabled && this.isActive } - // Form submission delegate - - formSubmissionStarted({ formElement }) { - markAsBusy(formElement, this.#findFrameElement(formElement)) + frameVisitStarted(frameVisit) { + this.#ignoringChangesToAttribute("complete", () => { + this.#frameVisit?.stop() + this.#frameVisit = frameVisit + this.element.removeAttribute("complete") + }) } - formSubmissionSucceededWithResponse(formSubmission, response) { - const frame = this.#findFrameElement(formSubmission.formElement, formSubmission.submitter) + async frameVisitSucceededWithResponse(frameVisit, fetchResponse) { + await this.loadResponse(fetchResponse, frameVisit) - frame.delegate.proposeVisitIfNavigatedWithAction(frame, getVisitAction(formSubmission.submitter, formSubmission.formElement, frame)) - frame.delegate.loadResponse(response) - - if (!formSubmission.isSafe) { + if (!frameVisit.isSafe) { session.clearCache() } } - formSubmissionFailedWithResponse(formSubmission, fetchResponse) { - this.element.delegate.loadResponse(fetchResponse) + async frameVisitFailedWithResponse(frameVisit, fetchResponse) { + await this.loadResponse(fetchResponse, frameVisit) + session.clearCache() } - formSubmissionErrored(formSubmission, error) { + frameVisitErrored(_frameVisit, fetchRequest, error) { console.error(error) + dispatch("turbo:fetch-request-error", { + target: this.element, + detail: { request: fetchRequest, error } + }) } - formSubmissionFinished({ formElement }) { - clearBusyState(formElement, this.#findFrameElement(formElement)) + frameVisitCompleted(_frameVisit) { + this.hasBeenLoaded = true } // View delegate @@ -294,83 +248,54 @@ export class FrameController { // Private - async #loadFrameResponse(fetchResponse, document) { - const newFrameElement = await this.extractForeignFrameElement(document.body) + async #loadFrameResponse(fetchResponse, pageSnapshot, frameVisit) { + const newFrameElement = await this.extractForeignFrameElement(pageSnapshot.element) 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() + this.changeHistory(frameVisit.action) await this.view.render(renderer) this.complete = true session.frameRendered(fetchResponse, this.element) session.frameLoaded(this.element) - await this.fetchResponseLoaded(fetchResponse) + await this.#proposeVisitIfNavigatedWithAction(frameVisit, fetchResponse) } else if (this.#willHandleFrameMissingFromResponse(fetchResponse)) { this.#handleFrameMissingFromResponse(fetchResponse) } } - async #visit(url) { - const request = new FetchRequest(this, FetchMethod.get, url, new URLSearchParams(), this.element) - - this.#currentFetchRequest?.cancel() - this.#currentFetchRequest = request - - return new Promise((resolve) => { - this.#resolveVisitPromise = () => { - this.#resolveVisitPromise = () => {} - this.#currentFetchRequest = null - resolve() - } - request.perform() - }) - } - - #navigateFrame(element, url, submitter) { - const frame = this.#findFrameElement(element, submitter) - - frame.delegate.proposeVisitIfNavigatedWithAction(frame, getVisitAction(submitter, element, frame)) - - this.#withCurrentNavigationElement(element, () => { - frame.src = url - }) + #navigateFrame(element, url) { + const frame = this.#findFrameElement(element) + frame.delegate.visit(FrameVisit.optionsForClick(element, expandURL(url))) } - proposeVisitIfNavigatedWithAction(frame, action = null) { - this.action = action - - if (this.action) { - const pageSnapshot = PageSnapshot.fromElement(frame).clone() - const { visitCachedSnapshot } = frame.delegate + async #proposeVisitIfNavigatedWithAction(frameVisit, fetchResponse) { + const { frameElement } = frameVisit - frame.delegate.fetchResponseLoaded = async (fetchResponse) => { - if (frame.src) { - const { statusCode, redirected } = fetchResponse - const responseHTML = await fetchResponse.responseHTML - const response = { statusCode, redirected, responseHTML } - const options = { - response, - visitCachedSnapshot, - willRender: false, - updateHistory: false, - restorationIdentifier: this.restorationIdentifier, - snapshot: pageSnapshot - } - - if (this.action) options.action = this.action - - session.visit(frame.src, options) - } + if (frameElement.src && frameVisit.action) { + const { statusCode, redirected } = fetchResponse + const responseHTML = await fetchResponse.responseHTML + const response = { statusCode, redirected, responseHTML } + const options = { + response, + visitCachedSnapshot: frameElement.delegate.visitCachedSnapshot, + willRender: false, + updateHistory: false, + restorationIdentifier: this.restorationIdentifier, + snapshot: frameVisit.snapshot, + action: frameVisit.action } + + session.visit(frameElement.src, options) } } - changeHistory() { - if (this.action) { - const method = getHistoryMethodForAction(this.action) + changeHistory(visitAction) { + if (visitAction) { + const method = getHistoryMethodForAction(visitAction) session.history.update(method, expandURL(this.element.src || ""), this.restorationIdentifier) } } @@ -384,7 +309,7 @@ export class FrameController { } #willHandleFrameMissingFromResponse(fetchResponse) { - this.element.setAttribute("complete", "") + this.complete = true const response = fetchResponse.response const visit = async (url, options) => { @@ -512,7 +437,7 @@ export class FrameController { } get isLoading() { - return this.formSubmission !== undefined || this.#resolveVisitPromise() !== undefined + return !!this.#frameVisit } get complete() { @@ -546,12 +471,6 @@ export class FrameController { callback() this.#ignoredAttributes.delete(attributeName) } - - #withCurrentNavigationElement(element, callback) { - this.currentNavigationElement = element - callback() - delete this.currentNavigationElement - } } function getFrameElementById(id) { diff --git a/src/core/frames/frame_visit.js b/src/core/frames/frame_visit.js new file mode 100644 index 000000000..834a7494a --- /dev/null +++ b/src/core/frames/frame_visit.js @@ -0,0 +1,152 @@ +import { expandURL } from "../url" +import { clearBusyState, getVisitAction, markAsBusy } from "../../util" +import { FetchMethod, FetchRequest } from "../../http/fetch_request" +import { FormSubmission } from "../drive/form_submission" +import { PageSnapshot } from "../drive/page_snapshot" +import { StreamMessage } from "../streams/stream_message" + +export class FrameVisit { + isFormSubmission = false + #resolveVisitPromise = () => {} + + static optionsForClick(element, url) { + const action = getVisitAction(element) + const acceptsStreamResponse = element.hasAttribute("data-turbo-stream") + + return { acceptsStreamResponse, action, url } + } + + static optionsForSubmit(form, submitter) { + const action = getVisitAction(form, submitter) + + return { action, submit: { form, submitter } } + } + + constructor(delegate, frameElement, options) { + this.delegate = delegate + this.frameElement = frameElement + this.previousURL = this.frameElement.src + + const { acceptsStreamResponse, action, url, submit } = (this.options = options) + + this.acceptsStreamResponse = acceptsStreamResponse || false + this.action = action || getVisitAction(this.frameElement) + + if (submit) { + const { fetchRequest } = (this.formSubmission = new FormSubmission(this, submit.form, submit.submitter)) + this.prepareRequest(fetchRequest) + this.isFormSubmission = true + this.isSafe = this.formSubmission.isSafe + } else if (url) { + this.fetchRequest = new FetchRequest(this, FetchMethod.get, expandURL(url), new URLSearchParams(), this.frameElement) + this.isSafe = true + } else { + throw new Error("FrameVisit must be constructed with either a url: or submit: option") + } + } + + async start() { + if (this.delegate.shouldVisitFrame(this)) { + if (this.action) { + this.snapshot = PageSnapshot.fromElement(this.frameElement).clone() + } + + if (this.formSubmission) { + await this.formSubmission.start() + } else { + await this.#performRequest() + } + + return this.frameElement.loaded + } else { + return Promise.resolve() + } + } + + stop() { + this.fetchRequest?.cancel() + this.formSubmission?.stop() + } + + // Fetch request delegate + + prepareRequest(fetchRequest) { + fetchRequest.headers["Turbo-Frame"] = this.frameElement.id + + if (this.acceptsStreamResponse || this.isFormSubmission) { + fetchRequest.acceptResponseType(StreamMessage.contentType) + } + } + + requestStarted(fetchRequest) { + this.delegate.frameVisitStarted(this) + + if (fetchRequest.target instanceof HTMLFormElement) { + markAsBusy(fetchRequest.target) + } + + markAsBusy(this.frameElement) + } + + requestPreventedHandlingResponse(_fetchRequest, _fetchResponse) { + this.#resolveVisitPromise() + } + + requestFinished(fetchRequest) { + clearBusyState(this.frameElement) + + if (fetchRequest.target instanceof HTMLFormElement) { + clearBusyState(fetchRequest.target) + } + + this.delegate.frameVisitCompleted(this) + } + + async requestSucceededWithResponse(_fetchRequest, fetchResponse) { + await this.delegate.frameVisitSucceededWithResponse(this, fetchResponse) + this.#resolveVisitPromise() + } + + async requestFailedWithResponse(_fetchRequest, fetchResponse) { + console.error(fetchResponse) + await this.delegate.frameVisitFailedWithResponse(this, fetchResponse) + this.#resolveVisitPromise() + } + + requestErrored(fetchRequest, error) { + this.delegate.frameVisitErrored(this, fetchRequest, error) + this.#resolveVisitPromise() + } + + // Form submission delegate + + formSubmissionStarted(formSubmission) { + this.requestStarted(formSubmission.fetchRequest) + } + + async formSubmissionSucceededWithResponse(formSubmission, fetchResponse) { + await this.requestSucceededWithResponse(formSubmission.fetchRequest, fetchResponse) + } + + async formSubmissionFailedWithResponse(formSubmission, fetchResponse) { + await this.requestFailedWithResponse(formSubmission.fetchRequest, fetchResponse) + } + + formSubmissionErrored(formSubmission, error) { + this.requestErrored(formSubmission.fetchRequest, error) + } + + formSubmissionFinished(formSubmission) { + this.requestFinished(formSubmission.fetchRequest) + } + + #performRequest() { + this.frameElement.loaded = new Promise((resolve) => { + this.#resolveVisitPromise = () => { + this.#resolveVisitPromise = () => {} + resolve() + } + this.fetchRequest?.perform() + }) + } +} diff --git a/src/core/session.js b/src/core/session.js index cdb978348..bde580993 100644 --- a/src/core/session.js +++ b/src/core/session.js @@ -98,10 +98,10 @@ export class Session { const frameElement = options.frame ? document.getElementById(options.frame) : null if (frameElement instanceof FrameElement) { - const action = options.action || getVisitAction(frameElement) - - frameElement.delegate.proposeVisitIfNavigatedWithAction(frameElement, action) - frameElement.src = location.toString() + frameElement.delegate.visit({ + url: location.toString(), + action: options.action || getVisitAction(frameElement) + }) } else { this.navigator.proposeVisit(expandURL(location), options) } diff --git a/src/tests/fixtures/tabs/three.html b/src/tests/fixtures/tabs/three.html index 517d0dc2b..f2749777a 100644 --- a/src/tests/fixtures/tabs/three.html +++ b/src/tests/fixtures/tabs/three.html @@ -2,11 +2,13 @@
-