mirror of
https://github.com/ppy/osu.git
synced 2024-12-13 08:32:57 +08:00
Merge pull request #28363 from peppy/external-link-open-trusted
Bypass external link dialog for links on the trusted osu! domain
This commit is contained in:
commit
b76ec9654d
@ -6,6 +6,7 @@ using NUnit.Framework;
|
|||||||
using osu.Framework.Graphics.Containers;
|
using osu.Framework.Graphics.Containers;
|
||||||
using osu.Framework.Testing;
|
using osu.Framework.Testing;
|
||||||
using osu.Game.Online.API.Requests.Responses;
|
using osu.Game.Online.API.Requests.Responses;
|
||||||
|
using osu.Game.Overlays;
|
||||||
using osu.Game.Screens.Menu;
|
using osu.Game.Screens.Menu;
|
||||||
using osuTK.Input;
|
using osuTK.Input;
|
||||||
|
|
||||||
@ -15,8 +16,14 @@ namespace osu.Game.Tests.Visual.Menus
|
|||||||
{
|
{
|
||||||
private OnlineMenuBanner onlineMenuBanner => Game.ChildrenOfType<OnlineMenuBanner>().Single();
|
private OnlineMenuBanner onlineMenuBanner => Game.ChildrenOfType<OnlineMenuBanner>().Single();
|
||||||
|
|
||||||
|
public override void SetUpSteps()
|
||||||
|
{
|
||||||
|
base.SetUpSteps();
|
||||||
|
AddStep("don't fetch online content", () => onlineMenuBanner.FetchOnlineContent = false);
|
||||||
|
}
|
||||||
|
|
||||||
[Test]
|
[Test]
|
||||||
public void TestOnlineMenuBanner()
|
public void TestOnlineMenuBannerTrusted()
|
||||||
{
|
{
|
||||||
AddStep("set online content", () => onlineMenuBanner.Current.Value = new APIMenuContent
|
AddStep("set online content", () => onlineMenuBanner.Current.Value = new APIMenuContent
|
||||||
{
|
{
|
||||||
@ -25,13 +32,51 @@ namespace osu.Game.Tests.Visual.Menus
|
|||||||
new APIMenuImage
|
new APIMenuImage
|
||||||
{
|
{
|
||||||
Image = @"https://assets.ppy.sh/main-menu/project-loved-2@2x.png",
|
Image = @"https://assets.ppy.sh/main-menu/project-loved-2@2x.png",
|
||||||
Url = @"https://osu.ppy.sh/home/news/2023-12-21-project-loved-december-2023",
|
Url = $@"{API.WebsiteRootUrl}/home/news/2023-12-21-project-loved-december-2023",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
AddAssert("system title not visible", () => onlineMenuBanner.State.Value, () => Is.EqualTo(Visibility.Hidden));
|
AddAssert("system title not visible", () => onlineMenuBanner.State.Value, () => Is.EqualTo(Visibility.Hidden));
|
||||||
AddStep("enter menu", () => InputManager.Key(Key.Enter));
|
AddStep("enter menu", () => InputManager.Key(Key.Enter));
|
||||||
AddUntilStep("system title visible", () => onlineMenuBanner.State.Value, () => Is.EqualTo(Visibility.Visible));
|
AddUntilStep("system title visible", () => onlineMenuBanner.State.Value, () => Is.EqualTo(Visibility.Visible));
|
||||||
|
AddUntilStep("image loaded", () => onlineMenuBanner.ChildrenOfType<OnlineMenuBanner.MenuImage>().FirstOrDefault()?.IsLoaded, () => Is.True);
|
||||||
|
|
||||||
|
AddStep("click banner", () =>
|
||||||
|
{
|
||||||
|
InputManager.MoveMouseTo(onlineMenuBanner);
|
||||||
|
InputManager.Click(MouseButton.Left);
|
||||||
|
});
|
||||||
|
|
||||||
|
// Might not catch every occurrence due to async nature, but works in manual testing and saves annoying test setup.
|
||||||
|
AddAssert("no dialog", () => Game.ChildrenOfType<DialogOverlay>().SingleOrDefault()?.CurrentDialog == null);
|
||||||
|
}
|
||||||
|
|
||||||
|
[Test]
|
||||||
|
public void TestOnlineMenuBannerUntrustedDomain()
|
||||||
|
{
|
||||||
|
AddStep("set online content", () => onlineMenuBanner.Current.Value = new APIMenuContent
|
||||||
|
{
|
||||||
|
Images = new[]
|
||||||
|
{
|
||||||
|
new APIMenuImage
|
||||||
|
{
|
||||||
|
Image = @"https://assets.ppy.sh/main-menu/project-loved-2@2x.png",
|
||||||
|
Url = @"https://google.com",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
AddAssert("system title not visible", () => onlineMenuBanner.State.Value, () => Is.EqualTo(Visibility.Hidden));
|
||||||
|
AddStep("enter menu", () => InputManager.Key(Key.Enter));
|
||||||
|
AddUntilStep("system title visible", () => onlineMenuBanner.State.Value, () => Is.EqualTo(Visibility.Visible));
|
||||||
|
AddUntilStep("image loaded", () => onlineMenuBanner.ChildrenOfType<OnlineMenuBanner.MenuImage>().FirstOrDefault()?.IsLoaded, () => Is.True);
|
||||||
|
|
||||||
|
AddStep("click banner", () =>
|
||||||
|
{
|
||||||
|
InputManager.MoveMouseTo(onlineMenuBanner);
|
||||||
|
InputManager.Click(MouseButton.Left);
|
||||||
|
});
|
||||||
|
|
||||||
|
AddUntilStep("wait for dialog", () => Game.ChildrenOfType<DialogOverlay>().SingleOrDefault()?.CurrentDialog != null);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,14 +5,14 @@ using osu.Framework.Localisation;
|
|||||||
|
|
||||||
namespace osu.Game.Localisation
|
namespace osu.Game.Localisation
|
||||||
{
|
{
|
||||||
public static class DeleteConfirmationDialogStrings
|
public static class DialogStrings
|
||||||
{
|
{
|
||||||
private const string prefix = @"osu.Game.Resources.Localisation.DeleteConfirmationDialog";
|
private const string prefix = @"osu.Game.Resources.Localisation.Dialog";
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// "Caution"
|
/// "Caution"
|
||||||
/// </summary>
|
/// </summary>
|
||||||
public static LocalisableString HeaderText => new TranslatableString(getKey(@"header_text"), @"Caution");
|
public static LocalisableString Caution => new TranslatableString(getKey(@"header_text"), @"Caution");
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// "Yes. Go for it."
|
/// "Yes. Go for it."
|
@ -8,8 +8,10 @@ using osu.Framework.Graphics;
|
|||||||
using osu.Framework.Graphics.Sprites;
|
using osu.Framework.Graphics.Sprites;
|
||||||
using osu.Framework.Platform;
|
using osu.Framework.Platform;
|
||||||
using osu.Game.Configuration;
|
using osu.Game.Configuration;
|
||||||
|
using osu.Game.Localisation;
|
||||||
using osu.Game.Overlays;
|
using osu.Game.Overlays;
|
||||||
using osu.Game.Overlays.Dialog;
|
using osu.Game.Overlays.Dialog;
|
||||||
|
using CommonStrings = osu.Game.Resources.Localisation.Web.CommonStrings;
|
||||||
|
|
||||||
namespace osu.Game.Online.Chat
|
namespace osu.Game.Online.Chat
|
||||||
{
|
{
|
||||||
@ -44,8 +46,8 @@ namespace osu.Game.Online.Chat
|
|||||||
{
|
{
|
||||||
public ExternalLinkDialog(string url, Action openExternalLinkAction, Action copyExternalLinkAction)
|
public ExternalLinkDialog(string url, Action openExternalLinkAction, Action copyExternalLinkAction)
|
||||||
{
|
{
|
||||||
HeaderText = "Just checking...";
|
HeaderText = DialogStrings.Caution;
|
||||||
BodyText = $"You are about to leave osu! and open the following link in a web browser:\n\n{url}";
|
BodyText = $"Are you sure you want to open the following link in a web browser?\n\n{url}";
|
||||||
|
|
||||||
Icon = FontAwesome.Solid.ExclamationTriangle;
|
Icon = FontAwesome.Solid.ExclamationTriangle;
|
||||||
|
|
||||||
@ -53,17 +55,17 @@ namespace osu.Game.Online.Chat
|
|||||||
{
|
{
|
||||||
new PopupDialogOkButton
|
new PopupDialogOkButton
|
||||||
{
|
{
|
||||||
Text = @"Yes. Go for it.",
|
Text = @"Open in browser",
|
||||||
Action = openExternalLinkAction
|
Action = openExternalLinkAction
|
||||||
},
|
},
|
||||||
new PopupDialogCancelButton
|
new PopupDialogCancelButton
|
||||||
{
|
{
|
||||||
Text = @"Copy URL to the clipboard instead.",
|
Text = @"Copy URL to the clipboard",
|
||||||
Action = copyExternalLinkAction
|
Action = copyExternalLinkAction
|
||||||
},
|
},
|
||||||
new PopupDialogCancelButton
|
new PopupDialogCancelButton
|
||||||
{
|
{
|
||||||
Text = @"No! Abort mission!"
|
Text = CommonStrings.ButtonsCancel,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -485,10 +485,19 @@ namespace osu.Game
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
public void OpenUrlExternally(string url, bool bypassExternalUrlWarning = false) => waitForReady(() => externalLinkOpener, _ =>
|
public void OpenUrlExternally(string url, bool forceBypassExternalUrlWarning = false) => waitForReady(() => externalLinkOpener, _ =>
|
||||||
{
|
{
|
||||||
|
bool isTrustedDomain;
|
||||||
|
|
||||||
if (url.StartsWith('/'))
|
if (url.StartsWith('/'))
|
||||||
url = $"{API.APIEndpointUrl}{url}";
|
{
|
||||||
|
url = $"{API.WebsiteRootUrl}{url}";
|
||||||
|
isTrustedDomain = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
isTrustedDomain = url.StartsWith(API.WebsiteRootUrl, StringComparison.Ordinal);
|
||||||
|
}
|
||||||
|
|
||||||
if (!url.CheckIsValidUrl())
|
if (!url.CheckIsValidUrl())
|
||||||
{
|
{
|
||||||
@ -500,7 +509,7 @@ namespace osu.Game
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
externalLinkOpener.OpenUrlExternally(url, bypassExternalUrlWarning);
|
externalLinkOpener.OpenUrlExternally(url, forceBypassExternalUrlWarning || isTrustedDomain);
|
||||||
});
|
});
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
|
@ -30,7 +30,7 @@ namespace osu.Game.Overlays.Dialog
|
|||||||
|
|
||||||
protected DangerousActionDialog()
|
protected DangerousActionDialog()
|
||||||
{
|
{
|
||||||
HeaderText = DeleteConfirmationDialogStrings.HeaderText;
|
HeaderText = DialogStrings.Caution;
|
||||||
|
|
||||||
Icon = FontAwesome.Regular.TrashAlt;
|
Icon = FontAwesome.Regular.TrashAlt;
|
||||||
|
|
||||||
@ -38,12 +38,12 @@ namespace osu.Game.Overlays.Dialog
|
|||||||
{
|
{
|
||||||
new PopupDialogDangerousButton
|
new PopupDialogDangerousButton
|
||||||
{
|
{
|
||||||
Text = DeleteConfirmationDialogStrings.Confirm,
|
Text = DialogStrings.Confirm,
|
||||||
Action = () => DangerousAction?.Invoke()
|
Action = () => DangerousAction?.Invoke()
|
||||||
},
|
},
|
||||||
new PopupDialogCancelButton
|
new PopupDialogCancelButton
|
||||||
{
|
{
|
||||||
Text = DeleteConfirmationDialogStrings.Cancel,
|
Text = DialogStrings.Cancel,
|
||||||
Action = () => CancelAction?.Invoke()
|
Action = () => CancelAction?.Invoke()
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -73,6 +73,9 @@ namespace osu.Game.Screens.Menu
|
|||||||
Task.Run(() => request.Perform())
|
Task.Run(() => request.Perform())
|
||||||
.ContinueWith(r =>
|
.ContinueWith(r =>
|
||||||
{
|
{
|
||||||
|
if (!FetchOnlineContent)
|
||||||
|
return;
|
||||||
|
|
||||||
if (r.IsCompletedSuccessfully)
|
if (r.IsCompletedSuccessfully)
|
||||||
Schedule(() => Current.Value = request.ResponseObject);
|
Schedule(() => Current.Value = request.ResponseObject);
|
||||||
|
|
||||||
@ -170,6 +173,11 @@ namespace osu.Game.Screens.Menu
|
|||||||
|
|
||||||
private Sprite flash = null!;
|
private Sprite flash = null!;
|
||||||
|
|
||||||
|
/// <remarks>
|
||||||
|
/// Overridden as a safety for <see cref="openUrlAction"/> functioning correctly.
|
||||||
|
/// </remarks>
|
||||||
|
public override bool IsPresent => base.IsPresent || Scheduler.HasPendingTasks;
|
||||||
|
|
||||||
private ScheduledDelegate? openUrlAction;
|
private ScheduledDelegate? openUrlAction;
|
||||||
|
|
||||||
public MenuImage(APIMenuImage image)
|
public MenuImage(APIMenuImage image)
|
||||||
|
Loading…
Reference in New Issue
Block a user