2026-05-10 09:42:29 -04:00
|
|
|
using System.Linq;
|
2026-05-07 11:09:56 -04:00
|
|
|
using System.Windows;
|
2026-05-10 14:39:04 -04:00
|
|
|
using System.Windows.Input;
|
2026-05-08 00:55:57 -04:00
|
|
|
using System.Windows.Shapes;
|
2026-05-08 01:05:26 -04:00
|
|
|
using TeamsISO.App.Services;
|
2026-05-07 11:40:49 -04:00
|
|
|
using TeamsISO.App.ViewModels;
|
2026-05-07 11:09:56 -04:00
|
|
|
|
|
|
|
|
namespace TeamsISO.App;
|
|
|
|
|
|
|
|
|
|
public partial class MainWindow : Window
|
|
|
|
|
{
|
|
|
|
|
public MainWindow()
|
|
|
|
|
{
|
|
|
|
|
InitializeComponent();
|
2026-05-08 00:55:57 -04:00
|
|
|
StateChanged += OnWindowStateChanged;
|
2026-05-08 13:59:14 -04:00
|
|
|
SourceInitialized += OnSourceInitialized;
|
|
|
|
|
Closing += OnClosing;
|
2026-05-07 11:09:56 -04:00
|
|
|
}
|
2026-05-07 11:40:49 -04:00
|
|
|
|
|
|
|
|
public MainWindow(MainViewModel viewModel) : this()
|
|
|
|
|
{
|
|
|
|
|
DataContext = viewModel;
|
|
|
|
|
}
|
2026-05-08 00:55:57 -04:00
|
|
|
|
2026-05-08 13:59:14 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Restore the window's previous placement after the HWND is created (so
|
|
|
|
|
/// SetWindowPos / WindowState transitions actually take effect). Falls
|
|
|
|
|
/// silently back to the XAML-default startup location if no snapshot exists.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void OnSourceInitialized(object? sender, EventArgs e)
|
|
|
|
|
{
|
|
|
|
|
WindowStateStore.TryApply(this);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>Persist the placement on close so next launch lands in the same spot.</summary>
|
|
|
|
|
private void OnClosing(object? sender, System.ComponentModel.CancelEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
WindowStateStore.Save(this);
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-08 00:55:57 -04:00
|
|
|
/// <summary>Custom min button — chrome'd window has no system caption buttons.</summary>
|
|
|
|
|
private void OnMinimize(object sender, RoutedEventArgs e) =>
|
|
|
|
|
WindowState = WindowState.Minimized;
|
|
|
|
|
|
|
|
|
|
/// <summary>Toggles maximize/restore. Bound to the maximize button + double-click on the drag region.</summary>
|
|
|
|
|
private void OnMaximizeRestore(object sender, RoutedEventArgs e) =>
|
|
|
|
|
WindowState = WindowState == WindowState.Maximized ? WindowState.Normal : WindowState.Maximized;
|
|
|
|
|
|
|
|
|
|
/// <summary>Custom close button.</summary>
|
|
|
|
|
private void OnClose(object sender, RoutedEventArgs e) => Close();
|
|
|
|
|
|
feat: app icon, FPS, drops counter, --version, About dialog, Stop Teams toggle
Six related polish items, all building on tonight's groundwork.
1. App icon: teamsiso.ico generated from dragon-mark.png at 7 sizes (16-256), wired as ApplicationIcon in the WPF csproj, MainWindow.Icon, AboutWindow.Icon, and ARPPRODUCTICON in the WiX MSI. Taskbar / window / Add-Remove-Programs all show the dragon mark now.
2. Running incoming FPS: ring buffer of last 30 frame timestamps in IsoPipeline; ComputeFps() returns moving-average rate. Surfaced on IsoHealthStats.IncomingFps and shown in the Source column of the participants DataGrid as 'WxH · 59.94 fps'. Resets cleanly on every supervisor restart.
3. Drops counter: FrameProcessor.Stats already aggregated FramesDropped (closest-frame strategy when the receiver outpaces the processor) and FramesDuplicated; just plumbed _liveProcessor through IsoPipeline so GetStats() can read them. Exposed in the Live column under the in/out counters as a coral-tinted 'drop N'.
4. Console --version flag: prints engine version (with embedded git SHA), .NET version, OS, NDI runtime banner, expected prefix, exit-code legend, plus a wilddragon.net link. Useful for support tickets.
5. About dialog: chromeless modal with the dragon mark + version / .NET / OS / NDI runtime fields and a link to wilddragon.net. Triggered by clicking the rail logo.
6. Teams launcher Stop toggle: TeamsLauncher gains IsRunning() and StopAll(). The rail's Teams button now toggles — if Teams is up, ask to close all Teams windows via WM_CLOSE; otherwise launch as before. Confirms before stopping so we don't kill the user's call mid-transition.
Tests: 74/74 unit + 9/9 NDI integration green throughout. MSI builds clean and now embeds the dragon icon for ARP.
2026-05-08 13:50:19 -04:00
|
|
|
/// <summary>Opens the About dialog — version, NDI runtime, build SHA.</summary>
|
|
|
|
|
private void OnAboutClick(object sender, RoutedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
var about = new AboutWindow { Owner = this };
|
|
|
|
|
about.ShowDialog();
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-10 09:42:29 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Opens the operator-presets dialog. Hands it the current participants
|
|
|
|
|
/// snapshot (so Save captures live state) and the engine controller (so
|
|
|
|
|
/// Apply can reconcile enable/disable). Owner is set so the chromeless
|
|
|
|
|
/// dialog centers over the main window and inherits z-order.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void OnPresetsClick(object sender, RoutedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
if (DataContext is not MainViewModel vm) return;
|
|
|
|
|
var dialog = new PresetsDialog(vm.Controller, vm.Participants.ToList(), vm.Toast)
|
|
|
|
|
{
|
|
|
|
|
Owner = this,
|
|
|
|
|
};
|
|
|
|
|
dialog.ShowDialog();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Tracks whether we have hidden Teams' windows so the next click reverses
|
|
|
|
|
/// the action. We treat this as "intent" rather than a query of OS state
|
|
|
|
|
/// because hidden windows still report as hidden if the operator manually
|
|
|
|
|
/// re-opens them and we only care about TeamsISO's own toggle history.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private bool _teamsWindowsHidden;
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Phase E.2 toggle. Hides every visible top-level Teams window on first
|
|
|
|
|
/// click; shows them again on the next. Surfaces the result via the toast
|
|
|
|
|
/// so the operator gets feedback even though the affected windows aren't
|
|
|
|
|
/// visible anymore.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void OnToggleTeamsWindowClick(object sender, RoutedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
if (!TeamsLauncher.IsRunning())
|
|
|
|
|
{
|
|
|
|
|
MessageBox.Show(
|
|
|
|
|
"Microsoft Teams isn't running. Click the camera icon above to launch it first.",
|
|
|
|
|
"TeamsISO — Hide / show Teams",
|
|
|
|
|
MessageBoxButton.OK,
|
|
|
|
|
MessageBoxImage.Information);
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var toast = (DataContext as MainViewModel)?.Toast;
|
|
|
|
|
if (_teamsWindowsHidden)
|
|
|
|
|
{
|
|
|
|
|
var shown = TeamsLauncher.ShowWindows();
|
|
|
|
|
_teamsWindowsHidden = false;
|
|
|
|
|
toast?.Show(shown > 0 ? $"Restored {shown} Teams window(s)" : "No Teams windows to restore");
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
var hidden = TeamsLauncher.HideWindows();
|
|
|
|
|
_teamsWindowsHidden = hidden > 0;
|
|
|
|
|
toast?.Show(hidden > 0 ? $"Hid {hidden} Teams window(s)" : "Teams has no visible windows yet");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-08 01:05:26 -04:00
|
|
|
/// <summary>
|
2026-05-10 14:39:04 -04:00
|
|
|
/// Three-state click behavior matching operator intuition:
|
|
|
|
|
/// 1. Teams not running → launch it via TeamsLauncher's fallback chain.
|
|
|
|
|
/// 2. Teams running but its windows are hidden (we toggled them off, OR
|
|
|
|
|
/// it launched into the tray) → restore the windows + foreground them.
|
|
|
|
|
/// This is the case the previous "ask to stop" dialog was ambushing —
|
|
|
|
|
/// operators don't think of a hidden Teams as "running and ready to
|
|
|
|
|
/// stop", they think of it as "I clicked Launch and nothing happened".
|
|
|
|
|
/// 3. Teams running with visible windows → bring the most recent one to
|
|
|
|
|
/// the foreground. (Stopping Teams is now a right-click action;
|
|
|
|
|
/// see OnLaunchTeamsRightClick.)
|
2026-05-08 01:05:26 -04:00
|
|
|
/// </summary>
|
|
|
|
|
private void OnLaunchTeamsClick(object sender, RoutedEventArgs e)
|
|
|
|
|
{
|
2026-05-10 14:39:04 -04:00
|
|
|
var toast = (DataContext as MainViewModel)?.Toast;
|
feat: app icon, FPS, drops counter, --version, About dialog, Stop Teams toggle
Six related polish items, all building on tonight's groundwork.
1. App icon: teamsiso.ico generated from dragon-mark.png at 7 sizes (16-256), wired as ApplicationIcon in the WPF csproj, MainWindow.Icon, AboutWindow.Icon, and ARPPRODUCTICON in the WiX MSI. Taskbar / window / Add-Remove-Programs all show the dragon mark now.
2. Running incoming FPS: ring buffer of last 30 frame timestamps in IsoPipeline; ComputeFps() returns moving-average rate. Surfaced on IsoHealthStats.IncomingFps and shown in the Source column of the participants DataGrid as 'WxH · 59.94 fps'. Resets cleanly on every supervisor restart.
3. Drops counter: FrameProcessor.Stats already aggregated FramesDropped (closest-frame strategy when the receiver outpaces the processor) and FramesDuplicated; just plumbed _liveProcessor through IsoPipeline so GetStats() can read them. Exposed in the Live column under the in/out counters as a coral-tinted 'drop N'.
4. Console --version flag: prints engine version (with embedded git SHA), .NET version, OS, NDI runtime banner, expected prefix, exit-code legend, plus a wilddragon.net link. Useful for support tickets.
5. About dialog: chromeless modal with the dragon mark + version / .NET / OS / NDI runtime fields and a link to wilddragon.net. Triggered by clicking the rail logo.
6. Teams launcher Stop toggle: TeamsLauncher gains IsRunning() and StopAll(). The rail's Teams button now toggles — if Teams is up, ask to close all Teams windows via WM_CLOSE; otherwise launch as before. Confirms before stopping so we don't kill the user's call mid-transition.
Tests: 74/74 unit + 9/9 NDI integration green throughout. MSI builds clean and now embeds the dragon icon for ARP.
2026-05-08 13:50:19 -04:00
|
|
|
|
2026-05-10 14:39:04 -04:00
|
|
|
if (!TeamsLauncher.IsRunning())
|
|
|
|
|
{
|
|
|
|
|
if (!TeamsLauncher.TryLaunch(out var error))
|
feat: app icon, FPS, drops counter, --version, About dialog, Stop Teams toggle
Six related polish items, all building on tonight's groundwork.
1. App icon: teamsiso.ico generated from dragon-mark.png at 7 sizes (16-256), wired as ApplicationIcon in the WPF csproj, MainWindow.Icon, AboutWindow.Icon, and ARPPRODUCTICON in the WiX MSI. Taskbar / window / Add-Remove-Programs all show the dragon mark now.
2. Running incoming FPS: ring buffer of last 30 frame timestamps in IsoPipeline; ComputeFps() returns moving-average rate. Surfaced on IsoHealthStats.IncomingFps and shown in the Source column of the participants DataGrid as 'WxH · 59.94 fps'. Resets cleanly on every supervisor restart.
3. Drops counter: FrameProcessor.Stats already aggregated FramesDropped (closest-frame strategy when the receiver outpaces the processor) and FramesDuplicated; just plumbed _liveProcessor through IsoPipeline so GetStats() can read them. Exposed in the Live column under the in/out counters as a coral-tinted 'drop N'.
4. Console --version flag: prints engine version (with embedded git SHA), .NET version, OS, NDI runtime banner, expected prefix, exit-code legend, plus a wilddragon.net link. Useful for support tickets.
5. About dialog: chromeless modal with the dragon mark + version / .NET / OS / NDI runtime fields and a link to wilddragon.net. Triggered by clicking the rail logo.
6. Teams launcher Stop toggle: TeamsLauncher gains IsRunning() and StopAll(). The rail's Teams button now toggles — if Teams is up, ask to close all Teams windows via WM_CLOSE; otherwise launch as before. Confirms before stopping so we don't kill the user's call mid-transition.
Tests: 74/74 unit + 9/9 NDI integration green throughout. MSI builds clean and now embeds the dragon icon for ARP.
2026-05-08 13:50:19 -04:00
|
|
|
{
|
|
|
|
|
MessageBox.Show(
|
2026-05-10 14:39:04 -04:00
|
|
|
$"Could not launch Microsoft Teams.\n\n{error}",
|
|
|
|
|
"TeamsISO — Launch Teams",
|
feat: app icon, FPS, drops counter, --version, About dialog, Stop Teams toggle
Six related polish items, all building on tonight's groundwork.
1. App icon: teamsiso.ico generated from dragon-mark.png at 7 sizes (16-256), wired as ApplicationIcon in the WPF csproj, MainWindow.Icon, AboutWindow.Icon, and ARPPRODUCTICON in the WiX MSI. Taskbar / window / Add-Remove-Programs all show the dragon mark now.
2. Running incoming FPS: ring buffer of last 30 frame timestamps in IsoPipeline; ComputeFps() returns moving-average rate. Surfaced on IsoHealthStats.IncomingFps and shown in the Source column of the participants DataGrid as 'WxH · 59.94 fps'. Resets cleanly on every supervisor restart.
3. Drops counter: FrameProcessor.Stats already aggregated FramesDropped (closest-frame strategy when the receiver outpaces the processor) and FramesDuplicated; just plumbed _liveProcessor through IsoPipeline so GetStats() can read them. Exposed in the Live column under the in/out counters as a coral-tinted 'drop N'.
4. Console --version flag: prints engine version (with embedded git SHA), .NET version, OS, NDI runtime banner, expected prefix, exit-code legend, plus a wilddragon.net link. Useful for support tickets.
5. About dialog: chromeless modal with the dragon mark + version / .NET / OS / NDI runtime fields and a link to wilddragon.net. Triggered by clicking the rail logo.
6. Teams launcher Stop toggle: TeamsLauncher gains IsRunning() and StopAll(). The rail's Teams button now toggles — if Teams is up, ask to close all Teams windows via WM_CLOSE; otherwise launch as before. Confirms before stopping so we don't kill the user's call mid-transition.
Tests: 74/74 unit + 9/9 NDI integration green throughout. MSI builds clean and now embeds the dragon icon for ARP.
2026-05-08 13:50:19 -04:00
|
|
|
MessageBoxButton.OK,
|
2026-05-10 14:39:04 -04:00
|
|
|
MessageBoxImage.Warning);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2026-05-10 20:35:00 -04:00
|
|
|
var autoHide = (DataContext as MainViewModel)?.Settings.AutoHideTeamsWindows ?? false;
|
|
|
|
|
toast?.Show(autoHide
|
|
|
|
|
? "Launching Microsoft Teams (will hide windows automatically)…"
|
|
|
|
|
: "Launching Microsoft Teams…");
|
|
|
|
|
if (autoHide)
|
|
|
|
|
{
|
|
|
|
|
_ = TeamsLauncher.AutoHideAfterLaunchAsync();
|
|
|
|
|
_teamsWindowsHidden = true;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
_teamsWindowsHidden = false;
|
|
|
|
|
}
|
feat: app icon, FPS, drops counter, --version, About dialog, Stop Teams toggle
Six related polish items, all building on tonight's groundwork.
1. App icon: teamsiso.ico generated from dragon-mark.png at 7 sizes (16-256), wired as ApplicationIcon in the WPF csproj, MainWindow.Icon, AboutWindow.Icon, and ARPPRODUCTICON in the WiX MSI. Taskbar / window / Add-Remove-Programs all show the dragon mark now.
2. Running incoming FPS: ring buffer of last 30 frame timestamps in IsoPipeline; ComputeFps() returns moving-average rate. Surfaced on IsoHealthStats.IncomingFps and shown in the Source column of the participants DataGrid as 'WxH · 59.94 fps'. Resets cleanly on every supervisor restart.
3. Drops counter: FrameProcessor.Stats already aggregated FramesDropped (closest-frame strategy when the receiver outpaces the processor) and FramesDuplicated; just plumbed _liveProcessor through IsoPipeline so GetStats() can read them. Exposed in the Live column under the in/out counters as a coral-tinted 'drop N'.
4. Console --version flag: prints engine version (with embedded git SHA), .NET version, OS, NDI runtime banner, expected prefix, exit-code legend, plus a wilddragon.net link. Useful for support tickets.
5. About dialog: chromeless modal with the dragon mark + version / .NET / OS / NDI runtime fields and a link to wilddragon.net. Triggered by clicking the rail logo.
6. Teams launcher Stop toggle: TeamsLauncher gains IsRunning() and StopAll(). The rail's Teams button now toggles — if Teams is up, ask to close all Teams windows via WM_CLOSE; otherwise launch as before. Confirms before stopping so we don't kill the user's call mid-transition.
Tests: 74/74 unit + 9/9 NDI integration green throughout. MSI builds clean and now embeds the dragon icon for ARP.
2026-05-08 13:50:19 -04:00
|
|
|
}
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-10 14:39:04 -04:00
|
|
|
// Teams is running. Always try to restore + foreground its window —
|
|
|
|
|
// if windows are already visible, ShowWindows is a SetForegroundWindow
|
|
|
|
|
// no-op besides bringing them forward; if they were hidden by our
|
|
|
|
|
// own toggle, this is the operator's intuitive "show me Teams" path.
|
|
|
|
|
var shown = TeamsLauncher.ShowWindows();
|
|
|
|
|
_teamsWindowsHidden = false;
|
|
|
|
|
toast?.Show(shown > 0
|
|
|
|
|
? $"Teams is already running — surfaced {shown} window(s)"
|
|
|
|
|
: "Teams is running but has no visible windows yet");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// <summary>
|
|
|
|
|
/// Right-click on the rail Launch button asks to stop Teams. Split out
|
|
|
|
|
/// from the left-click so a normal click is "open / surface" rather than
|
|
|
|
|
/// the previous "open OR ambush you with a stop dialog".
|
|
|
|
|
/// </summary>
|
2026-05-10 21:14:42 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Open the experimental Teams embed window. Operator enables the
|
|
|
|
|
/// preference first; this button materializes the host. See
|
|
|
|
|
/// <see cref="TeamsEmbedWindow"/> for the SetParent lifecycle.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void OnOpenEmbedWindowClick(object sender, RoutedEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
// Non-modal so the operator can keep using TeamsISO's controls.
|
|
|
|
|
// Owner = this so it minimizes / closes with TeamsISO.
|
|
|
|
|
var w = new TeamsEmbedWindow { Owner = this };
|
|
|
|
|
w.Show();
|
|
|
|
|
}
|
|
|
|
|
|
2026-05-10 14:39:04 -04:00
|
|
|
private void OnLaunchTeamsRightClick(object sender, MouseButtonEventArgs e)
|
|
|
|
|
{
|
|
|
|
|
if (!TeamsLauncher.IsRunning()) return;
|
|
|
|
|
|
|
|
|
|
var confirm = MessageBox.Show(
|
|
|
|
|
"Microsoft Teams is currently running.\n\nClose all Teams windows now?",
|
|
|
|
|
"TeamsISO — Stop Teams",
|
|
|
|
|
MessageBoxButton.YesNo,
|
|
|
|
|
MessageBoxImage.Question);
|
|
|
|
|
if (confirm != MessageBoxResult.Yes) return;
|
|
|
|
|
|
|
|
|
|
var asked = TeamsLauncher.StopAll();
|
|
|
|
|
if (TeamsLauncher.IsRunning())
|
2026-05-08 01:05:26 -04:00
|
|
|
{
|
|
|
|
|
MessageBox.Show(
|
2026-05-10 14:39:04 -04:00
|
|
|
asked == 0
|
|
|
|
|
? "No Teams windows responded to close."
|
|
|
|
|
: $"Sent close to {asked} Teams window(s); some may still be exiting.",
|
|
|
|
|
"TeamsISO — Stop Teams",
|
2026-05-08 01:05:26 -04:00
|
|
|
MessageBoxButton.OK,
|
2026-05-10 14:39:04 -04:00
|
|
|
MessageBoxImage.Information);
|
2026-05-08 01:05:26 -04:00
|
|
|
}
|
2026-05-10 14:39:04 -04:00
|
|
|
e.Handled = true;
|
2026-05-08 01:05:26 -04:00
|
|
|
}
|
|
|
|
|
|
2026-05-08 00:55:57 -04:00
|
|
|
/// <summary>
|
|
|
|
|
/// Swap the maximize-button glyph between the "single rectangle" (when normal) and the
|
|
|
|
|
/// "two-overlapping-rectangles" (when maximized) variants, matching the Windows 11
|
|
|
|
|
/// caption-button conventions.
|
|
|
|
|
/// </summary>
|
|
|
|
|
private void OnWindowStateChanged(object? sender, EventArgs e)
|
|
|
|
|
{
|
|
|
|
|
if (FindName("MaximizeIcon") is not Path icon) return;
|
|
|
|
|
icon.Data = WindowState == WindowState.Maximized
|
|
|
|
|
// Two-rectangle "restore" glyph
|
|
|
|
|
? System.Windows.Media.Geometry.Parse("M 2,0 L 10,0 L 10,8 M 0,2 L 8,2 L 8,10 L 0,10 Z")
|
|
|
|
|
// Single-rectangle "maximize" glyph
|
|
|
|
|
: System.Windows.Media.Geometry.Parse("M 0,0 L 10,0 L 10,10 L 0,10 Z");
|
|
|
|
|
}
|
2026-05-07 11:09:56 -04:00
|
|
|
}
|