// Copyright (c) ppy Pty Ltd . Licensed under the MIT Licence. // See the LICENCE file in the repository root for full licence text. #nullable disable using System; using System.Collections.Generic; using System.Diagnostics; using System.Threading; using System.Threading.Tasks; using JetBrains.Annotations; using osu.Framework.Allocation; using osu.Framework.Bindables; using osu.Framework.Graphics; using osu.Framework.Graphics.Containers; using osu.Framework.Input.Events; using osu.Game.Input.Bindings; using osu.Game.Online.API.Requests; using osu.Game.Online.API.Requests.Responses; using osu.Game.Overlays.Changelog; using osuTK.Graphics; namespace osu.Game.Overlays { public partial class ChangelogOverlay : OnlineOverlay { public readonly Bindable Current = new Bindable(); private List builds; protected List Streams; public ChangelogOverlay() : base(OverlayColourScheme.Purple, false) { } [BackgroundDependencyLoader] private void load() { Header.Build.BindTarget = Current; Current.BindValueChanged(e => { if (e.NewValue != null) loadContent(new ChangelogSingleBuild(e.NewValue)); else loadContent(new ChangelogListing(builds)); }); } protected override ChangelogHeader CreateHeader() => new ChangelogHeader { ListingSelected = ShowListing, }; protected override Color4 BackgroundColour => ColourProvider.Background4; public void ShowListing() { Current.Value = null; Show(); } /// /// Fetches and shows a specific build from a specific update stream. /// /// Must contain at least and /// . If and /// are specified, the header will instantly display them. public void ShowBuild([NotNull] APIChangelogBuild build) { ArgumentNullException.ThrowIfNull(build); Current.Value = build; Show(); } public void ShowBuild([NotNull] string updateStream, [NotNull] string version) { ArgumentNullException.ThrowIfNull(updateStream); ArgumentNullException.ThrowIfNull(version); Show(); performAfterFetch(() => { var build = builds.Find(b => b.Version == version && b.UpdateStream.Name == updateStream) ?? Streams.Find(s => s.Name == updateStream)?.LatestBuild; if (build != null) ShowBuild(build); }); } public override bool OnPressed(KeyBindingPressEvent e) { if (e.Repeat) return false; switch (e.Action) { case GlobalAction.Back: if (Current.Value == null) { Hide(); } else { Current.Value = null; } return true; } return false; } protected override void PopIn() { base.PopIn(); if (initialFetchTask == null) // fetch and refresh to show listing, if no other request was made via Show methods performAfterFetch(() => Current.TriggerChange()); } private Task initialFetchTask; private void performAfterFetch(Action action) { Debug.Assert(State.Value == Visibility.Visible); Schedule(() => { fetchListing()?.ContinueWith(_ => Schedule(action), TaskContinuationOptions.OnlyOnRanToCompletion); }); } private Task fetchListing() { if (initialFetchTask != null) return initialFetchTask; return initialFetchTask = Task.Run(async () => { var tcs = new TaskCompletionSource(); var req = new GetChangelogRequest(); req.Success += res => Schedule(() => { // remap streams to builds to ensure model equality res.Builds.ForEach(b => b.UpdateStream = res.Streams.Find(s => s.Id == b.UpdateStream.Id)); res.Streams.ForEach(s => s.LatestBuild.UpdateStream = res.Streams.Find(s2 => s2.Id == s.LatestBuild.UpdateStream.Id)); builds = res.Builds; Streams = res.Streams; Header.Populate(res.Streams); tcs.SetResult(true); }); req.Failure += e => { initialFetchTask = null; tcs.SetException(e); }; await API.PerformAsync(req).ConfigureAwait(false); return tcs.Task; }).Unwrap(); } private CancellationTokenSource loadContentCancellation; private void loadContent(ChangelogContent newContent) { Content.FadeTo(0.2f, 300, Easing.OutQuint); loadContentCancellation?.Cancel(); LoadComponentAsync(newContent, c => { Content.FadeIn(300, Easing.OutQuint); c.BuildSelected = ShowBuild; Child = c; }, (loadContentCancellation = new CancellationTokenSource()).Token); } } }