mirror of
https://git.do.srb2.org/STJr/UltimateZoneBuilder.git
synced 2024-11-26 22:01:45 +00:00
204982e5f8
Behavior can be configured in the "Toasts" tab in the preferences.
445 lines
14 KiB
C#
445 lines
14 KiB
C#
#region ================== Copyright (c) 2022 Boris Iwanski
|
|
|
|
/*
|
|
* This program is free software: you can redistribute it and/or modify
|
|
*
|
|
* it under the terms of the GNU General Public License as published by
|
|
*
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
*
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.See the
|
|
*
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program.If not, see<http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#endregion
|
|
|
|
#region ================== Namespaces
|
|
|
|
using System;
|
|
using System.Collections;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics;
|
|
using System.Drawing;
|
|
using System.Linq;
|
|
using System.Reflection;
|
|
using System.Runtime.CompilerServices;
|
|
using System.Windows.Forms;
|
|
using CodeImp.DoomBuilder.Config;
|
|
using CodeImp.DoomBuilder.Controls;
|
|
using CodeImp.DoomBuilder.IO;
|
|
using CodeImp.DoomBuilder.Windows;
|
|
|
|
#endregion
|
|
|
|
namespace CodeImp.DoomBuilder
|
|
{
|
|
public enum ToastType
|
|
{
|
|
INFO,
|
|
WARNING,
|
|
ERROR
|
|
}
|
|
|
|
public enum ToastAnchor
|
|
{
|
|
TOPLEFT = 1,
|
|
TOPRIGHT,
|
|
BOTTOMRIGHT,
|
|
BOTTOMLEFT
|
|
}
|
|
|
|
internal class ToastRegistryEntry
|
|
{
|
|
public bool Enabled { get; set; }
|
|
public string Name { get; set; }
|
|
public string Title { get; set; }
|
|
public string Description { get; set; }
|
|
|
|
public ToastRegistryEntry(string name, string title, string description, bool enabled)
|
|
{
|
|
Enabled = enabled;
|
|
Name = name;
|
|
Title = title;
|
|
Description = description;
|
|
}
|
|
}
|
|
|
|
public class ToastManager
|
|
{
|
|
#region ================== Static variables
|
|
|
|
public static readonly string TITLE_INFO = "Information";
|
|
public static readonly string TITLE_WARNING = "Warning";
|
|
public static readonly string TITLE_ERROR = "Error";
|
|
|
|
#endregion
|
|
|
|
#region ================== Variables
|
|
|
|
private List<ToastControl> toasts;
|
|
private Control bindcontrol;
|
|
private Timer timer;
|
|
private bool enabled;
|
|
private ToastAnchor anchor;
|
|
private long duration;
|
|
private Dictionary<string, ToastRegistryEntry> registry;
|
|
|
|
#endregion
|
|
|
|
#region ================== Properties
|
|
|
|
internal bool Enabled { get => enabled; set => enabled = value; }
|
|
internal ToastAnchor Anchor { get => anchor; set => anchor = value; }
|
|
internal long Duration { get => duration; set => duration = value; }
|
|
internal Dictionary<string, ToastRegistryEntry> Registry { get => registry; }
|
|
|
|
#endregion
|
|
|
|
#region ================== Constructors
|
|
|
|
public ToastManager(Control bindcontrol)
|
|
{
|
|
toasts = new List<ToastControl>();
|
|
|
|
this.bindcontrol = bindcontrol;
|
|
|
|
// Create the timer that will handle moving the toasts. Do not start it, though
|
|
timer = new Timer();
|
|
timer.Interval = 1; // Actually only called every 1/64 second, because Windows
|
|
timer.Tick += UpdateEvent;
|
|
|
|
// Create registry and load toasts from actions
|
|
registry = new Dictionary<string, ToastRegistryEntry>();
|
|
}
|
|
|
|
|
|
|
|
#endregion
|
|
|
|
#region ================== Events
|
|
|
|
private void UpdateEvent(object sender, EventArgs args)
|
|
{
|
|
if (toasts.Count == 0)
|
|
return;
|
|
|
|
// Go through all toasts and check if they should decay or not. Remove toasts that reached their lifetime
|
|
for (int i = toasts.Count - 1; i >= 0; i--)
|
|
{
|
|
toasts[i].CheckDecay();
|
|
|
|
if (!toasts[i].IsAlive())
|
|
{
|
|
bindcontrol.Controls.Remove(toasts[i]);
|
|
toasts[i].Dispose(); // Dispose, otherwise it'll leak
|
|
toasts.RemoveAt(i);
|
|
}
|
|
}
|
|
|
|
// No toasts left, so we should stop the timer
|
|
if (toasts.Count == 0)
|
|
{
|
|
timer.Stop();
|
|
return;
|
|
}
|
|
|
|
ToastControl ft = toasts[0];
|
|
|
|
// We only need to update the first toasts if it didn't reach it end position yet
|
|
bool needsupdate =
|
|
((anchor == ToastAnchor.TOPLEFT || anchor == ToastAnchor.TOPRIGHT) && ft.Location.Y != ft.Margin.Top)
|
|
||
|
|
((anchor == ToastAnchor.BOTTOMLEFT || anchor == ToastAnchor.BOTTOMRIGHT) && ft.Location.Y != bindcontrol.Height - ft.Height - ft.Margin.Bottom)
|
|
;
|
|
|
|
if(needsupdate)
|
|
{
|
|
int left;
|
|
int top;
|
|
|
|
if (anchor == ToastAnchor.TOPLEFT || anchor == ToastAnchor.BOTTOMLEFT)
|
|
left = ft.Margin.Right;
|
|
else
|
|
left = bindcontrol.Width - ft.Width - ft.Margin.Right;
|
|
|
|
// This moves the toast up or down a bit, depending on its anchor position. How fast this happens depends on
|
|
// the control's height, i.e. no matter the height a toast will always take the same time to slide in
|
|
// TODO: make it dependent on elapsed time
|
|
if (anchor == ToastAnchor.TOPLEFT || anchor == ToastAnchor.TOPRIGHT)
|
|
top = ft.Location.Y + ft.Height / 5;
|
|
else
|
|
top = ft.Location.Y - ft.Height / 5;
|
|
|
|
Point newLocation = new Point(left, top);
|
|
|
|
// If the movement overshot the final position snap it back to the final position
|
|
if ((anchor == ToastAnchor.BOTTOMLEFT || anchor == ToastAnchor.BOTTOMRIGHT) && newLocation.Y < bindcontrol.Height - ft.Height - ft.Margin.Bottom)
|
|
newLocation.Y = bindcontrol.Height - ft.Height - ft.Margin.Bottom;
|
|
else if ((anchor == ToastAnchor.TOPLEFT || anchor == ToastAnchor.TOPRIGHT) && newLocation.Y > ft.Margin.Top)
|
|
newLocation.Y = ft.Margin.Top;
|
|
|
|
ft.Location = newLocation;
|
|
}
|
|
|
|
if (toasts.Count > 1)
|
|
{
|
|
// Align all other toasts to their predecessor
|
|
for (int i = 1; i < toasts.Count; i++)
|
|
{
|
|
int top;
|
|
|
|
if (anchor == ToastAnchor.TOPLEFT || anchor == ToastAnchor.TOPRIGHT)
|
|
top = toasts[i - 1].Bottom + toasts[i - 1].Margin.Bottom;
|
|
else
|
|
top = toasts[i - 1].Location.Y - toasts[i].Height - toasts[i].Margin.Bottom;
|
|
|
|
toasts[i].Location = new Point(
|
|
ft.Location.X,
|
|
top
|
|
);
|
|
}
|
|
}
|
|
}
|
|
|
|
#endregion
|
|
|
|
#region ================== Methods
|
|
|
|
public void LoadSettings(Configuration cfg)
|
|
{
|
|
enabled = cfg.ReadSetting("toasts.enabled", true);
|
|
anchor = GetAnchorFromNumber(cfg.ReadSetting("toasts.anchor", 3));
|
|
duration = cfg.ReadSetting("toasts.duration", 3000);
|
|
|
|
// Make sure the duration is set to something sensible
|
|
if (duration <= 0)
|
|
duration = 3000;
|
|
|
|
IDictionary toastactionenableddict = cfg.ReadSetting("toasts.registry", new Hashtable());
|
|
foreach (string key in toastactionenableddict.Keys)
|
|
{
|
|
//string key = de.Key.ToString();
|
|
|
|
if (registry.ContainsKey(key))
|
|
registry[key].Enabled = cfg.ReadSetting($"toasts.registry.{key}", true);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Writes the settings to a configuration with a prefix.
|
|
/// </summary>
|
|
/// <param name="cfg">The Configuration</param>
|
|
/// <param name="prefix">The prefix</param>
|
|
public void WriteSettings(Configuration cfg)
|
|
{
|
|
cfg.WriteSetting("toasts.enabled", Enabled);
|
|
cfg.WriteSetting("toasts.anchor", (int)Anchor);
|
|
cfg.WriteSetting("toasts.duration", Duration);
|
|
|
|
foreach (string key in Registry.Keys)
|
|
{
|
|
// true is the default value, so we only need to save it if it's false
|
|
if (Registry[key].Enabled == false)
|
|
cfg.WriteSetting($"toasts.registry.{key}", false);
|
|
else
|
|
cfg.DeleteSetting($"toasts.registry.{key}");
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Registers toast from all defined actions.
|
|
/// </summary>
|
|
public void RegisterActions()
|
|
{
|
|
foreach (Actions.Action action in General.Actions.GetAllActions().Where(a => a.RegisterToast))
|
|
{
|
|
if (!registry.ContainsKey(action.Name))
|
|
registry[action.Name] = new ToastRegistryEntry(action.Name, action.Title, action.Description, true);
|
|
}
|
|
}
|
|
|
|
/// <summary>
|
|
/// Registers a toast by name. Automatically prepends the assembly name.
|
|
/// </summary>
|
|
/// <param name="name">Name of the toast (without assembly)</param>
|
|
/// <param name="title">Title to show in the toast preferences dialog</param>
|
|
/// <param name="description">Description to show in the toast preferences dialog</param>
|
|
[MethodImpl(MethodImplOptions.NoInlining)]
|
|
public void RegisterToast(string name, string title, string description)
|
|
{
|
|
string fullname = Assembly.GetCallingAssembly().GetName().Name.ToLowerInvariant() + $"_{name}";
|
|
|
|
if (registry.ContainsKey(fullname))
|
|
{
|
|
General.WriteLogLine($"Tried to register toast \"{fullname}\", but it is already registered");
|
|
return;
|
|
}
|
|
|
|
registry[fullname] = new ToastRegistryEntry(fullname, title, description, true);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Gets the ToastAnchor from a number. Makes sure the input is valid, otherwise returns a default.
|
|
/// </summary>
|
|
/// <param name="number">The number</param>
|
|
/// <returns>The appropriate ToastAnchor, or BOTTOMRIGHT if input is not valid</returns>
|
|
public static ToastAnchor GetAnchorFromNumber(int number)
|
|
{
|
|
return Enum.IsDefined(typeof(ToastAnchor), number) ? (ToastAnchor)number : ToastAnchor.BOTTOMRIGHT;
|
|
}
|
|
|
|
/// <summary>
|
|
/// Shows a new toast.
|
|
/// </summary>
|
|
/// <param name="type">Toast type</param>
|
|
/// <param name="message">The message body of the toast</param>
|
|
public void ShowToast(ToastType type, string title, string message, string shortmessage = "")
|
|
{
|
|
StatusType st = type == ToastType.INFO ? StatusType.Info : StatusType.Warning;
|
|
|
|
if (!enabled)
|
|
{
|
|
General.Interface.DisplayStatus(new StatusInfo(st, shortmessage));
|
|
return;
|
|
}
|
|
|
|
if (type == ToastType.WARNING)
|
|
title = "Warning";
|
|
else if (type == ToastType.ERROR)
|
|
title = "Error";
|
|
|
|
CreateToast(type, title, message);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Shows a new toast. Deducts the title from the type.
|
|
/// </summary>
|
|
/// <param name="name">Name of the toast</param>
|
|
/// <param name="type">Type of the toast</param>
|
|
/// <param name="message">Message to show</param>
|
|
/// <param name="statusinfo">StatusInfo to use when toasts are disabled</param>
|
|
[MethodImpl(MethodImplOptions.NoInlining)]
|
|
public void ShowToast(string name, ToastType type, string message, StatusInfo statusinfo)
|
|
{
|
|
string fullname = Assembly.GetCallingAssembly().GetName().Name.ToLowerInvariant() + $"_{name}";
|
|
string title = "Information";
|
|
|
|
if (type == ToastType.WARNING)
|
|
title = "Warning";
|
|
else if (type == ToastType.ERROR)
|
|
title = "Error";
|
|
|
|
CreateToast(fullname, type, title, message, statusinfo);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Shows a new toast.
|
|
/// </summary>
|
|
/// <param name="name">Name of the toast</param>
|
|
/// <param name="type">Type of the toast</param>
|
|
/// <param name="title">Title to show</param>
|
|
/// <param name="message">Message to show</param>
|
|
/// <param name="statusinfo">StatusInfo to use when toasts are disabled</param>
|
|
[MethodImpl(MethodImplOptions.NoInlining)]
|
|
public void ShowToast(string name, ToastType type, string title, string message, StatusInfo statusinfo)
|
|
{
|
|
string fullname = Assembly.GetCallingAssembly().GetName().Name.ToLowerInvariant() + $"_{name}";
|
|
|
|
CreateToast(fullname, type, title, message, statusinfo);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Shows a new toast.
|
|
/// </summary>
|
|
/// <param name="name">Name of the toast</param>
|
|
/// <param name="type">Type of the toast</param>
|
|
/// <param name="title">Title to show</param>
|
|
/// <param name="message">Message to show</param>
|
|
/// <param name="shortmessage">Message to show in the status bar if toasts are disabled. Should not include line breaks</param>
|
|
[MethodImpl(MethodImplOptions.NoInlining)]
|
|
public void ShowToast(string name, ToastType type, string title, string message, string shortmessage = null)
|
|
{
|
|
string fullname = Assembly.GetCallingAssembly().GetName().Name.ToLowerInvariant() + $"_{name}";
|
|
StatusType st = type == ToastType.INFO ? StatusType.Info : StatusType.Warning;
|
|
|
|
if (string.IsNullOrWhiteSpace(shortmessage))
|
|
shortmessage = message;
|
|
|
|
CreateToast(fullname, type, title, message, new StatusInfo(st, shortmessage));
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a toast.
|
|
/// </summary>
|
|
/// <param name="fullname">Full name (i.e. assembly and toast name) of the toast</param>
|
|
/// <param name="type">Type of the toast</param>
|
|
/// <param name="title">Title to show</param>
|
|
/// <param name="message">Message to show</param>
|
|
/// <param name="statusinfo">StatusInfo to use when toasts are disabled</param>
|
|
private void CreateToast(string fullname, ToastType type, string title, string message, StatusInfo statusinfo)
|
|
{
|
|
if (!enabled || registry[fullname]?.Enabled == false)
|
|
{
|
|
General.Interface.DisplayStatus(statusinfo);
|
|
return;
|
|
}
|
|
|
|
if (!registry.ContainsKey(fullname))
|
|
{
|
|
General.ErrorLogger.Add(ErrorType.Warning, $"Toast setting for \"{fullname}\" is not in the registry. Defaulting to show the toast.");
|
|
}
|
|
else if (registry[fullname].Enabled == false)
|
|
{
|
|
General.Interface.DisplayStatus(statusinfo);
|
|
return;
|
|
}
|
|
|
|
CreateToast(type, title, message);
|
|
}
|
|
|
|
/// <summary>
|
|
/// Creates a toast.
|
|
/// </summary>
|
|
/// <param name="type">Type of the toast</param>
|
|
/// <param name="title">Title to show</param>
|
|
/// <param name="message">Message to show</param>
|
|
private void CreateToast(ToastType type, string title, string message)
|
|
{
|
|
ToastControl tc = new ToastControl(type, title, message, duration);
|
|
|
|
// Set the initial y position of the control so that it's outside of the control the toast manager is bound to.
|
|
// No need to care about the x position, since that will be set in the update event anyway
|
|
if (anchor == ToastAnchor.TOPLEFT || anchor == ToastAnchor.TOPRIGHT)
|
|
tc.Location = new Point(0, -tc.Height);
|
|
else
|
|
tc.Location = new Point(0, bindcontrol.Height);
|
|
|
|
toasts.Insert(0, tc);
|
|
bindcontrol.Controls.Add(tc);
|
|
|
|
// Need to set the toast to be at the front, otherwise the new control would be behind the control the toast manager
|
|
// is bound to
|
|
bindcontrol.Controls.SetChildIndex(tc, 0);
|
|
|
|
// Start the timer so that the toast is moved into view
|
|
if (!timer.Enabled)
|
|
timer.Start();
|
|
|
|
// Play a sound for warnings and errors
|
|
if (type == ToastType.WARNING)
|
|
General.MessageBeep(MessageBeepType.Warning);
|
|
else if (type == ToastType.ERROR)
|
|
General.MessageBeep(MessageBeepType.Error);
|
|
}
|
|
|
|
#endregion
|
|
}
|
|
}
|