From 532a5af4669ce0c26f3091ec5b34eea0a78d5402 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Tue, 7 Jan 2025 10:40:23 -0800 Subject: [PATCH 01/36] rename file to match class name --- .../{WindowsConsole.cs => Win32Console.cs} | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename src/Consolonia.PlatformSupport/{WindowsConsole.cs => Win32Console.cs} (100%) diff --git a/src/Consolonia.PlatformSupport/WindowsConsole.cs b/src/Consolonia.PlatformSupport/Win32Console.cs similarity index 100% rename from src/Consolonia.PlatformSupport/WindowsConsole.cs rename to src/Consolonia.PlatformSupport/Win32Console.cs From 037a737650c5bf5c6709d231ca1c1f7c2149fbde Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Tue, 7 Jan 2025 10:56:16 -0800 Subject: [PATCH 02/36] Fix perf issue with windows and paste --- src/Consolonia.GuiCS/WindowsDriver.cs | 21 +++--- .../Win32Console.cs | 66 +++++++++++++------ 2 files changed, 57 insertions(+), 30 deletions(-) diff --git a/src/Consolonia.GuiCS/WindowsDriver.cs b/src/Consolonia.GuiCS/WindowsDriver.cs index 2ac79248..8679fcef 100644 --- a/src/Consolonia.GuiCS/WindowsDriver.cs +++ b/src/Consolonia.GuiCS/WindowsDriver.cs @@ -42,18 +42,21 @@ public CONSOLE_INPUT_MODE ConsoleMode } } + const int bufferSize = 65535; + private static INPUT_RECORD[] s_inputBuffer = new INPUT_RECORD[bufferSize]; + public INPUT_RECORD[] ReadConsoleInput() { - const int bufferSize = 1; - var records = new INPUT_RECORD[bufferSize]; - - if (!Kernel32.ReadConsoleInput(InputHandle, records, bufferSize, - out var numberEventsRead)) - throw GetLastError().GetException(); + lock (s_inputBuffer) + { - return numberEventsRead == 0 - ? Array.Empty() - : records; + if (!Kernel32.ReadConsoleInput(InputHandle, s_inputBuffer, bufferSize, + out var numberEventsRead)) + throw GetLastError().GetException(); + INPUT_RECORD[] recordsToReturn = new INPUT_RECORD[numberEventsRead]; + Array.Copy(s_inputBuffer, recordsToReturn, numberEventsRead); + return recordsToReturn; + } } } } \ No newline at end of file diff --git a/src/Consolonia.PlatformSupport/Win32Console.cs b/src/Consolonia.PlatformSupport/Win32Console.cs index cc99514c..1fb8c0b1 100644 --- a/src/Consolonia.PlatformSupport/Win32Console.cs +++ b/src/Consolonia.PlatformSupport/Win32Console.cs @@ -66,6 +66,14 @@ private static readonly FlagTranslator ]); + private static readonly KEY_EVENT_RECORD[] CtrlVKeyEvents = + [ + new KEY_EVENT_RECORD {bKeyDown = true,wVirtualKeyCode = 17, wVirtualScanCode = 29, dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED}, + new KEY_EVENT_RECORD {bKeyDown = true,wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED}, + new KEY_EVENT_RECORD {bKeyDown = false,wVirtualKeyCode = 86,wVirtualScanCode = 47, uChar = '\u0016', dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED}, + new KEY_EVENT_RECORD {bKeyDown = false,wVirtualKeyCode = 17,wVirtualScanCode = 29, dwControlKeyState = 0 } + ]; + private readonly WindowsConsole _windowsConsole; private MOUSE_BUTTON_STATE _mouseButtonsState = MOUSE_BUTTON_STATE.NONE; @@ -116,30 +124,46 @@ private void StartEventLoop() while (!Disposed /*inject ThreadAbortException*/) { PauseTask?.Wait(); - var readConsoleInput = _windowsConsole.ReadConsoleInput(); - if (!readConsoleInput.Any()) + var inputRecords = _windowsConsole.ReadConsoleInput(); + if (!inputRecords.Any()) throw new NotImplementedException(); - foreach (INPUT_RECORD inputRecord in readConsoleInput) - // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault - switch (inputRecord.EventType) + + // We only get multiple key events when console is translating + // CTRL+V to sequence of key strokes, so we are turning it back into + // CTRL+V key sequence. + bool isPaste = inputRecords.Where(ir => ir.EventType == EVENT_TYPE.KEY_EVENT).Count() > 1; + if (isPaste) + { + // simulate CTRL_V + foreach (var keyEvent in CtrlVKeyEvents) { - case EVENT_TYPE.WINDOW_BUFFER_SIZE_EVENT: - WINDOW_BUFFER_SIZE_RECORD windowBufferSize = inputRecord.Event.WindowBufferSizeEvent; - Size = new PixelBufferSize((ushort)windowBufferSize.dwSize.X, - (ushort)windowBufferSize.dwSize.Y); - break; - case EVENT_TYPE.FOCUS_EVENT: - FOCUS_EVENT_RECORD focusEvent = inputRecord.Event.FocusEvent; - RaiseFocusEvent(focusEvent.bSetFocus != 0); - break; - case EVENT_TYPE.KEY_EVENT: - HandleKeyInput(inputRecord.Event.KeyEvent); - break; - case EVENT_TYPE.MOUSE_EVENT: - MOUSE_EVENT_RECORD mouseEvent = inputRecord.Event.MouseEvent; - HandleMouseInput(mouseEvent); - break; + HandleKeyInput(keyEvent); } + } + else + { + foreach (INPUT_RECORD inputRecord in inputRecords) + // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault + switch (inputRecord.EventType) + { + case EVENT_TYPE.WINDOW_BUFFER_SIZE_EVENT: + WINDOW_BUFFER_SIZE_RECORD windowBufferSize = inputRecord.Event.WindowBufferSizeEvent; + Size = new PixelBufferSize((ushort)windowBufferSize.dwSize.X, + (ushort)windowBufferSize.dwSize.Y); + break; + case EVENT_TYPE.FOCUS_EVENT: + FOCUS_EVENT_RECORD focusEvent = inputRecord.Event.FocusEvent; + RaiseFocusEvent(focusEvent.bSetFocus != 0); + break; + case EVENT_TYPE.KEY_EVENT: + HandleKeyInput(inputRecord.Event.KeyEvent); + break; + case EVENT_TYPE.MOUSE_EVENT: + MOUSE_EVENT_RECORD mouseEvent = inputRecord.Event.MouseEvent; + HandleMouseInput(mouseEvent); + break; + } + } } }); } From 2195755096296d3518e09e253ced16dff4a3bcb7 Mon Sep 17 00:00:00 2001 From: Evgeny Gorbovoy Date: Wed, 8 Jan 2025 13:32:15 +0100 Subject: [PATCH 03/36] 1 000 000 characters test --- .../Infrastructure/ConsoleWindow.cs | 21 +++++++++++++++++++ 1 file changed, 21 insertions(+) diff --git a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs index ddc4dc1a..c47b77ec 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs @@ -2,6 +2,7 @@ using System.Collections.Generic; using System.Diagnostics; using System.Diagnostics.CodeAnalysis; +using System.Linq; using System.Threading; using System.Threading.Tasks; using Avalonia; @@ -397,6 +398,26 @@ private void OnConsoleOnResized() private async void ConsoleOnKeyEvent(Key key, char keyChar, RawInputModifiers rawInputModifiers, bool down, ulong timeStamp) { + if (keyChar is 'r') + { + Dispatcher.UIThread.Post(() => + { + Input!(new RawTextInputEventArgs(_myKeyboardDevice, + timeStamp, + _inputRoot, + string.Concat(Enumerable.Repeat( + @"This is long text. This is long text.This is long text.T +This is long text.This is long text.This is long text.This is long text.This +This is long text.This is long text.This is long text.This is long text.This is long text. is long text.This is long text. +his is long text.This is long text.This is long text.This is long text.This is lo +ng text.This is long text.This is +This is long text.This is long text.This is long text.This is long text. + long text.This is long text.This is long text.This is long text.This is long text.This is long text.This is long text.",2000)))); + }, DispatcherPriority.Input); + + return; + } + if (!down) { Dispatcher.UIThread.Post(() => From 86160a4028952d60ebb38db4e24aba07b8073e8b Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 8 Jan 2025 12:39:11 +0000 Subject: [PATCH 04/36] Automated JetBrains cleanup Co-authored-by: <+@users.noreply.github.com> --- .../Infrastructure/ConsoleWindow.cs | 9 +++--- .../Win32Console.cs | 31 ++++++++++++------- 2 files changed, 24 insertions(+), 16 deletions(-) diff --git a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs index c47b77ec..2dc6c8a3 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs @@ -406,18 +406,19 @@ private async void ConsoleOnKeyEvent(Key key, char keyChar, RawInputModifiers ra timeStamp, _inputRoot, string.Concat(Enumerable.Repeat( - @"This is long text. This is long text.This is long text.T + @"This is long text. This is long text.This is long text.T This is long text.This is long text.This is long text.This is long text.This This is long text.This is long text.This is long text.This is long text.This is long text. is long text.This is long text. his is long text.This is long text.This is long text.This is long text.This is lo ng text.This is long text.This is This is long text.This is long text.This is long text.This is long text. - long text.This is long text.This is long text.This is long text.This is long text.This is long text.This is long text.",2000)))); + long text.This is long text.This is long text.This is long text.This is long text.This is long text.This is long text.", + 2000)))); }, DispatcherPriority.Input); - + return; } - + if (!down) { Dispatcher.UIThread.Post(() => diff --git a/src/Consolonia.PlatformSupport/Win32Console.cs b/src/Consolonia.PlatformSupport/Win32Console.cs index 1fb8c0b1..7e748ef6 100644 --- a/src/Consolonia.PlatformSupport/Win32Console.cs +++ b/src/Consolonia.PlatformSupport/Win32Console.cs @@ -68,10 +68,22 @@ private static readonly FlagTranslator private static readonly KEY_EVENT_RECORD[] CtrlVKeyEvents = [ - new KEY_EVENT_RECORD {bKeyDown = true,wVirtualKeyCode = 17, wVirtualScanCode = 29, dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED}, - new KEY_EVENT_RECORD {bKeyDown = true,wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED}, - new KEY_EVENT_RECORD {bKeyDown = false,wVirtualKeyCode = 86,wVirtualScanCode = 47, uChar = '\u0016', dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED}, - new KEY_EVENT_RECORD {bKeyDown = false,wVirtualKeyCode = 17,wVirtualScanCode = 29, dwControlKeyState = 0 } + new() + { + bKeyDown = true, wVirtualKeyCode = 17, wVirtualScanCode = 29, + dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED + }, + new() + { + bKeyDown = true, wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', + dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED + }, + new() + { + bKeyDown = false, wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', + dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED + }, + new() { bKeyDown = false, wVirtualKeyCode = 17, wVirtualScanCode = 29, dwControlKeyState = 0 } ]; private readonly WindowsConsole _windowsConsole; @@ -133,21 +145,17 @@ private void StartEventLoop() // CTRL+V key sequence. bool isPaste = inputRecords.Where(ir => ir.EventType == EVENT_TYPE.KEY_EVENT).Count() > 1; if (isPaste) - { // simulate CTRL_V - foreach (var keyEvent in CtrlVKeyEvents) - { + foreach (KEY_EVENT_RECORD keyEvent in CtrlVKeyEvents) HandleKeyInput(keyEvent); - } - } else - { foreach (INPUT_RECORD inputRecord in inputRecords) // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault switch (inputRecord.EventType) { case EVENT_TYPE.WINDOW_BUFFER_SIZE_EVENT: - WINDOW_BUFFER_SIZE_RECORD windowBufferSize = inputRecord.Event.WindowBufferSizeEvent; + WINDOW_BUFFER_SIZE_RECORD + windowBufferSize = inputRecord.Event.WindowBufferSizeEvent; Size = new PixelBufferSize((ushort)windowBufferSize.dwSize.X, (ushort)windowBufferSize.dwSize.Y); break; @@ -163,7 +171,6 @@ private void StartEventLoop() HandleMouseInput(mouseEvent); break; } - } } }); } From 3061ac58933832f0dae9f444163c14d483bb9ef9 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 05:19:50 -0800 Subject: [PATCH 05/36] add scrollbar to gallery view pane make textbox elements scale to panel. --- .../Gallery/GalleryViews/GalleryTextBox.axaml | 108 +++++++++--------- .../View/ControlsListView.axaml | 5 +- 2 files changed, 57 insertions(+), 56 deletions(-) diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml index e22930b4..3f746530 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml @@ -6,69 +6,69 @@ mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryTextBox"> - - + + + + + + Custom context flyout + + + + + + + + + + - - - - - + + + + + - - - - - + + + + + - - + + - + \ No newline at end of file diff --git a/src/Consolonia.Gallery/View/ControlsListView.axaml b/src/Consolonia.Gallery/View/ControlsListView.axaml index 6350c69d..2f0ffa8f 100644 --- a/src/Consolonia.Gallery/View/ControlsListView.axaml +++ b/src/Consolonia.Gallery/View/ControlsListView.axaml @@ -43,9 +43,10 @@ + BorderBrush="{DynamicResource ThemeBorderBrush}" > + + From 0e924665dcce694999be6f54b1f8ed634cbb2d55 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 8 Jan 2025 13:26:31 +0000 Subject: [PATCH 06/36] Automated JetBrains cleanup Co-authored-by: <+@users.noreply.github.com> --- .../Gallery/GalleryViews/GalleryTextBox.axaml | 107 +++++++++--------- .../View/ControlsListView.axaml | 8 +- 2 files changed, 58 insertions(+), 57 deletions(-) diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml index 3f746530..23d69486 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml @@ -6,10 +6,11 @@ mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryTextBox"> - - - - - - Custom context flyout - - - - - - - - - - - - - - - + + + + + - - - - - + + + + + - - + + - + \ No newline at end of file diff --git a/src/Consolonia.Gallery/View/ControlsListView.axaml b/src/Consolonia.Gallery/View/ControlsListView.axaml index 2f0ffa8f..3097a458 100644 --- a/src/Consolonia.Gallery/View/ControlsListView.axaml +++ b/src/Consolonia.Gallery/View/ControlsListView.axaml @@ -42,10 +42,10 @@ - - + + Date: Wed, 8 Jan 2025 07:03:48 -0800 Subject: [PATCH 07/36] add paste clipboard comparison for emitting CTRL+V --- .../Infrastructure/ConsoleWindow.cs | 21 --- src/Consolonia.GuiCS/WindowsDriver.cs | 2 +- .../Win32Console.cs | 143 ++++++++++++++---- 3 files changed, 112 insertions(+), 54 deletions(-) diff --git a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs index 2dc6c8a3..266492ef 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs @@ -398,27 +398,6 @@ private void OnConsoleOnResized() private async void ConsoleOnKeyEvent(Key key, char keyChar, RawInputModifiers rawInputModifiers, bool down, ulong timeStamp) { - if (keyChar is 'r') - { - Dispatcher.UIThread.Post(() => - { - Input!(new RawTextInputEventArgs(_myKeyboardDevice, - timeStamp, - _inputRoot, - string.Concat(Enumerable.Repeat( - @"This is long text. This is long text.This is long text.T -This is long text.This is long text.This is long text.This is long text.This -This is long text.This is long text.This is long text.This is long text.This is long text. is long text.This is long text. -his is long text.This is long text.This is long text.This is long text.This is lo -ng text.This is long text.This is -This is long text.This is long text.This is long text.This is long text. - long text.This is long text.This is long text.This is long text.This is long text.This is long text.This is long text.", - 2000)))); - }, DispatcherPriority.Input); - - return; - } - if (!down) { Dispatcher.UIThread.Post(() => diff --git a/src/Consolonia.GuiCS/WindowsDriver.cs b/src/Consolonia.GuiCS/WindowsDriver.cs index 8679fcef..f9591cef 100644 --- a/src/Consolonia.GuiCS/WindowsDriver.cs +++ b/src/Consolonia.GuiCS/WindowsDriver.cs @@ -42,7 +42,7 @@ public CONSOLE_INPUT_MODE ConsoleMode } } - const int bufferSize = 65535; + const int bufferSize = 0xffff; private static INPUT_RECORD[] s_inputBuffer = new INPUT_RECORD[bufferSize]; public INPUT_RECORD[] ReadConsoleInput() diff --git a/src/Consolonia.PlatformSupport/Win32Console.cs b/src/Consolonia.PlatformSupport/Win32Console.cs index 7e748ef6..6eefbd6a 100644 --- a/src/Consolonia.PlatformSupport/Win32Console.cs +++ b/src/Consolonia.PlatformSupport/Win32Console.cs @@ -15,6 +15,9 @@ using Key = Avalonia.Input.Key; using Point = Avalonia.Point; using static Vanara.PInvoke.Kernel32; +using Avalonia.Input.Platform; +using System.Text; +using System.Collections.Generic; // ReSharper disable UnusedMember.Local #pragma warning disable CS0649 @@ -131,48 +134,124 @@ public override void PauseIO(Task task) private void StartEventLoop() { - Task.Run(() => + Task.Run(async () => { while (!Disposed /*inject ThreadAbortException*/) { PauseTask?.Wait(); var inputRecords = _windowsConsole.ReadConsoleInput(); - if (!inputRecords.Any()) - throw new NotImplementedException(); - - // We only get multiple key events when console is translating - // CTRL+V to sequence of key strokes, so we are turning it back into - // CTRL+V key sequence. - bool isPaste = inputRecords.Where(ir => ir.EventType == EVENT_TYPE.KEY_EVENT).Count() > 1; - if (isPaste) - // simulate CTRL_V - foreach (KEY_EVENT_RECORD keyEvent in CtrlVKeyEvents) - HandleKeyInput(keyEvent); + IClipboard clipboard = AvaloniaLocator.Current.GetService(); + if (clipboard != null && inputRecords.Where(evt => evt.EventType == EVENT_TYPE.KEY_EVENT).Skip(1).Any()) + { + // when console is translating CTRL+V to sequence of key strokes it comes in as multiple key events. + await ProcessClipboardInput(clipboard, inputRecords); + } else + { foreach (INPUT_RECORD inputRecord in inputRecords) - // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault - switch (inputRecord.EventType) + HandleInputRecord(inputRecord); + } + } + }); + } + + /// + /// Process clipboard input and compare to clipboard text to determine if we should paste clipboard text. + /// + /// + /// + /// + private async Task ProcessClipboardInput(IClipboard clipboard, INPUT_RECORD[] inputRecords) + { + var clipboardText = (await clipboard?.GetTextAsync()) ?? String.Empty; + if (clipboardText.Trim().Length == 0) + { + // no text in clipboard, just process input records + foreach (var inputRecord in inputRecords) + HandleInputRecord(inputRecord); + return; + } + + // KEY_EVENTS will emit \r instead of \n, so we need to remove \n from clipboard text + clipboardText = clipboardText.Replace("\n", String.Empty, StringComparison.Ordinal); + StringBuilder bufferText = new StringBuilder(); + List bufferedKeyEvents = new(); + + while (inputRecords.Any()) + { + // process all input records + for (int i = 0; i < inputRecords.Length; i++) + { + var inputRecord = inputRecords[i]; + if (inputRecord.EventType != EVENT_TYPE.KEY_EVENT) + { + // handle non-key board events + HandleInputRecord(inputRecord); + } + else + { + // capture the key event so we can play it back if we don't match clipboard text + bufferedKeyEvents.Add(inputRecord); + + // for key down events for chars that are not 0 (control keys) + if (inputRecord.Event.KeyEvent.bKeyDown && inputRecord.Event.KeyEvent.uChar != 0) + { + // append the char to the buffer text + bufferText.Append(inputRecord.Event.KeyEvent.uChar); + + var currentBufferText = bufferText.ToString(); + if (clipboardText.Trim() == currentBufferText.Trim()) + { + // buffered text matches clipboard, emit CTRL+V sequence and ignore buffered keyboard events + foreach (var ctrlVEvent in CtrlVKeyEvents) + HandleKeyInput(ctrlVEvent); + + // process remaining input records + for (++i; i < inputRecords.Length; i++) + HandleInputRecord(inputRecords[i]); + return; + } + else if (!clipboardText.StartsWith(currentBufferText, StringComparison.Ordinal)) { - case EVENT_TYPE.WINDOW_BUFFER_SIZE_EVENT: - WINDOW_BUFFER_SIZE_RECORD - windowBufferSize = inputRecord.Event.WindowBufferSizeEvent; - Size = new PixelBufferSize((ushort)windowBufferSize.dwSize.X, - (ushort)windowBufferSize.dwSize.Y); - break; - case EVENT_TYPE.FOCUS_EVENT: - FOCUS_EVENT_RECORD focusEvent = inputRecord.Event.FocusEvent; - RaiseFocusEvent(focusEvent.bSetFocus != 0); - break; - case EVENT_TYPE.KEY_EVENT: - HandleKeyInput(inputRecord.Event.KeyEvent); - break; - case EVENT_TYPE.MOUSE_EVENT: - MOUSE_EVENT_RECORD mouseEvent = inputRecord.Event.MouseEvent; - HandleMouseInput(mouseEvent); - break; + // buffered text doesn't match clipboard, emit buffered key events (we already played other events live) + foreach (var bufferedEvent in bufferedKeyEvents) + HandleInputRecord(bufferedEvent); + + // process remaining input records + for (++i; i < inputRecords.Length; i++) + HandleInputRecord(inputRecords[i]); + return; } + } + } } - }); + + inputRecords = _windowsConsole.ReadConsoleInput(); + } + } + + private void HandleInputRecord(INPUT_RECORD inputRecord) + { + // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault + switch (inputRecord.EventType) + { + case EVENT_TYPE.WINDOW_BUFFER_SIZE_EVENT: + WINDOW_BUFFER_SIZE_RECORD windowBufferSize = inputRecord.Event.WindowBufferSizeEvent; + Size = new PixelBufferSize((ushort)windowBufferSize.dwSize.X, + (ushort)windowBufferSize.dwSize.Y); + break; + case EVENT_TYPE.FOCUS_EVENT: + FOCUS_EVENT_RECORD focusEvent = inputRecord.Event.FocusEvent; + RaiseFocusEvent(focusEvent.bSetFocus != 0); + break; + case EVENT_TYPE.KEY_EVENT: + HandleKeyInput(inputRecord.Event.KeyEvent); + break; + case EVENT_TYPE.MOUSE_EVENT: + MOUSE_EVENT_RECORD mouseEvent = inputRecord.Event.MouseEvent; + HandleMouseInput(mouseEvent); + break; + } } // ReSharper disable ExpressionIsAlwaysNull From 378401911ee885406c81baa145a2fe6e1a350d31 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 8 Jan 2025 15:11:20 +0000 Subject: [PATCH 08/36] Automated JetBrains cleanup Co-authored-by: <+@users.noreply.github.com> --- .../Infrastructure/ConsoleWindow.cs | 1 - .../Win32Console.cs | 36 +++++++++---------- 2 files changed, 17 insertions(+), 20 deletions(-) diff --git a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs index 266492ef..ddc4dc1a 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs @@ -2,7 +2,6 @@ using System.Collections.Generic; using System.Diagnostics; using System.Diagnostics.CodeAnalysis; -using System.Linq; using System.Threading; using System.Threading.Tasks; using Avalonia; diff --git a/src/Consolonia.PlatformSupport/Win32Console.cs b/src/Consolonia.PlatformSupport/Win32Console.cs index 6eefbd6a..c0bc1b8c 100644 --- a/src/Consolonia.PlatformSupport/Win32Console.cs +++ b/src/Consolonia.PlatformSupport/Win32Console.cs @@ -1,12 +1,15 @@ using System; +using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.Linq; using System.Runtime.InteropServices; using System.Runtime.Versioning; +using System.Text; using System.Threading.Tasks; using Avalonia; using Avalonia.Input; +using Avalonia.Input.Platform; using Avalonia.Input.Raw; using Consolonia.Core.Drawing.PixelBufferImplementation; using Consolonia.Core.Infrastructure; @@ -15,9 +18,6 @@ using Key = Avalonia.Input.Key; using Point = Avalonia.Point; using static Vanara.PInvoke.Kernel32; -using Avalonia.Input.Platform; -using System.Text; -using System.Collections.Generic; // ReSharper disable UnusedMember.Local #pragma warning disable CS0649 @@ -140,41 +140,38 @@ private void StartEventLoop() { PauseTask?.Wait(); var inputRecords = _windowsConsole.ReadConsoleInput(); - IClipboard clipboard = AvaloniaLocator.Current.GetService(); - if (clipboard != null && inputRecords.Where(evt => evt.EventType == EVENT_TYPE.KEY_EVENT).Skip(1).Any()) - { + var clipboard = AvaloniaLocator.Current.GetService(); + if (clipboard != null && + inputRecords.Where(evt => evt.EventType == EVENT_TYPE.KEY_EVENT).Skip(1).Any()) // when console is translating CTRL+V to sequence of key strokes it comes in as multiple key events. await ProcessClipboardInput(clipboard, inputRecords); - } else - { foreach (INPUT_RECORD inputRecord in inputRecords) HandleInputRecord(inputRecord); - } } }); } /// - /// Process clipboard input and compare to clipboard text to determine if we should paste clipboard text. + /// Process clipboard input and compare to clipboard text to determine if we should paste clipboard text. /// /// /// /// private async Task ProcessClipboardInput(IClipboard clipboard, INPUT_RECORD[] inputRecords) { - var clipboardText = (await clipboard?.GetTextAsync()) ?? String.Empty; + string clipboardText = await clipboard?.GetTextAsync() ?? string.Empty; if (clipboardText.Trim().Length == 0) { // no text in clipboard, just process input records - foreach (var inputRecord in inputRecords) + foreach (INPUT_RECORD inputRecord in inputRecords) HandleInputRecord(inputRecord); return; } // KEY_EVENTS will emit \r instead of \n, so we need to remove \n from clipboard text - clipboardText = clipboardText.Replace("\n", String.Empty, StringComparison.Ordinal); - StringBuilder bufferText = new StringBuilder(); + clipboardText = clipboardText.Replace("\n", string.Empty, StringComparison.Ordinal); + var bufferText = new StringBuilder(); List bufferedKeyEvents = new(); while (inputRecords.Any()) @@ -182,7 +179,7 @@ private async Task ProcessClipboardInput(IClipboard clipboard, INPUT_RECORD[] in // process all input records for (int i = 0; i < inputRecords.Length; i++) { - var inputRecord = inputRecords[i]; + INPUT_RECORD inputRecord = inputRecords[i]; if (inputRecord.EventType != EVENT_TYPE.KEY_EVENT) { // handle non-key board events @@ -199,11 +196,11 @@ private async Task ProcessClipboardInput(IClipboard clipboard, INPUT_RECORD[] in // append the char to the buffer text bufferText.Append(inputRecord.Event.KeyEvent.uChar); - var currentBufferText = bufferText.ToString(); + string currentBufferText = bufferText.ToString(); if (clipboardText.Trim() == currentBufferText.Trim()) { // buffered text matches clipboard, emit CTRL+V sequence and ignore buffered keyboard events - foreach (var ctrlVEvent in CtrlVKeyEvents) + foreach (KEY_EVENT_RECORD ctrlVEvent in CtrlVKeyEvents) HandleKeyInput(ctrlVEvent); // process remaining input records @@ -211,10 +208,11 @@ private async Task ProcessClipboardInput(IClipboard clipboard, INPUT_RECORD[] in HandleInputRecord(inputRecords[i]); return; } - else if (!clipboardText.StartsWith(currentBufferText, StringComparison.Ordinal)) + + if (!clipboardText.StartsWith(currentBufferText, StringComparison.Ordinal)) { // buffered text doesn't match clipboard, emit buffered key events (we already played other events live) - foreach (var bufferedEvent in bufferedKeyEvents) + foreach (INPUT_RECORD bufferedEvent in bufferedKeyEvents) HandleInputRecord(bufferedEvent); // process remaining input records From d1dc7b9a744808de79111c07ce426592dd7e64c7 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 11:18:51 -0800 Subject: [PATCH 09/36] use simplified clipboard impl instead of avalonia platform. This should work on linux and macos as well --- src/Consolonia.Core/Consolonia.Core.csproj | 3 +- .../Infrastructure/ClipboardImpl.cs | 41 ++++++++++++++++ .../Infrastructure/ConsoloniaPlatform.cs | 47 ++++++++++--------- 3 files changed, 68 insertions(+), 23 deletions(-) create mode 100644 src/Consolonia.Core/Infrastructure/ClipboardImpl.cs diff --git a/src/Consolonia.Core/Consolonia.Core.csproj b/src/Consolonia.Core/Consolonia.Core.csproj index 6660e626..c3f0257b 100644 --- a/src/Consolonia.Core/Consolonia.Core.csproj +++ b/src/Consolonia.Core/Consolonia.Core.csproj @@ -6,7 +6,8 @@ - + + diff --git a/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs b/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs new file mode 100644 index 00000000..694e85c5 --- /dev/null +++ b/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs @@ -0,0 +1,41 @@ +using System; +using System.Threading.Tasks; +using Avalonia.Input; +using TextCopy; + +namespace Consolonia.Core.Infrastructure +{ + // TODO: Replace this with avalonia platform implementation + // + // The issue is that the current Avalonia Platform implementation is only easily accessible on win32 platform + // for linux and mac it has a ton of dependencies on the entire rendering subsystem. + // This is a temporary solution to get the clipboard working, but it is not a good solution for linux as it relies + // on invoking a shell command to get the clipboard content, and doesn't support GetData/SetData/GetFormats + public class ClipboardImpl : Avalonia.Input.Platform.IClipboard + { +#pragma warning disable CA1822 // Mark members as static + public Task ClearAsync() + => ClipboardService.SetTextAsync(string.Empty); + + public Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public Task GetTextAsync() + => ClipboardService.GetTextAsync(); + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public Task SetTextAsync(string text) + => ClipboardService.SetTextAsync(text); + } +} diff --git a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs index 8909e8c9..d8f94d78 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs @@ -59,34 +59,37 @@ public void Initialize() .Bind().ToConstant(new DummyIconLoader()) .Bind().ToSingleton() .Bind().ToSingleton() + .Bind().ToConstant(new ClipboardImpl()) + //.Bind().ToConstant(null) /*.Bind().ToTransient() todo: implement this navigation*/ //.Bind().ToConstant(new X11Clipboard(this)) //.Bind().ToConstant(new PlatformSettingsStub()) //.Bind().ToConstant(new GtkSystemDialog()) /*.Bind().ToConstant(new LinuxMountedVolumeInfoProvider())*/; - if (OperatingSystem.IsWindows()) - { - AvaloniaLocator.CurrentMutable.Bind() - .ToFunc(() => - { - Assembly assembly = Assembly.Load("Avalonia.Win32"); - ArgumentNullException.ThrowIfNull(assembly, "Avalonia.Win32"); - Type type = assembly.GetType(assembly.GetName().Name + ".ClipboardImpl"); - ArgumentNullException.ThrowIfNull(type, "ClipboardImpl"); - var clipboard = Activator.CreateInstance(type) as IClipboard; - ArgumentNullException.ThrowIfNull(clipboard, nameof(clipboard)); - return clipboard; - }); - } - else if (OperatingSystem.IsMacOS()) - { - // TODO: Implement or reuse MacOS clipboard - } - else if (OperatingSystem.IsLinux()) - { - // TODO: Implement or reuse X11 Clipboard - } + //if (OperatingSystem.IsWindows()) + //{ + // AvaloniaLocator.CurrentMutable.Bind() + // .ToFunc(() => + // { + // Assembly assembly = Assembly.Load("Avalonia.Win32"); + // ArgumentNullException.ThrowIfNull(assembly, "Avalonia.Win32"); + // Type type = assembly.GetType(assembly.GetName().Name + ".ClipboardImpl"); + // ArgumentNullException.ThrowIfNull(type, "ClipboardImpl"); + // var clipboard = Activator.CreateInstance(type) as IClipboard; + // ArgumentNullException.ThrowIfNull(clipboard, nameof(clipboard)); + // return clipboard; + // }); + //} + //else if (OperatingSystem.IsMacOS()) + //{ + // // TODO: Implement or reuse MacOS clipboard + //} + //else if (OperatingSystem.IsLinux()) + //{ + // //.Bind().ToConstant(new X11Clipboard(this)) + //} + } [DebuggerStepThrough] From da7b7a9e6ee3b0653cc07fca13a3e187a436bd40 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 11:41:47 -0800 Subject: [PATCH 10/36] remove extra files --- .../AvaloniaEdit.Demo.csproj | 43 ------------------- .../Consolonia.Editor/CEdit.csproj | 15 ------- src/Experimental/Consolonia.Editor/CEdit.sln | 22 ---------- 3 files changed, 80 deletions(-) delete mode 100644 src/Experimental/Consolonia.Editor/AvaloniaEdit.Demo.csproj delete mode 100644 src/Experimental/Consolonia.Editor/CEdit.csproj delete mode 100644 src/Experimental/Consolonia.Editor/CEdit.sln diff --git a/src/Experimental/Consolonia.Editor/AvaloniaEdit.Demo.csproj b/src/Experimental/Consolonia.Editor/AvaloniaEdit.Demo.csproj deleted file mode 100644 index 53b4c64e..00000000 --- a/src/Experimental/Consolonia.Editor/AvaloniaEdit.Demo.csproj +++ /dev/null @@ -1,43 +0,0 @@ - - - - WinExe - net6.0 - win7-x64;linux-x64;osx-x64 - False - DEBUG;TRACE - true - - - - - - - - - %(Filename) - Code - - - - - - - - - - - - - - - - - - - - - - - - diff --git a/src/Experimental/Consolonia.Editor/CEdit.csproj b/src/Experimental/Consolonia.Editor/CEdit.csproj deleted file mode 100644 index 1da2af0b..00000000 --- a/src/Experimental/Consolonia.Editor/CEdit.csproj +++ /dev/null @@ -1,15 +0,0 @@ - - - - Exe - net8.0 - enable - enable - - - - - - - - diff --git a/src/Experimental/Consolonia.Editor/CEdit.sln b/src/Experimental/Consolonia.Editor/CEdit.sln deleted file mode 100644 index e7c1b6c8..00000000 --- a/src/Experimental/Consolonia.Editor/CEdit.sln +++ /dev/null @@ -1,22 +0,0 @@ - -Microsoft Visual Studio Solution File, Format Version 12.00 -# Visual Studio Version 17 -VisualStudioVersion = 17.12.35527.113 d17.12 -MinimumVisualStudioVersion = 10.0.40219.1 -Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "CEdit", "CEdit.csproj", "{2E7D27E6-C5A0-4B2B-B3EE-18D03A4FCEDF}" -EndProject -Global - GlobalSection(SolutionConfigurationPlatforms) = preSolution - Debug|Any CPU = Debug|Any CPU - Release|Any CPU = Release|Any CPU - EndGlobalSection - GlobalSection(ProjectConfigurationPlatforms) = postSolution - {2E7D27E6-C5A0-4B2B-B3EE-18D03A4FCEDF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU - {2E7D27E6-C5A0-4B2B-B3EE-18D03A4FCEDF}.Debug|Any CPU.Build.0 = Debug|Any CPU - {2E7D27E6-C5A0-4B2B-B3EE-18D03A4FCEDF}.Release|Any CPU.ActiveCfg = Release|Any CPU - {2E7D27E6-C5A0-4B2B-B3EE-18D03A4FCEDF}.Release|Any CPU.Build.0 = Release|Any CPU - EndGlobalSection - GlobalSection(SolutionProperties) = preSolution - HideSolutionNode = FALSE - EndGlobalSection -EndGlobal From a85aef02e65564a4ece32170a0313bad6b0bac5b Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 11:50:30 -0800 Subject: [PATCH 11/36] cleanup --- src/Consolonia.Core/Helpers/Extensions.cs | 2 ++ src/Consolonia.Core/Infrastructure/ClipboardImpl.cs | 4 ++-- 2 files changed, 4 insertions(+), 2 deletions(-) diff --git a/src/Consolonia.Core/Helpers/Extensions.cs b/src/Consolonia.Core/Helpers/Extensions.cs index 0f9e1be8..4e147a6e 100644 --- a/src/Consolonia.Core/Helpers/Extensions.cs +++ b/src/Consolonia.Core/Helpers/Extensions.cs @@ -13,7 +13,9 @@ namespace Consolonia.Core.Helpers { +#pragma warning disable CA1742 public static class Extensions +#pragma warning restore CA1742 { public static IDisposable SubscribeAction( this IObservable> observable, diff --git a/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs b/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs index 694e85c5..f06c6456 100644 --- a/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs +++ b/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs @@ -11,7 +11,7 @@ namespace Consolonia.Core.Infrastructure // for linux and mac it has a ton of dependencies on the entire rendering subsystem. // This is a temporary solution to get the clipboard working, but it is not a good solution for linux as it relies // on invoking a shell command to get the clipboard content, and doesn't support GetData/SetData/GetFormats - public class ClipboardImpl : Avalonia.Input.Platform.IClipboard + internal class ClipboardImpl : Avalonia.Input.Platform.IClipboard { #pragma warning disable CA1822 // Mark members as static public Task ClearAsync() @@ -36,6 +36,6 @@ public Task SetDataObjectAsync(IDataObject data) } public Task SetTextAsync(string text) - => ClipboardService.SetTextAsync(text); + => ClipboardService.SetTextAsync(text ?? String.Empty); } } From 988380bbeddcad91f4b0ad6f0e1095f411b379ed Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 12:29:07 -0800 Subject: [PATCH 12/36] fix extensions naming error. (Why is this happening now?) --- .../Helpers/{Extensions.cs => UtilityExtensions.cs} | 4 +--- 1 file changed, 1 insertion(+), 3 deletions(-) rename src/Consolonia.Core/Helpers/{Extensions.cs => UtilityExtensions.cs} (98%) diff --git a/src/Consolonia.Core/Helpers/Extensions.cs b/src/Consolonia.Core/Helpers/UtilityExtensions.cs similarity index 98% rename from src/Consolonia.Core/Helpers/Extensions.cs rename to src/Consolonia.Core/Helpers/UtilityExtensions.cs index 4e147a6e..12dbf6d1 100644 --- a/src/Consolonia.Core/Helpers/Extensions.cs +++ b/src/Consolonia.Core/Helpers/UtilityExtensions.cs @@ -13,9 +13,7 @@ namespace Consolonia.Core.Helpers { -#pragma warning disable CA1742 - public static class Extensions -#pragma warning restore CA1742 + public static class UtilityExtensions { public static IDisposable SubscribeAction( this IObservable> observable, From 2090e28b8b8f863efc708e79c17daafc45a21c62 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Wed, 8 Jan 2025 20:35:58 +0000 Subject: [PATCH 13/36] Automated JetBrains cleanup Co-authored-by: <+@users.noreply.github.com> --- .../Infrastructure/ClipboardImpl.cs | 17 ++++++++++++----- .../Infrastructure/ConsoloniaPlatform.cs | 2 -- 2 files changed, 12 insertions(+), 7 deletions(-) diff --git a/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs b/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs index f06c6456..1b8b604a 100644 --- a/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs +++ b/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs @@ -2,6 +2,7 @@ using System.Threading.Tasks; using Avalonia.Input; using TextCopy; +using IClipboard = Avalonia.Input.Platform.IClipboard; namespace Consolonia.Core.Infrastructure { @@ -11,11 +12,13 @@ namespace Consolonia.Core.Infrastructure // for linux and mac it has a ton of dependencies on the entire rendering subsystem. // This is a temporary solution to get the clipboard working, but it is not a good solution for linux as it relies // on invoking a shell command to get the clipboard content, and doesn't support GetData/SetData/GetFormats - internal class ClipboardImpl : Avalonia.Input.Platform.IClipboard + internal class ClipboardImpl : IClipboard { #pragma warning disable CA1822 // Mark members as static public Task ClearAsync() - => ClipboardService.SetTextAsync(string.Empty); + { + return ClipboardService.SetTextAsync(string.Empty); + } public Task GetDataAsync(string format) { @@ -28,7 +31,9 @@ public Task GetFormatsAsync() } public Task GetTextAsync() - => ClipboardService.GetTextAsync(); + { + return ClipboardService.GetTextAsync(); + } public Task SetDataObjectAsync(IDataObject data) { @@ -36,6 +41,8 @@ public Task SetDataObjectAsync(IDataObject data) } public Task SetTextAsync(string text) - => ClipboardService.SetTextAsync(text ?? String.Empty); + { + return ClipboardService.SetTextAsync(text ?? string.Empty); + } } -} +} \ No newline at end of file diff --git a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs index d8f94d78..85631042 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs @@ -1,6 +1,5 @@ using System; using System.Diagnostics; -using System.Reflection; using Avalonia; using Avalonia.Controls.Platform; using Avalonia.Input; @@ -89,7 +88,6 @@ public void Initialize() //{ // //.Bind().ToConstant(new X11Clipboard(this)) //} - } [DebuggerStepThrough] From bb01b750a6008b475e8181c85c04b08e540e5bce Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 16:14:14 -0800 Subject: [PATCH 14/36] add textinput event --- .../Infrastructure/ConsoleBase.cs | 9 + .../Infrastructure/ConsoleWindow.cs | 13 + .../Infrastructure/IConsole.cs | 1 + src/Consolonia.Core/Text/Esc.cs | 2 +- .../CursesConsole.cs | 356 +++++++++--------- .../Win32Console.cs | 43 +-- 6 files changed, 233 insertions(+), 191 deletions(-) diff --git a/src/Consolonia.Core/Infrastructure/ConsoleBase.cs b/src/Consolonia.Core/Infrastructure/ConsoleBase.cs index 1a384a09..b7227e53 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoleBase.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoleBase.cs @@ -6,6 +6,7 @@ using Avalonia.Media; using Consolonia.Core.Drawing; using Consolonia.Core.Drawing.PixelBufferImplementation; +using Consolonia.Core.Text; namespace Consolonia.Core.Infrastructure { @@ -68,6 +69,7 @@ protected void StartSizeCheckTimerAsync(uint slowInterval = 1500) public event Action KeyEvent; public event Action MouseEvent; public event Action FocusEvent; + public event Action TextInputEvent; protected void RaiseMouseEvent(RawPointerEventType eventType, Point point, Vector? wheelDelta, RawInputModifiers modifiers) @@ -80,6 +82,11 @@ protected void RaiseKeyPress(Key key, char character, RawInputModifiers modifier KeyEvent?.Invoke(key, character, modifiers, down, timeStamp); } + protected void RaiseTextInput(string text, ulong timestamp) + { + TextInputEvent?.Invoke(text, timestamp); + } + protected void RaiseFocusEvent(bool focused) { FocusEvent?.Invoke(focused); @@ -123,6 +130,7 @@ public virtual void HideCaret() public virtual void PrepareConsole() { + WriteText(Esc.EnableBracketedPasteMode); _consoleOutput.PrepareConsole(); } @@ -135,6 +143,7 @@ public virtual void Print(PixelBufferCoordinate bufferPoint, Color background, C public virtual void RestoreConsole() { _consoleOutput.RestoreConsole(); + WriteText(Esc.DisableBracketedPasteMode); } public virtual void SetCaretPosition(PixelBufferCoordinate bufferPoint) diff --git a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs index ddc4dc1a..2387b19c 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoleWindow.cs @@ -38,6 +38,7 @@ public ConsoleWindow() Console = AvaloniaLocator.Current.GetService() ?? throw new NotImplementedException(); Console.Resized += OnConsoleOnResized; Console.KeyEvent += ConsoleOnKeyEvent; + Console.TextInputEvent += ConsoleOnTextInputEvent; Console.MouseEvent += ConsoleOnMouseEvent; Console.FocusEvent += ConsoleOnFocusEvent; Handle = null!; @@ -394,6 +395,18 @@ private void OnConsoleOnResized() }); } + private void ConsoleOnTextInputEvent(string text, ulong timeStamp) + { + Dispatcher.UIThread.Post(() => + { +#pragma warning disable CS0618 // Type or member is obsolete // todo: change to correct constructor, CFA20A9A-3A24-4187-9CA3-9DF0081124EE + var rawInputEventArgs = new RawTextInputEventArgs(_myKeyboardDevice, timeStamp, _inputRoot, text); +#pragma warning restore CS0618 // Type or member is obsolete + Input!(rawInputEventArgs); + }, DispatcherPriority.Input); + } + + private async void ConsoleOnKeyEvent(Key key, char keyChar, RawInputModifiers rawInputModifiers, bool down, ulong timeStamp) { diff --git a/src/Consolonia.Core/Infrastructure/IConsole.cs b/src/Consolonia.Core/Infrastructure/IConsole.cs index 3968f88a..ce276d36 100644 --- a/src/Consolonia.Core/Infrastructure/IConsole.cs +++ b/src/Consolonia.Core/Infrastructure/IConsole.cs @@ -32,6 +32,7 @@ public interface IConsole : IConsoleOutput public event Action Resized; event Action KeyEvent; + event Action TextInputEvent; event Action MouseEvent; event Action FocusEvent; diff --git a/src/Consolonia.Core/Text/Esc.cs b/src/Consolonia.Core/Text/Esc.cs index e0680db9..3f6f1e30 100644 --- a/src/Consolonia.Core/Text/Esc.cs +++ b/src/Consolonia.Core/Text/Esc.cs @@ -42,7 +42,7 @@ internal static class Esc // bracketed public const string EnableBracketedPasteMode = "\u001b[?2004h"; - public const string DisableBracktedPasteMode = "\u001b[?2004l"; + public const string DisableBracketedPasteMode = "\u001b[?2004l"; // mouse tracking public const string EnableMouseTracking = "\u001b[?1000h"; diff --git a/src/Consolonia.PlatformSupport/CursesConsole.cs b/src/Consolonia.PlatformSupport/CursesConsole.cs index 6cae63dd..135e4f33 100644 --- a/src/Consolonia.PlatformSupport/CursesConsole.cs +++ b/src/Consolonia.PlatformSupport/CursesConsole.cs @@ -8,6 +8,7 @@ using System; using System.Diagnostics; +using System.Text; using System.Threading; using System.Threading.Tasks; using Avalonia; @@ -163,7 +164,7 @@ private void ProcessInput() _keyModifiers = new KeyModifiers(); var k = Key.Unknown; - + if (code == Curses.KEY_CODE_YES) { switch (wch) @@ -216,182 +217,199 @@ private void ProcessInput() { // Special handling for ESC, we want to try to catch ESC+letter to simulate alt-letter as well as Alt-Fkey case 27: - { - Curses.timeout(200); - - code = Curses.get_wch(out int wch2); - - if (code == Curses.KEY_CODE_YES) k = Key.AltMask | MapCursesKey(wch); - - if (code == 0) { - //KeyEvent key; - - // The ESC-number handling, debatable. - // Simulates the AltMask itself by pressing Alt + Space. - if (wch2 == (int)Key.Space) - k = Key.AltMask; - else if (wch2 - (int)Key.Space >= (uint)Key.A && wch2 - (int)Key.Space <= (uint)Key.Z) - k = (Key)((uint)Key.AltMask + (wch2 - (int)Key.Space)); - else if (wch2 >= (uint)Key.A - 64 && wch2 <= (uint)Key.Z - 64) - k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + (wch2 + 64)); - else if (wch2 >= (uint)Key.D0 && wch2 <= (uint)Key.D9) - k = (Key)((uint)Key.AltMask + (uint)Key.D0 + (wch2 - (uint)Key.D0)); - else - switch (wch2) - { - case 27: - k = (Key)wch2; - break; - case Curses.KEY_CODE_SEQ: + Curses.timeout(200); + + code = Curses.get_wch(out int wch2); + + if (code == Curses.KEY_CODE_YES) k = Key.AltMask | MapCursesKey(wch); + + if (code == 0) + { + //KeyEvent key; + + // The ESC-number handling, debatable. + // Simulates the AltMask itself by pressing Alt + Space. + if (wch2 == (int)Key.Space) + k = Key.AltMask; + else if (wch2 - (int)Key.Space >= (uint)Key.A && wch2 - (int)Key.Space <= (uint)Key.Z) + k = (Key)((uint)Key.AltMask + (wch2 - (int)Key.Space)); + else if (wch2 >= (uint)Key.A - 64 && wch2 <= (uint)Key.Z - 64) + k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + (wch2 + 64)); + else if (wch2 >= (uint)Key.D0 && wch2 <= (uint)Key.D9) + k = (Key)((uint)Key.AltMask + (uint)Key.D0 + (wch2 - (uint)Key.D0)); + else + switch (wch2) { - int[] c = null; - while (code == 0) - { - code = Curses.get_wch(out wch2); - if (wch2 <= 0) continue; - int length = 1; - if (c != null) - length += c.Length; - Array.Resize(ref c, length); - - c[^1] = wch2; - } - - switch (c![0]) - { - case 49 when c[1] == 59 && c[2] == 55 && c[3] >= 80 && c[3] <= 83: - // Ctrl+Alt+(F1 - F4) - wch2 = c[3] + 185; - k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 - when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: - // Ctrl+Alt+(F5 - F8) - wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; - k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); - break; - case 50 - when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: - // Ctrl+Alt+(F9 - F12) - wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; - k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 when c[1] == 59 && c[2] == 56 && c[3] >= 80 && c[3] <= 83: - // Ctrl+Shift+Alt+(F1 - F4) - wch2 = c[3] + 185; - k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 - when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: - // Ctrl+Shift+Alt+(F5 - F8) - wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; - k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 50 - when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: - // Ctrl+Shift+Alt+(F9 - F12) - wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; - k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 when c[1] == 59 && c[2] == 52 && c[3] == 83: - // Shift+Alt+(F4) - wch2 = 268; - k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 - when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: - // Shift+Alt+(F5 - F8) - wch2 = c[1] < 55 ? c[1] + 216 : c[1] + 215; - k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 50 - when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: - // Shift+Alt+(F9 - F12) - wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; - k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 54 when c[1] == 59 && c[2] == 56 && c[3] == 126: - // Shift+Ctrl+Alt+KeyNPage - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageDown; - break; - case 53 when c[1] == 59 && c[2] == 56 && c[3] == 126: - // Shift+Ctrl+Alt+KeyPPage - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageUp; - break; - case 49 when c[1] == 59 && c[2] == 56 && c[3] == 72: - // Shift+Ctrl+Alt+KeyHome - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.Home; - break; - case 49 when c[1] == 59 && c[2] == 56 && c[3] == 70: - // Shift+Ctrl+Alt+KeyEnd - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.End; - break; - default: - k = MapCursesKey(wch2); + case 27: + k = (Key)wch2; + break; + case Curses.KEY_CODE_SEQ: + { + int[] c = null; + while (code == 0) + { + code = Curses.get_wch(out wch2); + if (wch2 <= 0) continue; + int length = 1; + if (c != null) + length += c.Length; + Array.Resize(ref c, length); + + c[^1] = wch2; + } + + switch (c![0]) + { + case 49 when c[1] == 59 && c[2] == 55 && c[3] >= 80 && c[3] <= 83: + // Ctrl+Alt+(F1 - F4) + wch2 = c[3] + 185; + k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 + when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: + // Ctrl+Alt+(F5 - F8) + wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; + k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); + break; + case 50 + when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: + // Ctrl+Alt+(F9 - F12) + wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; + k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 when c[1] == 59 && c[2] == 56 && c[3] >= 80 && c[3] <= 83: + // Ctrl+Shift+Alt+(F1 - F4) + wch2 = c[3] + 185; + k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 + when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: + // Ctrl+Shift+Alt+(F5 - F8) + wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; + k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 50 + when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: + // Ctrl+Shift+Alt+(F9 - F12) + wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; + k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 when c[1] == 59 && c[2] == 52 && c[3] == 83: + // Shift+Alt+(F4) + wch2 = 268; + k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 + when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: + // Shift+Alt+(F5 - F8) + wch2 = c[1] < 55 ? c[1] + 216 : c[1] + 215; + k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 50 + when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: + // Shift+Alt+(F9 - F12) + wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; + k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 54 when c[1] == 59 && c[2] == 56 && c[3] == 126: + // Shift+Ctrl+Alt+KeyNPage + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageDown; + break; + case 53 when c[1] == 59 && c[2] == 56 && c[3] == 126: + // Shift+Ctrl+Alt+KeyPPage + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageUp; + break; + case 49 when c[1] == 59 && c[2] == 56 && c[3] == 72: + // Shift+Ctrl+Alt+KeyHome + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.Home; + break; + case 49 when c[1] == 59 && c[2] == 56 && c[3] == 70: + // Shift+Ctrl+Alt+KeyEnd + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.End; + break; + + // ESC [200~ + case 50 when c[1] == 48 && c[2] == 48 && c[3] == 126: + StringBuilder sb = new StringBuilder(); + for (int i = 4; i < c.Length; i++) + { + sb.Append((char)c[i]); + } + var bufferText = sb.ToString(); + var index = bufferText.IndexOf("\u001b[201~", StringComparison.Ordinal); + if (index > 0) + { + var text = bufferText[..--index]; + RaiseTextInput(text, (ulong)Stopwatch.GetTimestamp()); + } + break; + + default: + k = MapCursesKey(wch2); + break; + } + break; - } + } + default: + { + // Unfortunately there are no way to differentiate Ctrl+Alt+alfa and Ctrl+Shift+Alt+alfa. + if (((Key)wch2 & Key.CtrlMask) != 0) _keyModifiers.Ctrl = true; + + if (wch2 == 0) + { + k = Key.CtrlMask | Key.AltMask | Key.Space; + } + // ReSharper disable once ConditionIsAlwaysTrueOrFalse todo: check why + else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) + { + _keyModifiers.Shift = true; + _keyModifiers.Alt = true; + } + else if (wch2 < 256) + { + k = (Key)wch2; + _keyModifiers.Alt = true; + } + else + { + k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + wch2); + } - break; - } - default: - { - // Unfortunately there are no way to differentiate Ctrl+Alt+alfa and Ctrl+Shift+Alt+alfa. - if (((Key)wch2 & Key.CtrlMask) != 0) _keyModifiers.Ctrl = true; - - if (wch2 == 0) - { - k = Key.CtrlMask | Key.AltMask | Key.Space; - } - // ReSharper disable once ConditionIsAlwaysTrueOrFalse todo: check why - else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) - { - _keyModifiers.Shift = true; - _keyModifiers.Alt = true; - } - else if (wch2 < 256) - { - k = (Key)wch2; - _keyModifiers.Alt = true; - } - else - { - k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + wch2); - } - - break; + break; + } } - } - } - else - { - k = Key.Esc; - } + } + else + { + k = Key.Esc; + } - break; - } + break; + } case Curses.KeyTab: k = MapCursesKey(wch); break; default: - { - // Unfortunately there are no way to differentiate Ctrl+alfa and Ctrl+Shift+alfa. - k = (Key)wch; - if (wch == 0) - { - k = Key.CtrlMask | Key.Space; - } - else if (wch >= (uint)Key.A - 64 && wch <= (uint)Key.Z - 64) - { - if ((Key)(wch + 64) != Key.J) k = Key.CtrlMask | (Key)(wch + 64); - } - else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) { - _keyModifiers.Shift = true; - } + // Unfortunately there are no way to differentiate Ctrl+alfa and Ctrl+Shift+alfa. + k = (Key)wch; + if (wch == 0) + { + k = Key.CtrlMask | Key.Space; + } + else if (wch >= (uint)Key.A - 64 && wch <= (uint)Key.Z - 64) + { + if ((Key)(wch + 64) != Key.J) k = Key.CtrlMask | (Key)(wch + 64); + } + else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) + { + _keyModifiers.Shift = true; + } - break; - } + break; + } } RaiseKeyPressInternal(k); @@ -415,10 +433,10 @@ private void RaiseKeyPressInternal(Key key) case ConsoleKey.NoName: return; case 0: - { - bool _ = Enum.TryParse(key.ToString(), true, out consoleKey); - break; - } + { + bool _ = Enum.TryParse(key.ToString(), true, out consoleKey); + break; + } } if (((uint)keyValue & (uint)Key.CharMask) > 27) diff --git a/src/Consolonia.PlatformSupport/Win32Console.cs b/src/Consolonia.PlatformSupport/Win32Console.cs index c0bc1b8c..e880b981 100644 --- a/src/Consolonia.PlatformSupport/Win32Console.cs +++ b/src/Consolonia.PlatformSupport/Win32Console.cs @@ -69,25 +69,25 @@ private static readonly FlagTranslator ]); - private static readonly KEY_EVENT_RECORD[] CtrlVKeyEvents = - [ - new() - { - bKeyDown = true, wVirtualKeyCode = 17, wVirtualScanCode = 29, - dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED - }, - new() - { - bKeyDown = true, wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', - dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED - }, - new() - { - bKeyDown = false, wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', - dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED - }, - new() { bKeyDown = false, wVirtualKeyCode = 17, wVirtualScanCode = 29, dwControlKeyState = 0 } - ]; + //private static readonly KEY_EVENT_RECORD[] CtrlVKeyEvents = + //[ + // new() + // { + // bKeyDown = true, wVirtualKeyCode = 17, wVirtualScanCode = 29, + // dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED + // }, + // new() + // { + // bKeyDown = true, wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', + // dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED + // }, + // new() + // { + // bKeyDown = false, wVirtualKeyCode = 86, wVirtualScanCode = 47, uChar = '\u0016', + // dwControlKeyState = CONTROL_KEY_STATE.LEFT_CTRL_PRESSED + // }, + // new() { bKeyDown = false, wVirtualKeyCode = 17, wVirtualScanCode = 29, dwControlKeyState = 0 } + //]; private readonly WindowsConsole _windowsConsole; @@ -200,8 +200,9 @@ private async Task ProcessClipboardInput(IClipboard clipboard, INPUT_RECORD[] in if (clipboardText.Trim() == currentBufferText.Trim()) { // buffered text matches clipboard, emit CTRL+V sequence and ignore buffered keyboard events - foreach (KEY_EVENT_RECORD ctrlVEvent in CtrlVKeyEvents) - HandleKeyInput(ctrlVEvent); + //foreach (KEY_EVENT_RECORD ctrlVEvent in CtrlVKeyEvents) + // HandleKeyInput(ctrlVEvent); + RaiseTextInput(currentBufferText, (ulong)Stopwatch.GetTimestamp()); // process remaining input records for (++i; i < inputRecords.Length; i++) From 3e7f3ef89bb640974ad0f6aa7fbccd14d1be5cbb Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 16:41:32 -0800 Subject: [PATCH 15/36] add new event to unit test --- src/Consolonia.NUnit/UnitTestConsole.cs | 1 + 1 file changed, 1 insertion(+) diff --git a/src/Consolonia.NUnit/UnitTestConsole.cs b/src/Consolonia.NUnit/UnitTestConsole.cs index 6463d0ca..8ef5e445 100644 --- a/src/Consolonia.NUnit/UnitTestConsole.cs +++ b/src/Consolonia.NUnit/UnitTestConsole.cs @@ -204,6 +204,7 @@ public void SetupLifetime(ClassicDesktopStyleApplicationLifetime lifetime) public event Action KeyEvent; public event Action MouseEvent; public event Action FocusEvent; + public event Action TextInputEvent; #pragma warning restore CS0067 } } \ No newline at end of file From a3c37ebc295e0d0d033d2cb6638a665b63461614 Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Thu, 9 Jan 2025 00:48:19 +0000 Subject: [PATCH 16/36] Automated JetBrains cleanup Co-authored-by: <+@users.noreply.github.com> --- .../CursesConsole.cs | 360 +++++++++--------- 1 file changed, 179 insertions(+), 181 deletions(-) diff --git a/src/Consolonia.PlatformSupport/CursesConsole.cs b/src/Consolonia.PlatformSupport/CursesConsole.cs index 135e4f33..4ecc2c6c 100644 --- a/src/Consolonia.PlatformSupport/CursesConsole.cs +++ b/src/Consolonia.PlatformSupport/CursesConsole.cs @@ -164,7 +164,7 @@ private void ProcessInput() _keyModifiers = new KeyModifiers(); var k = Key.Unknown; - + if (code == Curses.KEY_CODE_YES) { switch (wch) @@ -217,199 +217,197 @@ private void ProcessInput() { // Special handling for ESC, we want to try to catch ESC+letter to simulate alt-letter as well as Alt-Fkey case 27: + { + Curses.timeout(200); + + code = Curses.get_wch(out int wch2); + + if (code == Curses.KEY_CODE_YES) k = Key.AltMask | MapCursesKey(wch); + + if (code == 0) { - Curses.timeout(200); - - code = Curses.get_wch(out int wch2); - - if (code == Curses.KEY_CODE_YES) k = Key.AltMask | MapCursesKey(wch); - - if (code == 0) - { - //KeyEvent key; - - // The ESC-number handling, debatable. - // Simulates the AltMask itself by pressing Alt + Space. - if (wch2 == (int)Key.Space) - k = Key.AltMask; - else if (wch2 - (int)Key.Space >= (uint)Key.A && wch2 - (int)Key.Space <= (uint)Key.Z) - k = (Key)((uint)Key.AltMask + (wch2 - (int)Key.Space)); - else if (wch2 >= (uint)Key.A - 64 && wch2 <= (uint)Key.Z - 64) - k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + (wch2 + 64)); - else if (wch2 >= (uint)Key.D0 && wch2 <= (uint)Key.D9) - k = (Key)((uint)Key.AltMask + (uint)Key.D0 + (wch2 - (uint)Key.D0)); - else - switch (wch2) + //KeyEvent key; + + // The ESC-number handling, debatable. + // Simulates the AltMask itself by pressing Alt + Space. + if (wch2 == (int)Key.Space) + k = Key.AltMask; + else if (wch2 - (int)Key.Space >= (uint)Key.A && wch2 - (int)Key.Space <= (uint)Key.Z) + k = (Key)((uint)Key.AltMask + (wch2 - (int)Key.Space)); + else if (wch2 >= (uint)Key.A - 64 && wch2 <= (uint)Key.Z - 64) + k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + (wch2 + 64)); + else if (wch2 >= (uint)Key.D0 && wch2 <= (uint)Key.D9) + k = (Key)((uint)Key.AltMask + (uint)Key.D0 + (wch2 - (uint)Key.D0)); + else + switch (wch2) + { + case 27: + k = (Key)wch2; + break; + case Curses.KEY_CODE_SEQ: { - case 27: - k = (Key)wch2; - break; - case Curses.KEY_CODE_SEQ: - { - int[] c = null; - while (code == 0) - { - code = Curses.get_wch(out wch2); - if (wch2 <= 0) continue; - int length = 1; - if (c != null) - length += c.Length; - Array.Resize(ref c, length); - - c[^1] = wch2; - } + int[] c = null; + while (code == 0) + { + code = Curses.get_wch(out wch2); + if (wch2 <= 0) continue; + int length = 1; + if (c != null) + length += c.Length; + Array.Resize(ref c, length); + + c[^1] = wch2; + } + + switch (c![0]) + { + case 49 when c[1] == 59 && c[2] == 55 && c[3] >= 80 && c[3] <= 83: + // Ctrl+Alt+(F1 - F4) + wch2 = c[3] + 185; + k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 + when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: + // Ctrl+Alt+(F5 - F8) + wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; + k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); + break; + case 50 + when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: + // Ctrl+Alt+(F9 - F12) + wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; + k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 when c[1] == 59 && c[2] == 56 && c[3] >= 80 && c[3] <= 83: + // Ctrl+Shift+Alt+(F1 - F4) + wch2 = c[3] + 185; + k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 + when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: + // Ctrl+Shift+Alt+(F5 - F8) + wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; + k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 50 + when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: + // Ctrl+Shift+Alt+(F9 - F12) + wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; + k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 when c[1] == 59 && c[2] == 52 && c[3] == 83: + // Shift+Alt+(F4) + wch2 = 268; + k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 49 + when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: + // Shift+Alt+(F5 - F8) + wch2 = c[1] < 55 ? c[1] + 216 : c[1] + 215; + k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 50 + when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: + // Shift+Alt+(F9 - F12) + wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; + k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); + break; + case 54 when c[1] == 59 && c[2] == 56 && c[3] == 126: + // Shift+Ctrl+Alt+KeyNPage + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageDown; + break; + case 53 when c[1] == 59 && c[2] == 56 && c[3] == 126: + // Shift+Ctrl+Alt+KeyPPage + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageUp; + break; + case 49 when c[1] == 59 && c[2] == 56 && c[3] == 72: + // Shift+Ctrl+Alt+KeyHome + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.Home; + break; + case 49 when c[1] == 59 && c[2] == 56 && c[3] == 70: + // Shift+Ctrl+Alt+KeyEnd + k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.End; + break; - switch (c![0]) + // ESC [200~ + case 50 when c[1] == 48 && c[2] == 48 && c[3] == 126: + var sb = new StringBuilder(); + for (int i = 4; i < c.Length; i++) sb.Append((char)c[i]); + string bufferText = sb.ToString(); + int index = bufferText.IndexOf("\u001b[201~", StringComparison.Ordinal); + if (index > 0) { - case 49 when c[1] == 59 && c[2] == 55 && c[3] >= 80 && c[3] <= 83: - // Ctrl+Alt+(F1 - F4) - wch2 = c[3] + 185; - k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 - when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: - // Ctrl+Alt+(F5 - F8) - wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; - k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); - break; - case 50 - when c[2] == 59 && c[3] == 55 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: - // Ctrl+Alt+(F9 - F12) - wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; - k = Key.CtrlMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 when c[1] == 59 && c[2] == 56 && c[3] >= 80 && c[3] <= 83: - // Ctrl+Shift+Alt+(F1 - F4) - wch2 = c[3] + 185; - k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 - when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: - // Ctrl+Shift+Alt+(F5 - F8) - wch2 = c[1] == 53 ? c[1] + 216 : c[1] + 215; - k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 50 - when c[2] == 59 && c[3] == 56 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: - // Ctrl+Shift+Alt+(F9 - F12) - wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; - k = Key.CtrlMask | Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 when c[1] == 59 && c[2] == 52 && c[3] == 83: - // Shift+Alt+(F4) - wch2 = 268; - k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 49 - when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 53 && c[1] <= 57: - // Shift+Alt+(F5 - F8) - wch2 = c[1] < 55 ? c[1] + 216 : c[1] + 215; - k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 50 - when c[2] == 59 && c[3] == 52 && c[4] == 126 && c[1] >= 48 && c[1] <= 52: - // Shift+Alt+(F9 - F12) - wch2 = c[1] < 51 ? c[1] + 225 : c[1] + 224; - k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch2); - break; - case 54 when c[1] == 59 && c[2] == 56 && c[3] == 126: - // Shift+Ctrl+Alt+KeyNPage - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageDown; - break; - case 53 when c[1] == 59 && c[2] == 56 && c[3] == 126: - // Shift+Ctrl+Alt+KeyPPage - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.PageUp; - break; - case 49 when c[1] == 59 && c[2] == 56 && c[3] == 72: - // Shift+Ctrl+Alt+KeyHome - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.Home; - break; - case 49 when c[1] == 59 && c[2] == 56 && c[3] == 70: - // Shift+Ctrl+Alt+KeyEnd - k = Key.ShiftMask | Key.CtrlMask | Key.AltMask | Key.End; - break; - - // ESC [200~ - case 50 when c[1] == 48 && c[2] == 48 && c[3] == 126: - StringBuilder sb = new StringBuilder(); - for (int i = 4; i < c.Length; i++) - { - sb.Append((char)c[i]); - } - var bufferText = sb.ToString(); - var index = bufferText.IndexOf("\u001b[201~", StringComparison.Ordinal); - if (index > 0) - { - var text = bufferText[..--index]; - RaiseTextInput(text, (ulong)Stopwatch.GetTimestamp()); - } - break; - - default: - k = MapCursesKey(wch2); - break; + string text = bufferText[..--index]; + RaiseTextInput(text, (ulong)Stopwatch.GetTimestamp()); } break; - } - default: - { - // Unfortunately there are no way to differentiate Ctrl+Alt+alfa and Ctrl+Shift+Alt+alfa. - if (((Key)wch2 & Key.CtrlMask) != 0) _keyModifiers.Ctrl = true; - - if (wch2 == 0) - { - k = Key.CtrlMask | Key.AltMask | Key.Space; - } - // ReSharper disable once ConditionIsAlwaysTrueOrFalse todo: check why - else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) - { - _keyModifiers.Shift = true; - _keyModifiers.Alt = true; - } - else if (wch2 < 256) - { - k = (Key)wch2; - _keyModifiers.Alt = true; - } - else - { - k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + wch2); - } + default: + k = MapCursesKey(wch2); break; - } - } - } - else - { - k = Key.Esc; - } + } - break; + break; + } + default: + { + // Unfortunately there are no way to differentiate Ctrl+Alt+alfa and Ctrl+Shift+Alt+alfa. + if (((Key)wch2 & Key.CtrlMask) != 0) _keyModifiers.Ctrl = true; + + if (wch2 == 0) + { + k = Key.CtrlMask | Key.AltMask | Key.Space; + } + // ReSharper disable once ConditionIsAlwaysTrueOrFalse todo: check why + else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) + { + _keyModifiers.Shift = true; + _keyModifiers.Alt = true; + } + else if (wch2 < 256) + { + k = (Key)wch2; + _keyModifiers.Alt = true; + } + else + { + k = (Key)((uint)(Key.AltMask | Key.CtrlMask) + wch2); + } + + break; + } + } } + else + { + k = Key.Esc; + } + + break; + } case Curses.KeyTab: k = MapCursesKey(wch); break; default: + { + // Unfortunately there are no way to differentiate Ctrl+alfa and Ctrl+Shift+alfa. + k = (Key)wch; + if (wch == 0) { - // Unfortunately there are no way to differentiate Ctrl+alfa and Ctrl+Shift+alfa. - k = (Key)wch; - if (wch == 0) - { - k = Key.CtrlMask | Key.Space; - } - else if (wch >= (uint)Key.A - 64 && wch <= (uint)Key.Z - 64) - { - if ((Key)(wch + 64) != Key.J) k = Key.CtrlMask | (Key)(wch + 64); - } - else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) - { - _keyModifiers.Shift = true; - } - - break; + k = Key.CtrlMask | Key.Space; + } + else if (wch >= (uint)Key.A - 64 && wch <= (uint)Key.Z - 64) + { + if ((Key)(wch + 64) != Key.J) k = Key.CtrlMask | (Key)(wch + 64); } + else if (wch >= (uint)Key.A && wch <= (uint)Key.Z) + { + _keyModifiers.Shift = true; + } + + break; + } } RaiseKeyPressInternal(k); @@ -433,10 +431,10 @@ private void RaiseKeyPressInternal(Key key) case ConsoleKey.NoName: return; case 0: - { - bool _ = Enum.TryParse(key.ToString(), true, out consoleKey); - break; - } + { + bool _ = Enum.TryParse(key.ToString(), true, out consoleKey); + break; + } } if (((uint)keyValue & (uint)Key.CharMask) > 27) From a0de64a1d338749ce778d95596c56ab252e563e6 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Wed, 8 Jan 2025 17:40:47 -0800 Subject: [PATCH 17/36] switch to naive clipboard --- src/Consolonia.Core/Consolonia.Core.csproj | 1 - .../Infrastructure/ClipboardImpl.cs | 48 ------------------ .../Infrastructure/ConsoloniaPlatform.cs | 49 ++++++++++--------- .../Infrastructure/NaiveClipboard.cs | 44 +++++++++++++++++ .../TextBlockTests.cs | 4 +- 5 files changed, 72 insertions(+), 74 deletions(-) delete mode 100644 src/Consolonia.Core/Infrastructure/ClipboardImpl.cs create mode 100644 src/Consolonia.Core/Infrastructure/NaiveClipboard.cs diff --git a/src/Consolonia.Core/Consolonia.Core.csproj b/src/Consolonia.Core/Consolonia.Core.csproj index c3f0257b..bb4a9e0e 100644 --- a/src/Consolonia.Core/Consolonia.Core.csproj +++ b/src/Consolonia.Core/Consolonia.Core.csproj @@ -7,7 +7,6 @@ - diff --git a/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs b/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs deleted file mode 100644 index 1b8b604a..00000000 --- a/src/Consolonia.Core/Infrastructure/ClipboardImpl.cs +++ /dev/null @@ -1,48 +0,0 @@ -using System; -using System.Threading.Tasks; -using Avalonia.Input; -using TextCopy; -using IClipboard = Avalonia.Input.Platform.IClipboard; - -namespace Consolonia.Core.Infrastructure -{ - // TODO: Replace this with avalonia platform implementation - // - // The issue is that the current Avalonia Platform implementation is only easily accessible on win32 platform - // for linux and mac it has a ton of dependencies on the entire rendering subsystem. - // This is a temporary solution to get the clipboard working, but it is not a good solution for linux as it relies - // on invoking a shell command to get the clipboard content, and doesn't support GetData/SetData/GetFormats - internal class ClipboardImpl : IClipboard - { -#pragma warning disable CA1822 // Mark members as static - public Task ClearAsync() - { - return ClipboardService.SetTextAsync(string.Empty); - } - - public Task GetDataAsync(string format) - { - throw new NotImplementedException(); - } - - public Task GetFormatsAsync() - { - throw new NotImplementedException(); - } - - public Task GetTextAsync() - { - return ClipboardService.GetTextAsync(); - } - - public Task SetDataObjectAsync(IDataObject data) - { - throw new NotImplementedException(); - } - - public Task SetTextAsync(string text) - { - return ClipboardService.SetTextAsync(text ?? string.Empty); - } - } -} \ No newline at end of file diff --git a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs index 85631042..378620be 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs @@ -1,5 +1,6 @@ using System; using System.Diagnostics; +using System.Reflection; using Avalonia; using Avalonia.Controls.Platform; using Avalonia.Input; @@ -58,7 +59,6 @@ public void Initialize() .Bind().ToConstant(new DummyIconLoader()) .Bind().ToSingleton() .Bind().ToSingleton() - .Bind().ToConstant(new ClipboardImpl()) //.Bind().ToConstant(null) /*.Bind().ToTransient() todo: implement this navigation*/ //.Bind().ToConstant(new X11Clipboard(this)) @@ -66,28 +66,31 @@ public void Initialize() //.Bind().ToConstant(new GtkSystemDialog()) /*.Bind().ToConstant(new LinuxMountedVolumeInfoProvider())*/; - //if (OperatingSystem.IsWindows()) - //{ - // AvaloniaLocator.CurrentMutable.Bind() - // .ToFunc(() => - // { - // Assembly assembly = Assembly.Load("Avalonia.Win32"); - // ArgumentNullException.ThrowIfNull(assembly, "Avalonia.Win32"); - // Type type = assembly.GetType(assembly.GetName().Name + ".ClipboardImpl"); - // ArgumentNullException.ThrowIfNull(type, "ClipboardImpl"); - // var clipboard = Activator.CreateInstance(type) as IClipboard; - // ArgumentNullException.ThrowIfNull(clipboard, nameof(clipboard)); - // return clipboard; - // }); - //} - //else if (OperatingSystem.IsMacOS()) - //{ - // // TODO: Implement or reuse MacOS clipboard - //} - //else if (OperatingSystem.IsLinux()) - //{ - // //.Bind().ToConstant(new X11Clipboard(this)) - //} + if (OperatingSystem.IsWindows()) + { + AvaloniaLocator.CurrentMutable.Bind() + .ToFunc(() => + { + Assembly assembly = Assembly.Load("Avalonia.Win32"); + ArgumentNullException.ThrowIfNull(assembly, "Avalonia.Win32"); + Type type = assembly.GetType(assembly.GetName().Name + ".ClipboardImpl"); + ArgumentNullException.ThrowIfNull(type, "ClipboardImpl"); + var clipboard = Activator.CreateInstance(type) as IClipboard; + ArgumentNullException.ThrowIfNull(clipboard, nameof(clipboard)); + return clipboard; + }); + } + else if (OperatingSystem.IsMacOS()) + { + // TODO: Implement or reuse MacOS clipboard + AvaloniaLocator.CurrentMutable.Bind().ToConstant(new NaiveClipboard()); + } + else if (OperatingSystem.IsLinux()) + { + // TODO: Implement or reuse X11 clipboard + AvaloniaLocator.CurrentMutable.Bind().ToConstant(new NaiveClipboard()); + } + } [DebuggerStepThrough] diff --git a/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs b/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs new file mode 100644 index 00000000..fc9d8e5b --- /dev/null +++ b/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs @@ -0,0 +1,44 @@ +using System; +using System.Threading.Tasks; +using Avalonia.Input; +using Avalonia.Input.Platform; + +namespace Consolonia.Core.Infrastructure +{ + // TODO: Replace this with avalonia platform implementation + internal class NaiveClipboard : IClipboard + { + private string _text = String.Empty; + +#pragma warning disable CA1822 // Mark members as static + public async Task ClearAsync() + { + _text = String.Empty; + } + + public async Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + public async Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public async Task GetTextAsync() + { + return _text; + } + + public async Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public async Task SetTextAsync(string text) + { + _text = text; + } + } +} \ No newline at end of file diff --git a/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs b/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs index 86ecf972..c10b137a 100644 --- a/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs +++ b/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs @@ -43,8 +43,8 @@ public async Task HandlesMultilineText() { await UITest.KeyInput(Key.Tab); await UITest.AssertHasText( - "│Vivamus magna. Cras in mi at felis aliquet congue. Ut a │", - "│est eget ligula molestie gravida. Curabitur massa. Donec│"); + "│elit. Vivamus magna. Cras in mi at felis aliquet │", + "│congue. Ut a est eget ligula molestie gravida. │"); } [Test] From 2d23c1ebb0da5f53ef3ce453ea22c79b3e30fe69 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 09:48:06 -0800 Subject: [PATCH 18/36] implement cross platform clipboards --- .../Infrastructure/ConsoloniaPlatform.cs | 25 ---- .../Infrastructure/NaiveClipboard.cs | 15 +- .../Clipboard/ClipboardProcessRunner.cs | 114 +++++++++++++++ .../Clipboard/MacOSXClipboard.cs | 130 ++++++++++++++++++ .../Clipboard/WSLClipboard.cs | 98 +++++++++++++ .../Clipboard/X11Clipboard.cs | 52 +++++++ .../Clipboard/XClipClipboard.cs | 105 ++++++++++++++ .../Consolonia.PlatformSupport.csproj | 1 + .../PlatformSupportExtensions.cs | 31 ++++- 9 files changed, 540 insertions(+), 31 deletions(-) create mode 100644 src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs create mode 100644 src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs create mode 100644 src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs create mode 100644 src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs create mode 100644 src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs diff --git a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs index 378620be..acaa52a2 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs @@ -66,31 +66,6 @@ public void Initialize() //.Bind().ToConstant(new GtkSystemDialog()) /*.Bind().ToConstant(new LinuxMountedVolumeInfoProvider())*/; - if (OperatingSystem.IsWindows()) - { - AvaloniaLocator.CurrentMutable.Bind() - .ToFunc(() => - { - Assembly assembly = Assembly.Load("Avalonia.Win32"); - ArgumentNullException.ThrowIfNull(assembly, "Avalonia.Win32"); - Type type = assembly.GetType(assembly.GetName().Name + ".ClipboardImpl"); - ArgumentNullException.ThrowIfNull(type, "ClipboardImpl"); - var clipboard = Activator.CreateInstance(type) as IClipboard; - ArgumentNullException.ThrowIfNull(clipboard, nameof(clipboard)); - return clipboard; - }); - } - else if (OperatingSystem.IsMacOS()) - { - // TODO: Implement or reuse MacOS clipboard - AvaloniaLocator.CurrentMutable.Bind().ToConstant(new NaiveClipboard()); - } - else if (OperatingSystem.IsLinux()) - { - // TODO: Implement or reuse X11 clipboard - AvaloniaLocator.CurrentMutable.Bind().ToConstant(new NaiveClipboard()); - } - } [DebuggerStepThrough] diff --git a/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs b/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs index fc9d8e5b..27d324f4 100644 --- a/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs +++ b/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs @@ -5,39 +5,44 @@ namespace Consolonia.Core.Infrastructure { - // TODO: Replace this with avalonia platform implementation - internal class NaiveClipboard : IClipboard + /// + /// This clipboard only stores memory in the same process, so it is not useful for sharing data between processes. + /// + public class NaiveClipboard : IClipboard { private string _text = String.Empty; #pragma warning disable CA1822 // Mark members as static public async Task ClearAsync() { + await Task.CompletedTask; _text = String.Empty; } - public async Task GetDataAsync(string format) + public Task GetDataAsync(string format) { throw new NotImplementedException(); } - public async Task GetFormatsAsync() + public Task GetFormatsAsync() { throw new NotImplementedException(); } public async Task GetTextAsync() { + await Task.CompletedTask; return _text; } - public async Task SetDataObjectAsync(IDataObject data) + public Task SetDataObjectAsync(IDataObject data) { throw new NotImplementedException(); } public async Task SetTextAsync(string text) { + await Task.CompletedTask; _text = text; } } diff --git a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs new file mode 100644 index 00000000..044c0938 --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs @@ -0,0 +1,114 @@ +using System; +using System.Collections.Generic; +using System.Diagnostics; +using System.Linq; +using System.Text; +using System.Threading.Tasks; + +namespace Consolonia.PlatformSupport.Clipboard +{ + /// Provides cut, copy, and paste support for the OS clipboard. + /// + /// On Windows, the class uses the Windows Clipboard APIs via P/Invoke. + /// + /// On Linux, when not running under Windows Subsystem for Linux (WSL), the class uses + /// the xclip command line tool. If xclip is not installed, the clipboard will not work. + /// + /// + /// On Linux, when running under Windows Subsystem for Linux (WSL), the class launches + /// Windows' powershell.exe via WSL interop and uses the "Set-Clipboard" and "Get-Clipboard" Powershell CmdLets. + /// + /// + /// On the Mac, the class uses the MacO OS X pbcopy and pbpaste command line tools and + /// the Mac clipboard APIs vai P/Invoke. + /// + /// + + /// + /// Helper class for console drivers to invoke shell commands to interact with the clipboard. + /// + internal static class ClipboardProcessRunner + { + public static (int exitCode, string result) Bash( + string commandLine, + string inputText = "", + bool waitForOutput = false + ) + { + var arguments = $"-c \"{commandLine}\""; + (int exitCode, string result) = Process("bash", arguments, inputText, waitForOutput); + + return (exitCode, result.TrimEnd()); + } + + public static bool DoubleWaitForExit(this Process process) + { + bool result = process.WaitForExit(500); + + if (result) + { + process.WaitForExit(); + } + + return result; + } + + public static bool FileExists(this string value) { return !string.IsNullOrEmpty(value) && !value.Contains("not found"); } + + public static (int exitCode, string result) Process( + string cmd, + string arguments, + string input = null, + bool waitForOutput = true + ) + { + var output = string.Empty; + + using (var process = new Process + { + StartInfo = new() + { + FileName = cmd, + Arguments = arguments, + RedirectStandardOutput = true, + RedirectStandardError = true, + RedirectStandardInput = true, + UseShellExecute = false, + CreateNoWindow = true + } + }) + { + TaskCompletionSource eventHandled = new(); + process.Start(); + + if (!string.IsNullOrEmpty(input)) + { + process.StandardInput.Write(input); + process.StandardInput.Close(); + } + + if (!process.WaitForExit(5000)) + { + var timeoutError = + $@"Process timed out. Command line: {process.StartInfo.FileName} {process.StartInfo.Arguments}."; + + throw new TimeoutException(timeoutError); + } + + if (waitForOutput && process.StandardOutput.Peek() != -1) + { + output = process.StandardOutput.ReadToEnd(); + } + + if (process.ExitCode > 0) + { + output = $@"Process failed to run. Command line: {cmd} {arguments}. + Output: {output} + Error: {process.StandardError.ReadToEnd()}"; + } + + return (process.ExitCode, output); + } + } + } +} \ No newline at end of file diff --git a/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs new file mode 100644 index 00000000..d72ac701 --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs @@ -0,0 +1,130 @@ +using System.Runtime.InteropServices; +using System.Threading.Tasks; +using Avalonia.Input; +using Avalonia.Input.Platform; + +namespace Consolonia.PlatformSupport.Clipboard +{ + + /// + /// A clipboard implementation for MacOSX. This implementation uses the Mac clipboard API (via P/Invoke) to + /// copy/paste. The existence of the Mac pbcopy and pbpaste commands is used to determine if copy/paste is supported. + /// + internal class MacOSXClipboard : IClipboard + { + private readonly nint _allocRegister = sel_registerName("alloc"); + private readonly nint _clearContentsRegister = sel_registerName("clearContents"); + private readonly nint _generalPasteboard; + private readonly nint _generalPasteboardRegister = sel_registerName("generalPasteboard"); + private readonly nint _initWithUtf8Register = sel_registerName("initWithUTF8String:"); + private readonly nint _nsPasteboard = objc_getClass("NSPasteboard"); + private readonly nint _nsString = objc_getClass("NSString"); + private readonly nint _nsStringPboardType; + private readonly nint _setStringRegister = sel_registerName("setString:forType:"); + private readonly nint _stringForTypeRegister = sel_registerName("stringForType:"); + private readonly nint _utf8Register = sel_registerName("UTF8String"); + private readonly nint _utfTextType; + + public MacOSXClipboard() + { + _utfTextType = objc_msgSend( + objc_msgSend(_nsString, _allocRegister), + _initWithUtf8Register, + "public.utf8-plain-text" + ); + + _nsStringPboardType = objc_msgSend( + objc_msgSend(_nsString, _allocRegister), + _initWithUtf8Register, + "NSStringPboardType" + ); + _generalPasteboard = objc_msgSend(_nsPasteboard, _generalPasteboardRegister); + if (!CheckSupport()) + throw new System.Exception("clipboard operations are not supported pbcopy and pbpaste are not available on this system."); + } + + private static bool CheckSupport() + { + (int exitCode, string result) = ClipboardProcessRunner.Bash("which pbcopy", waitForOutput: true); + + if (exitCode != 0 || !result.FileExists()) + { + return false; + } + + (exitCode, result) = ClipboardProcessRunner.Bash("which pbpaste", waitForOutput: true); + + return exitCode == 0 && result.FileExists(); + } + + [DllImport("/System/Library/Frameworks/AppKit.framework/AppKit", CharSet = CharSet.Unicode)] + private static extern nint objc_getClass(string className); + + [DllImport("/System/Library/Frameworks/AppKit.framework/AppKit")] + private static extern nint objc_msgSend(nint receiver, nint selector); + + [DllImport("/System/Library/Frameworks/AppKit.framework/AppKit", CharSet = CharSet.Unicode)] + private static extern nint objc_msgSend(nint receiver, nint selector, string arg1); + + [DllImport("/System/Library/Frameworks/AppKit.framework/AppKit")] + private static extern nint objc_msgSend(nint receiver, nint selector, nint arg1); + + [DllImport("/System/Library/Frameworks/AppKit.framework/AppKit")] + private static extern nint objc_msgSend(nint receiver, nint selector, nint arg1, nint arg2); + + [DllImport("/System/Library/Frameworks/AppKit.framework/AppKit", CharSet = CharSet.Unicode)] + private static extern nint sel_registerName(string selectorName); + + public async Task GetTextAsync() + { + await Task.CompletedTask; + + nint ptr = objc_msgSend(_generalPasteboard, _stringForTypeRegister, _nsStringPboardType); + nint charArray = objc_msgSend(ptr, _utf8Register); + + return Marshal.PtrToStringAnsi(charArray); + } + + public async Task SetTextAsync(string text) + { + await Task.CompletedTask; + + nint str = default; + + try + { + str = objc_msgSend(objc_msgSend(_nsString, _allocRegister), _initWithUtf8Register, text); + objc_msgSend(_generalPasteboard, _clearContentsRegister); + objc_msgSend(_generalPasteboard, _setStringRegister, str, _utfTextType); + } + finally + { + if (str != default(nint)) + { + objc_msgSend(str, sel_registerName("release")); + } + } + } + + public Task ClearAsync() + { + return SetTextAsync(string.Empty); + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new System.NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new System.NotImplementedException(); + } + + public Task GetDataAsync(string format) + { + throw new System.NotImplementedException(); + } + } + +} diff --git a/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs new file mode 100644 index 00000000..90dc225f --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs @@ -0,0 +1,98 @@ +using System; +using System.Threading.Tasks; +using Avalonia.Input; +using Avalonia.Input.Platform; + +namespace Consolonia.PlatformSupport.Clipboard +{ + + /// + /// A clipboard implementation for Linux, when running under WSL. This implementation uses the Windows clipboard + /// to store the data, and uses Windows' powershell.exe (launched via WSL interop services) to set/get the Windows + /// clipboard. + /// + internal class WSLClipboard : IClipboard + { + private static string _powershellPath = string.Empty; + + private bool _isSupported; + public WSLClipboard() + { + if (string.IsNullOrEmpty(_powershellPath)) + { + // Specify pwsh.exe (not pwsh) to ensure we get the Windows version (invoked via WSL) + (int exitCode, string result) = ClipboardProcessRunner.Bash("which pwsh.exe", waitForOutput: true); + + if (exitCode > 0) + { + (exitCode, result) = ClipboardProcessRunner.Bash("which powershell.exe", waitForOutput: true); + } + + if (exitCode == 0) + { + _powershellPath = result; + } + } + + _isSupported = !string.IsNullOrEmpty(_powershellPath); + } + + public async Task ClearAsync() + { + await SetTextAsync(string.Empty); + } + + public Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public async Task GetTextAsync() + { + if (!_isSupported) + { + return string.Empty; + } + + (int exitCode, string output) = + ClipboardProcessRunner.Process(_powershellPath, "-noprofile -command \"Get-Clipboard\""); + + if (exitCode == 0) + { + return output; + } + + return string.Empty; + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public async Task SetTextAsync(string text) + { + await Task.CompletedTask; + + if (!_isSupported) + { + return; + } + + (int exitCode, string output) = ClipboardProcessRunner.Process( + _powershellPath, + $"-noprofile -command \"Set-Clipboard -Value \\\"{text}\\\"\"" + ); + + if (exitCode != 0) + { + throw new InvalidOperationException($"Failed to set clipboard text: {output} using powershell"); + } + } + } +} diff --git a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs new file mode 100644 index 00000000..ba69900a --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs @@ -0,0 +1,52 @@ +using System; +using System.Threading.Tasks; +using Avalonia.Input; +using Avalonia.Input.Platform; + +namespace Consolonia.PlatformSupport.Clipboard +{ + + /// + /// A clipboard implementation for X11; + /// + internal class X11Clipboard : IClipboard + { + public X11Clipboard() + { + } + + public async Task ClearAsync() + { + await Task.CompletedTask; + Medo.X11.X11Clipboard.Clipboard.Clear(); + } + + public Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public async Task GetTextAsync() + { + await Task.CompletedTask; + return Medo.X11.X11Clipboard.Clipboard.GetText(); + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public async Task SetTextAsync(string text) + { + await Task.CompletedTask; + + Medo.X11.X11Clipboard.Clipboard.SetText(text); + } + } +} diff --git a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs new file mode 100644 index 00000000..2c5b0ce0 --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs @@ -0,0 +1,105 @@ +using System; +using System.IO; +using System.Threading.Tasks; +using Avalonia.Input; +using Avalonia.Input.Platform; +using Unix.Terminal; + +namespace Consolonia.PlatformSupport.Clipboard +{ + + /// A clipboard implementation for Linux. This implementation uses the xclip command to access the clipboard. + /// If xclip is not installed, this implementation will not work. + internal class XClipClipboard : IClipboard + { + private string _xclipPath = string.Empty; + private readonly bool _isSupported; + + public XClipClipboard() + { + (int exitCode, string result) = ClipboardProcessRunner.Bash("which xclip", waitForOutput: true); + + if (exitCode == 0 && result.FileExists()) + { + _xclipPath = result; + + _isSupported = true; + } + else + _isSupported = false; + } + + + public Task ClearAsync() + { + throw new NotImplementedException(); + } + + public Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public async Task GetTextAsync() + { + if (!_isSupported) + { + throw new NotSupportedException("xclip is not installed."); + } + + string tempFileName = Path.GetTempFileName(); + var xclipargs = "-selection clipboard -o"; + + try + { + (int exitCode, string result) = + ClipboardProcessRunner.Bash($"{_xclipPath} {xclipargs} > {tempFileName}", waitForOutput: false); + + if (exitCode == 0) + { + return File.ReadAllText(tempFileName); + } + } + catch (Exception e) + { + throw new NotSupportedException($"\"{_xclipPath} {xclipargs}\" failed.", e); + } + finally + { + File.Delete(tempFileName); + } + + return string.Empty; + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public async Task SetTextAsync(string text) + { + await Task.CompletedTask; + if (!_isSupported) + { + throw new NotSupportedException("xclip is not installed."); + } + + var xclipargs = "-selection clipboard -i"; + + try + { + (int exitCode, _) = ClipboardProcessRunner.Bash($"{_xclipPath} {xclipargs}", text); + } + catch (Exception e) + { + throw new NotSupportedException($"\"{_xclipPath} {xclipargs} < {text}\" failed", e); + } + } + } +} diff --git a/src/Consolonia.PlatformSupport/Consolonia.PlatformSupport.csproj b/src/Consolonia.PlatformSupport/Consolonia.PlatformSupport.csproj index dbe245e5..75d0967a 100644 --- a/src/Consolonia.PlatformSupport/Consolonia.PlatformSupport.csproj +++ b/src/Consolonia.PlatformSupport/Consolonia.PlatformSupport.csproj @@ -1,6 +1,7 @@ + diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 54ca8123..0afdf253 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -1,11 +1,14 @@ using System; +using System.Reflection; using Avalonia; using Avalonia.Controls; +using Avalonia.Input.Platform; using Consolonia.Core.Drawing.PixelBufferImplementation; using Consolonia.Core.Drawing.PixelBufferImplementation.EgaConsoleColor; using Consolonia.Core.Dummy; using Consolonia.Core.Infrastructure; using Consolonia.PlatformSupport; +using Consolonia.PlatformSupport.Clipboard; // ReSharper disable CheckNamespace #pragma warning disable IDE0161 @@ -34,7 +37,33 @@ public static AppBuilder UseAutoDetectedConsole(this AppBuilder builder) _ => new DefaultNetConsole() }; - return builder.UseConsole(console).UseAutoDetectConsoleColorMode(); + return builder.UseConsole(console) + .UseAutoDectectedClipboard() + .UseAutoDetectConsoleColorMode(); + } + + public static AppBuilder UseAutoDectectedClipboard(this AppBuilder builder) + { + if (OperatingSystem.IsWindows()) + { + // we can consume the avalonia clipboard implementation because it's self contained enough for us to reach inside and pull it out. + Assembly assembly = Assembly.Load("Avalonia.Win32"); + ArgumentNullException.ThrowIfNull(assembly, "Avalonia.Win32"); + Type type = assembly.GetType(assembly.GetName().Name + ".ClipboardImpl"); + ArgumentNullException.ThrowIfNull(type, "ClipboardImpl"); + var clipboard = Activator.CreateInstance(type) as IClipboard; + return builder.With(clipboard ?? new NaiveClipboard()); + } + else if (OperatingSystem.IsMacOS()) + { + return builder.With(new MacOSXClipboard()); + } + else if (OperatingSystem.IsLinux()) + { + return builder.With(new X11Clipboard()); + } + else + return builder.With(new NaiveClipboard()); } public static AppBuilder UseAutoDetectConsoleColorMode(this AppBuilder builder) From 407dc9d3115979047d01752d5558e98a567a90a8 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConenll Date: Thu, 9 Jan 2025 10:00:16 -0800 Subject: [PATCH 19/36] update for wsl --- .gitignore | 1 + .../PlatformSupportExtensions.cs | 33 ++++++++++++++----- 2 files changed, 26 insertions(+), 8 deletions(-) diff --git a/.gitignore b/.gitignore index 1cff3edc..d2892cdd 100644 --- a/.gitignore +++ b/.gitignore @@ -767,3 +767,4 @@ fabric.properties *.sln.iml /src/Tools/Consolonia.PreviewHost/Properties/launchSettings.json +src/.vscode/launch.json diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 0afdf253..2581a16f 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -60,11 +60,28 @@ public static AppBuilder UseAutoDectectedClipboard(this AppBuilder builder) } else if (OperatingSystem.IsLinux()) { - return builder.With(new X11Clipboard()); + if (IsWSLPlatform()) + return builder.With(new WSLClipboard()); + else + return builder.With(new XClipClipboard()); } else return builder.With(new NaiveClipboard()); } + + private static bool IsWSLPlatform() + { + // xclip does not work on WSL, so we need to use the Windows clipboard vis Powershell + (int exitCode, string result) = ClipboardProcessRunner.Bash("uname -a", waitForOutput: true); + + if (exitCode == 0 && result.Contains("microsoft") && result.Contains("WSL")) + { + return true; + } + + return false; + } + public static AppBuilder UseAutoDetectConsoleColorMode(this AppBuilder builder) { @@ -75,13 +92,13 @@ public static AppBuilder UseAutoDetectConsoleColorMode(this AppBuilder builder) switch (Environment.OSVersion.Platform) { case PlatformID.Win32S or PlatformID.Win32Windows or PlatformID.Win32NT: - { - // if output is redirected, or we are a windows terminal we use the win32 ANSI based console. - if (Console.IsOutputRedirected || IsWindowsTerminal()) - result = new RgbConsoleColorMode(); - else - result = new EgaConsoleColorMode(); - } + { + // if output is redirected, or we are a windows terminal we use the win32 ANSI based console. + if (Console.IsOutputRedirected || IsWindowsTerminal()) + result = new RgbConsoleColorMode(); + else + result = new EgaConsoleColorMode(); + } break; case PlatformID.MacOSX: result = new RgbConsoleColorMode(); From b6237cdcb6e6a7d88d953b7cd9785dd934085acf Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 10:10:09 -0800 Subject: [PATCH 20/36] use X11CLipboard on linux non-wsl --- src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 2581a16f..69b9a972 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -63,7 +63,8 @@ public static AppBuilder UseAutoDectectedClipboard(this AppBuilder builder) if (IsWSLPlatform()) return builder.With(new WSLClipboard()); else - return builder.With(new XClipClipboard()); + //return builder.With(new XClipClipboard()); + return builder.With(new X11Clipboard()); } else return builder.With(new NaiveClipboard()); From ce09026ab4da1da0b01da2d2a325c00a364564bc Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 10:16:21 -0800 Subject: [PATCH 21/36] cleanup --- .../Clipboard/ClipboardProcessRunner.cs | 19 +---------------- .../Clipboard/MacOSXClipboard.cs | 3 ++- .../Clipboard/WSLClipboard.cs | 1 + .../Clipboard/XClipClipboard.cs | 2 +- .../PlatformSupportExtensions.cs | 21 +++++++++++++++++-- 5 files changed, 24 insertions(+), 22 deletions(-) diff --git a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs index 044c0938..d60e0939 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs @@ -7,23 +7,6 @@ namespace Consolonia.PlatformSupport.Clipboard { - /// Provides cut, copy, and paste support for the OS clipboard. - /// - /// On Windows, the class uses the Windows Clipboard APIs via P/Invoke. - /// - /// On Linux, when not running under Windows Subsystem for Linux (WSL), the class uses - /// the xclip command line tool. If xclip is not installed, the clipboard will not work. - /// - /// - /// On Linux, when running under Windows Subsystem for Linux (WSL), the class launches - /// Windows' powershell.exe via WSL interop and uses the "Set-Clipboard" and "Get-Clipboard" Powershell CmdLets. - /// - /// - /// On the Mac, the class uses the MacO OS X pbcopy and pbpaste command line tools and - /// the Mac clipboard APIs vai P/Invoke. - /// - /// - /// /// Helper class for console drivers to invoke shell commands to interact with the clipboard. /// @@ -53,7 +36,7 @@ public static bool DoubleWaitForExit(this Process process) return result; } - public static bool FileExists(this string value) { return !string.IsNullOrEmpty(value) && !value.Contains("not found"); } + public static bool FileExists(this string value) { return !string.IsNullOrEmpty(value) && !value.Contains("not found", StringComparison.Ordinal); } public static (int exitCode, string result) Process( string cmd, diff --git a/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs index d72ac701..dbae2be2 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs @@ -1,3 +1,4 @@ +using System; using System.Runtime.InteropServices; using System.Threading.Tasks; using Avalonia.Input; @@ -40,7 +41,7 @@ public MacOSXClipboard() ); _generalPasteboard = objc_msgSend(_nsPasteboard, _generalPasteboardRegister); if (!CheckSupport()) - throw new System.Exception("clipboard operations are not supported pbcopy and pbpaste are not available on this system."); + throw new NotSupportedException("clipboard operations are not supported pbcopy and pbpaste are not available on this system."); } private static bool CheckSupport() diff --git a/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs index 90dc225f..8025d551 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs @@ -54,6 +54,7 @@ public Task GetFormatsAsync() public async Task GetTextAsync() { + await Task.CompletedTask; if (!_isSupported) { return string.Empty; diff --git a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs index 2c5b0ce0..257257b4 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs @@ -62,7 +62,7 @@ public async Task GetTextAsync() if (exitCode == 0) { - return File.ReadAllText(tempFileName); + return await File.ReadAllTextAsync(tempFileName); } } catch (Exception e) diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 69b9a972..65c1ef51 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -42,6 +42,23 @@ public static AppBuilder UseAutoDetectedConsole(this AppBuilder builder) .UseAutoDetectConsoleColorMode(); } + + /// Provides cut, copy, and paste support for the OS clipboard. + /// + /// On Windows, the class uses the Avalonia Windows Clipboard . + /// + /// On Linux, when not running under Windows Subsystem for Linux (WSL), the class X11 + /// PInvoke calls. + /// + /// + /// On Linux, when running under Windows Subsystem for Linux (WSL), the class launches + /// Windows' powershell.exe via WSL interop and uses the "Set-Clipboard" and "Get-Clipboard" Powershell CmdLets. + /// + /// + /// On the Mac, the class uses the MacO OS X pbcopy and pbpaste command line tools and + /// the Mac clipboard APIs vai P/Invoke. + /// + /// public static AppBuilder UseAutoDectectedClipboard(this AppBuilder builder) { if (OperatingSystem.IsWindows()) @@ -69,13 +86,13 @@ public static AppBuilder UseAutoDectectedClipboard(this AppBuilder builder) else return builder.With(new NaiveClipboard()); } - + private static bool IsWSLPlatform() { // xclip does not work on WSL, so we need to use the Windows clipboard vis Powershell (int exitCode, string result) = ClipboardProcessRunner.Bash("uname -a", waitForOutput: true); - if (exitCode == 0 && result.Contains("microsoft") && result.Contains("WSL")) + if (exitCode == 0 && result.Contains("microsoft", StringComparison.OrdinalIgnoreCase) && result.Contains("WSL", StringComparison.OrdinalIgnoreCase)) { return true; } From b0fe14207069095c7a12d88ebdba9c88c04f6785 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 10:33:02 -0800 Subject: [PATCH 22/36] cleanup --- .../Clipboard/MacOSXClipboard.cs | 11 +++--- .../Clipboard/WSLClipboard.cs | 34 ++++++++----------- .../Clipboard/X11Clipboard.cs | 15 ++++---- .../Clipboard/XClipClipboard.cs | 4 +-- .../PlatformSupportExtensions.cs | 1 + 5 files changed, 29 insertions(+), 36 deletions(-) diff --git a/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs index dbae2be2..ead3a6d5 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs @@ -76,20 +76,16 @@ private static bool CheckSupport() [DllImport("/System/Library/Frameworks/AppKit.framework/AppKit", CharSet = CharSet.Unicode)] private static extern nint sel_registerName(string selectorName); - public async Task GetTextAsync() + public Task GetTextAsync() { - await Task.CompletedTask; - nint ptr = objc_msgSend(_generalPasteboard, _stringForTypeRegister, _nsStringPboardType); nint charArray = objc_msgSend(ptr, _utf8Register); - return Marshal.PtrToStringAnsi(charArray); + return Task.FromResult(Marshal.PtrToStringAnsi(charArray)); } - public async Task SetTextAsync(string text) + public Task SetTextAsync(string text) { - await Task.CompletedTask; - nint str = default; try @@ -105,6 +101,7 @@ public async Task SetTextAsync(string text) objc_msgSend(str, sel_registerName("release")); } } + return Task.CompletedTask; } public Task ClearAsync() diff --git a/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs index 8025d551..3cad6067 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs @@ -52,12 +52,11 @@ public Task GetFormatsAsync() throw new NotImplementedException(); } - public async Task GetTextAsync() + public Task GetTextAsync() { - await Task.CompletedTask; if (!_isSupported) { - return string.Empty; + return Task.FromResult(string.Empty); } (int exitCode, string output) = @@ -65,10 +64,10 @@ public async Task GetTextAsync() if (exitCode == 0) { - return output; + return Task.FromResult(output); } - return string.Empty; + return Task.FromResult(string.Empty); } public Task SetDataObjectAsync(IDataObject data) @@ -76,24 +75,21 @@ public Task SetDataObjectAsync(IDataObject data) throw new NotImplementedException(); } - public async Task SetTextAsync(string text) + public Task SetTextAsync(string text) { - await Task.CompletedTask; - - if (!_isSupported) + if (_isSupported) { - return; - } - - (int exitCode, string output) = ClipboardProcessRunner.Process( - _powershellPath, - $"-noprofile -command \"Set-Clipboard -Value \\\"{text}\\\"\"" - ); + (int exitCode, string output) = ClipboardProcessRunner.Process( + _powershellPath, + $"-noprofile -command \"Set-Clipboard -Value \\\"{text}\\\"\"" + ); - if (exitCode != 0) - { - throw new InvalidOperationException($"Failed to set clipboard text: {output} using powershell"); + if (exitCode != 0) + { + throw new InvalidOperationException($"Failed to set clipboard text: {output} using powershell"); + } } + return Task.CompletedTask; } } } diff --git a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs index ba69900a..db3a05fb 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs @@ -15,10 +15,10 @@ public X11Clipboard() { } - public async Task ClearAsync() + public Task ClearAsync() { - await Task.CompletedTask; Medo.X11.X11Clipboard.Clipboard.Clear(); + return Task.CompletedTask; } public Task GetDataAsync(string format) @@ -31,10 +31,10 @@ public Task GetFormatsAsync() throw new NotImplementedException(); } - public async Task GetTextAsync() + public Task GetTextAsync() { - await Task.CompletedTask; - return Medo.X11.X11Clipboard.Clipboard.GetText(); + var text = Medo.X11.X11Clipboard.Clipboard.GetText(); + return Task.FromResult(text); } public Task SetDataObjectAsync(IDataObject data) @@ -42,11 +42,10 @@ public Task SetDataObjectAsync(IDataObject data) throw new NotImplementedException(); } - public async Task SetTextAsync(string text) + public Task SetTextAsync(string text) { - await Task.CompletedTask; - Medo.X11.X11Clipboard.Clipboard.SetText(text); + return Task.CompletedTask; } } } diff --git a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs index 257257b4..cbdb0341 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs @@ -82,9 +82,8 @@ public Task SetDataObjectAsync(IDataObject data) throw new NotImplementedException(); } - public async Task SetTextAsync(string text) + public Task SetTextAsync(string text) { - await Task.CompletedTask; if (!_isSupported) { throw new NotSupportedException("xclip is not installed."); @@ -100,6 +99,7 @@ public async Task SetTextAsync(string text) { throw new NotSupportedException($"\"{_xclipPath} {xclipargs} < {text}\" failed", e); } + return Task.CompletedTask; } } } diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 65c1ef51..012e5618 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -80,6 +80,7 @@ public static AppBuilder UseAutoDectectedClipboard(this AppBuilder builder) if (IsWSLPlatform()) return builder.With(new WSLClipboard()); else + // alternatively use xclip CLI tool //return builder.With(new XClipClipboard()); return builder.With(new X11Clipboard()); } From e2d8827d2079cb0be51be2ba63a4d020a99d567a Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 10:35:23 -0800 Subject: [PATCH 23/36] fix typo in method name --- src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 012e5618..00d713ce 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -38,7 +38,7 @@ public static AppBuilder UseAutoDetectedConsole(this AppBuilder builder) }; return builder.UseConsole(console) - .UseAutoDectectedClipboard() + .UseAutoDetectClipboard() .UseAutoDetectConsoleColorMode(); } @@ -59,7 +59,7 @@ public static AppBuilder UseAutoDetectedConsole(this AppBuilder builder) /// the Mac clipboard APIs vai P/Invoke. /// /// - public static AppBuilder UseAutoDectectedClipboard(this AppBuilder builder) + public static AppBuilder UseAutoDetectClipboard(this AppBuilder builder) { if (OperatingSystem.IsWindows()) { From 55e5c7df187b17c13dc017b242b76f529ae0f829 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 11:13:40 -0800 Subject: [PATCH 24/36] add scrollbars to gallery pages --- .../GalleryViews/GalleryAnimatedLines.axaml | 242 +++++++++--------- .../Gallery/GalleryViews/GalleryBorders.axaml | 188 +++++++------- .../Gallery/GalleryViews/GalleryButton.axaml | 48 ++-- .../GalleryViews/GalleryCalendar.axaml | 90 +++---- .../GalleryViews/GalleryCalendarPicker.axaml | 82 +++--- .../GalleryViews/GalleryProgressBar.axaml | 140 +++++----- .../GalleryViews/GalleryRadioButton.axaml | 114 +++++---- .../GalleryViews/GalleryTextBlock.axaml | 176 ++++++------- .../Gallery/GalleryViews/GalleryTextBox.axaml | 104 ++++---- .../View/ControlsListView.axaml | 9 +- .../Base/GalleryTestsBaseBase.cs | 2 +- .../TextBlockTests.cs | 5 +- 12 files changed, 609 insertions(+), 591 deletions(-) diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml index 74de6bf9..c59640fc 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml @@ -5,128 +5,130 @@ mc:Ignorable="d" xmlns:consolonia="clr-namespace:Consolonia.Controls;assembly=Consolonia.Core" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryAnimatedLines"> - - - - - - - + + + + + + + + - - - - - - - - + + + + + + + + - - - - - - - - - + + + + + + + + + - - - - - - - - - + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendar.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendar.axaml index 75e32dae..a6fc5871 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendar.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendar.axaml @@ -4,50 +4,52 @@ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryCalendar"> - + - A calendar control for selecting dates + A calendar control for selecting dates - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendarPicker.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendarPicker.axaml index aac64940..1c3e3519 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendarPicker.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryCalendarPicker.axaml @@ -12,47 +12,49 @@ Value="0,0,0,1" /> - + - A control for selecting dates with a calendar drop-down - - - - - - - - - - - - - - - - - - - - - - - - - - + A control for selecting dates with a calendar drop-down + + + + + + + + + + + + + + + + + + + + + + + + + + + - + \ No newline at end of file diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryProgressBar.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryProgressBar.axaml index cc18bd76..4e07c17e 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryProgressBar.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryProgressBar.axaml @@ -4,76 +4,78 @@ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryProgressBar"> - - - - - - - - - + + + + + + + + + + - - - - - - + + + + + + - - - - - + + + + + + - + \ No newline at end of file diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryRadioButton.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryRadioButton.axaml index 250af196..65e0a337 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryRadioButton.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryRadioButton.axaml @@ -4,62 +4,64 @@ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryRadioButton"> - - - _Option 1 - O_ption 2 - Op_tion 3 - Disabled - - - - Three States: Option 1 - - - Three States: Option 2 - - - Three States: Option 3 - - - Disabled - - - - - Group A: Option 1 - - - Group A: Disabled - - Group B: Option 1 - - Group B: Option 3 - - + - - Group A: Option 2 - - Group B: Option 2 - - Group B: Option 4 - + Spacing="1"> + + _Option 1 + O_ption 2 + Op_tion 3 + Disabled + + + + Three States: Option 1 + + + Three States: Option 2 + + + Three States: Option 3 + + + Disabled + + + + + Group A: Option 1 + + + Group A: Disabled + + Group B: Option 1 + + Group B: Option 3 + + + + + Group A: Option 2 + + Group B: Option 2 + + Group B: Option 4 + + - + \ No newline at end of file diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml index 93210d92..8849f367 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml @@ -13,96 +13,98 @@ + - - - - - Selectable text: Loreum ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + + - - - + Foreground="Chartreuse" /> + + Selectable text: Loreum ipsum dolor sit amet, consectetur adipiscing elit. Vivamus magna. Cras in mi at felis aliquet congue. Ut a est eget ligula molestie gravida. Curabitur massa. Donec eleifend, libero at sagittis + + + + + + + + + - + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml index 23d69486..fa3ea823 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml @@ -6,11 +6,12 @@ mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryTextBox"> - + - - - - - Custom context flyout - - - - - - - - - - - - - - - + + + + + - - - - - + + + + + - - + + - + + \ No newline at end of file diff --git a/src/Consolonia.Gallery/View/ControlsListView.axaml b/src/Consolonia.Gallery/View/ControlsListView.axaml index 3097a458..6350c69d 100644 --- a/src/Consolonia.Gallery/View/ControlsListView.axaml +++ b/src/Consolonia.Gallery/View/ControlsListView.axaml @@ -42,11 +42,10 @@ - - - + diff --git a/src/Tests/Consolonia.Gallery.Tests/Base/GalleryTestsBaseBase.cs b/src/Tests/Consolonia.Gallery.Tests/Base/GalleryTestsBaseBase.cs index 027154fb..d3c226d0 100644 --- a/src/Tests/Consolonia.Gallery.Tests/Base/GalleryTestsBaseBase.cs +++ b/src/Tests/Consolonia.Gallery.Tests/Base/GalleryTestsBaseBase.cs @@ -12,7 +12,7 @@ namespace Consolonia.Gallery.Tests.Base internal class GalleryTestsBaseBase : ConsoloniaAppTestBase { protected GalleryTestsBaseBase(PixelBufferSize size = default) : base(size.IsEmpty - ? new PixelBufferSize(80, 40) + ? new PixelBufferSize(81, 40) : size) { Args = new string[2]; diff --git a/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs b/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs index c10b137a..2680e664 100644 --- a/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs +++ b/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs @@ -43,8 +43,9 @@ public async Task HandlesMultilineText() { await UITest.KeyInput(Key.Tab); await UITest.AssertHasText( - "│elit. Vivamus magna. Cras in mi at felis aliquet │", - "│congue. Ut a est eget ligula molestie gravida. │"); + "│Lorem ipsum dolor sit amet, consectetur adipiscing elit.│", + "│Vivamus magna. Cras in mi at felis aliquet congue. Ut a │", + "│est eget ligula molestie gravida. Curabitur massa. Donec│"); } [Test] From b1e1ff20d99f5336f8a0383cfcbb3dffd71cca50 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 11:28:35 -0800 Subject: [PATCH 25/36] lint --- .../Infrastructure/ConsoloniaPlatform.cs | 1 - .../Clipboard/ClipboardProcessRunner.cs | 55 +++++++++---------- .../Clipboard/XClipClipboard.cs | 7 ++- .../Consolonia.Editor/MainWindow.axaml.cs | 9 --- .../Resources/ResourceLoader.cs | 1 - 5 files changed, 30 insertions(+), 43 deletions(-) diff --git a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs index acaa52a2..7eeb6dc9 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs @@ -1,6 +1,5 @@ using System; using System.Diagnostics; -using System.Reflection; using Avalonia; using Avalonia.Controls.Platform; using Avalonia.Input; diff --git a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs index d60e0939..887f925a 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs @@ -1,9 +1,5 @@ using System; -using System.Collections.Generic; using System.Diagnostics; -using System.Linq; -using System.Text; -using System.Threading.Tasks; namespace Consolonia.PlatformSupport.Clipboard { @@ -47,7 +43,7 @@ public static (int exitCode, string result) Process( { var output = string.Empty; - using (var process = new Process + using var process = new Process { StartInfo = new() { @@ -59,39 +55,38 @@ public static (int exitCode, string result) Process( UseShellExecute = false, CreateNoWindow = true } - }) - { - TaskCompletionSource eventHandled = new(); - process.Start(); + }; - if (!string.IsNullOrEmpty(input)) - { - process.StandardInput.Write(input); - process.StandardInput.Close(); - } + // TaskCompletionSource eventHandled = new(); + process.Start(); - if (!process.WaitForExit(5000)) - { - var timeoutError = - $@"Process timed out. Command line: {process.StartInfo.FileName} {process.StartInfo.Arguments}."; + if (!string.IsNullOrEmpty(input)) + { + process.StandardInput.Write(input); + process.StandardInput.Close(); + } - throw new TimeoutException(timeoutError); - } + if (!process.WaitForExit(5000)) + { + var timeoutError = + $@"Process timed out. Command line: {process.StartInfo.FileName} {process.StartInfo.Arguments}."; - if (waitForOutput && process.StandardOutput.Peek() != -1) - { - output = process.StandardOutput.ReadToEnd(); - } + throw new TimeoutException(timeoutError); + } - if (process.ExitCode > 0) - { - output = $@"Process failed to run. Command line: {cmd} {arguments}. + if (waitForOutput && process.StandardOutput.Peek() != -1) + { + output = process.StandardOutput.ReadToEnd(); + } + + if (process.ExitCode > 0) + { + output = $@"Process failed to run. Command line: {cmd} {arguments}. Output: {output} Error: {process.StandardError.ReadToEnd()}"; - } - - return (process.ExitCode, output); } + + return (process.ExitCode, output); } } } \ No newline at end of file diff --git a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs index cbdb0341..dc43794d 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs @@ -3,7 +3,6 @@ using System.Threading.Tasks; using Avalonia.Input; using Avalonia.Input.Platform; -using Unix.Terminal; namespace Consolonia.PlatformSupport.Clipboard { @@ -57,7 +56,7 @@ public async Task GetTextAsync() try { - (int exitCode, string result) = + (int exitCode, string _) = ClipboardProcessRunner.Bash($"{_xclipPath} {xclipargs} > {tempFileName}", waitForOutput: false); if (exitCode == 0) @@ -94,6 +93,10 @@ public Task SetTextAsync(string text) try { (int exitCode, _) = ClipboardProcessRunner.Bash($"{_xclipPath} {xclipargs}", text); + if (exitCode != 0) + { + throw new NotSupportedException($"\"{_xclipPath} {xclipargs} < {text}\" failed"); + } } catch (Exception e) { diff --git a/src/Experimental/Consolonia.Editor/MainWindow.axaml.cs b/src/Experimental/Consolonia.Editor/MainWindow.axaml.cs index 2ce0485f..243a6145 100644 --- a/src/Experimental/Consolonia.Editor/MainWindow.axaml.cs +++ b/src/Experimental/Consolonia.Editor/MainWindow.axaml.cs @@ -1,21 +1,14 @@ -using System.ComponentModel; -using System.Runtime.CompilerServices; using Avalonia; using Avalonia.Controls; using Avalonia.Input; using Avalonia.Interactivity; -using Avalonia.Markup.Xaml; using Avalonia.Media; using AvaloniaEdit.CodeCompletion; using ConsoloniaEdit.Demo.Resources; using AvaloniaEdit.Document; -using AvaloniaEdit.Editing; using AvaloniaEdit.Folding; -using AvaloniaEdit.Rendering; using AvaloniaEdit.TextMate; using TextMateSharp.Grammars; -using AvaloniaEdit.Snippets; -using Snippet = AvaloniaEdit.Snippets.Snippet; using ConsoloniaEdit.Demo.ViewModels; using AvaloniaEdit; @@ -23,8 +16,6 @@ // ReSharper disable UnusedMember.Local namespace ConsoloniaEdit.Demo { - using Pair = KeyValuePair; - public partial class MainWindow : Window { private readonly TextEditor _textEditor; diff --git a/src/Experimental/Consolonia.Editor/Resources/ResourceLoader.cs b/src/Experimental/Consolonia.Editor/Resources/ResourceLoader.cs index 5eb07096..c1d723f9 100644 --- a/src/Experimental/Consolonia.Editor/Resources/ResourceLoader.cs +++ b/src/Experimental/Consolonia.Editor/Resources/ResourceLoader.cs @@ -1,4 +1,3 @@ -using System.IO; using System.Reflection; namespace ConsoloniaEdit.Demo.Resources From a157136fc987bdf34969db4e18c7e59f26e56812 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 11:41:07 -0800 Subject: [PATCH 26/36] null checks --- src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs index db3a05fb..45e19f74 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs @@ -34,7 +34,7 @@ public Task GetFormatsAsync() public Task GetTextAsync() { var text = Medo.X11.X11Clipboard.Clipboard.GetText(); - return Task.FromResult(text); + return Task.FromResult(text ?? String.Empty); } public Task SetDataObjectAsync(IDataObject data) @@ -44,7 +44,7 @@ public Task SetDataObjectAsync(IDataObject data) public Task SetTextAsync(string text) { - Medo.X11.X11Clipboard.Clipboard.SetText(text); + Medo.X11.X11Clipboard.Clipboard.SetText(text ?? String.Empty); return Task.CompletedTask; } } From 40cc7ec50dd9dce541843f917de9acfd88899734 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 11:41:36 -0800 Subject: [PATCH 27/36] rename IsWsl --- src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 00d713ce..93400135 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -77,7 +77,7 @@ public static AppBuilder UseAutoDetectClipboard(this AppBuilder builder) } else if (OperatingSystem.IsLinux()) { - if (IsWSLPlatform()) + if (IsWslPlatform()) return builder.With(new WSLClipboard()); else // alternatively use xclip CLI tool @@ -88,7 +88,7 @@ public static AppBuilder UseAutoDetectClipboard(this AppBuilder builder) return builder.With(new NaiveClipboard()); } - private static bool IsWSLPlatform() + private static bool IsWslPlatform() { // xclip does not work on WSL, so we need to use the Windows clipboard vis Powershell (int exitCode, string result) = ClipboardProcessRunner.Bash("uname -a", waitForOutput: true); From 8aecf8ec66cfb529583f3f488fb0522fbb9fbef1 Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 11:45:06 -0800 Subject: [PATCH 28/36] lint --- .../Clipboard/ClipboardProcessRunner.cs | 21 +++++++++---------- .../{MacOSXClipboard.cs => MacClipboard.cs} | 4 ++-- .../{WSLClipboard.cs => WslClipboard.cs} | 4 ++-- .../PlatformSupportExtensions.cs | 13 ++++++------ 4 files changed, 20 insertions(+), 22 deletions(-) rename src/Consolonia.PlatformSupport/Clipboard/{MacOSXClipboard.cs => MacClipboard.cs} (98%) rename src/Consolonia.PlatformSupport/Clipboard/{WSLClipboard.cs => WslClipboard.cs} (97%) diff --git a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs index 887f925a..e8a5f2bd 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs @@ -43,18 +43,17 @@ public static (int exitCode, string result) Process( { var output = string.Empty; - using var process = new Process + using var process = new Process(); + + process.StartInfo = new() { - StartInfo = new() - { - FileName = cmd, - Arguments = arguments, - RedirectStandardOutput = true, - RedirectStandardError = true, - RedirectStandardInput = true, - UseShellExecute = false, - CreateNoWindow = true - } + FileName = cmd, + Arguments = arguments, + RedirectStandardOutput = true, + RedirectStandardError = true, + RedirectStandardInput = true, + UseShellExecute = false, + CreateNoWindow = true }; // TaskCompletionSource eventHandled = new(); diff --git a/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/MacClipboard.cs similarity index 98% rename from src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs rename to src/Consolonia.PlatformSupport/Clipboard/MacClipboard.cs index ead3a6d5..bf4e62da 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/MacOSXClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/MacClipboard.cs @@ -11,7 +11,7 @@ namespace Consolonia.PlatformSupport.Clipboard /// A clipboard implementation for MacOSX. This implementation uses the Mac clipboard API (via P/Invoke) to /// copy/paste. The existence of the Mac pbcopy and pbpaste commands is used to determine if copy/paste is supported. /// - internal class MacOSXClipboard : IClipboard + internal class MacClipboard : IClipboard { private readonly nint _allocRegister = sel_registerName("alloc"); private readonly nint _clearContentsRegister = sel_registerName("clearContents"); @@ -26,7 +26,7 @@ internal class MacOSXClipboard : IClipboard private readonly nint _utf8Register = sel_registerName("UTF8String"); private readonly nint _utfTextType; - public MacOSXClipboard() + public MacClipboard() { _utfTextType = objc_msgSend( objc_msgSend(_nsString, _allocRegister), diff --git a/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/WslClipboard.cs similarity index 97% rename from src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs rename to src/Consolonia.PlatformSupport/Clipboard/WslClipboard.cs index 3cad6067..93931822 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/WSLClipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/WslClipboard.cs @@ -11,12 +11,12 @@ namespace Consolonia.PlatformSupport.Clipboard /// to store the data, and uses Windows' powershell.exe (launched via WSL interop services) to set/get the Windows /// clipboard. /// - internal class WSLClipboard : IClipboard + internal class WslClipboard : IClipboard { private static string _powershellPath = string.Empty; private bool _isSupported; - public WSLClipboard() + public WslClipboard() { if (string.IsNullOrEmpty(_powershellPath)) { diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 93400135..801be47b 100644 --- a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs +++ b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs @@ -45,17 +45,16 @@ public static AppBuilder UseAutoDetectedConsole(this AppBuilder builder) /// Provides cut, copy, and paste support for the OS clipboard. /// - /// On Windows, the class uses the Avalonia Windows Clipboard . + /// On Windows, we use the Avalonia Windows Clipboard . /// - /// On Linux, when not running under Windows Subsystem for Linux (WSL), the class X11 - /// PInvoke calls. + /// On Linux, when not running under Windows Subsystem for Linux (WSL), we use X11Clipboard to call X11 PInvoke calls. /// /// - /// On Linux, when running under Windows Subsystem for Linux (WSL), the class launches + /// On Linux, when running under Windows Subsystem for Linux (WSL), we use WslClipboard class launches /// Windows' powershell.exe via WSL interop and uses the "Set-Clipboard" and "Get-Clipboard" Powershell CmdLets. /// /// - /// On the Mac, the class uses the MacO OS X pbcopy and pbpaste command line tools and + /// On the Mac, we use MacClipboard class which uses the MacOS X pbcopy and pbpaste command line tools and /// the Mac clipboard APIs vai P/Invoke. /// /// @@ -73,12 +72,12 @@ public static AppBuilder UseAutoDetectClipboard(this AppBuilder builder) } else if (OperatingSystem.IsMacOS()) { - return builder.With(new MacOSXClipboard()); + return builder.With(new MacClipboard()); } else if (OperatingSystem.IsLinux()) { if (IsWslPlatform()) - return builder.With(new WSLClipboard()); + return builder.With(new WslClipboard()); else // alternatively use xclip CLI tool //return builder.With(new XClipClipboard()); From 6188132d25040ecbd0d5bf2134c664de5373cecb Mon Sep 17 00:00:00 2001 From: Tom Laird-McConnell Date: Thu, 9 Jan 2025 12:05:49 -0800 Subject: [PATCH 29/36] lint lint lint lint lint lint --- src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs index 45e19f74..e07f94b8 100644 --- a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs +++ b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs @@ -34,7 +34,7 @@ public Task GetFormatsAsync() public Task GetTextAsync() { var text = Medo.X11.X11Clipboard.Clipboard.GetText(); - return Task.FromResult(text ?? String.Empty); + return Task.FromResult(text); } public Task SetDataObjectAsync(IDataObject data) From f953816d5507be7090c36c9d139a6a72b81ef7bd Mon Sep 17 00:00:00 2001 From: "github-actions[bot]" <41898282+github-actions[bot]@users.noreply.github.com> Date: Thu, 9 Jan 2025 20:11:02 +0000 Subject: [PATCH 30/36] Automated JetBrains cleanup Co-authored-by: <+@users.noreply.github.com> --- .../Infrastructure/ConsoloniaPlatform.cs | 1 - .../Infrastructure/NaiveClipboard.cs | 7 +- .../GalleryViews/GalleryAnimatedLines.axaml | 2 +- .../Gallery/GalleryViews/GalleryBorders.axaml | 4 +- .../Gallery/GalleryViews/GalleryButton.axaml | 4 +- .../GalleryViews/GalleryCalendar.axaml | 4 +- .../GalleryViews/GalleryCalendarPicker.axaml | 4 +- .../GalleryViews/GalleryProgressBar.axaml | 2 +- .../GalleryViews/GalleryRadioButton.axaml | 4 +- .../GalleryViews/GalleryTextBlock.axaml | 2 +- .../Gallery/GalleryViews/GalleryTextBox.axaml | 8 +- .../Clipboard/ClipboardProcessRunner.cs | 27 +++--- .../Clipboard/MacClipboard.cs | 96 +++++++++---------- .../Clipboard/WslClipboard.cs | 32 ++----- .../Clipboard/X11Clipboard.cs | 11 +-- .../Clipboard/XClipClipboard.cs | 34 +++---- .../PlatformSupportExtensions.cs | 43 ++++----- 17 files changed, 121 insertions(+), 164 deletions(-) diff --git a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs index 7eeb6dc9..b6e49eef 100644 --- a/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs +++ b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs @@ -64,7 +64,6 @@ public void Initialize() //.Bind().ToConstant(new PlatformSettingsStub()) //.Bind().ToConstant(new GtkSystemDialog()) /*.Bind().ToConstant(new LinuxMountedVolumeInfoProvider())*/; - } [DebuggerStepThrough] diff --git a/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs b/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs index 27d324f4..046c069a 100644 --- a/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs +++ b/src/Consolonia.Core/Infrastructure/NaiveClipboard.cs @@ -6,17 +6,16 @@ namespace Consolonia.Core.Infrastructure { /// - /// This clipboard only stores memory in the same process, so it is not useful for sharing data between processes. + /// This clipboard only stores memory in the same process, so it is not useful for sharing data between processes. /// public class NaiveClipboard : IClipboard { - private string _text = String.Empty; - + private string _text = string.Empty; #pragma warning disable CA1822 // Mark members as static public async Task ClearAsync() { await Task.CompletedTask; - _text = String.Empty; + _text = string.Empty; } public Task GetDataAsync(string format) diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml index c59640fc..9bf5c8bf 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml @@ -5,7 +5,7 @@ mc:Ignorable="d" xmlns:consolonia="clr-namespace:Consolonia.Controls;assembly=Consolonia.Core" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryAnimatedLines"> - + diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryBorders.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryBorders.axaml index 1ea20187..35f8e037 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryBorders.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryBorders.axaml @@ -5,9 +5,9 @@ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryBorders"> - + + Spacing="1"> diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryButton.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryButton.axaml index 2d989b8d..29e344e1 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryButton.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryButton.axaml @@ -4,9 +4,9 @@ xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006" mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryButton"> - + + Spacing="1"> - + + Spacing="1"> A control for selecting dates with a calendar drop-down - + - + + Spacing="1"> _Option 1 diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml index 8849f367..da232148 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBlock.axaml @@ -13,7 +13,7 @@ - + diff --git a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml index fa3ea823..6aaeba7e 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml @@ -6,11 +6,11 @@ mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryTextBox"> - + + HorizontalAlignment="Stretch" + Spacing="1" + Margin="1,0">