mirror of
https://github.com/ppy/osu.git
synced 2025-01-13 15:33:21 +08:00
Merge pull request #20032 from peppy/toast-notification-tray
Add toast notification tray
This commit is contained in:
commit
6cadcc206b
@ -14,11 +14,10 @@ using osu.Framework.Audio;
|
||||
using osu.Framework.Graphics;
|
||||
using osu.Framework.Graphics.Containers;
|
||||
using osu.Framework.Localisation;
|
||||
using osu.Framework.Utils;
|
||||
using osu.Framework.Screens;
|
||||
using osu.Framework.Testing;
|
||||
using osu.Framework.Utils;
|
||||
using osu.Game.Configuration;
|
||||
using osu.Game.Graphics.Containers;
|
||||
using osu.Game.Overlays;
|
||||
using osu.Game.Overlays.Notifications;
|
||||
using osu.Game.Rulesets.Mods;
|
||||
@ -30,7 +29,6 @@ using osu.Game.Screens.Play;
|
||||
using osu.Game.Screens.Play.PlayerSettings;
|
||||
using osu.Game.Utils;
|
||||
using osuTK.Input;
|
||||
using SkipOverlay = osu.Game.Screens.Play.SkipOverlay;
|
||||
|
||||
namespace osu.Game.Tests.Visual.Gameplay
|
||||
{
|
||||
@ -83,6 +81,20 @@ namespace osu.Game.Tests.Visual.Gameplay
|
||||
[SetUp]
|
||||
public void Setup() => Schedule(() => player = null);
|
||||
|
||||
[SetUpSteps]
|
||||
public override void SetUpSteps()
|
||||
{
|
||||
base.SetUpSteps();
|
||||
|
||||
AddStep("read all notifications", () =>
|
||||
{
|
||||
notificationOverlay.Show();
|
||||
notificationOverlay.Hide();
|
||||
});
|
||||
|
||||
AddUntilStep("wait for no notifications", () => notificationOverlay.UnreadCount.Value, () => Is.EqualTo(0));
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Sets the input manager child to a new test player loader container instance.
|
||||
/// </summary>
|
||||
@ -287,16 +299,9 @@ namespace osu.Game.Tests.Visual.Gameplay
|
||||
|
||||
saveVolumes();
|
||||
|
||||
AddAssert("check for notification", () => notificationOverlay.UnreadCount.Value == 1);
|
||||
AddStep("click notification", () =>
|
||||
{
|
||||
var scrollContainer = (OsuScrollContainer)notificationOverlay.Children.Last();
|
||||
var flowContainer = scrollContainer.Children.OfType<FillFlowContainer<NotificationSection>>().First();
|
||||
var notification = flowContainer.First();
|
||||
AddAssert("check for notification", () => notificationOverlay.UnreadCount.Value, () => Is.EqualTo(1));
|
||||
|
||||
InputManager.MoveMouseTo(notification);
|
||||
InputManager.Click(MouseButton.Left);
|
||||
});
|
||||
clickNotificationIfAny();
|
||||
|
||||
AddAssert("check " + volumeName, assert);
|
||||
|
||||
@ -366,15 +371,7 @@ namespace osu.Game.Tests.Visual.Gameplay
|
||||
}));
|
||||
AddUntilStep("wait for player", () => player?.LoadState == LoadState.Ready);
|
||||
AddAssert($"notification {(shouldWarn ? "triggered" : "not triggered")}", () => notificationOverlay.UnreadCount.Value == (shouldWarn ? 1 : 0));
|
||||
AddStep("click notification", () =>
|
||||
{
|
||||
var scrollContainer = (OsuScrollContainer)notificationOverlay.Children.Last();
|
||||
var flowContainer = scrollContainer.Children.OfType<FillFlowContainer<NotificationSection>>().First();
|
||||
var notification = flowContainer.First();
|
||||
|
||||
InputManager.MoveMouseTo(notification);
|
||||
InputManager.Click(MouseButton.Left);
|
||||
});
|
||||
clickNotificationIfAny();
|
||||
AddUntilStep("wait for player load", () => player.IsLoaded);
|
||||
}
|
||||
|
||||
@ -439,6 +436,11 @@ namespace osu.Game.Tests.Visual.Gameplay
|
||||
AddUntilStep("skip button not visible", () => !checkSkipButtonVisible());
|
||||
}
|
||||
|
||||
private void clickNotificationIfAny()
|
||||
{
|
||||
AddStep("click notification", () => notificationOverlay.ChildrenOfType<Notification>().FirstOrDefault()?.TriggerClick());
|
||||
}
|
||||
|
||||
private EpilepsyWarning getWarning() => loader.ChildrenOfType<EpilepsyWarning>().SingleOrDefault();
|
||||
|
||||
private class TestPlayerLoader : PlayerLoader
|
||||
|
@ -52,6 +52,7 @@ namespace osu.Game.Tests.Visual.Menus
|
||||
},
|
||||
notifications = new NotificationOverlay
|
||||
{
|
||||
Depth = float.MinValue,
|
||||
Anchor = Anchor.TopRight,
|
||||
Origin = Anchor.TopRight,
|
||||
}
|
||||
@ -82,7 +83,14 @@ namespace osu.Game.Tests.Visual.Menus
|
||||
[Test]
|
||||
public virtual void TestPlayIntroWithFailingAudioDevice()
|
||||
{
|
||||
AddStep("hide notifications", () => notifications.Hide());
|
||||
AddStep("reset notifications", () =>
|
||||
{
|
||||
notifications.Show();
|
||||
notifications.Hide();
|
||||
});
|
||||
|
||||
AddUntilStep("wait for no notifications", () => notifications.UnreadCount.Value, () => Is.EqualTo(0));
|
||||
|
||||
AddStep("restart sequence", () =>
|
||||
{
|
||||
logo.FinishTransforms();
|
||||
|
@ -26,16 +26,7 @@ namespace osu.Game.Tests.Visual.Navigation
|
||||
public void TestImportantNotificationDoesntInterruptSetup()
|
||||
{
|
||||
AddStep("post important notification", () => Game.Notifications.Post(new SimpleNotification { Text = "Important notification" }));
|
||||
AddAssert("no notification posted", () => Game.Notifications.UnreadCount.Value == 0);
|
||||
AddAssert("first-run setup still visible", () => Game.FirstRunOverlay.State.Value == Visibility.Visible);
|
||||
|
||||
AddUntilStep("finish first-run setup", () =>
|
||||
{
|
||||
Game.FirstRunOverlay.NextButton.TriggerClick();
|
||||
return Game.FirstRunOverlay.State.Value == Visibility.Hidden;
|
||||
});
|
||||
AddWaitStep("wait for post delay", 5);
|
||||
AddAssert("notifications shown", () => Game.Notifications.State.Value == Visibility.Visible);
|
||||
AddAssert("notification posted", () => Game.Notifications.UnreadCount.Value == 1);
|
||||
}
|
||||
|
||||
|
@ -110,7 +110,8 @@ namespace osu.Game.Tests.Visual.UserInterface
|
||||
{
|
||||
AddStep(@"simple #1", sendHelloNotification);
|
||||
|
||||
AddAssert("Is visible", () => notificationOverlay.State.Value == Visibility.Visible);
|
||||
AddAssert("toast displayed", () => notificationOverlay.ToastCount == 1);
|
||||
AddAssert("is not visible", () => notificationOverlay.State.Value == Visibility.Hidden);
|
||||
|
||||
checkDisplayedCount(1);
|
||||
|
||||
@ -183,7 +184,7 @@ namespace osu.Game.Tests.Visual.UserInterface
|
||||
}
|
||||
|
||||
private void checkDisplayedCount(int expected) =>
|
||||
AddAssert($"Displayed count is {expected}", () => notificationOverlay.UnreadCount.Value == expected);
|
||||
AddUntilStep($"Displayed count is {expected}", () => notificationOverlay.UnreadCount.Value == expected);
|
||||
|
||||
private void sendDownloadProgress()
|
||||
{
|
||||
|
@ -804,8 +804,8 @@ namespace osu.Game
|
||||
Children = new Drawable[]
|
||||
{
|
||||
overlayContent = new Container { RelativeSizeAxes = Axes.Both },
|
||||
rightFloatingOverlayContent = new Container { RelativeSizeAxes = Axes.Both },
|
||||
leftFloatingOverlayContent = new Container { RelativeSizeAxes = Axes.Both },
|
||||
rightFloatingOverlayContent = new Container { RelativeSizeAxes = Axes.Both },
|
||||
}
|
||||
},
|
||||
topMostOverlayContent = new Container { RelativeSizeAxes = Axes.Both },
|
||||
|
@ -15,6 +15,7 @@ using osu.Framework.Threading;
|
||||
using osu.Game.Graphics.Containers;
|
||||
using osu.Game.Overlays.Notifications;
|
||||
using osu.Game.Resources.Localisation.Web;
|
||||
using osuTK;
|
||||
using NotificationsStrings = osu.Game.Localisation.NotificationsStrings;
|
||||
|
||||
namespace osu.Game.Overlays
|
||||
@ -37,15 +38,41 @@ namespace osu.Game.Overlays
|
||||
[Cached]
|
||||
private OverlayColourProvider colourProvider = new OverlayColourProvider(OverlayColourScheme.Purple);
|
||||
|
||||
private readonly IBindable<Visibility> firstRunSetupVisibility = new Bindable<Visibility>();
|
||||
public override bool ReceivePositionalInputAt(Vector2 screenSpacePos)
|
||||
{
|
||||
if (State.Value == Visibility.Visible)
|
||||
return base.ReceivePositionalInputAt(screenSpacePos);
|
||||
|
||||
if (toastTray.IsDisplayingToasts)
|
||||
return toastTray.ReceivePositionalInputAt(screenSpacePos);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
public override bool PropagatePositionalInputSubTree => base.PropagatePositionalInputSubTree || toastTray.IsDisplayingToasts;
|
||||
|
||||
private NotificationOverlayToastTray toastTray = null!;
|
||||
|
||||
private Container mainContent = null!;
|
||||
|
||||
[BackgroundDependencyLoader]
|
||||
private void load(FirstRunSetupOverlay? firstRunSetup)
|
||||
private void load()
|
||||
{
|
||||
X = WIDTH;
|
||||
Width = WIDTH;
|
||||
RelativeSizeAxes = Axes.Y;
|
||||
|
||||
Children = new Drawable[]
|
||||
{
|
||||
toastTray = new NotificationOverlayToastTray
|
||||
{
|
||||
ForwardNotificationToPermanentStore = addPermanently,
|
||||
Origin = Anchor.TopRight,
|
||||
},
|
||||
mainContent = new Container
|
||||
{
|
||||
AlwaysPresent = true,
|
||||
RelativeSizeAxes = Axes.Both,
|
||||
Children = new Drawable[]
|
||||
{
|
||||
new Box
|
||||
@ -72,23 +99,22 @@ namespace osu.Game.Overlays
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
};
|
||||
|
||||
if (firstRunSetup != null)
|
||||
firstRunSetupVisibility.BindTo(firstRunSetup.State);
|
||||
}
|
||||
|
||||
private ScheduledDelegate? notificationsEnabler;
|
||||
|
||||
private void updateProcessingMode()
|
||||
{
|
||||
bool enabled = (OverlayActivationMode.Value == OverlayActivation.All && firstRunSetupVisibility.Value != Visibility.Visible) || State.Value == Visibility.Visible;
|
||||
bool enabled = OverlayActivationMode.Value == OverlayActivation.All || State.Value == Visibility.Visible;
|
||||
|
||||
notificationsEnabler?.Cancel();
|
||||
|
||||
if (enabled)
|
||||
// we want a slight delay before toggling notifications on to avoid the user becoming overwhelmed.
|
||||
notificationsEnabler = Scheduler.AddDelayed(() => processingPosts = true, State.Value == Visibility.Visible ? 0 : 1000);
|
||||
notificationsEnabler = Scheduler.AddDelayed(() => processingPosts = true, State.Value == Visibility.Visible ? 0 : 100);
|
||||
else
|
||||
processingPosts = false;
|
||||
}
|
||||
@ -98,12 +124,13 @@ namespace osu.Game.Overlays
|
||||
base.LoadComplete();
|
||||
|
||||
State.BindValueChanged(_ => updateProcessingMode());
|
||||
firstRunSetupVisibility.BindValueChanged(_ => updateProcessingMode());
|
||||
OverlayActivationMode.BindValueChanged(_ => updateProcessingMode(), true);
|
||||
}
|
||||
|
||||
public IBindable<int> UnreadCount => unreadCount;
|
||||
|
||||
public int ToastCount => toastTray.UnreadCount;
|
||||
|
||||
private readonly BindableInt unreadCount = new BindableInt();
|
||||
|
||||
private int runningDepth;
|
||||
@ -127,18 +154,28 @@ namespace osu.Game.Overlays
|
||||
if (notification is IHasCompletionTarget hasCompletionTarget)
|
||||
hasCompletionTarget.CompletionTarget = Post;
|
||||
|
||||
var ourType = notification.GetType();
|
||||
playDebouncedSample(notification.PopInSampleName);
|
||||
|
||||
var section = sections.Children.FirstOrDefault(s => s.AcceptedNotificationTypes.Any(accept => accept.IsAssignableFrom(ourType)));
|
||||
section?.Add(notification, notification.DisplayOnTop ? -runningDepth : runningDepth);
|
||||
|
||||
if (notification.IsImportant)
|
||||
Show();
|
||||
if (State.Value == Visibility.Hidden)
|
||||
toastTray.Post(notification);
|
||||
else
|
||||
addPermanently(notification);
|
||||
|
||||
updateCounts();
|
||||
playDebouncedSample(notification.PopInSampleName);
|
||||
});
|
||||
|
||||
private void addPermanently(Notification notification)
|
||||
{
|
||||
var ourType = notification.GetType();
|
||||
int depth = notification.DisplayOnTop ? -runningDepth : runningDepth;
|
||||
|
||||
var section = sections.Children.First(s => s.AcceptedNotificationTypes.Any(accept => accept.IsAssignableFrom(ourType)));
|
||||
|
||||
section.Add(notification, depth);
|
||||
|
||||
updateCounts();
|
||||
}
|
||||
|
||||
protected override void Update()
|
||||
{
|
||||
base.Update();
|
||||
@ -152,7 +189,9 @@ namespace osu.Game.Overlays
|
||||
base.PopIn();
|
||||
|
||||
this.MoveToX(0, TRANSITION_LENGTH, Easing.OutQuint);
|
||||
this.FadeTo(1, TRANSITION_LENGTH, Easing.OutQuint);
|
||||
mainContent.FadeTo(1, TRANSITION_LENGTH, Easing.OutQuint);
|
||||
|
||||
toastTray.FlushAllToasts();
|
||||
}
|
||||
|
||||
protected override void PopOut()
|
||||
@ -162,7 +201,7 @@ namespace osu.Game.Overlays
|
||||
markAllRead();
|
||||
|
||||
this.MoveToX(WIDTH, TRANSITION_LENGTH, Easing.OutQuint);
|
||||
this.FadeTo(0, TRANSITION_LENGTH, Easing.OutQuint);
|
||||
mainContent.FadeTo(0, TRANSITION_LENGTH, Easing.OutQuint);
|
||||
}
|
||||
|
||||
private void notificationClosed()
|
||||
@ -183,16 +222,16 @@ namespace osu.Game.Overlays
|
||||
}
|
||||
}
|
||||
|
||||
private void updateCounts()
|
||||
{
|
||||
unreadCount.Value = sections.Select(c => c.UnreadCount).Sum();
|
||||
}
|
||||
|
||||
private void markAllRead()
|
||||
{
|
||||
sections.Children.ForEach(s => s.MarkAllRead());
|
||||
|
||||
toastTray.MarkAllRead();
|
||||
updateCounts();
|
||||
}
|
||||
|
||||
private void updateCounts()
|
||||
{
|
||||
unreadCount.Value = sections.Select(c => c.UnreadCount).Sum() + toastTray.UnreadCount;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
153
osu.Game/Overlays/NotificationOverlayToastTray.cs
Normal file
153
osu.Game/Overlays/NotificationOverlayToastTray.cs
Normal file
@ -0,0 +1,153 @@
|
||||
// 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.
|
||||
|
||||
using System;
|
||||
using System.Diagnostics;
|
||||
using System.Linq;
|
||||
using osu.Framework.Allocation;
|
||||
using osu.Framework.Extensions.Color4Extensions;
|
||||
using osu.Framework.Extensions.IEnumerableExtensions;
|
||||
using osu.Framework.Graphics;
|
||||
using osu.Framework.Graphics.Colour;
|
||||
using osu.Framework.Graphics.Containers;
|
||||
using osu.Framework.Graphics.Effects;
|
||||
using osu.Framework.Graphics.Shapes;
|
||||
using osu.Framework.Utils;
|
||||
using osu.Game.Overlays.Notifications;
|
||||
using osuTK;
|
||||
|
||||
namespace osu.Game.Overlays
|
||||
{
|
||||
/// <summary>
|
||||
/// A tray which attaches to the left of <see cref="NotificationOverlay"/> to show temporary toasts.
|
||||
/// </summary>
|
||||
public class NotificationOverlayToastTray : CompositeDrawable
|
||||
{
|
||||
public bool IsDisplayingToasts => toastFlow.Count > 0;
|
||||
|
||||
private FillFlowContainer<Notification> toastFlow = null!;
|
||||
private BufferedContainer toastContentBackground = null!;
|
||||
|
||||
[Resolved]
|
||||
private OverlayColourProvider colourProvider { get; set; } = null!;
|
||||
|
||||
public Action<Notification>? ForwardNotificationToPermanentStore { get; set; }
|
||||
|
||||
public int UnreadCount => toastFlow.Count(n => !n.WasClosed && !n.Read)
|
||||
+ InternalChildren.OfType<Notification>().Count(n => !n.WasClosed && !n.Read);
|
||||
|
||||
private int runningDepth;
|
||||
|
||||
[BackgroundDependencyLoader]
|
||||
private void load()
|
||||
{
|
||||
RelativeSizeAxes = Axes.X;
|
||||
AutoSizeAxes = Axes.Y;
|
||||
Padding = new MarginPadding(20);
|
||||
|
||||
InternalChildren = new Drawable[]
|
||||
{
|
||||
toastContentBackground = (new Box
|
||||
{
|
||||
Anchor = Anchor.TopRight,
|
||||
Origin = Anchor.TopRight,
|
||||
Colour = ColourInfo.GradientVertical(
|
||||
colourProvider.Background6.Opacity(0.7f),
|
||||
colourProvider.Background6.Opacity(0.5f)),
|
||||
RelativeSizeAxes = Axes.Both,
|
||||
}.WithEffect(new BlurEffect
|
||||
{
|
||||
PadExtent = true,
|
||||
Sigma = new Vector2(20),
|
||||
}).With(postEffectDrawable =>
|
||||
{
|
||||
postEffectDrawable.Scale = new Vector2(1.5f, 1);
|
||||
postEffectDrawable.Position += new Vector2(70, -50);
|
||||
postEffectDrawable.AutoSizeAxes = Axes.None;
|
||||
postEffectDrawable.RelativeSizeAxes = Axes.X;
|
||||
})),
|
||||
toastFlow = new AlwaysUpdateFillFlowContainer<Notification>
|
||||
{
|
||||
LayoutDuration = 150,
|
||||
LayoutEasing = Easing.OutQuart,
|
||||
Spacing = new Vector2(3),
|
||||
RelativeSizeAxes = Axes.X,
|
||||
AutoSizeAxes = Axes.Y,
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
public void MarkAllRead()
|
||||
{
|
||||
toastFlow.Children.ForEach(n => n.Read = true);
|
||||
InternalChildren.OfType<Notification>().ForEach(n => n.Read = true);
|
||||
}
|
||||
|
||||
public void FlushAllToasts()
|
||||
{
|
||||
foreach (var notification in toastFlow.ToArray())
|
||||
forwardNotification(notification);
|
||||
}
|
||||
|
||||
public void Post(Notification notification)
|
||||
{
|
||||
++runningDepth;
|
||||
|
||||
int depth = notification.DisplayOnTop ? -runningDepth : runningDepth;
|
||||
|
||||
toastFlow.Insert(depth, notification);
|
||||
|
||||
scheduleDismissal();
|
||||
|
||||
void scheduleDismissal() => Scheduler.AddDelayed(() =>
|
||||
{
|
||||
// Notification dismissed by user.
|
||||
if (notification.WasClosed)
|
||||
return;
|
||||
|
||||
// Notification forwarded away.
|
||||
if (notification.Parent != toastFlow)
|
||||
return;
|
||||
|
||||
// Notification hovered; delay dismissal.
|
||||
if (notification.IsHovered)
|
||||
{
|
||||
scheduleDismissal();
|
||||
return;
|
||||
}
|
||||
|
||||
// All looks good, forward away!
|
||||
forwardNotification(notification);
|
||||
}, notification.IsImportant ? 12000 : 2500);
|
||||
}
|
||||
|
||||
private void forwardNotification(Notification notification)
|
||||
{
|
||||
Debug.Assert(notification.Parent == toastFlow);
|
||||
|
||||
// Temporarily remove from flow so we can animate the position off to the right.
|
||||
toastFlow.Remove(notification);
|
||||
AddInternal(notification);
|
||||
|
||||
notification.MoveToOffset(new Vector2(400, 0), NotificationOverlay.TRANSITION_LENGTH, Easing.OutQuint);
|
||||
notification.FadeOut(NotificationOverlay.TRANSITION_LENGTH, Easing.OutQuint).OnComplete(_ =>
|
||||
{
|
||||
RemoveInternal(notification);
|
||||
ForwardNotificationToPermanentStore?.Invoke(notification);
|
||||
|
||||
notification.FadeIn(300, Easing.OutQuint);
|
||||
});
|
||||
}
|
||||
|
||||
protected override void Update()
|
||||
{
|
||||
base.Update();
|
||||
|
||||
float height = toastFlow.DrawHeight + 120;
|
||||
float alpha = IsDisplayingToasts ? MathHelper.Clamp(toastFlow.DrawHeight / 40, 0, 1) * toastFlow.Children.Max(n => n.Alpha) : 0;
|
||||
|
||||
toastContentBackground.Height = (float)Interpolation.DampContinuously(toastContentBackground.Height, height, 10, Clock.ElapsedFrameTime);
|
||||
toastContentBackground.Alpha = (float)Interpolation.DampContinuously(toastContentBackground.Alpha, alpha, 10, Clock.ElapsedFrameTime);
|
||||
}
|
||||
}
|
||||
}
|
@ -62,6 +62,8 @@ namespace osu.Game.Overlays.Notifications
|
||||
[Resolved]
|
||||
private OverlayColourProvider colourProvider { get; set; } = null!;
|
||||
|
||||
private readonly Box initialFlash;
|
||||
|
||||
private Box background = null!;
|
||||
|
||||
protected Notification()
|
||||
@ -134,6 +136,12 @@ namespace osu.Game.Overlays.Notifications
|
||||
}
|
||||
},
|
||||
},
|
||||
initialFlash = new Box
|
||||
{
|
||||
Colour = Color4.White.Opacity(0.8f),
|
||||
RelativeSizeAxes = Axes.Both,
|
||||
Blending = BlendingParameters.Additive,
|
||||
},
|
||||
}
|
||||
}
|
||||
};
|
||||
@ -178,6 +186,8 @@ namespace osu.Game.Overlays.Notifications
|
||||
|
||||
MainContent.MoveToX(DrawSize.X);
|
||||
MainContent.MoveToX(0, 500, Easing.OutQuint);
|
||||
|
||||
initialFlash.FadeOutFromOne(2000, Easing.OutQuart);
|
||||
}
|
||||
|
||||
public bool WasClosed;
|
||||
|
Loading…
Reference in New Issue
Block a user