mirror of
https://github.com/anomalyco/opencode.git
synced 2026-05-01 22:10:23 +00:00
150 lines
4.8 KiB
CSS
150 lines
4.8 KiB
CSS
/*
|
|
* TextReveal — mask-position wipe animation
|
|
*
|
|
* Instead of sliding text through a fixed mask (odometer style),
|
|
* the mask itself sweeps across each span to reveal/hide text.
|
|
*
|
|
* Direction: top-to-bottom. New text drops in from above, old text exits downward.
|
|
*
|
|
* Entering: gradient reveals top-to-bottom (top of text appears first).
|
|
* gradient(to bottom, white 33%, transparent 33%+edge)
|
|
* pos 0 100% = transparent covers element = hidden
|
|
* pos 0 0% = white covers element = visible
|
|
*
|
|
* Leaving: gradient hides top-to-bottom (top of text disappears first).
|
|
* gradient(to top, white 33%, transparent 33%+edge)
|
|
* pos 0 100% = white covers element = visible
|
|
* pos 0 0% = transparent covers element = hidden
|
|
*
|
|
* Both transition from 0 100% (swap) → 0 0% (settled).
|
|
*/
|
|
|
|
[data-component="text-reveal"] {
|
|
--_edge: var(--text-reveal-edge, 17%);
|
|
--_dur: var(--text-reveal-duration, 450ms);
|
|
--_spring: var(--text-reveal-spring, cubic-bezier(0.34, 1.08, 0.64, 1));
|
|
--_spring-soft: var(--text-reveal-spring-soft, cubic-bezier(0.34, 1, 0.64, 1));
|
|
--_travel: var(--text-reveal-travel, 0px);
|
|
|
|
display: inline-flex;
|
|
align-items: center;
|
|
min-width: 0;
|
|
overflow: visible;
|
|
|
|
[data-slot="text-reveal-track"] {
|
|
display: grid;
|
|
min-height: 20px;
|
|
line-height: 20px;
|
|
justify-items: start;
|
|
align-items: center;
|
|
overflow: visible;
|
|
transition: width var(--_dur) var(--_spring-soft);
|
|
}
|
|
|
|
[data-slot="text-reveal-entering"],
|
|
[data-slot="text-reveal-leaving"] {
|
|
grid-area: 1 / 1;
|
|
line-height: 20px;
|
|
white-space: nowrap;
|
|
justify-self: start;
|
|
text-align: start;
|
|
mask-size: 100% 300%;
|
|
-webkit-mask-size: 100% 300%;
|
|
mask-repeat: no-repeat;
|
|
-webkit-mask-repeat: no-repeat;
|
|
transition-duration: var(--_dur);
|
|
transition-timing-function: var(--_spring);
|
|
}
|
|
|
|
/* ── entering: reveal top-to-bottom ──
|
|
* Gradient(to top): white at bottom, transparent at top of mask.
|
|
* Settled pos 0 100% = white covers element = visible
|
|
* Swap pos 0 0% = transparent covers = hidden
|
|
* Slides from above: translateY(-travel) → translateY(0)
|
|
*/
|
|
[data-slot="text-reveal-entering"] {
|
|
mask-image: linear-gradient(to top, white 33%, transparent calc(33% + var(--_edge)));
|
|
-webkit-mask-image: linear-gradient(to top, white 33%, transparent calc(33% + var(--_edge)));
|
|
mask-position: 0 100%;
|
|
-webkit-mask-position: 0 100%;
|
|
transition-property:
|
|
mask-position,
|
|
-webkit-mask-position,
|
|
transform;
|
|
transform: translateY(0);
|
|
}
|
|
|
|
/* ── leaving: hide top-to-bottom + slide downward ──
|
|
* Gradient(to bottom): white at top, transparent at bottom of mask.
|
|
* Swap pos 0 0% = white covers element = visible
|
|
* Settled pos 0 100% = transparent covers = hidden
|
|
* Slides down: translateY(0) → translateY(travel)
|
|
*/
|
|
[data-slot="text-reveal-leaving"] {
|
|
mask-image: linear-gradient(to bottom, white 33%, transparent calc(33% + var(--_edge)));
|
|
-webkit-mask-image: linear-gradient(to bottom, white 33%, transparent calc(33% + var(--_edge)));
|
|
mask-position: 0 100%;
|
|
-webkit-mask-position: 0 100%;
|
|
transition-property:
|
|
mask-position,
|
|
-webkit-mask-position,
|
|
transform;
|
|
transform: translateY(var(--_travel));
|
|
}
|
|
|
|
/* ── swapping: instant reset ──
|
|
* Snap entering to hidden (above), leaving to visible (center).
|
|
*/
|
|
&[data-swapping="true"] [data-slot="text-reveal-entering"] {
|
|
mask-position: 0 0%;
|
|
-webkit-mask-position: 0 0%;
|
|
transform: translateY(calc(var(--_travel) * -1));
|
|
transition-duration: 0ms !important;
|
|
}
|
|
|
|
&[data-swapping="true"] [data-slot="text-reveal-leaving"] {
|
|
mask-position: 0 0%;
|
|
-webkit-mask-position: 0 0%;
|
|
transform: translateY(0);
|
|
transition-duration: 0ms !important;
|
|
}
|
|
|
|
/* ── not ready: kill all transitions ── */
|
|
&[data-ready="false"] [data-slot="text-reveal-track"] {
|
|
transition-duration: 0ms !important;
|
|
}
|
|
|
|
&[data-ready="false"] [data-slot="text-reveal-entering"],
|
|
&[data-ready="false"] [data-slot="text-reveal-leaving"] {
|
|
transition-duration: 0ms !important;
|
|
}
|
|
|
|
&[data-truncate="true"] {
|
|
width: 100%;
|
|
}
|
|
|
|
&[data-truncate="true"] [data-slot="text-reveal-track"] {
|
|
width: 100%;
|
|
min-width: 0;
|
|
overflow: hidden;
|
|
}
|
|
|
|
&[data-truncate="true"] [data-slot="text-reveal-entering"],
|
|
&[data-truncate="true"] [data-slot="text-reveal-leaving"] {
|
|
min-width: 0;
|
|
width: 100%;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
}
|
|
}
|
|
|
|
@media (prefers-reduced-motion: reduce) {
|
|
[data-component="text-reveal"] [data-slot="text-reveal-track"] {
|
|
transition-duration: 0ms !important;
|
|
}
|
|
|
|
[data-component="text-reveal"] [data-slot="text-reveal-entering"],
|
|
[data-component="text-reveal"] [data-slot="text-reveal-leaving"] {
|
|
transition-duration: 0ms !important;
|
|
}
|
|
}
|