- Allow configuring the secret used to compute fingerprints and checksums.
- Remove CSRF tokens - rely on same-origin/CORS instead
- Add
submitForm()
toTestLiveComponent
. - Add
live_action
Twig function
- Add parameter to
TestLiveComponent::call()
to add files to the request
- Add
modifier
option inLiveProp
so options can be modified at runtime. - Fix collections hydration with serializer in LiveComponents
- Add
loading
attribute to defer the rendering on the component after the page is rendered, either when the page loads (loading="defer"
) or when the component becomes visible in the viewport (loading="lazy"
). - Deprecate the
defer
attribute. - Add
UrlMapping
configuration object for URL bindings in LiveComponents
-
LiveComponents is now stable and no longer experimental 🥳
-
[BC BREAK] The
data-action-name
attribute behavior was removed in favor of using Stimulus "action parameters" anddata-live-action-param
. This is a breaking change if you were using thedata-action-name
attribute directly in your templates. #1418To upgrade your application, follow these changes:
<button data-action="live#action" - data-action-name="debounce(300)|save" + data-live-action-param="debounce(300)|save" >Save</button>
To pass arguments to an action, also use the Stimulus "action parameters" syntax:
<button data-action="live#action" - data-action-name="addItem(id={{ item.id }}, itemName=CustomItem)" + data-live-action-param="addItem" + data-live-id-param="{{ item.id }}" + data-live-item-name-param="CustomItem" >Add Item</button>
Additionally, the
prevent
modifier (e.g.prevent|save
) was removed. Replace this with the standard Stimulus:prevent
action option:<button - data-action="live#action + data-action="live#action:prevent" - data-action-name="prevent|save" + data-live-action-param="save" >Save</button>
-
[BC BREAK] The
data-event
attribute was removed in favor of using Stimulus "action parameters": renamedata-event
todata-live-event-param
. Additionally, if you were passing arguments to the event name, use action parameter attributes for those as well - e.g.data-live-foo-param="bar"
. #1418 -
Reverted setting
ignoreActiveValue: true
in Idiomorph #1548 -
New placeholder macro to generate defer/lazy skeleton #1532
-
improve TestLiveComponent::actingAs() #1461
-
Drop Twig 2 support #1436
-
Add better error message when hydrating dates #1431
-
Store TemplateMap in build_dir #1525
- [BC BREAK] The
data-live-id
attribute was changed toid
#1484 - Fixed child handling bug during re-rendering introduced with the new morphing library in 2.14.0 #1484
- Fix bug where the active input would maintain its value, but lose its cursor position #1501
- Restrict Twig 3.9 for now #1486
- Fixed a regression in the testing tools related to the default HTTP method change
- [BC BREAK] DOM morphing changed from
morphdom
toidiomorph
. As this is a different morphing library, there may be some edge cases where the morphing behavior is different. - Add support for URL binding in
LiveProp
- Allow multiple
LiveListener
attributes on a single method - Requests to LiveComponent are sent as POST by default
- Add method prop to AsLiveComponent to still allow GET requests, usage:
#[AsLiveComponent(method: 'get')]
- Add a new
urlReferenceType
parameter toAsLiveComponent
, which allows to generate different type URL (e.g. absolute) for the component Ajax calls - The
symfony/serializer
dependency is now optional - Added a
data-skip-morph
attribute to allow skipping morphing of an element (the element's attributes will be morphed, but its inner HTML will be overwritten instead of morphed) - Added an entry to the packages'
package.json
file so that@symfony/ux-live-component
will appear in the user'simportmap.php
file if using AssetMapper. This will allow using the JavaScript from the package without extra setup. - Fixed edge-case rendering bug where a 2nd Ajax request might start before the 1st finished processing
- Fix usage of
{% embed %}
with{% block %}
in<twig:>
components - Fixed
data-loading
not working when on root element of a component - Fixed error when
class
attributes contained a space at start or end - Fixed loading directives being matched in a child component
- Revert "Change JavaScript package to
type: module
"
- Add deferred/lazy rendering of Live Components.
- Fix option tag synchronization.
- Handle array-like objects when working with checkboxes.
- Add Symfony 7 support.
- Normalize "true" & "false" model values
- Fix DTO hydration from phpdoc typehints.
- Fix instantiating LiveComponentMetadata multiple times.
- Change JavaScript package to
type: module
. - Throwing an error when setting an invalid model name.
- Add support for (de)hydrating DTO classes in
LiveProp
. - Fixed
emit()
method ofTestLiveComponent
to properly test events. - Add
actionAs()
toTestLiveComponent
. - Fixed rendering bug when using Chrome's translation feature.
- Add
onUpdated()
hook forLiveProp
. - Fix support for Alpine.js & live components.
- Add helper for testing live components.
- Add initial file upload support.
- Respect
data-turbo="false"
when handling redirects. - Fix checksum calculation for deeply nested data.
- Add support for symfony/asset-mapper
- Increased the priority of
LiveComponentSubscriber
ControllerEvent
from 0 to 10 to fix incompatibility with SensioFrameworkExtraBundle.
- [BC BREAK]: The
exposed
option was changed towritable
inLiveProp
:
-#[LiveProp(exposed: ['email', 'plainPassword'])]
+#[LiveProp(writable: ['email', 'plainPassword'])]
public User $user;
-
[BC BREAK]:
LiveProp
values are no longer automatically (de)hydrated through Symfony's serializer. UseLiveProp(useSerializerForHydration: true)
to activate this. Also, aserializationContext
option was added toLiveProp
. -
[BC BREAK]: Child components are no longer automatically re-rendered when a parent component re-renders and the value of one of the props passed to the child has changed. Pass
acceptUpdatesFromParent: true
to anyLiveProp
on the child component to re-enable this behavior. -
Non-persisted entity objects can now be used with
LiveProp
: it will be serialized using the serializer. -
Better support for using arrays with
LiveProp
. -
Smart rendering system! If you have JavaScript that makes changes to the DOM inside a live component, those changes will now be kept when the component is re-rendered. This has limitations - see the documentation.
-
You can now
emit()
events to communicate between components. -
You can now dispatch DOM/browser events from components.
-
Boolean checkboxes are now supported. Of a checkbox does not have a
value
attribute, then the associatedLiveProp
will be set to a boolean when the input is checked/unchecked. -
A
format
option was added toLiveProp
to control howDateTime
properties are (de)hydrated. -
Added support for setting
writable
to a property that is an object (previously, only scalar values were supported). The object is passed through the serializer. -
Invalid data sent by the user is now handled in a robust way. Previously, if the user sent invalid data (e.g. a string for a
LiveProp
that has anint
type), the component update would break. Now, if the new data cannot be hydrated onto the object during a re-render, the last valid value is used. -
When using
ValidatableComponentTrait
, a new_errors
variable is sent to the template, which is easier to use! -
Several bug fixes to parent - child components - see #700.
-
Fixed handling of boolean attributes to a component - see #710.
-
Fixed performance calculating component fingerprint for large components.
-
[BC BREAK]: The "key" used to load the controller in your
assets/controllers.json
file changed fromtyped
tolive
. Update yourassets/controllers.json
file to change this key. -
Add a strategy for adding a Stimulus controller to a Twig component - #589.
-
Added a new
getCompontent()
function in JavaScript as the best way to find a Component object for a given element. -
Fixed various bugs related to child component handling - #596
-
Added a new
route
parameter toAsLiveComponent
, which allows to choose another route for Ajax calls. -
Add
assets/src
to.gitattributes
to exclude source TypeScript files from installing. -
TypeScript types are now included.
-
Added new
response:error
JavaScript component hook for custom handling Ajax errors - #587.
- [BC BREAK]: The path to
live_component.xml
changed and the import now MUST have aprefix
: you should update your route import accordingly (the name of the route also changed toux_live_component
):
# config/routes/ux_live_component.yaml
live_component:
- resource: '@LiveComponentBundle/Resources/config/routing/live_component.xml'
+ resource: '@LiveComponentBundle/config/routes.php'
+ prefix: /_components
- Removed
Content-Type
header when returning the empty response redirect. - Fixed bug when re-rendering SVG's (.#557)
-
[BEHAVIOR CHANGE] Previously, Ajax calls could happen in parallel (if you changed a model then triggered an action before the model update Ajax call finished, the action Ajax call would being in parallel). Now, if an Ajax call is currently happening, any future requests will wait until it finishes. Then, all queued changes (potentially multiple model updates or actions) will be sent all at once on the next request.
-
[BEHAVIOR CHANGE] Fields with
data-model
will now have theirvalue
set automatically when the component initially loads and re-renders. For example, previously you needed to manually set the value in your component template:<!-- BEFORE --> <input data-model="firstName" value="{{ firstName }}">
This is no longer necessary: Live Components will now set the value on load, which allows you to simply have the following in your template:
<!-- AFTER --> <input data-model="firstName">
-
[BEHAVIOR CHANGE] The way that child components re-render when a parent re-renders has changed, but shouldn't be drastically different. Child components will now avoid re-rendering if no "input" to the component changed and will maintain any writable
LiveProp
values after the re-render. Also, the re-render happens in a separate Ajax call after the parent has finished re-rendering. -
[BEHAVIOR CHANGE] If a model is updated, but the new value is equal to the old one, a re-render will now be avoided.
-
[BEHAVIOR CHANGE] Priority of
DoctrineObjectNormalizer
changed from 100 to -100 so that any custom normalizers are used before tryingDoctrineObjectNormalizer
. -
[BC BREAK] The
live:update-model
andlive:render
events are not longer dispatched. You can now use the "hook" system directly on theComponent
object/ -
[BC BREAK] The
LiveComponentHydrator::dehydrate()
method now returns aDehydratedComponent
object. -
Added a new JavaScript
Component
object, which is attached to the__component
property of all root component elements. -
the ability to add
data-loading
behavior, which is only activated when a specific action is triggered - e.g.<span data-loading="action(save)|show">Loading</span>
. -
Added the ability to add
data-loading
behavior, which is only activated when a specific model has been updated - e.g.<span data-loading="model(firstName)|show">Loading</span>
. -
Unexpected Ajax errors are now displayed in a modal to ease debugging! #467.
-
Fixed bug where sometimes a live component was broken after hitting "Back: in your browser - #436.
-
[BC BREAK] Previously, the
id
attribute was used withmorphdom
as the "node id" when updating the DOM after a render. This has changed todata-live-id
. This is useful when maintaining the correct order of a list of elements. -
[BC BREAK] If using
LiveCollectionType
, the name of the remove field changed frombutton_delete_prototype
tobutton_delete
and the add field changed frombutton_add_prototype
tobutton_add
. Additionally, theallow_add
andallow_delete
default values were changed fromfalse
totrue
. -
[BEHAVIOR CHANGE] If an action Ajax call is still processing and a model update occurs, the component will no longer re-render. The model will be updated internally, but not re-rendered (so, any model updates would effectively have the
|norender
modifier). See #419.
- [BC BREAK] The
data-action="live#update"
attribute must now be removed from nearly all elements. This is because LiveComponents now automatically listens to theinput
event on all elements with adata-model
attribute and updates the data. If you previously useddata-action="change->live#update"
to list on thechange
event, now you should use theon(change)
modifier insidedata-model
.
<!-- BEFORE -->
<input
data-model="max"
data-action="change->live#update"
>
<!-- AFTER -->
<input
data-model="on(change)|max"
>
- [BC BREAK] The
live#updateDefer
action was removed entirely. Now, to update a model without triggering a re-render, use thenorender
modifier fordata-model
:
<!-- BEFORE -->
<input
data-model="max"
data-action="live#updateDefer"
>
<!-- AFTER -->
<input
data-model="norender|max"
>
- [BC BREAK] The
name
attribute is no longer automatically used to update a model when a parent component hasdata-action="change->live#update"
. To make a form's fields behave like "model" fields (but using thename
attribute instead ofdata-model
) you need to add adata-model
attribute to the<form>
element around your fields (NOTE: the new attribute is automatically added to yourform
element when usingComponentWithFormTrait
):
<!-- BEFORE -->
<form data-action="change->live#update">
<input
name="max"
>
</form>
<!-- AFTER -->
<form data-model="on(change)|*">
<input
name="max"
>
</form>
-
The bundle now properly exposes a
live
controller, which can be imported via yourassets/controllers.json
file (like any other UX package). Previously, the controller needed to be imported and registered with Stimulus directly (usually in yourassets/bootstrap.js
file). That is no longer needed. -
Add a generic
LiveCollectionType
andLiveCollectionTrait
-
Allow to disable CSRF per component
-
Your component's live "data" is now send over Ajax as a JSON string. Previously data was sent as pure query parameters or as pure POST data. However, this made it impossible to keep certain data types, like distinguishing between
null
and''
. This has no impact on end-users. -
Added
data-live-ignore
attribute. If included in an element, that element will not be updated on re-render. -
ComponentWithFormTrait
no longer has asetForm()
method. But there is also no need to call it anymore. To pass an already-built form to your component, pass it as aform
var tocomponent()
. If you have a custommount()
, you no longer need to callsetForm()
or anything else. -
The Live Component AJAX endpoints now return HTML in all situations instead of JSON.
-
Ability to send live action arguments to backend
-
[BC BREAK] Remove
init_live_component()
twig function, use{{ attributes }}
instead:- <div {{ init_live_component() }}> + <div {{ attributes }}>
-
[BC BREAK] Replace property hydration system with
symfony/serializer
normalizers. This is a BC break if you've created custom hydrators. They'll need to be converted to normalizers. -
[BC BREAK] Rename
BeforeReRender
attribute toPreReRender
.
-
Support for
stimulus
version 2 was removed and support for@hotwired/stimulus
version 3 was added. See the @symfony/stimulus-bridge CHANGELOG for more details. -
Require live components have a default action (
__invoke()
by default) to enable controller annotations/attributes (ie@Security/@Cache
). AddedDefaultActionTrait
helper. -
When a model is updated, a new
live:update-model
event is dispatched. Parent components (in a parent-child component setup) listen to this and automatically try to update any model with a matching name. Adata-model-map
was also added to map child component model names to a parent - see #113. -
Child components are now re-rendered if the parent components passes new data to the child when rendering - see #113.
-
Minimum PHP version was bumped to 8.0 so that PHP 8 attributes could be used.
-
The
LiveComponentInterface
was dropped and replaced by theAsLiveComponent
attribute, which extends the newAsTwigComponent
from the TwigComponent library. All other annotations (e.g.@LiveProp
and@LiveAction
) were also replaced by PHP 8 attributes.
Before:
use App\Entity\Notification;
use App\Repository\NotificationRepository;
use Symfony\UX\LiveComponent\Attribute\LiveAction;
use Symfony\UX\LiveComponent\Attribute\LiveProp;
use Symfony\UX\LiveComponent\LiveComponentInterface;
final class NotificationComponent implements LiveComponentInterface
{
private NotificationRepository $repo;
/** @LiveProp */
public bool $expanded = false;
public function __construct(NotificationRepository $repo)
{
$this->repo = $repo;
}
/** @LiveAction */
public function toggle(): void
{
$this->expanded = !$this->expanded;
}
public function getNotifications(): array
{
return $this->repo->findAll();
}
public static function getComponentName(): string
{
return 'notification';
}
}
After:
use App\Entity\Notification;
use App\Repository\NotificationRepository;
use Symfony\UX\LiveComponent\Attribute\AsLiveComponent;
use Symfony\UX\LiveComponent\Attribute\LiveAction;
use Symfony\UX\LiveComponent\Attribute\LiveProp;
#[AsLiveComponent('notification')]
final class NotificationComponent
{
private NotificationRepository $repo;
#[LiveProp]
public bool $expanded = false;
public function __construct(NotificationRepository $repo)
{
$this->repo = $repo;
}
#[LiveAction]
public function toggle(): void
{
$this->expanded = !$this->expanded;
}
public function getNotifications(): array
{
return $this->repo->findAll();
}
}
- The LiveComponent library was introduced!