Skip to content

Last updated: First published:

Managing App State with <ReplacementSwap/>

This page gives some details on how The Bag keeps Starlight’s app state up to date during view transitions.

Visit the corresponding section in the overview to see where we are in the big picture of The Bag’s Starlight support.

The Tightly Coupled approach

As you know from the section on updating script state adding the <ClientRouter /> component is the simpler part. Ensuring that script state is up to date might be more of a challenge. From what you saw there, it is best practice to use the lifecycle events to update the state of your scripts.

Using lifecycle events assumes that you add them to your own code. Or at least code you can control. I have no idea about the details of the scripts running inside Starlight. And I definitively do not want to change them. It would be possible to deeply integrate support for view transitions into Starlight, but that would be a pretty tight coupling. This could be a burden for both projects, especially since the 👜 Bag of Tricks ✨ is still young and not very stable yet.

The Loosely Coupled Solution

So let’s turn the tables and look for a solution that is more loosely coupled: The core problem with script states and DOM updates is that the scripts don’t recognize when the DOM is replaced with a new version. They still point to the old elements and have no idea about the new ones. What if we could avoid this situation altogether?

The key assumption here is that the only thing that changes when you navigate a Starlight site is the main content area and the page’s table of contents to the right.

Static and dynamic areas of a Starlight site

Static and dynamic areas of a Starlight site

The other assumption is that all scrips reference elements in the application frame but not in the main content area or table of content.

The Assumptions

If these two assumptions are true, we have no problems with the script state and do not need script updates.

  1. Only the main content area and page table of contents changes during navigation.
  2. Script state does not depend on elements in those areas.

Keeping the Frame Part of the DOM …

During swap() we would simply copy the application frame to the next page. Doing this with transition:persist requires a deep knowledge of the structure of a Starlight page and is more vulnerable to changes in that structure.

… with <ReplacementSwap>

The much better way is not to copy the frame but to leave it as is and just swap the main content area and the table of contents. For Starlight this is even more simple: There is a single <div> with class .main-frame which holds both.

So the solution is to use The Bag’s <ReplacementSwap> component and just swap div.main-frame on view transitions.

This automatically also adds some positive effects to Starlight as the app state now survives navigation.

  • If you have a long side bar thescroll position in that sidebar does not get lost on navigation
  • When you revisit the search bar, it remembered your last search!
<ReplacementSwap rootAttributesToPreserve="data-theme" />

See this line in its context in `astro-vtbot/components/starlight/Base.astro.

The rootAttributesToPreserve attribute prevents Starlight’s current theme flag on the <html> element from being overridden on transitions.

Setting data-vtbot-replace="main"

The <ReplacementSwap> component needs to be told what part of the DOM to replace. And with what. We could add the required data-vtbot-replace attribute to Starlight’s built-in PageFrame component or override that component with out patched custom version. This all would make a lot of work when updating to the next Starlight version.

The solution that is implemented in The Bag’s Starlight support is to add the data-vtbot-replace="main" attribute dynamically after the loader has fetched the new DOM. Here is the function that does that:

astro-vtbot/components/starlight/StarlightConnector.astro
function markMainFrameForReplacementSwap(doc: Document) {
doc.body.querySelector(STARLIGHT_MAIN_FRAME)?.setAttribute('data-vtbot-replace', 'main');
}

Revealed: This reveals what Lines 12 & 13 of the StarlightConnector do.

There is one caveat in this solution: it does not work for view transitions from outside your Starlight site. This is even true for navigation between two Starlight sites. The reason is that <ReplacementSwap/> does not find an initialized app frame on the ingoing edge and falls back to Astro’s built-in swap(). The solution would be to do a full reload on transitions to your sit from the outside. Component support coming soon.

Conclusion

So now we have a solution to keep the app state of a Starlight site up to date across navigation, where we do not need to change any of Starlight’s scripts, and we do not even need to know what they are and what they do.

This is very important for robustness and maintainability of the solution.

Spoiler: The first assumptions does not hold when we navigate to pages with a different template, especially the splash template. Read more about the consequences on the next page!