2021-04-21 09:21:07 +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.
|
|
|
|
|
2021-06-04 02:29:36 +00:00
|
|
|
using System;
|
2021-05-20 06:58:17 +00:00
|
|
|
using System.Linq;
|
2021-05-19 12:09:03 +00:00
|
|
|
using System.Threading;
|
|
|
|
using osu.Framework.Allocation;
|
|
|
|
using osu.Framework.Bindables;
|
|
|
|
using osu.Framework.Graphics;
|
2022-03-04 04:30:08 +00:00
|
|
|
using osu.Game.Extensions;
|
2024-08-13 03:50:33 +00:00
|
|
|
using osu.Game.Localisation;
|
2021-05-19 12:09:03 +00:00
|
|
|
using osu.Game.Online.API;
|
|
|
|
using osu.Game.Online.API.Requests;
|
|
|
|
using osu.Game.Online.API.Requests.Responses;
|
2021-04-21 09:21:07 +00:00
|
|
|
using osu.Game.Overlays.Wiki;
|
|
|
|
|
|
|
|
namespace osu.Game.Overlays
|
|
|
|
{
|
|
|
|
public partial class WikiOverlay : OnlineOverlay<WikiHeader>
|
|
|
|
{
|
2024-01-08 22:24:56 +00:00
|
|
|
public const string INDEX_PATH = @"Main_page";
|
2021-05-19 12:09:03 +00:00
|
|
|
|
2022-12-23 13:17:42 +00:00
|
|
|
public string CurrentPath => path.Value;
|
|
|
|
|
2024-01-08 22:24:56 +00:00
|
|
|
private readonly Bindable<string> path = new Bindable<string>(INDEX_PATH);
|
2024-08-13 07:08:06 +00:00
|
|
|
private readonly Bindable<APIWikiPage?> wikiData = new Bindable<APIWikiPage?>();
|
|
|
|
private readonly IBindable<Language> language = new Bindable<Language>();
|
2021-05-19 12:09:03 +00:00
|
|
|
|
|
|
|
[Resolved]
|
2024-08-13 07:08:06 +00:00
|
|
|
private IAPIProvider api { get; set; } = null!;
|
2021-05-19 12:09:03 +00:00
|
|
|
|
2024-08-13 07:08:06 +00:00
|
|
|
[Resolved]
|
|
|
|
private OsuGameBase game { get; set; } = null!;
|
2021-05-19 12:09:03 +00:00
|
|
|
|
2024-08-13 07:08:06 +00:00
|
|
|
private GetWikiRequest? request;
|
|
|
|
private CancellationTokenSource? cancellationToken;
|
|
|
|
private WikiArticlePage? articlePage;
|
2021-05-19 12:09:03 +00:00
|
|
|
|
|
|
|
private bool displayUpdateRequired = true;
|
|
|
|
|
2021-04-21 09:21:07 +00:00
|
|
|
public WikiOverlay()
|
|
|
|
: base(OverlayColourScheme.Orange, false)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2024-08-13 07:08:06 +00:00
|
|
|
protected override void LoadComplete()
|
2024-08-13 03:50:33 +00:00
|
|
|
{
|
2024-08-13 07:08:06 +00:00
|
|
|
base.LoadComplete();
|
|
|
|
|
|
|
|
path.BindValueChanged(onPathChanged);
|
|
|
|
wikiData.BindTo(Header.WikiPageData);
|
|
|
|
|
|
|
|
language.BindTo(game.CurrentLanguage);
|
|
|
|
language.BindValueChanged(onLangChanged);
|
2024-08-13 03:50:33 +00:00
|
|
|
}
|
|
|
|
|
2024-01-08 22:24:56 +00:00
|
|
|
public void ShowPage(string pagePath = INDEX_PATH)
|
2021-06-01 05:01:08 +00:00
|
|
|
{
|
|
|
|
path.Value = pagePath.Trim('/');
|
|
|
|
Show();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override WikiHeader CreateHeader() => new WikiHeader
|
|
|
|
{
|
|
|
|
ShowIndexPage = () => ShowPage(),
|
|
|
|
ShowParentPage = showParentPage,
|
|
|
|
};
|
|
|
|
|
|
|
|
protected override void PopIn()
|
|
|
|
{
|
|
|
|
base.PopIn();
|
|
|
|
|
|
|
|
if (displayUpdateRequired)
|
|
|
|
{
|
|
|
|
path.TriggerChange();
|
|
|
|
displayUpdateRequired = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected override void PopOutComplete()
|
|
|
|
{
|
|
|
|
base.PopOutComplete();
|
|
|
|
displayUpdateRequired = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected void LoadDisplay(Drawable display)
|
|
|
|
{
|
|
|
|
ScrollFlow.ScrollToStart();
|
|
|
|
LoadComponentAsync(display, loaded =>
|
|
|
|
{
|
|
|
|
Child = loaded;
|
|
|
|
Loading.Hide();
|
|
|
|
}, (cancellationToken = new CancellationTokenSource()).Token);
|
|
|
|
}
|
|
|
|
|
2021-06-04 02:29:36 +00:00
|
|
|
protected override void UpdateAfterChildren()
|
|
|
|
{
|
|
|
|
base.UpdateAfterChildren();
|
|
|
|
|
|
|
|
if (articlePage != null)
|
|
|
|
{
|
|
|
|
articlePage.SidebarContainer.Height = DrawHeight;
|
|
|
|
articlePage.SidebarContainer.Y = Math.Clamp(ScrollFlow.Current - Header.DrawHeight, 0, Math.Max(ScrollFlow.ScrollContent.DrawHeight - DrawHeight - Header.DrawHeight, 0));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-08-13 05:34:11 +00:00
|
|
|
private void loadPage(string path, Language lang)
|
2021-05-19 12:09:03 +00:00
|
|
|
{
|
|
|
|
cancellationToken?.Cancel();
|
|
|
|
request?.Cancel();
|
|
|
|
|
2024-08-13 03:50:33 +00:00
|
|
|
// Language code + path, or just path1 + path2 in case
|
2024-08-13 05:34:11 +00:00
|
|
|
string[] values = path.Split('/', 2);
|
2022-03-04 04:30:08 +00:00
|
|
|
|
2024-08-13 05:34:11 +00:00
|
|
|
if (values.Length > 1 && LanguageExtensions.TryParseCultureCode(values[0], out var parsedLang))
|
|
|
|
request = new GetWikiRequest(values[1], parsedLang);
|
2022-03-04 04:30:08 +00:00
|
|
|
else
|
2024-08-13 05:34:11 +00:00
|
|
|
request = new GetWikiRequest(path, lang);
|
2021-05-19 12:09:03 +00:00
|
|
|
|
|
|
|
Loading.Show();
|
|
|
|
|
|
|
|
request.Success += response => Schedule(() => onSuccess(response));
|
2022-12-23 08:45:40 +00:00
|
|
|
request.Failure += ex =>
|
|
|
|
{
|
|
|
|
if (ex is not OperationCanceledException)
|
2022-12-23 15:42:46 +00:00
|
|
|
Schedule(onFail, request.Path);
|
2022-12-23 08:45:40 +00:00
|
|
|
};
|
2021-05-19 12:09:03 +00:00
|
|
|
|
|
|
|
api.PerformAsync(request);
|
|
|
|
}
|
|
|
|
|
2024-08-13 05:34:11 +00:00
|
|
|
private void onPathChanged(ValueChangedEvent<string> e)
|
|
|
|
{
|
|
|
|
// the path could change as a result of redirecting to a newer location of the same page.
|
|
|
|
// we already have the correct wiki data, so we can safely return here.
|
|
|
|
if (e.NewValue == wikiData.Value?.Path)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (e.NewValue == "error")
|
|
|
|
return;
|
|
|
|
|
|
|
|
loadPage(e.NewValue, language.Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
private void onLangChanged(ValueChangedEvent<Language> e)
|
|
|
|
{
|
|
|
|
// Path unmodified, just reload the page with new language value.
|
|
|
|
loadPage(path.Value, e.NewValue);
|
|
|
|
}
|
|
|
|
|
2021-05-19 12:09:03 +00:00
|
|
|
private void onSuccess(APIWikiPage response)
|
|
|
|
{
|
|
|
|
wikiData.Value = response;
|
2022-07-20 05:58:24 +00:00
|
|
|
path.Value = response.Path;
|
2021-05-25 07:20:04 +00:00
|
|
|
|
2024-01-09 04:13:32 +00:00
|
|
|
if (response.Layout.Equals(INDEX_PATH, StringComparison.OrdinalIgnoreCase))
|
2021-05-25 07:20:04 +00:00
|
|
|
{
|
|
|
|
LoadDisplay(new WikiMainPage
|
|
|
|
{
|
2021-05-25 07:37:14 +00:00
|
|
|
Markdown = response.Markdown,
|
|
|
|
Padding = new MarginPadding
|
|
|
|
{
|
|
|
|
Vertical = 20,
|
2023-04-03 00:12:27 +00:00
|
|
|
Horizontal = HORIZONTAL_PADDING,
|
2021-05-25 07:37:14 +00:00
|
|
|
},
|
2021-05-25 07:20:04 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
else
|
2021-05-19 12:09:03 +00:00
|
|
|
{
|
2021-06-04 02:29:36 +00:00
|
|
|
LoadDisplay(articlePage = new WikiArticlePage($@"{api.WebsiteRootUrl}/wiki/{path.Value}/", response.Markdown));
|
2021-05-25 07:20:04 +00:00
|
|
|
}
|
2021-05-19 12:09:03 +00:00
|
|
|
}
|
|
|
|
|
2022-12-23 15:42:46 +00:00
|
|
|
private void onFail(string originalPath)
|
2021-06-03 12:40:54 +00:00
|
|
|
{
|
2023-05-08 06:20:17 +00:00
|
|
|
wikiData.Value = null;
|
2022-12-23 13:17:42 +00:00
|
|
|
path.Value = "error";
|
2023-05-08 06:20:17 +00:00
|
|
|
|
2021-06-04 16:59:11 +00:00
|
|
|
LoadDisplay(articlePage = new WikiArticlePage($@"{api.WebsiteRootUrl}/wiki/",
|
2024-01-08 22:24:56 +00:00
|
|
|
$"Something went wrong when trying to fetch page \"{originalPath}\".\n\n[Return to the main page]({INDEX_PATH})."));
|
2021-06-03 12:40:54 +00:00
|
|
|
}
|
|
|
|
|
2021-05-20 06:58:17 +00:00
|
|
|
private void showParentPage()
|
|
|
|
{
|
2021-10-27 04:04:41 +00:00
|
|
|
string parentPath = string.Join("/", path.Value.Split('/').SkipLast(1));
|
2021-05-20 06:58:17 +00:00
|
|
|
ShowPage(parentPath);
|
|
|
|
}
|
|
|
|
|
2021-05-19 12:09:03 +00:00
|
|
|
protected override void Dispose(bool isDisposing)
|
|
|
|
{
|
|
|
|
cancellationToken?.Cancel();
|
|
|
|
request?.Cancel();
|
|
|
|
base.Dispose(isDisposing);
|
|
|
|
}
|
2021-04-21 09:21:07 +00:00
|
|
|
}
|
|
|
|
}
|