Skip to content

Commit

Permalink
Delay Visit page scrolling
Browse files Browse the repository at this point in the history
Closes hotwired#400

Before finishing a Visit by scrolling to an element, wait for an
additional animation frame (with the `nextAnimationFrame()` utility
which calls [requestAnimationFrame][] behind the scenes).

[requestAnimationFrame]: https://developer.mozilla.org/en-US/docs/Web/API/window/requestAnimationFrame
  • Loading branch information
seanpdoyle committed Nov 27, 2021
1 parent aa9724d commit 33f562b
Show file tree
Hide file tree
Showing 5 changed files with 83 additions and 2 deletions.
6 changes: 4 additions & 2 deletions src/core/drive/visit.ts
Original file line number Diff line number Diff line change
Expand Up @@ -218,6 +218,7 @@ export class Visit implements FetchRequestDelegate {
if (this.view.renderPromise) await this.view.renderPromise
if (isSuccessful(statusCode) && responseHTML != null) {
await this.view.renderPage(PageSnapshot.fromHTMLString(responseHTML), false, this.willRender)
await this.performScroll()
this.adapter.visitRendered(this)
this.complete()
} else {
Expand Down Expand Up @@ -260,6 +261,7 @@ export class Visit implements FetchRequestDelegate {
} else {
if (this.view.renderPromise) await this.view.renderPromise
await this.view.renderPage(snapshot, isPreview, this.willRender)
await this.performScroll()
this.adapter.visitRendered(this)
if (!isPreview) {
this.complete()
Expand All @@ -283,6 +285,7 @@ export class Visit implements FetchRequestDelegate {
if (this.isSamePage) {
this.render(async () => {
this.cacheSnapshot()
await this.performScroll()
this.adapter.visitRendered(this)
})
}
Expand Down Expand Up @@ -329,7 +332,7 @@ export class Visit implements FetchRequestDelegate {

// Scrolling

performScroll() {
async performScroll() {
if (!this.scrolled) {
if (this.action == "restore") {
this.scrollToRestoredPosition() || this.scrollToAnchor() || this.view.scrollToTop()
Expand Down Expand Up @@ -408,7 +411,6 @@ export class Visit implements FetchRequestDelegate {
})
await callback()
delete this.frame
this.performScroll()
}

cancelRender() {
Expand Down
20 changes: 20 additions & 0 deletions src/tests/fixtures/scroll/one.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Scroll: One</title>
<script src="/dist/turbo.es2017-umd.js" data-turbo-track="reload"></script>
<script src="/src/tests/fixtures/test.js"></script>
<style>
html { scroll-behavior: smooth; }
.push-below-fold { margin-top: 100vh; }
</style>
</head>
<body>
<h1>Scroll: One</h1>
<hr class="push-below-fold">
<a id="one-below-fold" href="/src/tests/fixtures/scroll/two.html#two-below-fold">two.html#two-below-fold</a>
<hr class="push-below-fold">
<hr class="push-below-fold">
</body>
</html>
21 changes: 21 additions & 0 deletions src/tests/fixtures/scroll/two.html
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8">
<title>Scroll: Two</title>
<script src="/dist/turbo.es2017-umd.js" data-turbo-track="reload"></script>
<script src="/src/tests/fixtures/test.js"></script>
<style>
html { scroll-behavior: smooth; }
.push-below-fold { margin-top: 100vh; }
</style>
</head>
<body>
<h1>Scroll: Two</h1>
<hr class="push-below-fold">
<a id="two-below-fold" href="/src/tests/fixtures/scroll/one.html#one-below-fold">one.html#one-below-fold</a>
<hr class="push-below-fold">
<hr class="push-below-fold">
<hr class="push-below-fold">
</body>
</html>
7 changes: 7 additions & 0 deletions src/tests/fixtures/visit.html
Original file line number Diff line number Diff line change
Expand Up @@ -5,13 +5,20 @@
<title>Turbo</title>
<script src="/dist/turbo.es2017-umd.js" data-turbo-track="reload"></script>
<script src="/src/tests/fixtures/test.js"></script>
<style>
.push-below-fold { margin-top: 100vh; }
</style>
</head>
<body>
<section>
<h1>Visit</h1>
<p><a id="same-origin-link" href="/src/tests/fixtures/one.html">Same-origin link</a></p>
<p><a id="same-origin-link-search-params" href="/src/tests/fixtures/one.html?key=value">Same-origin link with ?key=value</a></p>
<p><a id="sample-response" href="/src/tests/fixtures/one.html">Sample response</a></p>
<p><a id="same-page-link" href="/src/tests/fixtures/visit.html">Same page link</a></p>
<hr class="push-below-fold">
<p><a id="below-the-fold-link" href="/src/tests/fixtures/one.html">one.html</a></p>
<hr class="push-below-fold">
</section>
</body>
</html>
31 changes: 31 additions & 0 deletions src/tests/functional/visit_tests.ts
Original file line number Diff line number Diff line change
Expand Up @@ -120,6 +120,37 @@ export class VisitTests extends TurboDriveTestCase {
this.assert.notOk(await this.hasSelector("some-cached-element"))
}

async "test can scroll to element after click-initiated turbo:visit"() {
const id = "below-the-fold-link"
await this.evaluate((id: string) => {
addEventListener("turbo:load", () => document.getElementById(id)?.scrollIntoView())
}, id)

this.assert(await this.isScrolledToTop, "starts unscrolled")

await this.clickSelector("#same-page-link")
await this.nextEventNamed("turbo:load")
await this.nextBeat

this.assert(await this.isScrolledToSelector("#" + id), "scrolls after click-initiated turbo:load")
}

async "test can scroll to element after history-initiated turbo:visit"() {
const id = "below-the-fold-link"
await this.evaluate((id: string) => {
addEventListener("turbo:load", () => document.getElementById(id)?.scrollIntoView())
}, id)

await this.scrollToSelector("#" + id)
await this.clickSelector("#" + id)
await this.nextEventNamed("turbo:load")
await this.goBack()
await this.nextEventNamed("turbo:load")
await this.nextBeat

this.assert(await this.isScrolledToSelector("#" + id), "scrolls after history-initiated turbo:load")
}

async visitLocation(location: string) {
this.remote.execute((location: string) => window.Turbo.visit(location), [location])
}
Expand Down

0 comments on commit 33f562b

Please sign in to comment.