1
0
mirror of https://github.com/ppy/osu.git synced 2025-01-25 21:32:57 +08:00
osu-lazer/osu.Game/Overlays/Notifications/ProgressNotification.cs

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

335 lines
10 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 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
using System;
using System.Threading;
2017-02-10 15:26:43 +08:00
using osu.Framework.Allocation;
using osu.Framework.Audio;
using osu.Framework.Audio.Sample;
using osu.Framework.Extensions.Color4Extensions;
2017-02-10 15:26:43 +08:00
using osu.Framework.Graphics;
using osu.Framework.Graphics.Colour;
2017-02-10 15:26:43 +08:00
using osu.Framework.Graphics.Containers;
using osu.Framework.Graphics.Shapes;
using osu.Framework.Graphics.Sprites;
using osu.Framework.Localisation;
2017-02-10 15:26:43 +08:00
using osu.Game.Graphics;
using osu.Game.Graphics.Containers;
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
2017-02-10 15:26:43 +08:00
namespace osu.Game.Overlays.Notifications
{
public partial class ProgressNotification : Notification, IHasCompletionTarget
{
private const float loading_spinner_size = 22;
public Func<bool>? CancelRequested { get; set; }
/// <summary>
/// Whether the operation represented by the <see cref="ProgressNotification"/> is still ongoing.
/// </summary>
public bool Ongoing => State != ProgressNotificationState.Completed && State != ProgressNotificationState.Cancelled;
protected override bool AllowFlingDismiss => false;
public override string PopOutSampleName => State is ProgressNotificationState.Cancelled ? base.PopOutSampleName : "";
/// <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;
public override LocalisableString Text
{
2021-11-09 20:34:36 +08:00
get => text;
set
{
text = value;
Scheduler.AddOnce(t => textDrawable.Text = t, text);
2021-11-09 20:34:36 +08:00
}
}
2018-04-13 17:19:50 +08:00
public LocalisableString CompletionText { get; set; } = "Task has completed!";
2018-04-13 17:19:50 +08:00
private float progress;
2017-02-10 15:26:43 +08:00
public float Progress
{
get => progress;
set
{
progress = value;
2022-08-30 16:33:08 +08:00
Scheduler.AddOnce(p => progressBar.Progress = p, progress);
}
2017-02-10 15:26:43 +08:00
}
2018-04-13 17:19:50 +08:00
protected override IconUsage CloseButtonIcon => FontAwesome.Solid.Times;
[Resolved]
private OverlayColourProvider colourProvider { get; set; } = null!;
protected override void LoadComplete()
{
base.LoadComplete();
2018-04-13 17:19:50 +08:00
2020-05-05 09:31:11 +08:00
// we may have received changes before we were displayed.
Scheduler.AddOnce(updateState);
}
2018-04-13 17:19:50 +08:00
private readonly CancellationTokenSource cancellationTokenSource = new CancellationTokenSource();
public CancellationToken CancellationToken => cancellationTokenSource.Token;
public ProgressNotificationState State
2017-02-10 15:26:43 +08:00
{
get => state;
set
{
if (state == value) return;
state = value;
Scheduler.AddOnce(updateState);
attemptPostCompletion();
}
}
private void updateState()
{
const double colour_fade_duration = 200;
switch (state)
{
case ProgressNotificationState.Queued:
Light.Colour = colourQueued;
Light.Pulsate = false;
progressBar.Active = false;
IconContent.FadeColour(ColourInfo.GradientVertical(colourQueued, colourQueued.Lighten(0.5f)), colour_fade_duration);
loadingSpinner.Show();
break;
case ProgressNotificationState.Active:
Light.Colour = colourActive;
Light.Pulsate = true;
progressBar.Active = true;
IconContent.FadeColour(ColourInfo.GradientVertical(colourActive, colourActive.Lighten(0.5f)), colour_fade_duration);
loadingSpinner.Show();
break;
case ProgressNotificationState.Cancelled:
cancellationTokenSource.Cancel();
IconContent.FadeColour(ColourInfo.GradientVertical(Color4.Gray, Color4.Gray.Lighten(0.5f)), colour_fade_duration);
cancelSample?.Play();
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);
Light.Colour = colourCancelled;
Light.Pulsate = false;
progressBar.Active = false;
break;
case ProgressNotificationState.Completed:
loadingSpinner.Hide();
attemptPostCompletion();
break;
}
2017-02-10 15:26:43 +08:00
}
2018-04-13 17:19:50 +08:00
private int completionSent;
/// <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;
// 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)
return;
CompletionTarget.Invoke(CreateCompletionNotification());
Close(false);
}
2017-02-10 15:26:43 +08:00
private ProgressNotificationState state;
2018-04-13 17:19:50 +08:00
protected virtual Notification CreateCompletionNotification() => new ProgressCompletionNotification
{
Activated = CompletionClickAction,
Text = CompletionText
};
2018-04-13 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
public override bool DisplayOnTop => false;
2018-04-13 17:19:50 +08:00
public override bool IsImportant => false;
private readonly ProgressBar progressBar;
2017-02-10 15:26:43 +08:00
private Color4 colourQueued;
private Color4 colourActive;
private Color4 colourCancelled;
2018-04-13 17:19:50 +08:00
private LoadingSpinner loadingSpinner = null!;
private Sample? cancelSample;
private readonly TextFlowContainer textDrawable;
2018-04-13 17:19:50 +08:00
2017-02-12 18:39:54 +08:00
public ProgressNotification()
2017-02-10 15:26:43 +08:00
{
Content.Add(textDrawable = new OsuTextFlowContainer(t => t.Font = t.Font.With(size: 14, weight: FontWeight.Medium))
2017-02-10 15:26:43 +08:00
{
AutoSizeAxes = Axes.Y,
RelativeSizeAxes = Axes.X,
});
2018-04-13 17:19:50 +08:00
MainContent.Add(progressBar = new ProgressBar
2017-02-10 15:26:43 +08:00
{
Origin = Anchor.BottomLeft,
Anchor = Anchor.BottomLeft,
RelativeSizeAxes = Axes.X,
});
2018-04-13 17:19:50 +08:00
// make some extra space for the progress bar.
IconContent.Margin = new MarginPadding { Bottom = 5 };
2017-02-10 15:26:43 +08:00
State = ProgressNotificationState.Queued;
2018-04-13 17:19:50 +08:00
// don't close on click by default.
Activated = () => false;
2017-02-10 15:26:43 +08:00
}
2018-04-13 17:19:50 +08:00
2017-02-12 18:39:54 +08:00
[BackgroundDependencyLoader]
private void load(OsuColour colours, AudioManager audioManager)
2017-02-12 18:39:54 +08:00
{
colourQueued = colours.YellowDark;
colourActive = colours.Blue;
colourCancelled = colours.Red;
IconContent.AddRange(new Drawable[]
{
new Box
{
RelativeSizeAxes = Axes.Both,
Colour = colourProvider.Background5,
Depth = float.MaxValue,
},
loadingSpinner = new LoadingSpinner
{
Size = new Vector2(loading_spinner_size),
}
});
cancelSample = audioManager.Samples.Get(@"UI/notification-cancel");
2017-02-12 18:39:54 +08:00
}
2018-04-13 17:19:50 +08:00
public override void Close(bool runFlingAnimation)
2017-02-10 15:26:43 +08:00
{
switch (State)
{
case ProgressNotificationState.Completed:
2017-02-10 15:26:43 +08:00
case ProgressNotificationState.Cancelled:
base.Close(runFlingAnimation);
2017-02-10 15:26:43 +08:00
break;
2019-04-01 11:44:46 +08:00
2017-02-10 15:26:43 +08:00
case ProgressNotificationState.Active:
case ProgressNotificationState.Queued:
if (CancelRequested?.Invoke() != false)
State = ProgressNotificationState.Cancelled;
2017-02-10 15:26:43 +08:00
break;
}
}
2018-04-13 17:19:50 +08:00
2017-03-07 09:59:19 +08:00
private partial class ProgressBar : Container
2017-02-10 15:26:43 +08:00
{
private readonly Box box;
2018-04-13 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
private Color4 colourActive;
private Color4 colourInactive;
2018-04-13 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
private float progress;
2019-02-28 12:31:40 +08:00
2017-02-10 15:26:43 +08:00
public float Progress
{
get => progress;
2017-02-10 15:26:43 +08:00
set
{
if (progress == value) return;
2018-04-13 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
progress = value;
2017-07-23 02:50:25 +08:00
box.ResizeTo(new Vector2(progress, 1), 100, Easing.OutQuad);
2017-02-10 15:26:43 +08:00
}
}
2018-04-13 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
private bool active;
2018-04-13 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
public bool Active
{
get => active;
2017-02-10 15:26:43 +08:00
set
{
active = value;
this.FadeColour(active ? colourActive : colourInactive, 100);
2017-02-10 15:26:43 +08:00
}
}
2018-04-13 17:19:50 +08:00
public ProgressBar()
2017-02-10 15:26:43 +08:00
{
Children = new[]
{
box = new Box
{
RelativeSizeAxes = Axes.Both,
Width = 0,
}
};
}
2018-04-13 17:19:50 +08:00
[BackgroundDependencyLoader]
private void load(OsuColour colours)
{
colourActive = colours.Blue;
Colour = colourInactive = OsuColour.Gray(0.5f);
Height = 5;
}
2017-02-10 15:26:43 +08:00
}
}
2018-04-13 17:19:50 +08:00
2017-02-10 15:26:43 +08:00
public enum ProgressNotificationState
{
Queued,
Active,
Completed,
Cancelled
}
}