Build a Zero JavaScript User Experience ;)
ZSX is a dependency free progressive enhancement library for server-rendered web applications. Build slick web apps with semantic HTML, CSS, links and forms — In any backend language.
Start with HTML and end with an amazing user experience!
<!-- Swap out the element from the response of a link click into the page -->
<a href="./zxswap.html?message=hello" zx-swap="#targetContent">Say Hello</a>
ZSX is spiritually closest to Unpoly, and similar to frameworks like HTMX, Twinspark, and Turbo. However, it has an opinionated feature set for building highly maintainable yet interactive web applications. ZSX is more like a toolbox of HTML/HTTP enhancements, rather than a client side framework.
ChartSQL Studio Editor was the original source of ZSX and funds its development.
- Quick Start
- Features
- HTML API
- Events API
- CSS Reference
- ZSX Design Goals
- Developing Applications
- Cookbook
- Common Issues
- Roadmap
Include the zsx.js script. It can be in the head or end of the body. There is zsx.css
which contains styles and animations for visual elements like zx-loader
.
<head>
<!-- Style sheet for ZSX loading indicator styles -->
<link rel="stylesheet" href="dist/zsx.css">
<script src="dist/zsx.js"></script>
</head>
<body>
<!-- ... You can also include the script in the body -->
<script src="dist/zsx.js"></script>
</body>
<body>
<!-- set zx-script-skip="true" so that body swaps do not execute this again -->
<script zx-script-skip="true">
let zsxjs = new ZsxJs();
zsxjs.init(document, {
/* options */
});
</script>
</body>
↑ top
<!--- SERVER SIDE RENDERED HANDLEBARS TEMPLATE --->
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<link rel="stylesheet" href="dist/zsx.css">
<script src="dist/zsx.js"></script>
<title>ZSX - QuickStart</title>
</head>
<body>
<h1>ZSX QuickStart</h1>
<a href="?hello=true" zx-swap="#targetContent">Hello</a>
<a href="?hello=false" zx-swap="#targetContent">Good Bye</a>
<div id="targetContent" style="width: 300px; height:100px;">
{{#if url.hello}}
Hello World
{{else}}
Goodbye World
{{/if}}
</div>
<script zx-script-skip="true">
let zsxjs = new ZsxJs();
zsxjs.init(document, {
/* options */
});
</script>
</body>
</html>
↑ top
ZSX upgrades links, forms and buttons to make server rendered applications more responsive and user friendly.
ZSX adds improvements across three areas: Page Fragment Updates, Enhanced Interactivity, and Navigation & State Management. Together these features allow developers to build slick server rendered applications with HTML markup.
Page Fragment Updates: Updating the DOM without full page reloads
Enhanced Interactivity: Improving the user's visual experience
- Page Jump Supression -
zx-jump-guard
- Visual Loading Indicators —
zx-loader
- Scroll Elements Into View —
zx-scroll-to
- Action Confirmation Dialogs —
zx-dialog-confirm
- App-Style Links —
zx-link-mode
Navigation and State Management: Maintaining the user's state
Dynamically update parts of the page by swapping elements based on their ID, class, or tag in response to link clicks or form submissions.
The content swap feature allows you to update portions of your page without a full page reload, enhancing performance and user experience.
See zx-swap
↑ top | Features | next section → HTML API
When swapping content, keep client side elements, javascript, canvas, or media content that should not be changed. You can mark which child elements need to be maintained.
See zx-keep
↑ top | Features | next section → HTML API
Minimize disruptive page jumps from removed content.
When elements are removed from the DOM, the page may abruptly jump upward if the combined height of the remaining content and the viewport is less than the current scroll position. ZSX can guard against these jumps.
See zx-jump-guard
↑ top | Features | next section → HTML API
Easily add loading indicators to all of your links and buttons to improve responsiveness
Links
Links get a background color that moves from left to right to simulate a progress loading indicator
Buttons
Buttons get a background color that moves from left to right to simulate a progress loading indicator
See zx-loader
↑ top | Features | next section → HTML API
Scroll newly swapped elements into the viewport. You can define custom scroll targets to ensure important content is brought into focus.
See zx-scroll-to
↑ top | Features | next section → HTML API
Synchronize all links on the page to match the parameters of the clicked link.
URL syncrhonization allows you to swap small portions of the page, but ensure all other links on the page match the correct parameters.
See zx-sync-params
↑ top | Features | next section → HTML API
All links and forms that redirect to GET update the browser history and allow the user to navigate back to previous URLs.
This feature is automatic and does not require any zx-attributes
↑ top | Features | next section → HTML API
Present a confirmation dialog before proceeding with destructive actions.
↑ top | Features | next section → HTML API
Convert traditional <a>
tags into app-like buttons or interactive controls that blend seamlessly into your application's interface. App Links prevent the display of URL tooltips and the standard context menu, providing a cleaner, more cohesive user experience for full screen apps.
See zx-link-mode
↑ top | Features | next section → HTML API
ZSX works by adding attributes like zx-swap
to your existing HTML markup.
attribute | Description |
---|---|
zx-dialog-confirm | Confirmation question before proceeding with the click |
zx-jump-guard | Guards against page jumps from removed content |
zx-keep | Keeps specified content in the DOM after a parent element is swapped |
zx-link-mode | Whether to render the link as a browser (default) link or an application clickable element |
zx-loader | Specify that the link or button should have a loading indicator |
zx-script-skip | Whether to ignore executing a script tag during a swap |
zx-scroll-to | Where to sroll to after the content swap |
zx-swap | Swaps target selector content from the response of a link click or form post |
zx-sync-params | Syncronizes URL parameters across links after a click |
string
Generates a confirmation dialog modal to confirm the action before proceeding with the link or button click.
Used on Tags: <a>
, <button>
Valid Values: Any text string
There are additional attributes you can add to control the content:
- zx-dialog-confirm-title: Sets a title for the dialog (defaults to 'Confirm')
- zx-dialog-confirm-yes: The text of the yes/ok button
- zx-dialog-confirm-no: The text of the no/cancel button
<a href="?hello=true" zx-dialog-confirm="Continue?">Hello</a>
<form method="post" action="/foo" zx-swap="#container">
<button zx-dialog-confirm="Continue?">Hello</button>
</form>
You should ask users for confirmation of actions that are sensitive, cannot be easily undone, or have side effects. For example, deleting a record which cannot be recovered. Typically this will be used on form buttons (because links/GET should not make permanent changes).
↑ top | Features | HTML Api | next section → Events
true | false
Enables or disables the use of a spacer element to guard against page jumps when content is expected to shrink or be removed.
Used on Tags: <a>
, <button>
, <form>
Valid Values:
true
: Activates the spacer to maintain page stability and prevent jumps during content changes.false
: Deactivates the spacer, allowing the page to adjust naturally, which may result in jumps.
<a href="/shrinkContent" zx-swap="#targetContent" zx-jump-guard="true">Shrink Content</a>
<form action="/home/echo" method="POST" zx-swap="#container" zx-jump-guard="true">
<button id="button" type="submit">Form Swap</button>
</form>
The button within a form can also have zx-jump-guard, which will override the <form>
value if it exists
<form action="/home/echo" method="POST" zx-swap="#container" zx-jump-guard="false">
<button id="button" type="submit" zx-jump-guard="true">Form Swap</button>
</form>
When elements are removed from the DOM, the page may abruptly jump upward if the combined height of the remaining content and the viewport is less than the current scroll position.
zx-jump-guard
manages the content height of elements that are being swapped to ensure that the page does not jump.
Calculating the page size to prevent jumps is an expensive (about 20ms) operation per swap. Therefore only add zx-jump-guard
where you expect content may be removed or shrink.
- Use
zx-jump-guard="true"
for links, buttons or forms that swap content likely to cause large decreases in content size, ensuring that user experience remains smooth and uninterrupted. - Set
zx-jump-guard="false"
disables zx-jump-guard
↑ top | Features | HTML Api | next section → Events
true | false
Keeps an an element and its descendants unchanged when a parent is swapped out. Useful for maintaining page level content like video, audio forms or other interactive content that should not be swapped.
Used on tags: any html element
Values Values:
true
: Keep the element when swapping out the parentfalse
: Do not keep the element when swapping out the parent
<a href="/" zx-swap="#container">Update</a>
<div id="container">
Updated Content
<div id="someContent" zx-keep="true">
Kept Conent
</div>
</div>
Element with zx-keep requires that an Id be set
To perform a zx-keep, ZSX needs to move, swap and restore the elements, which can be a costly operation. In conjunction with targeted use of zx-swap, use zx-keep on the inner most elements that are necessary to retain.
<iframe>
Moving and restoring iframe element with zx-keep does not work. The iframe will be reloaded when it is restored into the DOM.
↑ top | Features | HTML Api | next section → Events
browser | app
Specifies whether a link should behave like a regular browser link, or adopt a more application-like interaction.
Used on Tags: <a>
Valid Values:
"browser"
: (default link behavior) Maintains standard browser link behavior, including displaying URL tooltips and offering content menus. This is the default."app"
: Suppresses the URL tooltip and modifies the link's behavior to mimic a form button or application-like interaction.
<a href="https://example.com/action" zx-link-mode="app">App-like Link</a>
This works exactly like a typical link and is provided in case we provide more types in the future
<a href="https://example.com/page" zx-link-mode="browser">Standard Link</a>
Browsers provide some default features for links like tooltips, right click menu, and clicked and unclicked states. For some application where we want a desktop like app experience, disabling these default browser behaviors is desired.
When zx-link-mode="app"
the link is just clickable text. You handle all additional styling for the link.
You'll typically want app style links when links wrap more complex context than just a test string, like cards, list items or links that are styled as buttons.
↑ top | Features | HTML Api | next section → Events
true | false | cursor-wait | cursor-progress
Adds a visual progress indicator to buttons and links for requests that take time to complete.
Used on Tags: <a>
, <button>
Valid Values:
- true: Add the default loading indicator when clicked
- false: Do not add a loading indicator
- cursor-wait: Change the cursor to the browsers default wait cursor
- cursor-progress: Change the cursor to the browsers default progress cursor
<a>
loading indicator<a>
link with 'wait' cursor<a>
link with 'progress' cursor<button>
loading indicator
Animates the background of the text with a progress indicator. Useful for App-Style links or links that look like buttons.
<a href="/path" zx-swap="#target" zx-loader="true">Link with Loader</a>
Uses the browsers default wait cursor style. Useful for regular links where you want to disuade clicking the element again.
<a href="/path" zx-swap="#target" zx-loader="cursor-wait">Link with Wait Cursor</a>
Hover to Show Cursor
You can override the .zx-loading-cursor-wait
class if you need to customize the icon
.zx-loading-cursor-wait {
cursor: wait !important;
}
Uses the browsers default progress cursor style. Useful for regular links where it's okay for the user to click again
<a href="/path" zx-swap="#target" zx-loader="cursor-wait">Link with Progress Cursor</a>
Hover to Show Cursor
You can override the .zx-loading-cursor-progress
class if you need to customize the icon
.zx-loading-cursor-progress {
cursor: progress !important;
}
Animates the button with a simulated progress indicator and disables the button until the request completes
<button type="submit" zx-loader="true">
Loading Indicator
</button>
Use zx-loader
for any action which is not, or nearly not immediate. You shouldn't add loaders to elements which do not have much delay because the quick flash of the loader is more distracting.
We provide different types of loading styles that serve different UX purposes.
- progress: Used to imply that an operation is proceeding. The simulated progress indicator gives a feeling that the application is moving forward.
- cursor-wait: Used to imply that the application is processing and the user should not click again.
- cursor-progress: Used to imply that the application is processing and the user CAN click again.
↑ top | Features | HTML Api | next section → Events
true | false
Tells ZSX to skip executing an inline <script>
tag received in the response
Used on Tags: <a>
, <button>
Valid Values:
true
: Skip executing the<script>
tagfalse
: Executes the script tag, this is the same as not existing
<body>
<a href="/" zx-swap="body">Update!</a>
Content {{now}}
<script zx-script-skip="true">
//Execute only on initial page load
</script>
</body>
Uze zx-script-skip
when you want to specify that a script should not be executed after performing a zx-swap
. This is necessary in cases when the javascript should only have been executed on initial page load. For example setting up global libraries, like zsx.
↑ top | Features | HTML Api | next section → Events
true | false | top | CSS selector
Tells ZSX to scroll to a particular element after completing a zx-swap. This can improve the user experience when the swapped elements may be out of view or we want to focus the new elements.
Used on Tags: <a>
, <button>
Valid Values:
true | false
: When true, scroll to the first zx-swap selector element. When false, this disables zx-scroll-to and also a link URL hash target.#idSelector
: Scroll to the provided Id, regardless of the zx-swap selectors.classSelector
: Any valid CSS selector. The first item returned will be the target of the scroll"top"
: Scroll to the top of the page
Default Browser hash(#) identifier
When zx-swap is applied, if there is an id hash in the URL, it will scroll to that ID. This is the browser's semantic way to scroll to.
<a href="/page#elementId" zx-swap="#elementId">Link</a>
Scrolling to the Swap Target
When zx-scroll-to="true"
it will scroll to the location of the zx-swap
<a href="?hello=true" zx-swap="#targetContainer" zx-scroll-to="true">Hello</a>
Sometimes as a result of adding a new item to the page, we will want to scroll to that item, except we do not know a unique ID for it because it does not exist yet. You can use a complex class selector to scroll to the last (or first) item.
<button type="submit" zx-scroll-to=":nth-last-child(1 of .classList)">
By default, ZSX handles links to different pages (base paths) by scrolling to the top of the page. This matches default browser behavior and user expectations when switching pages. In effect, when the base path changes, it is as if you set zx-scroll-to="top"
.
You can change this by setting zx-scoll-to="false" to disable scrolling to top when the base path changes, or any other zx-scroll-to value.
<a href="/page1" zx-swap="#targetContainer" zx-scroll-to="false">Hello</a>
<a href="/page2" zx-swap="#targetContainer" zx-scroll-to="false">Hello</a>
The reason for this behavior is that typically, in a multi-page appication, each page route is significantly different content. The usual experience is to start that page at the top of the content.
ZSX follows the browser/application level scroll setting. You can enable smooth scrolling by setting the scroll-behavior
at the page level with CSS.
<html lang="en" style="scroll-behavior: smooth;">
Sometimes you want the scroll location to be just above the target element. You define this in CSS on the element that will be scrolled to with CSS.
<div id="myElement" style="scroll-margin-top: 20px;"></div>
Use zx-scroll-to
sparingly, only when necessary to highlight an element that the user must perform in a workflow, or important information that they must see after an action.
↑ top | Features | HTML Api | next section → Events
#idSelector | .classSelector | tag-selector | list of CSS selectors
Swap out content in the current page with content from the response HTML.Specify the CSS selectors to target.
Used on tags: <a>
, <form>
Valid Values:
#id
: An HTML element ID to swap.class
: A set of class elements to swaphtml-tag
: A set of HTML tags to swaplist
: A comma separated list of selectors to swap
Swap a single element by target id
<a href="?hello=true" zx-swap="#targetContent">Hello</a>
Swap all elements matching the target class
<a href="?hello=true" zx-swap=".targetContent">Hello</a>
When a class target is defined, each element needs a unique id in order to disambiguate the elements
Swaps the elements matching the given tag
<a href="?hello=true" zx-swap="body">Hello</a>
When a tag target is defined, if there are more than one result, each tag will need its own Id
Swap multiple targets by providing a comma separated list
<a href="?hello=true" zx-swap="#targetContent,#target2,.classTarget">Hello</a>
Swap out the element from the form post content
<form action="/" method="POST" zx-swap="#anchorTargetContainer">
<button type="Submit">Form Swap</button>
</form>
zx-swap
is the primary feature of ZSX. As a result of every link or form action, you describe one or more elements to swap from the response HTML.
You should liberally use zx-swap throughout the application. For best performance, specify as narrow a target as possible. See minimizing zx-swap targets
zx-swap
swaps the innerHTML of the target and merges the attributes. This will preserve event handlers and references of the old element, but any child event handlers will not be preserved.
Reattach event handlers using zsx-zx-swap-after
event
Listen for the zx-swap event and reattach the event handlers manually for any children after the swap is complete. See zsx-zx-swap.after
Forms can either be a GET request or a POST. Typically when a POST, the server should redirect back to a GET (POST-REDIRECT-GET style) so that the browser does not perform duplicate submits.
ZSX will update the URL history to the value of any GET redirect. POST redirects are not added to the history.
↑ top | Features | HTML Api | next section → Events
string list
Synchronizes parameters from an <a>
link click with other links on the page.
Used on tags: <a>
Valid Values:
param
: A single param to be synchronized with other linkslist
: A comma separated list of params to sync with other links
<a href="?hello=true" zx-swap="#targetContent" zx-sync-params="hello" >Hello</a>
<a href="?hello=false" zx-swap="#targetContent" zx-sync-params="hello" >Goodbye</a>
<a href="?hello=false" zx-swap="#targetContent" >Other Link</a>
<a href="?hello=true&foo=bar" zx-swap="#targetContent" zx-sync-params="hello,foo" >Hello</a>
You use zx-sync-params
when you need to sync all links using the parameter, even when only a small subset of the page content is updated.
This is necessary when the URL contains important state that should persist across subsequent clicks, regardless of swapped content.
The zx-sync-params
logic is:
- WHEN a link with
zx-sync-params
is clicked - FOR each link on the page
- FOR each of the parameters in the clicked
zx-sync-params
- IF that parameter is in the other link's
href
- THEN update the parameter to the value of the parameter in the clicked link.
- UNLESS the other link itself contains a
zx-sync-params
with the same parameter name
zx-sync-params
can take a single parameter or multiple parameters.
Bypassing Link Update
Other links on the page which should maintain their parameter value can be excluded by having their own zx-sync-params
.
For example, consider an element that needs to be visible or hidden. One link shows it, the other link hides it, all other links need to reflect the last action
<a href="?showMessage=true" zx-swap="#pageContent" zx-sync-params="showMessage" >Show</a>
<a href="?showMessage=false" zx-swap="#pageContent" zx-sync-params="showMessage" >Hide</a>
<a href="?showMessage=false" zx-swap="#pageContent" >Other Link</a>
<div id="pageContent">
Hellow World,
{{#if url.showMessage}}
<div>
It is a good day!
</div>
{{/if}}
</div>
Because both the show and hide links have a zx-sync-params
, that means the inverse link is ignored when updated. Therefore only the 'Other Link' gets updated.
↑ top | Features | HTML Api | next section → Events
ZSX fires the following events that your application can listen to and perform additional actions.
Event | Description |
---|---|
zsx.zx-swap.after | After a content zx-swap for one selector has been completed. |
Fired after the swap for a selector is completed. If the zx-swap contained multiple selectors, a zsx.zx-swap.after
event is emitted for each selector.
event.detail {object}
oldElement
The old element that was swapped out. No longer exists in the DOMnewElement
Live DOM reference to the new element that was swapped inselector
The selector that was used to locate the elements
// Example resetting bootstrap tooltips and popovers after a swap
document.addEventListener('zsx.zx-swap.after', function(event) {
// Respond to zsx.zx-swap.after event
})
Use the zsx.zx-swap.after
to process the content after ZSX is finished replacing it. You might use this to re-attach event listeners and state from other libraries or purposes.
See Cookbook Restoring Events and Features After Swap
↑ top | Features | HTML Api | next section → CSS Reference
The following CSS classes are created by zsx.css for the visual features. You can override these style classes if needed.
CSS | Purpose |
---|---|
.zx-link-app | Added to links that have zx-link=mode="app" to add default pointer styles. |
.zx-loading-cursor-wait | The cursor style applied when zx-loader="cursor-wait" is used |
.zx-loading-cursor-progress | The cursor style applied when zx-loader="cursor-progress" is used |
↑ top | Features | HTML Api | Events | next section → ZSX Design Goals
ZSX has an opinionated philosophy in regards to web application architecture. These opinions dictate it's featureset. The ZSX Design Goals will help you determine if ZSX is right for your development style.
ZSX is designed to enhance server rendered applications without breaking user’s expectations of browser behavior.
We built ChartSQL Studio with ZSX and followed these guiding principles:
- URLs and HTTP is the Right Architecture
- Links, Forms, and Buttons Are All You (mostly) Need
- Web Components Are Not Necessary
- Use Minimal Javascript
- Assume Full Page Rendering
- Explicit Link and Form Handling
- Must Be Able to Hard Refresh
Use Cases, Future Development plans & Alternatives to ZSX
Regardless of how you render the front end of the application, we believe that URLs and HTTP (GET and POST) is broadly the right application architecture.
Web applications should be addressible by URLs (links) and backend state changes should be communicated to the server with POSTs (forms).
Applications should be decomposed into pages (/entity1, /entity2) representing different resources, following a generally RESTful style.
It follows that HTML, links and forms are the natural way to work with the web application architecture.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
Modern HTML contains everything necessary for a user to interract with 99% of web applications in an accessible and semantic way. It is not necessary to reinvent form controls or client side routing mechanisms. Virtually any DOM driven user experience can be maniuplated with just links, forms and buttons. Many developers are not even aware of modern features of HTML.
More advanced visual fidelity can be achieved with minor javascript, canvas and WebGL functionality. But when you are working with the DOM, rely on links, forms and buttons unless it can't be achieved any other way.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
A lot of 'modern' front end development seeks to componetize HTML. In fact 'web components' in the general sense is not a new technology. Some of the earliest web frameworks like JSP were very component-like. However, unless you are creating a "component library" for others to use, we do not believe web components bring enough benefits.
Building HTML UIs requires a lot of markup. Componetizing that markup does not lead to significant user improvement or developer ergonomics, in our opinion.
We believe that it is easiest and most maintainable to build web UIs in basic HTML pages that can be read as straight HTML.
ZSX exists to enhance that experience while staying true to the nature of HTML and HTTP.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
ZSX takes an minimalist approach to JavaScript. While some JS is necessary, we think JS heavy applications are not good UX.
We believe the best UX is based on fundamental HTML/HTTP, links and forms, and that JavaScript-first frontends are overly complex and brittle to maintain.
In a ZSX application, you should be able to look at the rendered HTML and follow the links and forms to understand the application interactions.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
In a ZSX application, the server by default always fully renders pages. ZSX provides features to hot swap elements on the page and avoid a full browser page reload.
Avoiding a full page reload improves performance of the browser. Browser performance is improved because it doesn't need to reflow the entire document and re-execute scripts. This performance improvement is perceived by the user as more responsive.
Full server renders might seem costly, but it greatly improves the maintainability of applications:
- Most pages in an application are fast enough that it doesn't matter.
- Forces developers to think through the first page load experience up front.
- Dissuades client side hydration which we consider an anti-pattern.
- "Premature optimization is the root of all evil" - You don't need to spend time optimizing the performance of pages that are seldom used.
- Caching of pages can be done server-side in a more robust manner
The payload size of compressed HTML over the wire is not a significant factor in perceived latency compared to server response time and client side processing.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
As compared to some similar libraries, ZSX does not automatically enhance all links or forms, because:
- The application will be most responsive when you swap the smallest amount of content necessary
- You should be able to inspect the HTML and understand what each link/form behavior is without having to know hidden information
- You should think through exactly which content needs to be updated after every click or form post.
Therefore when using ZSX, for every link or form, you explicitly decide which content needs to be updated, and which zx-* features to apply.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
A litmus test for web applications with proper UX is “can you hard refresh.” Many applications exhibit poor user experience on hard refresh:
- It takes a long time for the page to load
- Time to interaction is delayed as client side features are hydrated
- There may be a lot of flashing and layout thrashing as elements are loaded in
- States that were clicked through may be lost
This leads to many lost features native to browsers:
- Cannot share links
- Cannot bookmark links
- Cannot open a link in a new tab
- Cannot duplicate a tab
- Cannot refresh to see style changes when developing
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
Use ZSX for server rendered applications that rely on links, forms and HTML. Often referred to as 'multi-page applications'
If your application is fully SPA, then ZSX is not a good fit.
ZSX is a small library and will work well with other JS libraries.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
ZSX will only add enhancement features that we find solve common tasks in our applications, without breaking fundemental browser architecture. The HTML specification is continually evolving with new features that make browser apps more responsive and user-friendly. We are betting on HTML/CSS/HTTP and Server Side Rendering as the core of applications.
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
ZSX is similar to HTMX and Unpoly.js in its fundamental philosophy of server rendered applications. However it differs from them in important ways:
HTMX seeks to add new non-standard “hypermedia” features to HTML. Essentially, it allows turning any element into a link, button, GET/POST/PATCH. This is not semantic and means that applications built in HTMX cannot work with JavaScript turned off.
We explicitly do not like that any element on the page can be an interaction point, instead Links, Forms, and Buttons Are All You (mostly) Need.
Unpoly.js is probably the closest philosophically to ZSX. However we consider it’s feature set more complex than we desire. We believe there are only a few necessary conventions that we need. Unpoly drifts away from semantic HTML with features like layers. In comparison, ZSX only enhances fundamental browser behavior.
Unpoly also has a lot of configurations for targeting different elements, parents, children, appending, prepending and automatically choosing ambiguous cases. We consider these uncessary.
Instead, with ZSX, you control your HTML, the IDs, and classes you wish to target. This makes it easier to maintain your HTML and understand your content updates. You can always inspect the source HTML to understand precisely what will happen.
Turbo is also another strategy to “dramatically reduce the amount of custom JavaScript” but it has different goals. It provides features like ‘Frames’ and ‘Streams’ to add additional architectural options. We believe that Turbo goes too far beyond the idea that URLs and HTTP is the Right Architecture
↑ top | Features | HTML Api | Events | ZSX Design Goals | next section → Developing Applications
Additional architectural tips and tricks for building maintainable and interactive web applications with ZSX
- Start with the Raw HTML
- Understanding Application State
- User Interactions
- Using Loading Indicators
- Using Animations
- Performance Optimization
When building your application, start with your HTML, Links and FORMs without any zx-* attributes. Get your basic workflows and content correct. When you are confident in the features, then enhance the UX by adding in attributes like zx-swap
, zx-loader
, zx-dialog-confirm
to improve the slickness of your application.
State for an application UI can live in different architectural layers. Where you decide to store your state depends on how it needs to be accessed and interact with browser capabilities.
One of the most challenging decisions in designing an application is deciding just where the UI state should live. This guide gives some requirements and recommendations.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Use the Database/Backend when the UI state must survive browser sessions or application restarts. The application will load and render the database/backend state into the HTML.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Use Server Sessions when the UI state must not be shareable, does not need to be permenantly persisteted, but must survive page refreshes. Mutating session state requires interacting with the server. Your server will maintain the session variables and render them into HTML. With session state, all open tabs for a single browser instance will share the same state.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Use URL Parameters when the UI state can be shareable, disclosable, and survive refreshes. URL state should be used whenever possible to match users expectations of browser functionality. URL state allows the page to be shareable, bookmarkable, deep linked, and opened in multiple tabs and multiple browsers independently.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Use Cookies when the UI state must not be shareable across users or browsers, and can be set by the client without posting to the server. Cookies allow bi-directional synchronization of state with the browser and server. The server can use the cookies to control rendering output, and the client can also set the cookies (whereas the browser cannot set session data without a request to the server)
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Use localStorage when the UI state must not be shareable, and does not or must not be shared with the server for rendering. The client can set and restore values on page load into localStorage.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Use Javascript/Page Stage when you need small client side features like tooktops. Page state is kept in runtime variables or HTML data-* attributes. Page State is never rendered by the server or shared with it. Page state should be kept to a minimum so that the convention of URL refreshing and link sharing is maintained.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Links and Forms are the bedrock of HTML and HTTP interaction through a browser. Links (<a href>) represent GET requests and Forms (<form>) represent POST requests.
You should build your application interaction elements entirely out of links and forms wherever possible. This is the most maintainable, accessible and straightforward way to build on the web.
ZSX enhances links and forms by providing features to control how to render the response content. It is the same response content that would have rendered had JS been turned off. But with ZSX, you can speed up your responsiveness of links and forms.
Minor interactions like hover states, tooltips and alerts can be entiely client side.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Loading indicators should be used whenever the server response time can be greater than 100ms.
We do not automatically add loading indicators to every link/button because when they are not needed, they add unecessary distraction.
See zx-loader
for optionally adding loading indicators.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
In general, using animations and page transitions is an anti-pattern. The best UX is: Instantly change that which the user should notice
The fastest change possible is simply to update the content. Animating content in and out is initially slick, but unecessary.
If you use any animations, they must do the following:
- Start immediately on the user click
- The user must never have to wait for the animation itself to finish to click the next action that is available.
This means that you cannot wait to start animations after a request has finished. It's too late and will make the application feel sluggish
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Most of the processing time is waiting on one of three operations: network requests, DOM swaps, and inline javascript. Decreasing the amount of time for these operations will have the biggest impact on application responsiveness.
Follow these techniques in this order to improve the performance of your application:
- Optimize Time to Response
- Minimize zx-swap Targets
- Decrease Inline Javascript
- Syncrhonizing Links
- Server Side Fragments
You should always seek to first optimize the backend server performance. Requests which respond in < 100ms are going to feel almost instantaneous to the user. Fast server responses improve both full page loads, and subsequent zx-swap loads.
Note: We find that the amount of raw HTML returned is not a significant factor, just response time. So a lot of HTML fully cached and quick from the server is a very viable optimization.
The time it takes to place new HTML into the DOM is many time larger than it takes to parse the HTML. While parsing takes 1-10ms, DOM placement per swap target will be 5-100ms.
Therefore you want to reduce the amount of HTML to be swapped, and the number of swaps that need to take place.
Ideally, have each zx-swap
only target the smallest parent that is necessary to update the content. Don't have more zx-swap
targets than is necessary. Don't just zx-swap
the body tag, instead think carefully about updating only specific page fragments.
Don't use zx-swap
to replace link URLs on the page, as swapping is an expensive operation. You can use zx-sync-params
feature to update all links on the page to match the clicked URL.
zx-swap
also executes any inline javascript and can be a source of unexpected delay. If the javascript is not necessary to run on every swap, move it outside of the target.
COMING SOON:
The final optimization is to have the server return partial HTML based on the zx-swap
target that was requested. Because this adds complexity to the backend to maintain two rendering paths, we use this sparingly. Relying too much on server side fragments means that the initial page loads will be slow, which is not good UX. Also the amount of HTML returned is not a critical factor compared to the server response time. So only use this method when you can significantly decrease the server response time.
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | next section → Cookbook
Solving common application design tasks with ZSX
If you have other javascript that needs to be attached to the swapped elements, you can do so by listening on zsx.zx-swap.after
event
// Example resetting bootstrap tooltips and popovers after a swap
document.addEventListener('zsx.zx-swap.after', function(event) {
// Get the new element from the swap event
var element = event.detail.newElement;
// Remove any existing tooltips and popovers if they exist
var tooltips = element.querySelectorAll('.tooltip');
var popovers = element.querySelectorAll('.popover');
tooltips.forEach(function(tooltip){
tooltip.remove();
});
popovers.forEach(function(popover){
popover.remove();
});
// Select and add the tooltips
var tooltipTriggerList = element.querySelectorAll('data-bs-toggle="tooltip"]')
tooltipTriggerList.forEach(function(tooltipTriggerEl){
return new bootstrap.Tooltip(tooltipTriggerEl)
})
// Select and add the popovers
var popoverTriggerList = element.querySelectorAll('[data-bs-toggle="popover"]')
popoverTriggerList.forEach(function (popoverTriggerEl) {
return new bootstrap.Popover(popoverTriggerEl)
});
})
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | Cookbook | next section → Roadmap
Features or improvements to be completed
Feature |
---|
Enhance zx-dialog-confirm to work from either buttons or forms |
↑ top | Features | HTML Api | Events | ZSX Design Goals | Developing Applications | Cookbook | Roadmap | next section → Common Issues
The following are common errors that ZSX will throw when misusing features
When your zx-swap directive results in multiple elements, each element needs a unique id in order to be able to determine which content area is which
We are not currently taking other contributors but you can open issues and we will address them.