Files
Will Miao 67869f19ff feat(early-access): implement EA filtering and UI improvements
Add Early Access version support with filtering and improved UI:

Backend:
- Add is_early_access and early_access_ends_at fields to ModelVersionRecord
- Implement two-phase EA detection (bulk API + single API enrichment)
- Add hide_early_access_updates setting to filter EA updates
- Update has_update() and has_updates_bulk() to respect EA filter setting
- Add _enrich_early_access_details() for precise EA time fetching
- Fix setting propagation through base_model_service and model_update_service

Frontend:
- Add smart relative time display for EA (in Xh, in Xd, or date)
- Replace EA label with clock icon in metadata (fa-clock)
- Show Download button with bolt icon for EA versions (fa-bolt)
- Change EA badge color to #F59F00 (CivitAI Buzz theme)
- Fix toggle UI for hide_early_access_updates setting
- Add translation keys for EA time formatting

Tests:
- Update all tests to pass with new EA functionality
- Add test coverage for EA filtering logic

Closes #815
2026-02-20 10:32:51 +08:00

438 lines
10 KiB
CSS

.model-versions-tab {
display: flex;
flex-direction: column;
gap: var(--space-3);
padding: var(--space-2) 0;
}
.versions-toolbar {
display: flex;
flex-wrap: wrap;
align-items: flex-end;
justify-content: space-between;
gap: var(--space-2);
padding: var(--space-2);
background: color-mix(in oklch, var(--lora-surface) 70%, transparent);
border: 1px solid var(--lora-border);
border-radius: var(--border-radius-sm);
}
.versions-toolbar-info h3 {
margin: 0 0 4px;
font-size: 1.05rem;
font-weight: 600;
color: var(--text-color);
}
.sr-only {
position: absolute;
width: 1px;
height: 1px;
padding: 0;
margin: -1px;
overflow: hidden;
clip: rect(0, 0, 0, 0);
border: 0;
}
.versions-toolbar-info p {
margin: 0;
font-size: 0.85rem;
color: var(--text-muted);
}
.versions-toolbar-info-heading {
display: flex;
align-items: center;
gap: var(--space-2);
}
.versions-toolbar-actions {
display: flex;
flex-wrap: wrap;
gap: var(--space-1);
}
.versions-toolbar-btn {
appearance: none;
border-radius: var(--border-radius-xs);
padding: 8px 14px;
font-size: 0.85rem;
font-weight: 600;
cursor: pointer;
border: 1px solid transparent;
transition: background-color 0.2s ease, border-color 0.2s ease, color 0.2s ease, transform 0.2s ease;
}
.versions-toolbar-btn-primary {
background: var(--lora-accent);
color: #fff;
border-color: color-mix(in oklch, var(--lora-accent) 70%, transparent);
}
.versions-toolbar-btn-primary:hover:not(:disabled) {
transform: translateY(-1px);
background: color-mix(in oklch, var(--lora-accent) 85%, transparent);
}
.versions-toolbar-btn-secondary {
background: transparent;
color: var(--text-muted);
border-color: var(--border-color);
}
.versions-toolbar-btn-secondary:hover:not(:disabled) {
color: var(--text-color);
}
.versions-filter-toggle {
appearance: none;
border: 1px solid var(--border-color);
border-radius: var(--border-radius-sm);
padding: 0;
margin-bottom: 4px;
width: 30px;
height: 30px;
background: color-mix(in oklch, var(--card-bg) 80%, var(--bg-color));
align-self: center;
display: inline-flex;
align-items: center;
justify-content: center;
color: var(--text-muted);
transition: border-color 0.2s ease, background 0.2s ease, color 0.2s ease, transform 0.2s ease;
position: relative;
cursor: pointer;
}
.versions-filter-toggle i {
font-size: 1rem;
}
.versions-filter-toggle:hover:not(:disabled) {
border-color: var(--text-color);
color: var(--text-color);
transform: translateY(-1px);
}
.versions-filter-toggle[data-filter-active="true"] {
border-color: color-mix(in oklch, var(--lora-accent) 65%, transparent);
color: var(--lora-accent);
background: color-mix(in oklch, var(--lora-accent) 20%, var(--card-bg) 80%);
}
.versions-toolbar-btn:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.versions-list {
display: flex;
flex-direction: column;
gap: var(--space-2);
}
.version-divider {
height: 1px;
background: var(--border-color);
margin: var(--space-1) 0;
}
.model-version-row {
display: grid;
grid-template-columns: 124px 1fr auto;
align-items: center;
gap: var(--space-2);
padding: var(--space-2);
background: color-mix(in oklch, var(--card-bg) 92%, var(--bg-color) 8%);
border: 1px solid var(--lora-border);
border-radius: var(--border-radius-sm);
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.04);
transition: border-color 0.2s ease, box-shadow 0.2s ease, transform 0.2s ease;
}
[data-theme="dark"] .model-version-row {
background: color-mix(in oklch, var(--card-bg) 88%, black 12%);
}
.model-version-row:hover {
transform: translateY(-1px);
box-shadow: 0 4px 16px rgba(0, 0, 0, 0.08);
}
.model-version-row.is-clickable {
cursor: pointer;
}
.model-version-row.is-current {
border-color: var(--lora-accent);
box-shadow: 0 0 0 1px color-mix(in oklch, var(--lora-accent) 65%, transparent),
0 10px 22px rgba(0, 0, 0, 0.12);
}
.version-media {
width: 124px;
height: 88px;
border-radius: var(--border-radius-xs);
overflow: hidden;
background: rgba(0, 0, 0, 0.03);
display: flex;
align-items: center;
justify-content: center;
border: 1px solid color-mix(in oklch, var(--border-color) 70%, transparent);
}
.version-media img,
.version-media video {
width: 100%;
height: 100%;
object-fit: cover;
}
.version-media img {
/* Bias cropping toward the upper region to keep faces visible */
object-position: center 20%;
}
.version-media video {
background: #000;
}
.version-media-placeholder {
font-size: 0.85rem;
color: var(--text-muted);
border-style: dashed;
border-width: 1px;
}
.version-details {
display: flex;
flex-direction: column;
gap: 8px;
min-width: 0;
}
.version-title {
display: flex;
align-items: center;
gap: 8px;
font-weight: 600;
font-size: 0.95rem;
}
.versions-tab-version-name {
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
max-width: 100%;
}
.version-badges {
display: flex;
flex-wrap: wrap;
gap: 6px;
}
.version-badge {
display: inline-flex;
align-items: center;
gap: 4px;
padding: 3px 8px;
border-radius: 999px;
border: 1px solid transparent;
font-size: 0.7rem;
font-weight: 600;
letter-spacing: 0.02em;
}
.version-badge-info {
background: color-mix(in oklch, var(--badge-update-bg) 25%, transparent);
color: var(--badge-update-bg);
border-color: color-mix(in oklch, var(--badge-update-bg) 55%, transparent);
}
.version-badge-success {
background: color-mix(in oklch, var(--lora-success) 25%, transparent);
color: var(--lora-success);
border-color: color-mix(in oklch, var(--lora-success) 50%, transparent);
}
.version-badge-muted {
background: color-mix(in oklch, var(--text-muted) 18%, transparent);
color: var(--text-muted);
border-color: color-mix(in oklch, var(--text-muted) 40%, transparent);
}
.version-badge-current {
background: color-mix(in oklch, var(--lora-accent) 22%, transparent);
color: var(--lora-accent);
border-color: color-mix(in oklch, var(--lora-accent) 55%, transparent);
}
.version-meta {
font-size: 0.8rem;
color: var(--text-muted);
display: flex;
flex-wrap: wrap;
gap: 10px;
}
.version-meta-item {
display: inline-flex;
align-items: center;
gap: 4px;
}
.version-meta-primary {
font-weight: 600;
color: color-mix(in oklch, var(--text-color) 88%, var(--lora-accent) 12%);
}
.version-meta-separator {
color: color-mix(in oklch, var(--text-muted) 90%, var(--text-color) 10%);
}
.version-actions {
display: flex;
flex-direction: column;
gap: 6px;
align-items: flex-end;
}
.version-action {
min-width: 128px;
padding: 7px 12px;
border-radius: var(--border-radius-xs);
border: 1px solid transparent;
font-size: 0.8rem;
font-weight: 600;
cursor: pointer;
transition: background-color 0.2s ease, color 0.2s ease, border-color 0.2s ease, transform 0.2s ease;
}
.version-action-primary {
background: var(--lora-accent);
color: #fff;
border-color: color-mix(in oklch, var(--lora-accent) 65%, transparent);
}
.version-action-primary:hover {
transform: translateY(-1px);
background: color-mix(in oklch, var(--lora-accent) 85%, transparent);
}
.version-action-danger {
background: transparent;
border-color: color-mix(in oklch, var(--lora-error) 60%, transparent);
color: var(--lora-error);
}
.version-action-danger:hover {
background: color-mix(in oklch, var(--lora-error) 12%, transparent);
}
.version-action-ghost {
background: transparent;
border-color: var(--border-color);
color: var(--text-color);
}
.version-action-ghost:hover {
background: color-mix(in oklch, var(--lora-surface) 35%, transparent);
}
.version-action:disabled {
opacity: 0.6;
cursor: not-allowed;
}
.versions-loading-state,
.versions-empty,
.versions-error {
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
gap: 8px;
padding: var(--space-3);
border: 1px dashed var(--lora-border);
border-radius: var(--border-radius-sm);
color: var(--text-muted);
text-align: center;
}
.versions-error {
border-style: solid;
border-color: color-mix(in oklch, var(--lora-error) 45%, transparent);
color: var(--lora-error);
}
.versions-empty i,
.versions-error i {
font-size: 1.25rem;
}
@media (max-width: 900px) {
.model-version-row {
grid-template-columns: 1fr;
align-items: stretch;
}
.version-actions {
flex-direction: row;
justify-content: flex-end;
flex-wrap: wrap;
}
.version-action {
min-width: 0;
}
}
/* Early Access styles - Buzz theme color (#F59F00) */
.version-badge-early-access {
background: color-mix(in oklch, #F59F00 25%, transparent);
color: #E67700;
border-color: color-mix(in oklch, #F59F00 55%, transparent);
}
[data-theme="dark"] .version-badge-early-access {
background: color-mix(in oklch, #F59F00 20%, transparent);
color: #F59F00;
border-color: color-mix(in oklch, #F59F00 45%, transparent);
}
.version-meta-ea {
color: #E67700;
font-weight: 600;
}
[data-theme="dark"] .version-meta-ea {
color: #F59F00;
}
/* Early Access row - gray out effect */
.model-version-row.is-early-access {
opacity: 0.85;
filter: grayscale(40%);
transition: opacity 0.2s ease, filter 0.2s ease;
}
.model-version-row.is-early-access:hover {
opacity: 0.95;
filter: grayscale(25%);
}
/* Early Access download button - Buzz theme color (#F59F00) */
.version-action-early-access {
background: color-mix(in oklch, #F59F00 15%, transparent);
color: #E67700;
border-color: color-mix(in oklch, #F59F00 50%, transparent);
cursor: not-allowed;
}
[data-theme="dark"] .version-action-early-access {
background: color-mix(in oklch, #F59F00 12%, transparent);
color: #F59F00;
border-color: color-mix(in oklch, #F59F00 40%, transparent);
}