With turbo-rails 7.3.0 (latest version at the time of this post):
Let’s say you use “advance” mode with Turbo frames which allows you to update a URL as you navigate between frames. This is very handy in a number of use cases such as having a sidebar with a main content area. Each sidebar link could render a new frame in the main content area to avoid rendering an expensive sidebar.
I ran into this exact use case with a video platform where I want to use frames to avoid reloading an <iframe> to have a permanent video that doesn’t lose its playback position and to also avoid rendering an expensive query to render a table of contents. Turbo frames feels like it was designed for this exact purpose!
The problem with the above is Turbo frames only swaps the frame (by nature) but when navigating between URLs you would expect the following <head> tags to be updated:
Not having these updates is kind of a deal breaker for using Turbo frames with “advance” where you change the URL since having those <head> fields not get updated could penalize you from Google and other crawlers. It also makes your site appear to be buggy. For example here’s an issue from the exercism.io site which uses frames in this way but doesn’t get updated properly Browser tab shows wrong title · Issue #6636 · exercism/exercism · GitHub.
Any thoughts or suggestions on how to handle this in a clean way?
Unfortunately it didn’t update upon page transition.
Although I am using a custom layout, if I explicitly reference this new frame layout then it updates.
My custom layout has:
<% content_for :content do %>
<main>
<p>A whole bunch of things are here, which I omit for the sake of this post.</p>
</main>
<% end %>
<%= render template: 'layouts/application' %>
If I replace application with turbo_rails/frame then it updates but then the pages are missing a bunch of content since it’s not using my custom layout.
Turbo still excludes everything outside of the frame itself, except for specific items like Turbo-specific meta tags that it checks for. So although this change sounds a bit like what you’re describing, it’s not really the same thing, unfortunately.
Okay, so, this is not a good practice, and is not approved by HTML standard, and might break in future browsers, but putting the new title tag inside the frame tag works:
# layouts/_tab.html.erb
<%= turbo_frame_tag :tab_content do %>
<head><title> <%= content_for :head_title %> </title></head>
<%= yield %>
<% end %>
Thanks. There’s also the meta description, rel canonical tag, opengraph tags and more that could change between transitioning pages with Turbo Frames.
I’m kind of surprised there’s no clean way to handle this yet at the library level since Turbo Frame page transitions with URL changes is a built-in feature of the library.
While this works for the title, it doesn’t seem to be working since the meta description tag has no “content” but a content attribute. I’m still looking for a solution here if anyone has one…
I haven’t tried this technique yet, but I thought about handling this with Stimulus. Have a data-controller on the <html> tag itself, a target on the <title> and any <meta> tags you want to target, and an action that listens for turbo:frame-render.
Now you just need to include the title/meta updates you want to make somewhere in the responding turbo-frame’s HTML. You could put some data attributes on any element in the response and have the turbo:frame-render listener look for those data attributes and update the targets (if the data attributes are found):
update(event) {
const frame = event.target
const title = frame.querySelector("[data-title]")?.dataSet?.title
const description = frame.querySelector("[data-description]")?.dataSet?.description
if (title) {
this.titleTarget.textContent = title
}
if (description) {
this.descriptionTarget.setAttribute("content", description)
}
}
You should be able to hook into turbo:before-stream-render as well and do the same thing with turbo-stream responses.