wire:model modifiers (.blur, .change, .debounce) only control network request timing. Ephemeral (client-side) state ALWAYS syncs immediately as user types. Users can't delay ephemeral sync until blur/enter/change.
Split modifiers into two layers: modifiers BEFORE .live control ephemeral sync timing (forwarded to x-model), modifiers AFTER .live control network timing. Breaking change for v4.
wire:model {{-- ephemeral immediate, no network --}}
wire:model.blur {{-- ephemeral immediate, network on blur --}}
wire:model.live {{-- ephemeral immediate, network debounced --}}
wire:model.live.debounce.500ms {{-- ephemeral immediate, network debounced 500ms --}}- All modifiers only affect network layer
- No way to control ephemeral sync timing
.blur/.changedon't affect when x-model syncs
wire:model {{-- ephemeral immediate, no network --}}
wire:model.blur {{-- ephemeral on blur, no network --}}
wire:model.change {{-- ephemeral on change, no network --}}
wire:model.enter {{-- ephemeral on enter, no network --}}
wire:model.blur.enter {{-- ephemeral on blur OR enter, no network --}}
wire:model.live {{-- ephemeral immediate, network debounced --}}
wire:model.live.blur {{-- ephemeral immediate, network on blur (OLD .blur behavior) --}}
wire:model.blur.live {{-- ephemeral on blur, network on blur --}}
wire:model.blur.live.debounce.500ms {{-- ephemeral on blur, network debounced 500ms after --}}- Modifiers before
.live→ forwarded to x-model (controls ephemeral sync) - Modifiers after
.live→ handled by Livewire (controls network) - Everything chains sequentially: ephemeral syncs first, then network fires
js/directives/wire-model.js— main implementation (lines 1-186)node_modules/alpinejs/src/directives/x-model.js— reference for how Alpine handles.blur/.change/.enter(lines 58-98)src/Features/SupportDataBinding/BrowserTest.php— existing browser tests
let isLive = modifiers.includes('live')
let isLazy = modifiers.includes('lazy') || modifiers.includes('change')
let onBlur = modifiers.includes('blur')
// ... these only control WHEN update() is called for network
// x-model always gets modifiers minus 'lazy','defer','debounce','throttle'
// meaning .blur/.change still get forwarded but DON'T affect ephemeral sync
// because wire:model uses get/set object pattern, not event listenersProblem: getModifierTail() forwards .blur to x-model, but wire:model uses getter/setter pattern (x-model.blur() { get(), set() }), not Alpine's event-listener approach. So Alpine's .blur modifier has no effect.
-
Parse modifiers into two groups — split at
.liveboundaryephemeralModifiers= everything before.livenetworkModifiers= everything after.live
-
Create extraction function with clean interface:
let { ephemeralModifiers, // forwarded to x-model tail onEphemeralSync, // callback when ephemeral state changes shouldSendNetwork, // fn to check if network should fire } = parseWireModelModifiers(modifiers)
-
Forward ephemeral modifiers to x-model — these get added to
x-model.blur.change.enteretc. -
Add event listeners for network triggers — if network modifiers include
.blur, add@blurhandler. Same for.change,.enter. -
Handle debounce/throttle at network layer — wrap network callbacks in debounce/throttle based on network modifiers.
-
x-model set() callback triggers network — when x-model syncs (based on its modifiers), check if
.liveis present and call network update.
directive('model', ({ el, directive, component, cleanup }) => {
let { expression, modifiers } = directive
// Split modifiers at .live boundary
let liveIndex = modifiers.indexOf('live')
let ephemeralModifiers = liveIndex === -1 ? modifiers : modifiers.slice(0, liveIndex)
let networkModifiers = liveIndex === -1 ? [] : modifiers.slice(liveIndex + 1)
let isLive = liveIndex !== -1
// Build x-model modifier tail from ephemeral modifiers
let xModelTail = buildModifierTail(ephemeralModifiers)
// Build network update function with debounce/throttle
let networkUpdate = buildNetworkUpdate(networkModifiers, component, expression, el, directive)
// Determine network trigger events
let networkTriggers = extractNetworkTriggers(networkModifiers)
Alpine.bind(el, {
['@blur']() {
if (networkTriggers.blur) networkUpdate()
},
['@change']() {
if (networkTriggers.change) networkUpdate()
},
['@keydown.enter']() {
if (networkTriggers.enter) networkUpdate()
},
['x-model' + xModelTail]() {
return {
get() {
return dataGet(component.$wire, expression)
},
set(value) {
dataSet(component.$wire, expression, value)
// If live with no specific triggers, fire on every sync
if (isLive && networkTriggers.immediate) {
networkUpdate()
}
},
}
}
})
})- Breaking change in v4 — acceptable since v4 is ~2 weeks old
.liveis the delimiter — everything before controls ephemeral, after controls network- Modifiers compound —
.blur.entermeans blur OR enter (both trigger sync) - Network inherits ephemeral timing by default — if no network modifiers, network fires when ephemeral syncs
.livealone = debounced immediate — current behavior, ephemeral immediate + network debounced.live.blur= ephemeral immediate, network on blur — preserves old.blurbehavior for migration- Empty network modifiers with
.live— means "fire network when ephemeral syncs" (for.blur.live)
src/Features/SupportDataBinding/BrowserTest.php (add to existing file)
Single consolidated test with multiple scenarios:
wire:model— ephemeral syncs on input, no networkwire:model.blur— ephemeral syncs on blur only, no networkwire:model.change— ephemeral syncs on change only, no networkwire:model.enter— ephemeral syncs on enter only, no networkwire:model.blur.enter— ephemeral syncs on blur OR enter, no networkwire:model.live— ephemeral immediate, network debouncedwire:model.live.blur— ephemeral immediate, network on blur (old behavior)wire:model.blur.live— ephemeral on blur, network on blurwire:model.blur.live.debounce.500ms— ephemeral on blur, network debouncedwire:model.blur.live.throttle.500ms— ephemeral on blur, network throttled
Each scenario: type into input, verify ephemeral state via x-text="$wire.prop", verify server state via {{ $prop }}, trigger blur/enter/change, verify both states.
-
Test ephemeral-only blur
- Create input with
wire:model.blur="title" - Type "hello" — ephemeral should NOT update
- Click away — ephemeral updates to "hello"
- Server still shows old value
- Create input with
-
Test network on blur (old behavior)
- Create input with
wire:model.live.blur="title" - Type "hello" — ephemeral updates immediately
- Click away — network request fires, server updates
- Create input with
-
Test full blur.live
- Create input with
wire:model.blur.live="title" - Type "hello" — ephemeral stays unchanged
- Click away — ephemeral updates, network fires, server updates
- Create input with
File: docs/wire-model.md
Section to modify: "Customizing update timing" (line ~51) and "Reference" table (line ~291)
Add:
- Explain modifier layering concept
- Update examples to show new syntax
- Add migration note about breaking change
- Update modifier reference table
Commit message:
feat(wire:model): control ephemeral sync timing with modifiers
BREAKING: Modifiers before .live now control ephemeral (x-model) sync
timing, not just network timing. Previous .blur behavior is now .live.blur.
- wire:model.blur syncs ephemeral on blur (no network)
- wire:model.live.blur syncs ephemeral immediate, network on blur
- wire:model.blur.live syncs both on blur
- Supports .blur, .change, .enter and combinations
Migration: Replace wire:model.blur with wire:model.live.blur for old behavior.
Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
PR Title:
feat(wire:model)!: control ephemeral sync timing with modifiers
PR Description:
## Summary
- Modifiers before `.live` control ephemeral (x-model) sync timing
- Modifiers after `.live` control network timing
- BREAKING: `wire:model.blur` now delays ephemeral sync, use `.live.blur` for old behavior
## Test plan
- [ ] `wire:model.blur` delays ephemeral sync until blur
- [ ] `wire:model.live.blur` matches old `.blur` behavior
- [ ] `wire:model.blur.live` delays both ephemeral and network until blur
- [ ] Compound modifiers work (`.blur.enter`)
- [ ] Debounce/throttle work on network layer