feat: card crumples to small ball on drag pickup so buckets expand fully
This commit is contained in:
parent
d82db402a3
commit
f8aafb2974
1 changed files with 76 additions and 5 deletions
|
|
@ -1,6 +1,7 @@
|
||||||
<template>
|
<template>
|
||||||
<div
|
<div
|
||||||
class="card-stack"
|
class="card-stack"
|
||||||
|
:class="{ 'bucket-mode': isBucketMode && motion.rich.value }"
|
||||||
ref="stackEl"
|
ref="stackEl"
|
||||||
:draggable="motion.rich.value"
|
:draggable="motion.rich.value"
|
||||||
@dragstart="onDragStart"
|
@dragstart="onDragStart"
|
||||||
|
|
@ -86,7 +87,37 @@ const cardStyle = computed(() => {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
function onDragStart() { emit('drag-start') }
|
function onDragStart(e: DragEvent) {
|
||||||
|
if (motion.rich.value && e.dataTransfer) {
|
||||||
|
// Custom drag ghost: small crumpled-paper ball
|
||||||
|
const ghost = document.createElement('div')
|
||||||
|
ghost.setAttribute('aria-hidden', 'true')
|
||||||
|
Object.assign(ghost.style, {
|
||||||
|
position: 'fixed',
|
||||||
|
top: '-200px',
|
||||||
|
left: '0',
|
||||||
|
width: '80px',
|
||||||
|
height: '80px',
|
||||||
|
borderRadius: '50%',
|
||||||
|
background: '#e4ebf5',
|
||||||
|
border: '3px solid #2A6080',
|
||||||
|
display: 'flex',
|
||||||
|
alignItems: 'center',
|
||||||
|
justifyContent: 'center',
|
||||||
|
fontSize: '1.75rem',
|
||||||
|
boxShadow: '0 4px 20px rgba(0,0,0,0.25)',
|
||||||
|
transform: 'rotate(-5deg)',
|
||||||
|
})
|
||||||
|
ghost.textContent = '✉️'
|
||||||
|
document.body.appendChild(ghost)
|
||||||
|
e.dataTransfer.setDragImage(ghost, 40, 40)
|
||||||
|
// Remove after browser captures — RAF is too fast, 0ms timeout works
|
||||||
|
setTimeout(() => {
|
||||||
|
if (document.body.contains(ghost)) document.body.removeChild(ghost)
|
||||||
|
}, 0)
|
||||||
|
}
|
||||||
|
emit('drag-start')
|
||||||
|
}
|
||||||
function onDragEnd() { emit('drag-end') }
|
function onDragEnd() { emit('drag-end') }
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
|
|
@ -94,6 +125,32 @@ function onDragEnd() { emit('drag-end') }
|
||||||
.card-stack {
|
.card-stack {
|
||||||
position: relative;
|
position: relative;
|
||||||
min-height: 200px;
|
min-height: 200px;
|
||||||
|
max-height: 2000px; /* effectively unlimited — needed for max-height transition */
|
||||||
|
overflow: hidden;
|
||||||
|
transition:
|
||||||
|
max-height 280ms cubic-bezier(0.34, 1.56, 0.64, 1),
|
||||||
|
min-height 280ms cubic-bezier(0.34, 1.56, 0.64, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* Bucket mode: collapse card stack to a small pill so buckets get more room */
|
||||||
|
.card-stack.bucket-mode {
|
||||||
|
min-height: 0;
|
||||||
|
max-height: 90px;
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: flex-start;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-stack.bucket-mode .card-shadow {
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 180ms ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.card-stack.bucket-mode .card-wrapper {
|
||||||
|
transform: scale(0.25) rotate(-4deg);
|
||||||
|
transform-origin: top center;
|
||||||
|
border-radius: 50% !important;
|
||||||
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.card-shadow {
|
.card-shadow {
|
||||||
|
|
@ -102,6 +159,7 @@ function onDragEnd() { emit('drag-end') }
|
||||||
border-radius: var(--radius-card, 1rem);
|
border-radius: var(--radius-card, 1rem);
|
||||||
background: var(--color-surface-raised, #fff);
|
background: var(--color-surface-raised, #fff);
|
||||||
border: 1px solid var(--color-border, #e0e4ed);
|
border: 1px solid var(--color-border, #e0e4ed);
|
||||||
|
transition: opacity 180ms ease;
|
||||||
}
|
}
|
||||||
.card-shadow-1 { transform: translateY(8px) scale(0.97); opacity: 0.6; }
|
.card-shadow-1 { transform: translateY(8px) scale(0.97); opacity: 0.6; }
|
||||||
.card-shadow-2 { transform: translateY(16px) scale(0.94); opacity: 0.35; }
|
.card-shadow-2 { transform: translateY(16px) scale(0.94); opacity: 0.35; }
|
||||||
|
|
@ -112,16 +170,22 @@ function onDragEnd() { emit('drag-end') }
|
||||||
border-radius: var(--radius-card, 1rem);
|
border-radius: var(--radius-card, 1rem);
|
||||||
background: var(--color-surface-raised, #fff);
|
background: var(--color-surface-raised, #fff);
|
||||||
will-change: transform, opacity;
|
will-change: transform, opacity;
|
||||||
|
transition:
|
||||||
|
transform 280ms cubic-bezier(0.34, 1.56, 0.64, 1),
|
||||||
|
border-radius 280ms cubic-bezier(0.34, 1.56, 0.64, 1),
|
||||||
|
opacity 280ms ease;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Dismissal animations — only active under .rich-motion on root */
|
/* Dismissal animations — dismiss class is only applied during the motion.rich await window,
|
||||||
:global(.rich-motion) .card-wrapper.dismiss-label {
|
so no ancestor guard needed; :global(.rich-motion) was being miscompiled by Vue's scoped
|
||||||
|
CSS transformer (dropping the descendant selector entirely). */
|
||||||
|
.card-wrapper.dismiss-label {
|
||||||
animation: fileAway var(--card-dismiss, 350ms ease-in) forwards;
|
animation: fileAway var(--card-dismiss, 350ms ease-in) forwards;
|
||||||
}
|
}
|
||||||
:global(.rich-motion) .card-wrapper.dismiss-discard {
|
.card-wrapper.dismiss-discard {
|
||||||
animation: crumple var(--card-dismiss, 350ms ease-in) forwards;
|
animation: crumple var(--card-dismiss, 350ms ease-in) forwards;
|
||||||
}
|
}
|
||||||
:global(.rich-motion) .card-wrapper.dismiss-skip {
|
.card-wrapper.dismiss-skip {
|
||||||
animation: slideUnder var(--card-skip, 300ms ease-out) forwards;
|
animation: slideUnder var(--card-skip, 300ms ease-out) forwards;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -135,4 +199,11 @@ function onDragEnd() { emit('drag-end') }
|
||||||
@keyframes slideUnder {
|
@keyframes slideUnder {
|
||||||
to { transform: translateX(110%) rotate(5deg); opacity: 0; }
|
to { transform: translateX(110%) rotate(5deg); opacity: 0; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@media (prefers-reduced-motion: reduce) {
|
||||||
|
.card-stack,
|
||||||
|
.card-stack.bucket-mode .card-wrapper {
|
||||||
|
transition: none;
|
||||||
|
}
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue