af724ce570
Replaces flat RELEASES/SESSIONS/MIXES nav with ARCHIVE dropdown (PageRoute.Children,
one-level cap, dual-role node). Adds /archive overview, /cuts (AlbumsView + medium
filter; /albums redirects), /sessions + /sessions/{id} (hero-dominant), /mixes +
/mixes/{id} (MixWaveformVisualizer full-page background). Extracts ReleaseDetailScaffold
from TrackDetail (invariant trio). PersistentComponentState bridge on all new pages.
Click-to-seek seam designed on MixWaveformVisualizer (inert until wired).
53 lines
2.1 KiB
C#
53 lines
2.1 KiB
C#
using DeepDrftModels.DTOs;
|
|
using DeepDrftPublic.Client.ViewModels;
|
|
using Microsoft.AspNetCore.Components;
|
|
|
|
namespace DeepDrftPublic.Client.Pages;
|
|
|
|
/// <summary>
|
|
/// Shared load + prerender-bridge logic for the single-release detail pages (Session, Mix).
|
|
/// Subclasses supply only their markup; this base loads the release through
|
|
/// <see cref="ReleaseDetailViewModel"/> and bridges the prerendered release across the prerender ->
|
|
/// WASM seam so the WASM pass does not re-fetch (see the TracksView seam). The playable track is
|
|
/// re-resolved on a restore miss only.
|
|
/// </summary>
|
|
public abstract class ReleaseDetailBase : ComponentBase, IDisposable
|
|
{
|
|
[Parameter] public long Id { get; set; }
|
|
[Inject] public required ReleaseDetailViewModel ViewModel { get; set; }
|
|
[Inject] public required PersistentComponentState PersistentState { get; set; }
|
|
|
|
private PersistingComponentStateSubscription _persistingSubscription;
|
|
|
|
// Distinct keys per medium so a Session restore never lands on a Mix page.
|
|
protected abstract string PersistKey { get; }
|
|
|
|
protected override async Task OnInitializedAsync()
|
|
{
|
|
_persistingSubscription = PersistentState.RegisterOnPersisting(Persist);
|
|
|
|
// The bridged payload carries both the release and its resolved track so the interactive
|
|
// pass renders identically without a second round-trip.
|
|
if (PersistentState.TryTakeFromJson<BridgedDetail>(PersistKey, out var restored) && restored?.Release is not null)
|
|
{
|
|
ViewModel.Restore(restored.Release, restored.Track);
|
|
}
|
|
else
|
|
{
|
|
await ViewModel.Load(Id);
|
|
}
|
|
}
|
|
|
|
private Task Persist()
|
|
{
|
|
if (ViewModel.Release is not null)
|
|
PersistentState.PersistAsJson(PersistKey, new BridgedDetail(ViewModel.Release, ViewModel.Track));
|
|
return Task.CompletedTask;
|
|
}
|
|
|
|
public void Dispose() => _persistingSubscription.Dispose();
|
|
|
|
// JSON-serializable bridge payload. Round-trips through PersistentComponentState's serializer.
|
|
protected sealed record BridgedDetail(ReleaseDto Release, TrackDto? Track);
|
|
}
|