Add view transitions for event filter layout animation

This commit is contained in:
Jürgen Mummert
2026-04-05 13:36:58 +02:00
parent f014595bc5
commit 8a422aa4ee
@@ -37,6 +37,21 @@
<p id="eventfilter-status" class="visually-hidden" aria-live="polite"></p> <p id="eventfilter-status" class="visually-hidden" aria-live="polite"></p>
</div> </div>
<style>
.event-filter-target-list {
view-transition-name: event-filter-list;
}
@media (prefers-reduced-motion: no-preference) {
::view-transition-group(event-filter-list),
::view-transition-old(event-filter-list),
::view-transition-new(event-filter-list) {
animation-duration: 240ms;
animation-timing-function: ease;
}
}
</style>
<script type="module"> <script type="module">
const filters = document.getElementById('eventfilters'); const filters = document.getElementById('eventfilters');
@@ -116,9 +131,32 @@
const animationMs = 220; const animationMs = 220;
let hideTimers = new WeakMap(); let hideTimers = new WeakMap();
const supportsViewTransitions = typeof document.startViewTransition === 'function';
let isViewTransitionMutation = false;
let currentFilter = { type: 'all', value: '' }; let currentFilter = { type: 'all', value: '' };
let suppressedChangeEvents = 0; let suppressedChangeEvents = 0;
const runWithLayoutTransition = (mutation) => {
if (!supportsViewTransitions) {
mutation();
return;
}
try {
document.startViewTransition(() => {
isViewTransitionMutation = true;
try {
mutation();
} finally {
isViewTransitionMutation = false;
}
});
} catch (error) {
mutation();
}
};
const hasOptionValue = (selectElement, value) => { const hasOptionValue = (selectElement, value) => {
if (!selectElement) { if (!selectElement) {
return false; return false;
@@ -253,7 +291,16 @@
const showEvent = (eventItem) => { const showEvent = (eventItem) => {
clearHideTimer(eventItem); clearHideTimer(eventItem);
eventItem.hidden = false;
if (eventItem.hidden) {
if (supportsViewTransitions && !isViewTransitionMutation) {
runWithLayoutTransition(() => {
eventItem.hidden = false;
});
} else {
eventItem.hidden = false;
}
}
requestAnimationFrame(() => { requestAnimationFrame(() => {
eventItem.classList.remove('is-filtered-out'); eventItem.classList.remove('is-filtered-out');
@@ -265,7 +312,14 @@
eventItem.classList.add('is-filtered-out'); eventItem.classList.add('is-filtered-out');
const timer = window.setTimeout(() => { const timer = window.setTimeout(() => {
eventItem.hidden = true; if (supportsViewTransitions && !isViewTransitionMutation) {
runWithLayoutTransition(() => {
eventItem.hidden = true;
});
} else {
eventItem.hidden = true;
}
hideTimers.delete(eventItem); hideTimers.delete(eventItem);
}, animationMs); }, animationMs);
@@ -332,7 +386,7 @@
return; return;
} }
const visibleCount = events.filter((eventItem) => !eventItem.hidden).length; const visibleCount = events.filter((eventItem) => matches(eventItem, filterState)).length;
let filterText = 'alle'; let filterText = 'alle';
if (filterState.type === 'tag' && tagSelect) { if (filterState.type === 'tag' && tagSelect) {
@@ -357,12 +411,14 @@
currentFilter = filterState; currentFilter = filterState;
setActiveControl(filterState); setActiveControl(filterState);
events.forEach((eventItem) => { runWithLayoutTransition(() => {
if (matches(eventItem, filterState)) { events.forEach((eventItem) => {
showEvent(eventItem); if (matches(eventItem, filterState)) {
} else { showEvent(eventItem);
hideEvent(eventItem); } else {
} hideEvent(eventItem);
}
});
}); });
updateStatus(filterState); updateStatus(filterState);