Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
Loading...
You can alter CBWIRE and Livewire's default behavior by overriding settings in your config/ColdBox.cfc file.
Automatically include Livewire's CSS and JavaScript assets. This removes the need to manually add references to wireStyles() in your <head> and wireScripts() at the end of </body> in your ColdBox layout file. Defaults to true.
The maximum amount of time allowed for uploads to complete.
Sets the URI endpoint where CBWIRE posts its updates to the server. You might need to change this if you do not have URL rewriting enabled.
Determines if CSRF token protection is enabled or not. Defaults to false. This will be changed to 'true' in CBWIRE 5.
CBWIRE uses CSRF tokens to protect incoming requests from bad actors. You can set the wirebox mapping to determine what storage provider is used when storing the CSRF tokens. Defaults to SessionStorage@cbstorages.
Override to change the URL root path to CBWIRE.
When set to true, any that contain strings will be automatically trimmed on updates. Great for form inputs. Defaults to false.
You can enable it directly on your if you don't want to set this globally.
The relative folder path where are stored. Defaults to 'wires'.
When set to true, it displays a progress bar at the top of the page using to load pages. Defaults to true. Set to false to disable the progress bar altogether.
Use to adjust the progress bar color when using .
CBWIRE is a ColdBox module that uses Livewire and Alpine.js to help you build modern, reactive CFML applications in record time without building backend APIs.
Let's insert a counter element into our Main layout using wire( "Counter" ).
You now have a reactive counter that increments when you click the plus button without any page refreshing or writing JavaScript! 🤯
CBWIRE renders the component with default values (starting at 0).
A button click triggers Livewire.js to send a request to the server.
CBWIRE processes the request, runs increment()
, and updates the data.
The updated HTML is sent back in the response.
Livewire.js refreshes the display using DOM comparison.
We developed a responsive counter.
We avoided writing any JavaScript code.
We didn't need to create an API.
There was no need for page refreshes.
We skipped using webpack or dealing with JavaScript compilation.
The example shows how easily components interact with your CFML back-end. But for quick UI updates, JavaScript can handle changes without unnecessary server requests. While we don’t need a request to increment the counter, we might if we want to save it. This is where CBWIRE and Alpine work perfectly together.
Below, let's change our component to use Alpine.js and add a method to save the counter.
The counter updates instantly on the client side, sending a server request only when clicking "Save" to store the value in the session. On page load, onMount()
sets the counter from the session. Using $wire
, Alpine communicates with the component, giving full control over server requests. CBWIRE is redefining CFML development—we think you'll love it!
Download and install . From your CLI, start CommandBox by typing 'box'. Then run the following:
Let's define our Counter .
Finally, define our counter .
We stayed entirely within the CFML environment.
is a lightweight JavaScript framework that was built for simplicity and designed to work alongside Livewire ( and therefore CBWIRE ).
CBWIRE uses the JavaScript bits from and for its DOM diffing and client-side functionality. CBWIRE wouldn't exist without the incredible work of , the creator of both Livewire and Alpine. CBWIRE was created to bring these excellent tools into the ColdBox and CFML ecosystem.
The CBWIRE module for ColdBox is written and maintained by , , and .
Please consider becoming one of our lovingly esteemed .
05/16/2024
Upgraded Livewire.js to version 3
Alpine.js automatically included with CBWIRE
Lazy loading components
Execute JS from actions using js() method
Single-page applications now with wire:navigate
Stream content using wire:stream
Transitions using wire:transition
Request bundling to cut down on server requests
Complete rewrite of CBWIRE engine
Removed 'enableTurbo' global setting
Removed 'throwOnMissingSetter' global setting
None
12/20/2023
Automatically populate data properties with passed parameters when onMount() isn't defined Contributed by Michael Rigsby
Add support for calling wire( "MyComponent@module" ). Contributed by Michael Rigsby
Add ability to dispatch browser events
Add ability to call event.method() calls from CBWIRE templates
None
You can get started with CBWIRE with a few initial steps.
Adobe ColdFusion 2021+ or Lucee 5+
ColdBox 6+
Within the root of your project, run:
If you want the latest bleeding edge, run:
For CBWIRE to work, your layout must include Livewire's CSS and JavaScript assets. CBWIRE 4 automatically adds these assets to your layout.
If you want to disable this functionality and manually include the CSS and JavaScript yourself, you can set the autoInjectAssets setting to false in your ColdBox.cfc.
Then, you can manually add the CSS and JavaScript using the wireStyles() and wireScripts() methods in your layout file.
Your template must include Livewire's CSS and JavaScript; otherwise, CBWIRE will not work. If you are trying to use CBWIRE and it's not working as expected, this is the first place to check.
Install .
09/25/2023
Add configuration property to include CBWIRE styling and JavaScript assets automatically. Remove the need to add wireScripts() and wireStyles() to templateBugs
Add onUpdate() and onUpdateProperty() lifecycle hooks
Add the ability to refresh all child components from the parent component when an action is called
Add the ability to call CBWIRE UDFs from the cbwire template as computed properties are too limiting.
Single file component file names are not unique and could create conflicts in high-traffic applications
Clear any compiled single file components on ColdBox fwreinit
Add params arguments as a substitute for parameters argument when calling onMount
Add 'cacheSingleFileComponents' setting to control if single file components are cached
Cleanup unnecessary variables available to templates to avoid potential collisions
Add the ability to call application helpers defined in ColdBox.cfc or any installed modules from CBWIRE templates.
Add resetExcept()
Fix child components not re-rendering properly on subsequent requests
HTML comments in component templates cause re-renders not to work.
The user requests a page from the server that includes our Counter component.
ColdBox receives the request.
CBWIRE parses our component and initially renders it with its default values.
The rendered component is sent to the browser. It's important to understand that nothing extraordinary has happened at this point. We've returned HTML to the browser from an incoming request. Next is when things get interesting.
The user clicks the button.
CBWIRE uses Livewire.js to intercept the user's click on the client side and send an XHR request to the server.
CBWIRE runs the increment() method, updating the data properties.
CBWIRE re-renders the template.
CBWIRE returns the re-rendered template to the browser.
CBWIRE uses Livewire.js DOM diffing technology to update the component on the page.
This diagram shows the Counter we created in the and how CBWIRE re-renders our template and updates the DOM.
05/18/2023
Add Inline Components ( NEW SYNTAX )
Make components module-aware and allow components to be created within modules
Remove cbValidation as a dependency of CBWIRE
Increase component rendering performance by bypassing unnecessary ColdBox view caching and lookups
Refactor core engine for better performance and easier maintenance
Remove computedPropertiesProxy as it's no longer needed
Replace calling computed properties in templates using #args.computed.[property]()# with #property()# instead.
Replace calling data properties in templates using #args.[property]()# with #property# instead.
Using new component format, validation references this.constraints. Change to just 'constraints'.
Change incorrect emitTo( event, component ) signature to instead by emitTo( component, event )
Empty string and null values are not being correctly passed to Livewire
Fix the ability to locate Wires using full paths such as appMapping.wires.SomeComponent
Component actions that update data properties don't always update the DOM
Calling .see() and .dontSee() in component tests doesn't return the test object for additional method chaining
Overriding computed properties not working when writing component tests
In this section, you will find the release notes for each version we release under this major version. If you are looking for the release notes of previous major versions use the version switcher at the top left of this documentation book.
11/26/2022
Ability to output component template directly from onRender() method instead of defining a .cfm template in wire/views.
Computed properties that do not return a value result in the error 'variable [VALUE] doesn't exist
Nested components are causing the template rendering only to render the last nested template
Struct values are not being passed to the template and are instead being replaced with an empty string
CBWIRE doesn't work when the ColdBox app is in a subdirectory
Getting errors when rendering component templates in the latest version of ColdBox
Nested components are not rendering
01/09/2022
Implement lifecycle hook onHydrate().
Implement lifecycle hook onHydrate[Property]().
Implement an automatic trim() for all data properties.
Implement the ability to interact with CBWIRE component from JavaScript using cbwire.find( '#args._id#' ).
Add configuration setting 'enableTurbo' to automatically include everything needed to work with Turbo for single-page applications.
Add the ability to call reset() without passing a key to reset all data properties to their original values.
Implement onMount() method instead of mount().
DocBox-generated docs are failing because of file structure.
Listeners are fired immediately when calling emit() when the listener is defined on the same component, which they shouldn't.
onHydrate() is firing after actions are performed.
Computed properties are not being rendered before actions are called.
Coming soon
Your templates must have a single outer element for CBWIRE to bind to your component and update the DOM properly. This can be any valid HTML element. Below, we are using a div element.
Below is a template with one outer element ( good ) and another with two outer elements ( bad ).
By default, CBWIRE will look in the ./wires folder for a .cfm file with the same name as your component.
To override the default implicit rendering, we can tell CBWIRE where our component template is by defining a onRender() method on our component and using the template() method.
We can pass parameters when calling template().
In this case, someVar becomes available to our template.
You can also return your template inline like so:
You access computed properties in your template by invoking the method.
You can access any global helper methods defined in your ColdBox application and any modules you have installed.
For example, suppose you've installed the cbi8n module ( an internalization module for ColdBox ). You can access its global helper methods in your template, such as the $r() method for displaying text in various languages.
Templates are your ' HTML and consist of valid HTML/CFML tags. This includes <cfif>, <cfloop>, etc.
You can change this default location in the settings.
You can access your from your template by calling #propertyName#.
You define on your component by using the computed annotation.
Computed properties are cached are are executed when first called. See .
Visit to find ColdBox modules for your application.
CBWIRE has dramatically improved since its v2.x days. Here, we will outline the core differences between CBWIRE 2 and CBWIRE 4 to assist you with upgrading.
Here's a quick list of features in 4.x you'll get when upgrading:
Easier access to data properties, component methods, and computed properties from your templates.
Single-file components
Define your component in a single .CFM file
Upgraded DOM diffing engine via Livewire.js
Ability to pass keys with your wires to control when CBWIRE renders your components.
#wire( name="MyForm", key="my-form" )#
New HTML Directives
$wire object allows you to access and interact with your wire from JavaScript directly
In 2.x, you had to place wireStyles() and wireScripts() in your layout like this to include CBWIRE's JavaScript and CSS assets.
This is now automatically handled for you by default.
In 2.x, you had to include Alpine.js into your project via CDN or Webpak manually.
Alpine.js is now automatically included with CBWIRE 4.
CBWIRE and Alpine.js have such a deep integration together that automatically including the two makes sense, and this follows the direction that Livewire 3 took. There is no longer a need to pull Alpine.js into your project.
Removing any previous installations or references to Alpine.js is important since it is now included with CBWIRE. Otherwise, you will run into various rendering errors, and your Alpine components will likely run twice.
In 4.x, you can fully interact with your components using Alpine and the magical $wire object.
In 2.x, you defined your components with a .CFC file and .cfm template.
You can still use separate files in 4.x, but you can also place everything in a single .cfm file. Use whichever pattern you prefer.
You still render your components using the wire() method, but the signature has changed.
In 2.x, the signature was:
In 4.x, the signature is:
Key differences:
The arguments componaneName and parameters were given shorter names.
In 2.x, data properties were referenced in your templates using #args.prop#.
In 4.x, you can access the data property without args.
You can still reference your props using args for backward compatibility, but it's optional.
Any methods you define in your component you can access from the template.
In 2.x, computed properties were defined with a computed struct in your component.
And you would access them in your templates like this:
In 4.x, we define computed properties like this:
And we access them in our templates like this:
In 2.x, you could communicate with your components using emit().
In 4.x, this has been replaced with dispatch().
You can also dispatch from JavaScript using $wire.$dispatch and cbwire.dispatch().
Official support for Adobe CF 2018 was dropped.
In 2.x, you defined your validation constraints using this.constraints.
In 4.x, you use constraints or variables.constraints.
In 2.x, relocating was done with the relocate() method. It's been replaced with redirect();
We no longer recommend using the Turbolinks library for SPA-style applications.
Components are sections of your application that you want to be reactive to user input.
You can render a component using the wire() method.
You can render wires from folders and subfolders outside of the default ./wires folder.
You can also reference components within another ColdBox module by using the @module syntax.
You can pass data into a component as an additional argument using wire().
By passing in parameters, you can create reusable UI components that are unique but similar in functionality. For example, you could make a button component that you reuse throughout your app.
Passed-in properties must have a data type of string, boolean, numeric, date, array, or struct.
There may be areas of your application where you need to use wire() but don't have access to it. You can use the CBWIREController object to insert wires anywhere you need.
is included
with placeholders
If you still want to use these methods in your layout manually, you are welcome to do so, but you will need to update your ColdBox.cfc module settings. See .
See the section of these docs. Learn more about Alpine at .
You can provide a key, which is essential for correct rendering when nesting components ( See )
You can lazy load components ( See )
See .
TurboLinks has been replaced with . By adding a simple wire:navigate to your links, you can quickly load pages in the background and prevent re-rendering all your JS and CSS assets.
Components are made up of , , , and a
By default, components are placed in the ./wires folder in the project root. You can change this location using the wiresLocation .
Parameters are passed into your component's method. This is an optional method you can add.
Properties you pass into your component as params will be automatically populated if onMount() is not defined and a matching is found.
You can nest components as much as you need by simply calling wire() from within a ( See ).
08/30/2022
File Uploads
Ability to write unit tests for components
Ability to override the default 'wires' folder location
noRender() method to prevent template rendering during actions
Implement dirty property tracking
Invoke computed properties during template rendering and only execute once per request
Ability to specify a 'moduleRootURI' setting to change the URI path for cbwire
Upgrade Livewire JS to v2.10.6
Disable browser caching on XHR responses
Reduce payload bloat by removing unnecessary data in XHR requests and responses
Move internal methods to a separate Engine object to avoid collisions with user-defined methods
Reject incoming XHR request if 'X-Livewire' HTTP Header is not present
Dependency injection capabilities to cbwire components
Update to match Livewire's current incoming and outgoing HTTP responses
Ability to use Turbo to create Single-page Applications ( SPAs )
Option to specify the component's template path using this.template instead of defining renderIt() method.
Specify null values for data properties
Calling reset( "someProperty" ) throws error
Browser back history doesn't work
On subsequent renderings of components, it's changing the unique id and causing DOM diff issues
Livewire expects params to be an array
Not passing parameters when calling update methods
Events and listeners provide an elegant means for your components to communicate. You can dispatch events from one component, listen for them on another, execute actions, and re-render the listener.
You can send parameters as your dispatching an event.
You can dispatch events to components of a specific component using dispatchTo().
You can dispatch events to the component that fired the event using dispatchSelf().
You register event listeners by adding a listeners struct to your component.
Listeners are a key/value pair where the key is the event to listen for, and the value is the action to invoke on the component.
You can also dispatch browser events using the dispatch() method.
You can listen for the event using vanilla JavaScript:
The lifecycle methods are executed in the following order when a component is initially loaded:
onMount()
onRender()
Lifecycle methods are executed in this order for subsequent AJAX requests.
onHydrate[DataProperty]()
onHydrate()
onUpdate[DataProperty]()
onUpdate()
Fire actions
onRender()
It runs only once when a component is initially wired. This can inject data values into your component via params you pass in when calling wire(), or pulling in values from the RC or PRC scopes.
onMount() only fires when the component is initially rendered and does not fire on subsequent requests when your component re-renders. This can cause issues referencing things such as the RC or PRC scope. If you pass in values with the RC or PRC scope, you must store them as data properties to ensure they are available to your component in subsequent requests.
It runs on all requests before rendering your component. This gives you more control if needed when rendering. There is also a renderIt() alias, which does the same.
Runs on subsequent requests after a specific data property is hydrated but before computed properties are rendered, before an action is performed, or before the component is rendered.
JavaScript parses your data properties. Your data properties can only store JavaScript-friendly values: strings, numerics, arrays, structs, or booleans.
Single or double quotes must surround property names.
JavaScript is a case-sensitive language, and CFML isn't. To preserve the casing of your property names, you must surround them with quotes.
Don't do this.
Data properties are visible to JavaScript. You SHOULD NOT store sensitive data in them.
You can reset individual, some, or all data properties.
You also can reset all properties EXCEPT the properties you pass.
Here is a basic example of how to use it:
Some examples of events you can listen for include:
click
wire:click
keydown
wire:keydown
submit
wire:submit
On some elements, such as forms or links, you need to add a .prevent modifier to prevent the browser's default behavior. Otherwise, the browser will cause the page to reload and you will get unintended results.
You can pass parameters to actions such as actionName( arg1, arg2, arg3 ).
The parameter is then passed through to your actions via function arguments.
$refresh
Will re-render the component without firing any action
$set( 'dataproperty', value )
Shortcut to update the value of a property
$toggle( 'dataproperty' )
Shortcut to toggle boolean properties off and on
Consider the example below.
You can dispatch events directly in your component using dispatch().
You can dispatch events from your using $dispatch().
You can also dispatch using the .
If you use , you can listen for an event like this.
CBWIRE provides lifecycle methods you can hook into to update and render your .
Runs on subsequent requests after a component is hydrated but before are rendered, before a is updated or is performed, or before the component is rendered.
Runs on subsequent requests after any is updated using wire:model or $set.
Runs on subsequent requests after a is updated using wire:model or $set. It only runs when the targeted data property is updated.
Data Properties hold the state of our component and are defined with a data structure in your . Each data property is assigned a default value.
Using the data structure, you can access data properties from within your component .
You can access data properties within your using #propertyName#.
You can reset all data properties to their original default value inside your using reset().
Actions are methods on your that either change the component's or perform some routine, such as updating your database or anything you can dream up in CFML.
Livewire listens for browser events and invokes actions on your using directives. These directives are used in your HTML and follow the format: wire:[browser event]=[action].
There are a few magic actions already created on your .
You can also pass parameters.
Computed Properties are dynamic properties that are cached per component rendering.
They are declared as functions within your component with a computed attribute added.
They are cached.
You can define Computed Properties on your components as functions with the computed attribute added.
You can access Computed Properties in your component template using propertyName().
Computed Properties cache their results for the lifetime of the request. It will only execute once if you reference your Computed Property three times in your component template or from within a component action.
You can prevent caching on a computed property by passing a false argument when invoking it.
CBWIRE includes Alpine.js, and uses Livewire.js for all client-side functionality and DOM diffing. These tools give you complete control over client-side functionality while providing ways to interact with your component server-side.
Alpine.js is included with CBWIRE and was designed to work beautifully with your CBWIRE components.
The $wire object is one of the most powerful features in Livewire.
With $wire, you get a JavaScript representation of your server-side CBWIRE component that you can interact with. You can change data properties, call actions, access parent components, fire events, and much more.
Below are the properties and methods of the $wire object:
You can execute bespoke JavaScript in your CBWIRE component using <cbwire:script>.
Here's a complete example:
Unlike assets, which are loaded only once per page, scripts are evaluated for every component instance.
You can load style assets on the page with your component using <cbwire:assets>.
You can access your component's $wire object within your scripts block.
You can access the livewire:init and livewire:initialized events as Livewire is loading.
You can interact with Livewire from external scripts using the Livewire global object.
Livewire allows you to register custom directives using Livewire.directive().
A commit is made every time a CBWIRE component is sent to the server. You can hook into an individual commit like this:
Using the request hook, you can hook into the entire HTTP request, which may contain multiple commits.
If the default page expiration dialog isn't suitable for your application, you can use the request hook to implement a custom solution.
Let's combine these into a single file.
There are only a few differences with the single-file format.
The contents of your .cfc file are instead placed inside a <cfscript> block
The // @startWire and // @endWire comment markers have been added so that CBWIRE can parse the file.
You must add the //@startWire and //@endWire markers when using single-file format.
There is a slight performance hit when using the single file format because CBWIRE has to parse out the sections of your file. However, this hit is minimal because of the internal caching CBWIRE uses.
Test your components using TestBox.
CBWIRE includes a beautiful test API to test your front-end components quickly.
You can invoke your component by calling wire() and then chain additional state changes and assertions.
Set a data property to the specified value.
Set a computed property to the specified closure.
Toggles a data property between true and false.
Calls an action. An optional array of parameters can be provided.
Emits an event and fires any listeners that are defined on the component. An optional array of parameters can be provided, which will be passed on to listeners.
Verifies a value can be found in the current component rendering. Otherwise, the test fails.
Verifies a value is not found in the current component rendering. Otherwise, the test fails.
Verifies a data property matches a specified value. Otherwise, the test fails.
Verifies a data property does not match a specified value. Otherwise, test fails.
Computed Properties are similar to with some key differences:
They can return any CFML data type, not just values that can be parsed by JavaScript like .
Computed properties are meant to return values and not change state such as modifying . If you need to update a data property, use instead.
You can also access Computed Properties from within your .
Using Alpine.js is as easy as declaring an x-data in your . You can declare client-side properties, actions, and interact with your component using the $wire object.
Learn more on the features page.
Here's an example of loading a date picker called :
Here's the implementation for :
You can bind your to input elements within your using .
See the page for complete documentation.
When are updated using wire:model, CBWIRE has several methods you can hook into such as onUpdate and onHyrdate ( See ).
are typically built with a .cfc file and a .cfm file. You can combine these into a single .cfm file, similar to how components in Vue.js are built.
Let's look at the example from the that uses two separate files.
Set your data properties with incoming query string values.
It can sometimes be helpful to update the browser's query string when your component's state changes.
Let's say we are building a component to search articles.
We can automatically populate the search property above from the URL query string by adding a queryString array to our component.
Now when we access the page with 'search' included in the query string (/search-articles?search=some+string), the search property will have a default value of "some string".
In addition to this, when the user starts typing into our search field, the URL displayed in the browser will be instantly updated.
Coming soon.
A slow component can slow down the loading of an entire page. CBWIRE's lazy loading feature allows you to delay loading your components until the page is fully rendered.
Let's create a slow component that users will hate.
Changing this to a lazy-loaded component is as simple as using wire( lazy=true ).
This will cause Livewire to skip this component on the initial page load. Once visible in the viewport, Livewire will request a network to load the component fully.
As your components load, you often want to display some indicator, such as a spinner, to your users. You can do this easily using the placeholder() method.
The placeholder and the main component must use the same root element type, such as a div. You will get rendering errors if you don't use the same root element.
You can render a ColdBox view for a placeholder using the view() method.
Below are the currently available constraints.
You can also run validations against any constraints you provide by passing them to the validate() method, which returns a ValidateResult object.
You can use hasErrors() and getErrors() to check for validation errors that have occurred and to display all errors.
To check for errors on a specific data property, you can use hasError( field) and getError( field )
You can use ColdBox's validation engine and to validate your forms.
To install cbValidation, run the following command in :
You can define constraints within your by populating a constraints struct. Once constraints are defined, CBWIRE will automatically validate your component on each request.
Review the for additional information.
You can manually validate your using the validate() or validateOrFail() methods.
Silently fails and prevents further processing of the current .
In your , you have several helper methods for displaying errors.
This is a great way to keep your business logic out of your components.
You can achieve this by defining a CFC property in your component and using inject.
You can use getInstance() to access a dependency from within your actions.
Here is the method signature for getInstance():
You can access any dependencies your component may have using , ColdBox's robust dependency injection framework.
Learn about the full capabilities of WireBox at .
You can listen for click events within your using wire:click and provide an to run when clicked.
You can tell CBWIRE to ignore updating parts of your using wire:ignore. This is especially useful when using Alpine.js and other third-party libraries.
Using wire:key is essential, especially when looping over elements, to ensure that Livewire's DOM diffing correctly identifies what has changed and needs updating in the browser.
Consider this listing of posts.
As posts are removed, CBWIRE re-renders our template, and then Livewire must figure out how to update the DOM. We can help Livewire know exactly what has changed using wire:key.
It's important to ensure your key names are unique throughout the page, not just within your template. Otherwise, you may encounter weird rendering issues.
You can use wire:confirm in your to prompt users for confirmation before executing actions. This can be useful when dealing with potentially irreversible actions such as deletions or updates.
You can use Livewire's wire:navigate feature to speed up page navigation and give your users a SPA-like experience.
When any link with wire:navigate is clicked, Livewire intercepts the click. Instead of allowing the browser to perform a full page visit, Livewire fetches the page in the background and swaps it with the current page. This results in much faster and smoother page navigation.
Here's a breakdown of what happens:
The user clicks a link
Livewire prevents the browser from visiting the new page
Livewire requests the page in the background and shows a loading bar at the top of the page
Once Livewire receives the HTML for the new page, it replaces the current URL, <title> tag, and <body> contents.
This technique results in much faster page load times, often twice as fast, and makes your application feel like a JavaScript-powered single-page application.
You can add the .hover modifier to make Livewire prefetch a page when the user hovers over a link. This will ensure that the page has already been downloaded from the server when the user clicks on the link, making the contents load faster in the browser.
Prefetching on hover increases server usage.
Sometimes, you need to persist parts of a user interface between page loads, such as audio or video players.
Here is an example of an audio player being persisted across pages.
By default, Livewire preserves the scroll position of a page when navigating between pages. However, there may be times when you want to preserve the scroll position of an individual element between page loads.
To do this, you must add Livewire's wire:scroll to the element containing a scrollbar.
Each page navigation triggers three lifecycle hooks.
livewire:navigate
livewire:navigating
livewire:navigated
Attaching an event listener to the document will not be removed when you navigate to a different page. This can lead to unexpected behavior if you need code to run only after navigating to a specific page.
You can remove an event listener after it runs by passing the option {once: true} as a third parameter to the addEventListener function.
You can use Livewire.navigation() to trigger a visit to a new page using JavaScript.
When navigating pages using wire:navigate, any <script> tags in the <head> are evaluated once when the page is initially loaded.
This creates a problem for some analytics software as these tools rely on a <script> snippet evaluated on every page change, not just the first.
You can ensure your script tag is tracked properly using data-spa="auto".
When navigating to a new page using wire:navigate, it feels like the browser has changed pages. However, from the browser's perspective, you are technically still on the original page.
Here are a few caveats to consider.
Scripts in <body> are re-evaluated on each page load. You can use data-navigate-once to tell Livewire only to evaluate once.
Head assets are blocking. When you navigate to a new page that contains assets like <script src="..."> in the head tag, those will be fetched and processed before the navigation is complete and the new page is swapped in.
Using the available JavaScript hooks, you can create your own custom progress loader. Here's an example you could place in your layout file to show a custom loader.
You can instruct Livewire to use its wire:navigate functionality to load the new page when using CBWIRE's redirect() method by passing redirectUsingNavigate as true.
You can achieve this using Alpine's plugin, which is included with CBWIRE.
Persisted elements must be placed outside your CBWIRE . A common practice is to put these in your main layout.
Livewire will show a progress bar at the top of the page when a page takes longer than 150ms to load. You can customize the bar's color or disable it altogether in your .
When the user submits the form, the server receives a request with the updated data property values from the inputs. An email is sent, and the template is re-rendered with the updated values.
You can use wire:model.live to send property updates to the server as a user types.
You can customize the debounce using wire:model.live.debounce.Xms.
Livewire provides various modifiers to control when server requests are sent.
Below are all the available modifiers:
.live
Send updates as user types
.blur
Only send updates on blur event
.change
Only send updates on the change event
.lazy
Alias for .change
.debounce.Xms
Debounce sending updates for X milliseconds
.throttle.Xms
Throttle network request updates by X milliseconds
.number
Cast text input to an integer on the server
.boolean
Cast text input to a boolean on the server
.fill
Use the initial value the "value" HTML attribute provides during page load.
If the content data property is initialized with a string, Livewire will fill the textarea with its value. Don't do this.
Checkboxes with multiple inputs are stored as an array.
You can dynamically populate these with values.
You can use wire:model in your to bind to with form inputs.
When using wire:model, Livewire will only send a network request when an action is performed, like with or . However, you may want to update the server more frequently for things like real-time validation. In those instances, use wire:model.live.
Below shows how to bind various input fields with your .
If you have a dropdown that is dependent on the value of another dropdown and needs to update, you can use to ensure it updates.
Polling is a straightforward yet effective technique used in web applications to continuously request updates from the server at regular intervals. This method is essential for keeping page content fresh without the complexity of technologies like WebSockets.
In CBWIRE, implementing polling is as simple as adding wire:poll to an element in your component.
Here’s an example of a Subscriber Count component that dynamically updates the user's subscriber count:
In your subscriberCount view file (wires/subscriberCount.cfm):
Normally, the subscriber count would only update upon page refresh. However, with wire:poll, the refreshSubscribers() method is called every 2.5 seconds, ensuring the count is always up-to-date.
Polling can be resource-intensive, particularly with many users. To manage this, CBWIRE allows you to control the polling interval:
CBWIRE intelligently throttles polling when the webpage is in the background. This reduces server load significantly. However, if continuous polling is necessary, you can use the .keep-alive modifier:
This ensures polling continues even when the tab is not active.
To optimize performance further, you can use the .visible modifier, which makes CBWIRE poll only when the element is visible on the user’s screen:
This is useful for content lower on a page, as it starts polling only when scrolled into view and stops when it's no longer visible.
You can redirect users in your actions using redirect().
As the user begins typing in the name or email field, CBWIRE will detect the data properties that have changed on the client side and display a reminder to click submit. After the user clicks submit, the data properties are synchronized, and the reminder is removed.
You can invert the behavior of wire:dirty by using the .remove modifier:
You can target specific data properties using wire:dirty and wire:target.
You can toggle classes on elements using wire:dirty.class.
When the user types into the name field, a yellow border is displayed, letting the user know the changes have not been saved.
You can use wire:dirty with to display elements when have been updated on the client side but not the server side.
You can run an once your component is rendered in the browser using wire:init. This can be helpful when you don't want to hold up loading the entire page but want to load some data immediately after the page loads.
The need for wire:init has largely been replaced by CBWIRE's , but still exists and can be used if you prefer.
CBWIRE provides offline state management, making it easy to toggle UI elements when the user is offline or online.
You can use wire:offline to display elements when CBWIRE detects that the user is offline.
Copy
You can append .class to your wire:offline directive and specify a CSS class to toggle when the user is offline.
Copy
Use .remove to specify classes you want to be removed when offline.
Copy
CBWIRE offers a smooth way to show or hide elements on your webpage with the wire:transition directive. This feature enhances user experience by making elements transition in and out smoothly, rather than just popping into view.
For instance, let’s say you’re working on a component that shows comments on a blog post. You want to give users the option to view comments with a nice visual effect. Here’s how you can set it up:
In your view, you can use wire:transition to handle the visibility of the comments section:
With wire:transition, when you click "Show comments," the comments section will gracefully fade into view instead of appearing abruptly.
Currently, wire:transition should be applied to a single conditional element and doesn’t work as expected with a list of dynamic elements, like individual comments in a loop. This is due to the limitations of how transitions are handled in the underlying framework.
By default, elements with wire:transition will fade (change opacity) and slightly scale. Here’s what that looks like:
Fading in: Opacity transitions from 0 to 100%.
Scaling in: Transforms from slightly smaller to full size.
CBWIRE allows you to customize these transitions to fit your needs. You can specify whether to only apply the transition when elements enter or leave the DOM, adjust the duration and delay of the transitions, or only use certain effects like opacity or scaling:
.in: Only apply when the element appears.
.out: Only apply when the element disappears.
.duration.[?ms]: Set how long the transition takes in milliseconds.
.opacity: Only use the opacity transition for a simple fade effect without scaling.
.scale: Apply a scaling effect.
.origin.[top|bottom|left|right]: Set the origin point for the scale effect, useful for dropdowns or popups.
For example, to make a dropdown menu that fades out and scales down from the top when closed, you might set up your transition like this:
As you use CBWIRE, you are likely to run into issues from time to time. The most common issues are rendering issues. Here, we try to address the most common problems and show how to solve them.
Consider the following widget example.
Let's include our widget somewhere on our site and add lazy loading.
What should happen on page load is our placeholder is shown first, and then our widget renders to the page. Instead, we get a weird error like this.
Snapshot missing on Livewire component with id K8SDFLSDF902KSDFLASKJFASDFLJ.
The Livewire JavaScript error isn't super helpful in identifying the problem because the real issue is a mismatch between your widget's outer element and its placeholder's outer element.
Notice our widget's template above has an outer <div> but our placeholder has an outer <section> element. This will not work. Your placeholder and your corresponding template must have the same outer element. Otherwise, Livewire's DOM diffing engine gets confused.
The fix would be to make them both have outer <div> tags.
Consider the following:
Notice we have an alert that is displayed when there are errors in our errorList
variable.
Depending on the additional sections within your page and how deeply nested each section is, Livewire might have difficulty detecting the alert has been added on re-rendering and that a DOM update is required.
We can let Livewire know there is a conditional section by adding an if-BLOCK comment.
While Alpine is lightweight and simple to use, it's easy to get tripped up when getting started.
See if you can spot what is wrong with this code.
The code above will result in a JavaScript error.
The problem is our use of double quotes for our name property. Because our x-data block is surrounded by double quotes, using double quotes inside the block isn't proper syntax.
Change to using single quotes instead, and all is well again.
You can run into rendering issues when using Alpine's <template> tag with an x-if if you are also including a component within the tag using wire().
For example:
As you toggle the loading value from true to false, you may see a JavaScript error similar to this. The component ID will be different
Unable to find component K8SDFLSDF902KSDFLASKJFASDFLJ.
The issue above is that Alpine.js' x-if directive completely removes the contents inside from the DOM and recreates them when true. Livewire.js' DOM diffing engine is expecting the spinner to be there, and it can't find it, hence the error.
Luckily, the simple fix is to change your <template x-if> to something like a <div> and use x-show instead.
You'll still achieve the desired result and Livewire.js can track the spinner accurately.
When an is executed on your components, the component is re-rendered, and then it's Livewire's job to figure out how to update the DOM. For the most part, the DOM updates are completed without issue. However, you may run into problems with conditional areas ( if-else statements ) in your .
Coming soon
After clicking the save button, you will briefly see the saving loading indicator. Once the save completes, the loading indicator goes away.
You can instead show elements by default and hide them during requests to the server using wire:loading.remove.
Now the save button will disappear until the save completes.
You can toggle classes using wire:loading.class.
You can also remove classes using wire:loading.class.remove.
You can toggle attributes using wire:loading.attr.
By default, wire:loading will fire when ANY action is called. You can target specific actions using wire:target.
The saving loading indicator is displayed when the save button is pressed and not when the reset button is pressed.
You can provide multiple targets to wire:target.
You can specify parameters to match against when using wire:target="action(param)".
Above, our loading indicator is only displayed for the post that is being removed.
Multiple action parameters are currently not supported. This will not work.
You can use wire:loading with wire:model.live to display elements during data property updates.
You can exclude targets using wire:target.except.
By default, CBWIRE uses display: none to hide elements and display: inline-block to show elements.
When toggling elements with a display value other than inline-block, you can use wire:loading.[value].
Below are the available display values:
Sometimes, loading indicators are so fast that they only display on the screen briefly before being removed. This can be jarring and distracting to users.
You can delay showing an indicator using wire:loading.delay.
This will prevent the indicator from showing unless the server request takes 200ms or more.
There are built-in interval aliases you can use as well.
Using wire:loading, you can show and hide elements in your while a request is sent to the server.
Provide your users with file uploads and thumbnail previews without page refreshes.
CBWIRE makes uploading and storing files easy.
Here's an example of a simple Wire that handles uploading a photo:
Handling file inputs is no different than handling any other input type with CBWIRE. Add wire:model to the input tag and CBWIRE will take care of the rest.
Several things are happening under the hood to make file uploads work:
CBWIRE makes an initial request to the server to get a temporary "signed" upload URL.
Once the URL is received, JavaScript then does the actual "upload" to the signed URL, storing the upload in a temporary directory designated by CBWIRE and returning the new temporary file's unique hash ID.
Once the file is uploaded, and the unique hash ID is generated, CBWIRE makes a final request to the server, telling it to "set" the desired data property to the upload file as an instance of FileUpload ( see below ).
Now the data property (in this case, photo ) is set to an instance of FileUpload@cbwire and is ready to be stored or validated at any point.
CBWIRE will automatically upload your files and set your associated data property to an instance of FileUpload@cbwire. Here are some of the methods you can use that are helpful.
getComp()
Returns an instance of your current Wire.
getParams()
Returns any params that are passed in, including the data property name.
get()
Returns the contents of the file uploaded.
getBase64()
Returns base64 encoded string of the file.
getBase64Src()
Returns a base64 src string that can be used with <img tag>. It is recommended to use this carefully because with larger objects, this can slow down CBWIRE response times.
getTemporaryStoragePath()
Returns the temporary storage path where the file is stored by CBWIRE.
getMetaPath()
Returns the file path to meta information of the uploaded file. You can find additional details here like the name of the file when it was uploaded, the client file extension, etc.
getMeta()
Returns all captured meta information on the file upload.
getSize()
Returns the size of the file upload.
getMimeType()
Returns the mime type of the file upload.
isImage()
Returns true if this is an image.
getPreviewURL()
Provides a URL to preview the file uploaded. It only works with image uploads.
destroy()
Deletes the temporary storage and metadata of the file upload. It's essential to use this after storing the file upload in permanent storage.
You can display a loading indicator scoped to the file input during upload like so:
The "Uploading..." message will be shown as the file is uploading and then hidden when the upload is finished.
This works with the entire Loading States API.