2019-01-24 16:43:03 +08: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-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
using System;
|
2019-05-28 17:59:21 +08:00
|
|
|
|
using System.Threading;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Framework.Allocation;
|
2023-07-05 16:27:42 +08:00
|
|
|
|
using osu.Framework.Audio;
|
|
|
|
|
using osu.Framework.Audio.Sample;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
using osu.Framework.Extensions.Color4Extensions;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Framework.Graphics;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
using osu.Framework.Graphics.Colour;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Framework.Graphics.Containers;
|
|
|
|
|
using osu.Framework.Graphics.Shapes;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
using osu.Framework.Graphics.Sprites;
|
2021-10-31 00:50:13 +08:00
|
|
|
|
using osu.Framework.Localisation;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
using osu.Game.Graphics;
|
|
|
|
|
using osu.Game.Graphics.Containers;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
using osu.Game.Graphics.UserInterface;
|
2018-11-20 15:51:59 +08:00
|
|
|
|
using osuTK;
|
|
|
|
|
using osuTK.Graphics;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
|
|
|
|
namespace osu.Game.Overlays.Notifications
|
|
|
|
|
{
|
2022-11-24 13:32:20 +08:00
|
|
|
|
public partial class ProgressNotification : Notification, IHasCompletionTarget
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2021-10-28 15:34:12 +08:00
|
|
|
|
private const float loading_spinner_size = 22;
|
|
|
|
|
|
2022-08-30 14:51:46 +08:00
|
|
|
|
public Func<bool>? CancelRequested { get; set; }
|
|
|
|
|
|
2023-07-08 20:11:58 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// Whether the operation represented by the <see cref="ProgressNotification"/> is still ongoing.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public bool Ongoing => State != ProgressNotificationState.Completed && State != ProgressNotificationState.Cancelled;
|
2023-07-07 12:10:49 +08:00
|
|
|
|
|
2022-09-26 13:30:40 +08:00
|
|
|
|
protected override bool AllowFlingDismiss => false;
|
|
|
|
|
|
2023-07-07 00:50:15 +08:00
|
|
|
|
public override string PopOutSampleName => State is ProgressNotificationState.Cancelled ? base.PopOutSampleName : "";
|
|
|
|
|
|
2022-08-30 14:51:46 +08:00
|
|
|
|
/// <summary>
|
|
|
|
|
/// The function to post completion notifications back to.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Action<Notification>? CompletionTarget { get; set; }
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// An action to complete when the completion notification is clicked. Return true to close.
|
|
|
|
|
/// </summary>
|
|
|
|
|
public Func<bool>? CompletionClickAction { get; set; }
|
|
|
|
|
|
2021-11-09 20:34:36 +08:00
|
|
|
|
private LocalisableString text;
|
|
|
|
|
|
2022-01-25 22:38:46 +08:00
|
|
|
|
public override LocalisableString Text
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2021-11-09 20:34:36 +08:00
|
|
|
|
get => text;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
text = value;
|
|
|
|
|
Schedule(() => textDrawable.Text = text);
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2023-02-17 21:32:03 +08:00
|
|
|
|
public LocalisableString CompletionText { get; set; } = "Task has completed!";
|
2018-04-13 17:19:50 +08:00
|
|
|
|
|
2019-06-20 18:05:33 +08:00
|
|
|
|
private float progress;
|
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
public float Progress
|
|
|
|
|
{
|
2019-06-20 18:05:33 +08:00
|
|
|
|
get => progress;
|
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
progress = value;
|
2022-08-30 16:33:08 +08:00
|
|
|
|
Scheduler.AddOnce(p => progressBar.Progress = p, progress);
|
2019-06-20 18:05:33 +08:00
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-08-30 16:37:43 +08:00
|
|
|
|
protected override IconUsage CloseButtonIcon => FontAwesome.Solid.Times;
|
|
|
|
|
|
|
|
|
|
[Resolved]
|
|
|
|
|
private OverlayColourProvider colourProvider { get; set; } = null!;
|
2021-10-14 16:52:19 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
protected override void LoadComplete()
|
|
|
|
|
{
|
|
|
|
|
base.LoadComplete();
|
|
|
|
|
|
2020-05-05 09:31:11 +08:00
|
|
|
|
// we may have received changes before we were displayed.
|
2022-09-06 03:07:49 +08:00
|
|
|
|
Scheduler.AddOnce(updateState);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-28 17:59:21 +08:00
|
|
|
|
private readonly CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
|
|
|
|
|
|
|
|
|
|
public CancellationToken CancellationToken => cancellationTokenSource.Token;
|
|
|
|
|
|
2019-06-20 18:05:33 +08:00
|
|
|
|
public ProgressNotificationState State
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
2019-02-28 12:58:19 +08:00
|
|
|
|
get => state;
|
2019-06-20 18:05:33 +08:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (state == value) return;
|
|
|
|
|
|
|
|
|
|
state = value;
|
|
|
|
|
|
2022-09-06 03:07:49 +08:00
|
|
|
|
Scheduler.AddOnce(updateState);
|
|
|
|
|
attemptPostCompletion();
|
2019-06-20 18:05:33 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private void updateState()
|
|
|
|
|
{
|
2021-10-28 15:34:12 +08:00
|
|
|
|
const double colour_fade_duration = 200;
|
|
|
|
|
|
2019-06-20 18:05:33 +08:00
|
|
|
|
switch (state)
|
|
|
|
|
{
|
|
|
|
|
case ProgressNotificationState.Queued:
|
|
|
|
|
Light.Colour = colourQueued;
|
|
|
|
|
Light.Pulsate = false;
|
|
|
|
|
progressBar.Active = false;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
|
2022-08-30 16:40:35 +08:00
|
|
|
|
IconContent.FadeColour(ColourInfo.GradientVertical(colourQueued, colourQueued.Lighten(0.5f)), colour_fade_duration);
|
2021-10-28 15:34:12 +08:00
|
|
|
|
loadingSpinner.Show();
|
2019-06-20 18:05:33 +08:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ProgressNotificationState.Active:
|
|
|
|
|
Light.Colour = colourActive;
|
|
|
|
|
Light.Pulsate = true;
|
|
|
|
|
progressBar.Active = true;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
|
2022-08-30 16:40:35 +08:00
|
|
|
|
IconContent.FadeColour(ColourInfo.GradientVertical(colourActive, colourActive.Lighten(0.5f)), colour_fade_duration);
|
2021-10-28 15:34:12 +08:00
|
|
|
|
loadingSpinner.Show();
|
2019-06-20 18:05:33 +08:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ProgressNotificationState.Cancelled:
|
|
|
|
|
cancellationTokenSource.Cancel();
|
|
|
|
|
|
2022-08-30 16:40:35 +08:00
|
|
|
|
IconContent.FadeColour(ColourInfo.GradientVertical(Color4.Gray, Color4.Gray.Lighten(0.5f)), colour_fade_duration);
|
2023-07-05 16:27:42 +08:00
|
|
|
|
cancelSample?.Play();
|
2021-10-28 15:34:12 +08:00
|
|
|
|
loadingSpinner.Hide();
|
|
|
|
|
|
|
|
|
|
var icon = new SpriteIcon
|
|
|
|
|
{
|
|
|
|
|
Icon = FontAwesome.Solid.Ban,
|
|
|
|
|
Anchor = Anchor.Centre,
|
|
|
|
|
Origin = Anchor.Centre,
|
|
|
|
|
Size = new Vector2(loading_spinner_size),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
IconContent.Add(icon);
|
|
|
|
|
|
|
|
|
|
icon.FadeInFromZero(200, Easing.OutQuint);
|
|
|
|
|
|
2019-06-20 18:05:33 +08:00
|
|
|
|
Light.Colour = colourCancelled;
|
|
|
|
|
Light.Pulsate = false;
|
|
|
|
|
progressBar.Active = false;
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
case ProgressNotificationState.Completed:
|
2021-10-28 15:34:12 +08:00
|
|
|
|
loadingSpinner.Hide();
|
2022-09-16 16:54:44 +08:00
|
|
|
|
attemptPostCompletion();
|
2019-06-20 18:05:33 +08:00
|
|
|
|
break;
|
|
|
|
|
}
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-11-02 13:47:56 +08:00
|
|
|
|
private int completionSent;
|
2022-09-06 03:07:49 +08:00
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Attempt to post a completion notification.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void attemptPostCompletion()
|
|
|
|
|
{
|
|
|
|
|
if (state != ProgressNotificationState.Completed) return;
|
|
|
|
|
|
|
|
|
|
// This notification may not have been posted yet (and thus may not have a target to post the completion to).
|
|
|
|
|
// Completion posting will be re-attempted in a scheduled invocation.
|
|
|
|
|
if (CompletionTarget == null)
|
|
|
|
|
return;
|
|
|
|
|
|
2022-11-02 13:47:56 +08:00
|
|
|
|
// Thread-safe barrier, as this may be called by a web request and also scheduled to the update thread at the same time.
|
2022-11-02 14:47:30 +08:00
|
|
|
|
if (Interlocked.Exchange(ref completionSent, 1) == 1)
|
2022-09-06 03:07:49 +08:00
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
CompletionTarget.Invoke(CreateCompletionNotification());
|
2022-09-16 15:36:56 +08:00
|
|
|
|
|
|
|
|
|
Close(false);
|
2022-09-06 03:07:49 +08:00
|
|
|
|
}
|
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
private ProgressNotificationState state;
|
|
|
|
|
|
|
|
|
|
protected virtual Notification CreateCompletionNotification() => new ProgressCompletionNotification
|
|
|
|
|
{
|
|
|
|
|
Activated = CompletionClickAction,
|
|
|
|
|
Text = CompletionText
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
public override bool DisplayOnTop => false;
|
|
|
|
|
|
2022-09-01 23:58:10 +08:00
|
|
|
|
public override bool IsImportant => false;
|
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
private readonly ProgressBar progressBar;
|
|
|
|
|
private Color4 colourQueued;
|
|
|
|
|
private Color4 colourActive;
|
|
|
|
|
private Color4 colourCancelled;
|
|
|
|
|
|
2022-08-30 14:51:46 +08:00
|
|
|
|
private LoadingSpinner loadingSpinner = null!;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
|
2023-07-05 16:27:42 +08:00
|
|
|
|
private Sample? cancelSample;
|
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
private readonly TextFlowContainer textDrawable;
|
|
|
|
|
|
|
|
|
|
public ProgressNotification()
|
|
|
|
|
{
|
2022-08-31 11:49:28 +08:00
|
|
|
|
Content.Add(textDrawable = new OsuTextFlowContainer(t => t.Font = t.Font.With(size: 14, weight: FontWeight.Medium))
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
AutoSizeAxes = Axes.Y,
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
});
|
|
|
|
|
|
2022-08-30 16:49:29 +08:00
|
|
|
|
MainContent.Add(progressBar = new ProgressBar
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
Origin = Anchor.BottomLeft,
|
|
|
|
|
Anchor = Anchor.BottomLeft,
|
|
|
|
|
RelativeSizeAxes = Axes.X,
|
|
|
|
|
});
|
|
|
|
|
|
2021-10-28 15:34:12 +08:00
|
|
|
|
// make some extra space for the progress bar.
|
|
|
|
|
IconContent.Margin = new MarginPadding { Bottom = 5 };
|
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
State = ProgressNotificationState.Queued;
|
|
|
|
|
|
|
|
|
|
// don't close on click by default.
|
|
|
|
|
Activated = () => false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
2023-07-05 16:27:42 +08:00
|
|
|
|
private void load(OsuColour colours, AudioManager audioManager)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
colourQueued = colours.YellowDark;
|
|
|
|
|
colourActive = colours.Blue;
|
|
|
|
|
colourCancelled = colours.Red;
|
2021-10-28 15:34:12 +08:00
|
|
|
|
|
|
|
|
|
IconContent.AddRange(new Drawable[]
|
|
|
|
|
{
|
2022-08-30 16:40:35 +08:00
|
|
|
|
new Box
|
2021-10-28 15:34:12 +08:00
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
2022-08-30 16:40:35 +08:00
|
|
|
|
Colour = colourProvider.Background5,
|
2022-09-12 13:58:46 +08:00
|
|
|
|
Depth = float.MaxValue,
|
2021-10-28 15:34:12 +08:00
|
|
|
|
},
|
|
|
|
|
loadingSpinner = new LoadingSpinner
|
|
|
|
|
{
|
|
|
|
|
Size = new Vector2(loading_spinner_size),
|
|
|
|
|
}
|
|
|
|
|
});
|
2023-07-05 16:27:42 +08:00
|
|
|
|
|
|
|
|
|
cancelSample = audioManager.Samples.Get(@"UI/notification-cancel");
|
2018-04-13 17:19:50 +08:00
|
|
|
|
}
|
|
|
|
|
|
2022-09-12 17:57:18 +08:00
|
|
|
|
public override void Close(bool runFlingAnimation)
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
switch (State)
|
|
|
|
|
{
|
2022-09-16 15:36:56 +08:00
|
|
|
|
case ProgressNotificationState.Completed:
|
2018-04-13 17:19:50 +08:00
|
|
|
|
case ProgressNotificationState.Cancelled:
|
2022-09-12 17:57:18 +08:00
|
|
|
|
base.Close(runFlingAnimation);
|
2018-04-13 17:19:50 +08:00
|
|
|
|
break;
|
2019-04-01 11:44:46 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
case ProgressNotificationState.Active:
|
|
|
|
|
case ProgressNotificationState.Queued:
|
|
|
|
|
if (CancelRequested?.Invoke() != false)
|
|
|
|
|
State = ProgressNotificationState.Cancelled;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2022-11-24 13:32:20 +08:00
|
|
|
|
private partial class ProgressBar : Container
|
2018-04-13 17:19:50 +08:00
|
|
|
|
{
|
|
|
|
|
private readonly Box box;
|
|
|
|
|
|
|
|
|
|
private Color4 colourActive;
|
|
|
|
|
private Color4 colourInactive;
|
|
|
|
|
|
|
|
|
|
private float progress;
|
2019-02-28 12:31:40 +08:00
|
|
|
|
|
2018-04-13 17:19:50 +08:00
|
|
|
|
public float Progress
|
|
|
|
|
{
|
2019-02-28 12:58:19 +08:00
|
|
|
|
get => progress;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
if (progress == value) return;
|
|
|
|
|
|
|
|
|
|
progress = value;
|
|
|
|
|
box.ResizeTo(new Vector2(progress, 1), 100, Easing.OutQuad);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private bool active;
|
|
|
|
|
|
|
|
|
|
public bool Active
|
|
|
|
|
{
|
2019-02-28 12:58:19 +08:00
|
|
|
|
get => active;
|
2018-04-13 17:19:50 +08:00
|
|
|
|
set
|
|
|
|
|
{
|
|
|
|
|
active = value;
|
|
|
|
|
this.FadeColour(active ? colourActive : colourInactive, 100);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public ProgressBar()
|
|
|
|
|
{
|
|
|
|
|
Children = new[]
|
|
|
|
|
{
|
|
|
|
|
box = new Box
|
|
|
|
|
{
|
|
|
|
|
RelativeSizeAxes = Axes.Both,
|
|
|
|
|
Width = 0,
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
[BackgroundDependencyLoader]
|
|
|
|
|
private void load(OsuColour colours)
|
|
|
|
|
{
|
|
|
|
|
colourActive = colours.Blue;
|
|
|
|
|
Colour = colourInactive = OsuColour.Gray(0.5f);
|
|
|
|
|
Height = 5;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public enum ProgressNotificationState
|
|
|
|
|
{
|
|
|
|
|
Queued,
|
|
|
|
|
Active,
|
|
|
|
|
Completed,
|
|
|
|
|
Cancelled
|
|
|
|
|
}
|
|
|
|
|
}
|