using Observatory.Framework; using Observatory.Framework.Files.Journal; using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System.Linq; using System.Runtime.InteropServices; using System.Text; using System.Threading.Tasks; using System.Windows.Forms; namespace Observatory.UI { public partial class NotificationForm : Form { private Color _color; private readonly Guid _guid; private readonly System.Timers.Timer _timer; private bool _defaultPosition = true; private Point _originalLocation; protected override bool ShowWithoutActivation => true; protected override CreateParams CreateParams { get { CreateParams cp = base.CreateParams; cp.ExStyle |= 0x00000008; // WS_EX_TOPMOST return cp; } } public NotificationForm(Guid guid, NotificationArgs args) { _guid = guid; _color = Color.FromArgb((int)Properties.Core.Default.NativeNotifyColour); CreationTime = DateTime.Now; InitializeComponent(); Title.Paint += DrawText; Body.Paint += DrawText; if (System.Environment.OSVersion.Version.Major >= 6 && DwmHelper.IsCompositionEnabled()) { if (Environment.OSVersion.Version.Major > 6) { DwmHelper.Windows10EnableBlurBehind(Handle); } else { DwmHelper.WindowEnableBlurBehind(Handle); } // For some reason this causes the window to become all white on my own // PC. Looks very similar to strange system-specific all-white behaviour // of Avalonia. // DwmHelper.WindowBorderlessDropShadow(Handle, 2); } Title.ForeColor = _color; Title.Text = args.Title; Title.Font = new Font(Properties.Core.Default.NativeNotifyFont, 18); Body.ForeColor = _color; Body.Text = args.Detail; Body.Font = new Font(Properties.Core.Default.NativeNotifyFont, 14); Paint += DrawBorder; AdjustPosition(args.XPos / 100, args.YPos / 100); _timer = new(); _timer.Elapsed += CloseNotification; if (args.Timeout != 0) { _timer.Interval = args.Timeout == -1 ? Properties.Core.Default.NativeNotifyTimeout : args.Timeout; _timer.Start(); } } private void NotificationForm_FormClosed(object? sender, FormClosedEventArgs e) { throw new NotImplementedException(); } public DateTime CreationTime { get; private init; } public void Update(NotificationArgs notificationArgs) { // Catch Cross-thread access and invoke try { Title.Text = notificationArgs.Title; Body.Text = notificationArgs.Detail; } catch { try { Invoke(() => { Title.Text = notificationArgs.Title; Body.Text = notificationArgs.Detail; }); } catch (Exception ex) { throw new Exception("Notification Update Failure, please inform Vithigar. Details: " + ex.Message); } } } private void AdjustPosition(double x = -1.0, double y = -1.0) { int screen = Properties.Core.Default.NativeNotifyScreen; int corner = Properties.Core.Default.NativeNotifyCorner; Rectangle screenBounds; if (screen == -1 || screen > Screen.AllScreens.Length) if (Screen.AllScreens.Length == 1) screenBounds = Screen.GetBounds(this); else screenBounds = Screen.PrimaryScreen.Bounds; else screenBounds = Screen.AllScreens[screen - 1].Bounds; if (x >= 0 && y >= 0) { _defaultPosition = false; int xLocation = Convert.ToInt32(screenBounds.Width * x); int yLocation = Convert.ToInt32(screenBounds.Height * x); Location = Point.Add(screenBounds.Location, new Size(xLocation, yLocation)); } else { _defaultPosition = true; switch (corner) { default: case 0: Location = Point.Add( new Point(screenBounds.Right, screenBounds.Bottom), new Size(-(Width + 50), -(Height + 50))); break; case 1: Location = Point.Add( new Point(screenBounds.Left, screenBounds.Bottom), new Size(50, -(Height + 50))); break; case 2: Location = Point.Add( new Point(screenBounds.Right, screenBounds.Top), new Size(-(Width + 50), 50)); break; case 3: Location = Point.Add( new Point(screenBounds.Left, screenBounds.Top), new Size(50, 00)); break; } _originalLocation = new Point(Location.X, Location.Y); } } private void DrawBorder(object? sender, PaintEventArgs e) { using (Pen pen = new Pen(_color)) { pen.Width = 6; e.Graphics.DrawLine(pen, 0, 0, Width, 0); e.Graphics.DrawLine(pen, 0, 0, 0, Height); e.Graphics.DrawLine(pen, 0, Height, Width, Height); e.Graphics.DrawLine(pen, Width, 0, Width, Height); } } protected override void WndProc(ref Message m) { switch (m.Msg) { case DwmHelper.WM_DWMCOMPOSITIONCHANGED: if (System.Environment.OSVersion.Version.Major >= 6 && DwmHelper.IsCompositionEnabled()) { var policy = DwmHelper.DWMNCRENDERINGPOLICY.Enabled; DwmHelper.WindowSetAttribute(Handle, DwmHelper.DWMWINDOWATTRIBUTE.NCRenderingPolicy, (int)policy); DwmHelper.WindowBorderlessDropShadow(Handle, 2); m.Result = IntPtr.Zero; } break; case 0x0084: m.Result = (IntPtr)(-1); return; default: break; } base.WndProc(ref m); } private void DrawText(object? sender, PaintEventArgs e) { if (sender != null) { var label = (Label)sender; e.Graphics.Clear(Color.FromArgb(64, 64, 64)); using (var sf = new StringFormat()) using (var brush = new SolidBrush(label.ForeColor)) { sf.Alignment = sf.LineAlignment = StringAlignment.Near; e.Graphics.TextRenderingHint = System.Drawing.Text.TextRenderingHint.AntiAliasGridFit; e.Graphics.DrawString(label.Text, label.Font, brush, label.ClientRectangle, sf); } } } public Guid Guid { get => _guid; } public void AdjustOffset(bool increase) { if (_defaultPosition) { if (increase || Location != _originalLocation) { var corner = Properties.Core.Default.NativeNotifyCorner; if ((corner >= 2 && increase) || (corner <= 1 && !increase)) { Location = new Point(Location.X, Location.Y + Height); } else { Location = new Point(Location.X, Location.Y - Height); } } } } private void CloseNotification(object? sender, System.Timers.ElapsedEventArgs e) { // Catch Cross-thread access and invoke try { Close(); } catch { try { Invoke(() => Close()); } catch (Exception ex) { throw new Exception("Notification Close Failure, please inform Vithigar. Details: " + ex.Message); } } _timer.Stop(); _timer.Dispose(); } } }