2019-05-13 08:14:52 +00:00
|
|
|
|
// Copyright (c) ppy Pty Ltd <contact@ppy.sh>. Licensed under the MIT Licence.
|
|
|
|
|
// See the LICENCE file in the repository root for full licence text.
|
2018-07-16 21:50:22 +00:00
|
|
|
|
|
2019-05-21 04:34:35 +00:00
|
|
|
|
using System;
|
2019-05-17 08:15:51 +00:00
|
|
|
|
using System.Collections.Generic;
|
2019-05-17 08:47:28 +00:00
|
|
|
|
using System.Threading;
|
2019-05-23 02:38:13 +00:00
|
|
|
|
using System.Threading.Tasks;
|
2019-05-21 04:34:35 +00:00
|
|
|
|
using JetBrains.Annotations;
|
2018-07-19 17:07:24 +00:00
|
|
|
|
using osu.Framework.Allocation;
|
2018-07-23 21:15:14 +00:00
|
|
|
|
using osu.Framework.Audio;
|
|
|
|
|
using osu.Framework.Audio.Sample;
|
2019-05-21 04:34:35 +00:00
|
|
|
|
using osu.Framework.Bindables;
|
2018-07-16 21:50:22 +00:00
|
|
|
|
using osu.Framework.Graphics;
|
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Framework.Graphics.Shapes;
|
|
|
|
|
using osu.Game.Graphics;
|
|
|
|
|
using osu.Game.Graphics.Containers;
|
2018-07-18 17:32:15 +00:00
|
|
|
|
using osu.Game.Input.Bindings;
|
2018-07-19 17:07:24 +00:00
|
|
|
|
using osu.Game.Online.API.Requests;
|
2018-07-23 19:38:14 +00:00
|
|
|
|
using osu.Game.Online.API.Requests.Responses;
|
2018-07-16 21:50:22 +00:00
|
|
|
|
using osu.Game.Overlays.Changelog;
|
|
|
|
|
|
|
|
|
|
namespace osu.Game.Overlays
|
|
|
|
|
{
|
2019-05-14 05:18:36 +00:00
|
|
|
|
public class ChangelogOverlay : FullscreenOverlay
|
2018-07-16 21:50:22 +00:00
|
|
|
|
{
|
2019-05-23 02:23:24 +00:00
|
|
|
|
public readonly Bindable<APIChangelogBuild> Current = new Bindable<APIChangelogBuild>();
|
|
|
|
|
|
2019-05-14 05:18:36 +00:00
|
|
|
|
private ChangelogHeader header;
|
2019-05-12 15:36:05 +00:00
|
|
|
|
|
2019-05-17 08:47:28 +00:00
|
|
|
|
private Container<ChangelogContent> content;
|
2018-07-22 16:35:29 +00:00
|
|
|
|
|
2018-07-24 16:41:47 +00:00
|
|
|
|
private SampleChannel sampleBack;
|
|
|
|
|
|
2019-05-17 08:15:51 +00:00
|
|
|
|
private List<APIChangelogBuild> builds;
|
|
|
|
|
|
2019-05-23 09:54:42 +00:00
|
|
|
|
private List<APIUpdateStream> streams;
|
|
|
|
|
|
2019-05-14 05:18:36 +00:00
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load(AudioManager audio, OsuColour colour)
|
2018-07-16 21:50:22 +00:00
|
|
|
|
{
|
2019-05-17 08:20:50 +00:00
|
|
|
|
Waves.FirstWaveColour = colour.GreyVioletLight;
|
|
|
|
|
Waves.SecondWaveColour = colour.GreyViolet;
|
|
|
|
|
Waves.ThirdWaveColour = colour.GreyVioletDark;
|
|
|
|
|
Waves.FourthWaveColour = colour.GreyVioletDarker;
|
2018-07-16 21:50:22 +00:00
|
|
|
|
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
|
|
|
|
new Box
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
2019-05-23 02:23:24 +00:00
|
|
|
|
Colour = colour.PurpleDarkAlternative,
|
2018-07-16 21:50:22 +00:00
|
|
|
|
},
|
2019-06-14 06:55:32 +00:00
|
|
|
|
new OsuScrollContainer
|
2018-07-16 21:50:22 +00:00
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
|
ScrollbarVisible = false,
|
|
|
|
|
Child = new ReverseChildIDFillFlowContainer<Drawable>
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
Direction = FillDirection.Vertical,
|
|
|
|
|
Children = new Drawable[]
|
|
|
|
|
{
|
2019-05-21 03:52:50 +00:00
|
|
|
|
header = new ChangelogHeader
|
|
|
|
|
{
|
|
|
|
|
ListingSelected = ShowListing,
|
|
|
|
|
},
|
2019-05-17 08:47:28 +00:00
|
|
|
|
content = new Container<ChangelogContent>
|
2019-05-17 08:15:51 +00:00
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
}
|
2018-07-16 21:50:22 +00:00
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
},
|
|
|
|
|
};
|
2019-05-15 08:55:26 +00:00
|
|
|
|
|
2019-06-04 02:04:28 +00:00
|
|
|
|
sampleBack = audio.Samples.Get(@"UI/generic-select-soft");
|
2018-07-18 17:32:15 +00:00
|
|
|
|
|
2019-05-21 04:34:35 +00:00
|
|
|
|
header.Current.BindTo(Current);
|
2018-07-16 21:50:22 +00:00
|
|
|
|
|
2019-05-21 04:34:35 +00:00
|
|
|
|
Current.BindValueChanged(e =>
|
|
|
|
|
{
|
|
|
|
|
if (e.NewValue != null)
|
|
|
|
|
loadContent(new ChangelogSingleBuild(e.NewValue));
|
|
|
|
|
else
|
|
|
|
|
loadContent(new ChangelogListing(builds));
|
|
|
|
|
});
|
2018-07-16 21:50:22 +00:00
|
|
|
|
}
|
2018-07-19 17:07:24 +00:00
|
|
|
|
|
2019-05-23 09:54:42 +00:00
|
|
|
|
public void ShowListing()
|
|
|
|
|
{
|
|
|
|
|
Current.Value = null;
|
2019-06-11 05:28:52 +00:00
|
|
|
|
Show();
|
2019-05-23 09:54:42 +00:00
|
|
|
|
}
|
2019-05-23 02:23:24 +00:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Fetches and shows a specific build from a specific update stream.
|
|
|
|
|
/// </summary>
|
|
|
|
|
/// <param name="build">Must contain at least <see cref="APIUpdateStream.Name"/> and
|
|
|
|
|
/// <see cref="APIChangelogBuild.Version"/>. If <see cref="APIUpdateStream.DisplayName"/> and
|
|
|
|
|
/// <see cref="APIChangelogBuild.DisplayVersion"/> are specified, the header will instantly display them.</param>
|
|
|
|
|
public void ShowBuild([NotNull] APIChangelogBuild build)
|
|
|
|
|
{
|
|
|
|
|
if (build == null) throw new ArgumentNullException(nameof(build));
|
|
|
|
|
|
|
|
|
|
Current.Value = build;
|
2019-06-11 05:28:52 +00:00
|
|
|
|
Show();
|
2019-05-23 09:54:42 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void ShowBuild([NotNull] string updateStream, [NotNull] string version)
|
|
|
|
|
{
|
|
|
|
|
if (updateStream == null) throw new ArgumentNullException(nameof(updateStream));
|
|
|
|
|
if (version == null) throw new ArgumentNullException(nameof(version));
|
|
|
|
|
|
|
|
|
|
performAfterFetch(() =>
|
|
|
|
|
{
|
|
|
|
|
var build = builds.Find(b => b.Version == version && b.UpdateStream.Name == updateStream)
|
|
|
|
|
?? streams.Find(s => s.Name == updateStream)?.LatestBuild;
|
|
|
|
|
|
|
|
|
|
if (build != null)
|
2019-05-31 04:54:40 +00:00
|
|
|
|
ShowBuild(build);
|
2019-05-23 09:54:42 +00:00
|
|
|
|
});
|
|
|
|
|
|
2019-06-11 05:28:52 +00:00
|
|
|
|
Show();
|
2019-05-23 02:23:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2018-07-24 16:41:47 +00:00
|
|
|
|
public override bool OnPressed(GlobalAction action)
|
2018-07-22 18:27:50 +00:00
|
|
|
|
{
|
2018-07-24 16:41:47 +00:00
|
|
|
|
switch (action)
|
|
|
|
|
{
|
|
|
|
|
case GlobalAction.Back:
|
2019-05-21 04:34:35 +00:00
|
|
|
|
if (Current.Value == null)
|
2018-07-24 22:12:05 +00:00
|
|
|
|
{
|
2019-06-11 05:28:52 +00:00
|
|
|
|
Hide();
|
2018-07-24 22:12:05 +00:00
|
|
|
|
}
|
2018-07-24 16:41:47 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
2019-05-21 04:34:35 +00:00
|
|
|
|
Current.Value = null;
|
2018-07-24 16:41:47 +00:00
|
|
|
|
sampleBack?.Play();
|
|
|
|
|
}
|
2019-05-12 15:36:05 +00:00
|
|
|
|
|
2018-07-24 16:41:47 +00:00
|
|
|
|
return true;
|
|
|
|
|
}
|
2018-07-22 18:27:50 +00:00
|
|
|
|
|
2018-07-24 16:41:47 +00:00
|
|
|
|
return false;
|
2018-07-19 17:07:24 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-21 04:34:35 +00:00
|
|
|
|
protected override void PopIn()
|
2018-07-22 16:35:29 +00:00
|
|
|
|
{
|
2019-05-21 04:34:35 +00:00
|
|
|
|
base.PopIn();
|
2018-07-23 19:38:14 +00:00
|
|
|
|
|
2019-05-23 09:54:42 +00:00
|
|
|
|
if (initialFetchTask == null)
|
|
|
|
|
// fetch and refresh to show listing, if no other request was made via Show methods
|
|
|
|
|
performAfterFetch(() => Current.TriggerChange());
|
2019-05-17 08:47:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-23 09:54:42 +00:00
|
|
|
|
private Task initialFetchTask;
|
2019-05-17 08:49:05 +00:00
|
|
|
|
|
2019-05-23 09:54:42 +00:00
|
|
|
|
private void performAfterFetch(Action action) => fetchListing()?.ContinueWith(_ => Schedule(action));
|
|
|
|
|
|
|
|
|
|
private Task fetchListing()
|
2019-05-17 08:47:28 +00:00
|
|
|
|
{
|
2019-05-23 09:54:42 +00:00
|
|
|
|
if (initialFetchTask != null)
|
|
|
|
|
return initialFetchTask;
|
|
|
|
|
|
2019-05-23 10:08:44 +00:00
|
|
|
|
return initialFetchTask = Task.Run(async () =>
|
2019-05-17 08:47:28 +00:00
|
|
|
|
{
|
2019-05-23 10:08:44 +00:00
|
|
|
|
var tcs = new TaskCompletionSource<bool>();
|
|
|
|
|
|
2019-05-23 02:38:13 +00:00
|
|
|
|
var req = new GetChangelogRequest();
|
2019-10-31 06:04:13 +00:00
|
|
|
|
|
2019-09-13 08:21:47 +00:00
|
|
|
|
req.Success += res => Schedule(() =>
|
2019-05-23 02:38:13 +00:00
|
|
|
|
{
|
|
|
|
|
// 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));
|
2019-05-17 08:47:28 +00:00
|
|
|
|
|
2019-05-23 02:38:13 +00:00
|
|
|
|
builds = res.Builds;
|
2019-05-23 09:54:42 +00:00
|
|
|
|
streams = res.Streams;
|
|
|
|
|
|
2019-05-23 02:38:13 +00:00
|
|
|
|
header.Streams.Populate(res.Streams);
|
2019-05-14 05:18:36 +00:00
|
|
|
|
|
2019-05-23 09:54:42 +00:00
|
|
|
|
tcs.SetResult(true);
|
2019-09-13 08:21:47 +00:00
|
|
|
|
});
|
2019-10-31 06:04:13 +00:00
|
|
|
|
|
|
|
|
|
req.Failure += _ =>
|
|
|
|
|
{
|
|
|
|
|
initialFetchTask = null;
|
|
|
|
|
tcs.SetResult(false);
|
|
|
|
|
};
|
|
|
|
|
|
2019-11-29 11:03:14 +00:00
|
|
|
|
await API.PerformAsync(req);
|
2019-05-23 09:54:42 +00:00
|
|
|
|
|
2019-05-23 10:08:44 +00:00
|
|
|
|
await tcs.Task;
|
|
|
|
|
});
|
2019-05-17 08:47:28 +00:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-31 04:54:40 +00:00
|
|
|
|
private CancellationTokenSource loadContentCancellation;
|
2019-05-17 08:47:28 +00:00
|
|
|
|
|
|
|
|
|
private void loadContent(ChangelogContent newContent)
|
|
|
|
|
{
|
|
|
|
|
content.FadeTo(0.2f, 300, Easing.OutQuint);
|
|
|
|
|
|
2019-05-31 04:54:40 +00:00
|
|
|
|
loadContentCancellation?.Cancel();
|
2019-05-17 08:47:28 +00:00
|
|
|
|
|
|
|
|
|
LoadComponentAsync(newContent, c =>
|
2019-05-15 09:21:06 +00:00
|
|
|
|
{
|
2019-05-17 08:47:28 +00:00
|
|
|
|
content.FadeIn(300, Easing.OutQuint);
|
|
|
|
|
|
|
|
|
|
c.BuildSelected = ShowBuild;
|
|
|
|
|
content.Child = c;
|
2019-05-31 04:54:40 +00:00
|
|
|
|
}, (loadContentCancellation = new CancellationTokenSource()).Token);
|
2018-07-19 17:07:24 +00:00
|
|
|
|
}
|
2018-07-16 21:50:22 +00:00
|
|
|
|
}
|
|
|
|
|
}
|