osu/osu.Game/Overlays/Notifications/Notification.cs

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

500 lines
16 KiB
C#
Raw Normal View History

// 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-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
using System;
2022-09-11 12:32:22 +00:00
using System.Linq;
2017-02-10 07:26:43 +00:00
using osu.Framework.Allocation;
using osu.Framework.Extensions.Color4Extensions;
2017-02-10 07:26:43 +00:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Colour;
using osu.Framework.Graphics.Containers;
2019-04-02 05:51:28 +00:00
using osu.Framework.Graphics.Effects;
2022-09-11 12:32:22 +00:00
using osu.Framework.Graphics.Primitives;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
2018-10-02 03:02:47 +00:00
using osu.Framework.Input.Events;
using osu.Framework.Localisation;
2022-09-11 12:32:22 +00:00
using osu.Framework.Utils;
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
using osuTK;
using osuTK.Graphics;
using osuTK.Input;
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
namespace osu.Game.Overlays.Notifications
{
public abstract class Notification : Container
{
/// <summary>
/// User requested close.
2017-02-10 07:26:43 +00:00
/// </summary>
public event Action? Closed;
2018-04-13 09:19:50 +00:00
public abstract LocalisableString Text { get; set; }
2018-07-13 12:25:08 +00:00
/// <summary>
/// Whether this notification should forcefully display itself.
/// </summary>
2018-07-16 04:00:21 +00:00
public virtual bool IsImportant => true;
2018-07-13 12:25:08 +00:00
2017-02-10 07:26:43 +00:00
/// <summary>
/// Run on user activating the notification. Return true to close.
/// </summary>
public Func<bool>? Activated;
2018-04-13 09:19:50 +00:00
public Action? ForwardToOverlay { get; set; }
2017-02-10 07:26:43 +00:00
/// <summary>
/// Should we show at the top of our section on display?
/// </summary>
public virtual bool DisplayOnTop => true;
2018-04-13 09:19:50 +00:00
public virtual string PopInSampleName => "UI/notification-pop-in";
2021-01-15 06:21:50 +00:00
2017-02-10 07:26:43 +00:00
protected NotificationLight Light;
2022-08-30 08:33:08 +00:00
2017-02-10 07:26:43 +00:00
protected Container IconContent;
2022-08-30 08:33:08 +00:00
private readonly Container content;
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
protected override Container<Drawable> Content => content;
2018-04-13 09:19:50 +00:00
protected Container MainContent;
2018-04-13 09:19:50 +00:00
2022-09-11 12:32:22 +00:00
private readonly DragContainer dragContainer;
2017-03-07 01:59:19 +00:00
public virtual bool Read { get; set; }
2018-04-13 09:19:50 +00:00
public new bool IsDragged => dragContainer.IsDragged;
protected virtual IconUsage CloseButtonIcon => FontAwesome.Solid.Check;
[Resolved]
private OverlayColourProvider colourProvider { get; set; } = null!;
public override bool PropagatePositionalInputSubTree => base.PropagatePositionalInputSubTree && !WasClosed;
private bool isInToastTray;
/// <summary>
/// Whether this notification is in the <see cref="NotificationOverlayToastTray"/>.
/// </summary>
public bool IsInToastTray
{
get => isInToastTray;
set
{
isInToastTray = value;
if (!isInToastTray)
{
dragContainer.ResetPosition();
if (!Read)
Light.FadeIn(100);
}
}
}
private readonly Box initialFlash;
private Box background = null!;
2017-03-09 05:01:08 +00:00
protected Notification()
2017-02-10 07:26:43 +00:00
{
RelativeSizeAxes = Axes.X;
AutoSizeAxes = Axes.Y;
2018-04-13 09:19:50 +00:00
2022-08-30 08:33:08 +00:00
InternalChildren = new Drawable[]
2017-02-10 07:26:43 +00:00
{
Light = new NotificationLight
{
Alpha = 0,
2017-02-10 07:26:43 +00:00
Margin = new MarginPadding { Right = 5 },
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreRight,
},
2022-09-11 12:32:22 +00:00
dragContainer = new DragContainer(this)
{
// Use margin instead of FillFlow spacing to fix extra padding appearing when notification shrinks
// in height.
Padding = new MarginPadding { Vertical = 3f },
2022-09-11 12:32:22 +00:00
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
}.WithChild(MainContent = new Container
2017-02-10 07:26:43 +00:00
{
CornerRadius = 6,
2017-02-10 07:26:43 +00:00
Masking = true,
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
AutoSizeDuration = 400,
AutoSizeEasing = Easing.OutQuint,
2017-02-10 07:26:43 +00:00
Children = new Drawable[]
{
2022-08-30 08:33:08 +00:00
new GridContainer
2017-02-10 07:26:43 +00:00
{
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
2022-08-30 08:33:08 +00:00
RowDimensions = new[]
2017-02-10 07:26:43 +00:00
{
2022-08-30 08:33:08 +00:00
new Dimension(GridSizeMode.AutoSize, minSize: 60)
},
ColumnDimensions = new[]
{
new Dimension(GridSizeMode.AutoSize),
new Dimension(),
new Dimension(GridSizeMode.AutoSize),
},
Content = new[]
{
new Drawable[]
2017-02-10 07:26:43 +00:00
{
2022-08-30 08:33:08 +00:00
IconContent = new Container
{
Width = 40,
RelativeSizeAxes = Axes.Y,
},
new Container
2017-02-10 07:26:43 +00:00
{
2022-08-30 08:33:08 +00:00
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
Padding = new MarginPadding(10),
Children = new Drawable[]
{
content = new Container
{
Masking = true,
2022-08-30 08:33:08 +00:00
RelativeSizeAxes = Axes.X,
AutoSizeAxes = Axes.Y,
},
}
2017-02-10 07:26:43 +00:00
},
new CloseButton(CloseButtonIcon)
2022-08-30 08:33:08 +00:00
{
Action = () => Close(true),
2022-08-30 08:33:08 +00:00
Anchor = Anchor.TopRight,
Origin = Anchor.TopRight,
}
2017-02-10 07:26:43 +00:00
}
},
2022-08-30 08:33:08 +00:00
},
initialFlash = new Box
{
Colour = Color4.White.Opacity(0.8f),
RelativeSizeAxes = Axes.Both,
Blending = BlendingParameters.Additive,
},
2017-02-10 07:26:43 +00:00
}
2022-09-11 12:32:22 +00:00
})
2022-08-30 08:33:08 +00:00
};
2017-02-10 07:26:43 +00:00
}
2018-04-13 09:19:50 +00:00
2022-08-30 08:33:08 +00:00
[BackgroundDependencyLoader]
private void load()
2017-02-10 07:26:43 +00:00
{
MainContent.Add(background = new Box
2022-08-30 08:33:08 +00:00
{
RelativeSizeAxes = Axes.Both,
Colour = colourProvider.Background3,
Depth = float.MaxValue
});
2017-02-10 07:26:43 +00:00
}
2018-04-13 09:19:50 +00:00
protected override bool OnHover(HoverEvent e)
{
background.FadeColour(colourProvider.Background2, 200, Easing.OutQuint);
return base.OnHover(e);
}
protected override void OnHoverLost(HoverLostEvent e)
{
background.FadeColour(colourProvider.Background3, 200, Easing.OutQuint);
base.OnHoverLost(e);
}
protected override bool OnMouseDown(MouseDownEvent e)
2017-02-10 07:26:43 +00:00
{
// right click doesn't trigger OnClick so we need to handle here until that changes.
if (e.Button != MouseButton.Left)
{
Close(true);
return true;
}
return base.OnMouseDown(e);
}
protected override bool OnClick(ClickEvent e)
{
// Clicking with anything but left button should dismiss but not perform the activation action.
if (e.Button == MouseButton.Left)
Activated?.Invoke();
2018-04-13 09:19:50 +00:00
Close(false);
2017-02-10 07:26:43 +00:00
return true;
}
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
protected override void LoadComplete()
{
base.LoadComplete();
2021-01-15 06:21:50 +00:00
this.FadeInFromZero(200);
2022-08-30 08:33:08 +00:00
MainContent.MoveToX(DrawSize.X);
MainContent.MoveToX(0, 500, Easing.OutQuint);
initialFlash.FadeOutFromOne(2000, Easing.OutQuart);
2017-02-10 07:26:43 +00:00
}
2018-04-13 09:19:50 +00:00
public bool WasClosed;
2018-04-13 09:19:50 +00:00
public virtual void Close(bool runFlingAnimation)
2017-02-10 07:26:43 +00:00
{
if (WasClosed) return;
2019-02-28 04:31:40 +00:00
WasClosed = true;
2018-04-13 09:19:50 +00:00
if (runFlingAnimation && dragContainer.FlingLeft())
2022-09-11 12:32:22 +00:00
this.FadeOut(600, Easing.In);
else
this.FadeOut(100);
Closed?.Invoke();
2017-02-10 07:26:43 +00:00
Expire();
}
2018-04-13 09:19:50 +00:00
2022-09-11 12:32:22 +00:00
private class DragContainer : Container
{
private Vector2 velocity;
private Vector2 lastPosition;
private readonly Notification notification;
public DragContainer(Notification notification)
{
this.notification = notification;
}
public override RectangleF BoundingBox
{
get
{
var childBounding = Children.First().BoundingBox;
if (X < 0) childBounding *= new Vector2(1, Math.Max(0, 1 + (X / 300)));
if (Y > 0) childBounding *= new Vector2(1, Math.Max(0, 1 - (Y / 200)));
2022-09-11 12:32:22 +00:00
return childBounding;
}
}
protected override bool OnDragStart(DragStartEvent e) => notification.IsInToastTray;
2022-09-11 12:32:22 +00:00
protected override void OnDrag(DragEvent e)
{
if (!notification.IsInToastTray)
return;
2022-09-11 12:32:22 +00:00
Vector2 change = e.MousePosition - e.MouseDownPosition;
// Diminish the drag distance as we go further to simulate "rubber band" feeling.
change *= change.Length <= 0 ? 0 : MathF.Pow(change.Length, 0.8f) / change.Length;
2022-09-11 12:32:22 +00:00
// Only apply Y change if dragging to the left.
if (change.X >= 0)
2022-09-11 12:32:22 +00:00
change.Y = 0;
else
change.Y *= (float)Interpolation.ApplyEasing(Easing.InOutQuart, Math.Min(1, -change.X / 200));
2022-09-11 12:32:22 +00:00
this.MoveTo(change);
}
protected override void OnDragEnd(DragEndEvent e)
{
if (Rotation < -10 || velocity.X < -0.3f)
notification.Close(true);
else if (X > 30 || velocity.X > 0.3f)
notification.ForwardToOverlay?.Invoke();
2022-09-11 12:32:22 +00:00
else
ResetPosition();
2022-09-11 12:32:22 +00:00
base.OnDragEnd(e);
}
private bool flinging;
protected override void UpdateAfterChildren()
{
base.UpdateAfterChildren();
Rotation = Math.Min(0, X * 0.1f);
if (flinging)
{
velocity.Y += (float)Clock.ElapsedFrameTime * 0.005f;
Position += (float)Clock.ElapsedFrameTime * velocity;
}
else if (Clock.ElapsedFrameTime > 0)
2022-09-11 12:32:22 +00:00
{
Vector2 change = (Position - lastPosition) / (float)Clock.ElapsedFrameTime;
if (velocity.X == 0)
velocity = change;
else
{
velocity = new Vector2(
(float)Interpolation.DampContinuously(velocity.X, change.X, 40, Clock.ElapsedFrameTime),
(float)Interpolation.DampContinuously(velocity.Y, change.Y, 40, Clock.ElapsedFrameTime)
);
}
lastPosition = Position;
}
}
public bool FlingLeft()
{
if (!notification.IsInToastTray)
2022-09-11 12:32:22 +00:00
return false;
if (flinging)
return true;
if (velocity.X > -0.3f)
velocity.X = -0.3f - 0.5f * RNG.NextSingle();
flinging = true;
ClearTransforms();
return true;
}
public void ResetPosition()
{
this.MoveTo(Vector2.Zero, 800, Easing.OutElastic);
this.RotateTo(0, 800, Easing.OutElastic);
}
2022-09-11 12:32:22 +00:00
}
internal class CloseButton : OsuClickableContainer
2017-02-10 07:26:43 +00:00
{
2022-08-30 08:33:08 +00:00
private SpriteIcon icon = null!;
private Box background = null!;
2018-04-13 09:19:50 +00:00
private readonly IconUsage iconUsage;
2022-08-30 08:33:08 +00:00
[Resolved]
private OverlayColourProvider colourProvider { get; set; } = null!;
public CloseButton(IconUsage iconUsage)
{
this.iconUsage = iconUsage;
}
2022-08-30 08:33:08 +00:00
[BackgroundDependencyLoader]
private void load()
2017-02-10 07:26:43 +00:00
{
2022-08-30 08:33:08 +00:00
RelativeSizeAxes = Axes.Y;
Width = 28;
2018-04-13 09:19:50 +00:00
2022-08-30 08:33:08 +00:00
Children = new Drawable[]
2017-02-10 07:26:43 +00:00
{
2022-08-30 08:33:08 +00:00
background = new Box
{
Colour = OsuColour.Gray(0).Opacity(0.15f),
2022-08-30 08:33:08 +00:00
Alpha = 0,
RelativeSizeAxes = Axes.Both,
},
icon = new SpriteIcon
2017-02-10 07:26:43 +00:00
{
Anchor = Anchor.Centre,
Origin = Anchor.Centre,
Icon = iconUsage,
2022-08-30 08:33:08 +00:00
Size = new Vector2(12),
Colour = colourProvider.Foreground1,
2017-02-10 07:26:43 +00:00
}
};
}
2018-04-13 09:19:50 +00:00
2018-10-02 03:02:47 +00:00
protected override bool OnHover(HoverEvent e)
2017-02-10 07:26:43 +00:00
{
background.FadeIn(200, Easing.OutQuint);
icon.FadeColour(colourProvider.Content1, 200, Easing.OutQuint);
2018-10-02 03:02:47 +00:00
return base.OnHover(e);
2017-02-10 07:26:43 +00:00
}
2018-04-13 09:19:50 +00:00
2018-10-02 03:02:47 +00:00
protected override void OnHoverLost(HoverLostEvent e)
2017-02-10 07:26:43 +00:00
{
background.FadeOut(200, Easing.OutQuint);
icon.FadeColour(colourProvider.Foreground1, 200, Easing.OutQuint);
2018-10-02 03:02:47 +00:00
base.OnHoverLost(e);
2017-02-10 07:26:43 +00:00
}
}
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
public class NotificationLight : Container
{
private bool pulsate;
private Container pulsateLayer = null!;
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
public bool Pulsate
{
get => pulsate;
2017-02-10 07:26:43 +00:00
set
{
if (pulsate == value) return;
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
pulsate = value;
2018-04-13 09:19:50 +00:00
2017-02-25 12:12:39 +00:00
pulsateLayer.ClearTransforms();
2017-02-10 07:26:43 +00:00
pulsateLayer.Alpha = 1;
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
if (pulsate)
{
const float length = 1000;
pulsateLayer.Loop(length / 2,
2017-07-22 18:50:25 +00:00
p => p.FadeTo(0.4f, length, Easing.In).Then().FadeTo(1, length, Easing.Out)
);
2017-02-10 07:26:43 +00:00
}
}
}
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
public new SRGBColour Colour
{
set
{
base.Colour = value;
2017-06-12 03:48:47 +00:00
pulsateLayer.EdgeEffect = new EdgeEffectParameters
2017-02-10 07:26:43 +00:00
{
Colour = ((Color4)value).Opacity(0.5f), //todo: avoid cast
Type = EdgeEffectType.Glow,
Radius = 12,
Roundness = 12,
};
}
}
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
[BackgroundDependencyLoader]
private void load()
2017-02-10 07:26:43 +00:00
{
Size = new Vector2(6, 15);
2018-04-13 09:19:50 +00:00
2017-02-10 07:26:43 +00:00
Children = new[]
{
pulsateLayer = new CircularContainer
{
Anchor = Anchor.CentreLeft,
Origin = Anchor.CentreLeft,
Masking = true,
RelativeSizeAxes = Axes.Both,
Children = new[]
{
new Box
{
RelativeSizeAxes = Axes.Both,
},
}
}
};
}
}
}
2017-12-25 09:31:32 +00:00
}