Turbolinks makes navigating your web application faster. Get the performance benefits of a single-page application without the added complexity of a client-side JavaScript framework. Use HTML to render your views on the server side and link to pages as usual. When you follow a link, Turbolinks automatically fetches the page, swaps in its <body>
, and merges its <head>
, all without incurring the cost of a full page load.
Developed for the all-new Basecamp 3, Turbolinks 5 is a complete rewrite that adds support for iOS and Android hybrid applications. This preview release is API-stable, but official documentation remains a work in progress.
- Add the
turbolinks
gem, version 5, to your Gemfile:gem 'turbolinks', '~> 5'
- Run
bundle install
. - Add
//= require turbolinks
to your JavaScript manifest file (usually found atapp/assets/javascripts/application.js
).
Simply include dist/turbolinks.js
in your app's JavaScript bundle.
At Basecamp, we’re big believers in the hybrid approach to building native applications: server-generated web views wrapped in, and enhanced by, native navigation controls. And Turbolinks’ page replacement strategy is the key ingredient to making our web views fast.
We were able to integrate Turbolinks into our hybrid apps for Basecamp 3, but not with the level of fidelity we expected from a native application. Eventually, we determined we’d need to redesign Turbolinks with more than just the browser in mind.
Rewriting Turbolinks from the ground up let us isolate the browser-specific behavior behind a pluggable adapter interface, with hooks in place for explicit control over every step of navigation. The rewrite also gave us a fresh slate to simplify the API, revisit past philosophical decisions, and trim some technical baggage.
Version 2.5.3 is the most recent official release of Turbolinks; version 3 has been in development for some time without an official release. We are preserving the original code base as Turbolinks Classic, and all existing issues and pull requests remain at turbolinks/turbolinks-classic on GitHub.
We wanted to signify that this rewrite is a major leap with a backwards-incompatible API. While we could have gone with version 4, we thought 5 had a nice ring to it, given that it coincides with the upcoming Rails 5 release.
Partial replacement is an API introduced in the unreleased Turbolinks 3 which allows you to pick and choose individual elements for replacement during navigation.
It’s our opinion that partial replacement is mostly orthogonal to the responsibilities of navigating between pages. Partial replacement drastically expands the scope of Turbolinks’ API, and we feel the tradeoff between performance and complexity it introduces is not in line with the Turbolinks philosophy.
However, we’ve kept Turbolinks 3’s concept of “permanent” elements via the data-turbolinks-permanent
annotation. Placing this annotation on any element with an ID allows that element to persist across page changes. We think this feature gives you 90% of the benefits of partial replacement with 10% of its complexity.
Consider using Turbolinks 5 now if you are starting work on a new application and don’t mind consulting the source code when something doesn’t work.
If you have an existing application built with Turbolinks Classic, you may wish to wait until the final release of Turbolinks 5 before upgrading.
Note that the API has changed. In particular, all of the Turbolinks Classic events have been renamed, and several—including page:update
—have been removed. We’ve made available a basic compatibility shim for Turbolinks Classic events for use during the transition.
Our iOS and Android adapters let you build hybrid apps which combine native navigation patterns with a single shared web view.
We plan on open-sourcing these adapters in the next few weeks. To see the adapters in action, check out our Basecamp 3 for iOS and Basecamp 3 for Android apps.
As with previous versions of Turbolinks, you may encounter problems with third-party libraries which do the following:
- Add event listeners directly to page elements (including
<body>
) - Install behavior on
DOMContentLoaded
orwindow.onload
To work around these issues, prefer using event delegation on document.documentElement
, document
, or window
, and consider using MutationObserver
to install behavior on elements as they’re added to the page.
Also note that libraries which feature Turbolinks Classic integration may not work as expected with Turbolinks 5.
Copyright © 2016 Basecamp, LLC