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.Core/Helpers/Extensions.cs b/src/Consolonia.Core/Helpers/UtilityExtensions.cs similarity index 99% rename from src/Consolonia.Core/Helpers/Extensions.cs rename to src/Consolonia.Core/Helpers/UtilityExtensions.cs index 0f9e1be8..12dbf6d1 100644 --- a/src/Consolonia.Core/Helpers/Extensions.cs +++ b/src/Consolonia.Core/Helpers/UtilityExtensions.cs @@ -13,7 +13,7 @@ namespace Consolonia.Core.Helpers { - public static class Extensions + public static class UtilityExtensions { public static IDisposable SubscribeAction( this IObservable> observable, 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/ConsoloniaPlatform.cs b/src/Consolonia.Core/Infrastructure/ConsoloniaPlatform.cs index 8909e8c9..b6e49eef 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; @@ -59,34 +58,12 @@ public void Initialize() .Bind().ToConstant(new DummyIconLoader()) .Bind().ToSingleton() .Bind().ToSingleton() + //.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 - } } [DebuggerStepThrough] 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/Infrastructure/InprocessClipboard.cs b/src/Consolonia.Core/Infrastructure/InprocessClipboard.cs new file mode 100644 index 00000000..559d243b --- /dev/null +++ b/src/Consolonia.Core/Infrastructure/InprocessClipboard.cs @@ -0,0 +1,47 @@ +using System; +using System.Threading.Tasks; +using Avalonia.Input; +using Avalonia.Input.Platform; + +namespace Consolonia.Core.Infrastructure +{ + /// + /// This clipboard only stores memory in the same process, so it is not useful for sharing data between processes. + /// + public class InprocessClipboard : 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 Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public Task GetTextAsync() + { + return Task.FromResult(_text); + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public Task SetTextAsync(string text) + { + _text = text; + return Task.CompletedTask; + } + } +} \ No newline at end of file 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.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryAnimatedLines.axaml index 74de6bf9..9bf5c8bf 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..d2cfb568 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..3b53782e 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..6d754771 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..e7bbd815 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 + + _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 + + - - - 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..da232148 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 e22930b4..6aaeba7e 100644 --- a/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml +++ b/src/Consolonia.Gallery/Gallery/GalleryViews/GalleryTextBox.axaml @@ -6,69 +6,72 @@ mc:Ignorable="d" x:Class="Consolonia.Gallery.Gallery.GalleryViews.GalleryTextBox"> - - + + + + + + + Custom context flyout + + + + + + + + + + - - - - - + + + + + - - - - - + + + + + - - + + - + + \ No newline at end of file diff --git a/src/Consolonia.GuiCS/WindowsDriver.cs b/src/Consolonia.GuiCS/WindowsDriver.cs index 2ac79248..4acba090 100644 --- a/src/Consolonia.GuiCS/WindowsDriver.cs +++ b/src/Consolonia.GuiCS/WindowsDriver.cs @@ -42,18 +42,17 @@ public CONSOLE_INPUT_MODE ConsoleMode } } + const int bufferSize = 0xffff; + 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, + if (!Kernel32.ReadConsoleInput(InputHandle, s_inputBuffer, bufferSize, out var numberEventsRead)) throw GetLastError().GetException(); - - return numberEventsRead == 0 - ? Array.Empty() - : records; + 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.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 diff --git a/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs new file mode 100644 index 00000000..2fadd967 --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/ClipboardProcessRunner.cs @@ -0,0 +1,79 @@ +using System; +using System.Diagnostics; +using System.Threading; + +namespace Consolonia.PlatformSupport.Clipboard +{ + /// + /// 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 + ) + { + string arguments = $"-c \"{commandLine}\""; + (int exitCode, string result) = Process("bash", arguments, inputText, waitForOutput); + + return (exitCode, result.TrimEnd()); + } + + 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, + string arguments, + string input = null, + bool waitForOutput = true, + int timeout = Timeout.Infinite + ) + { + string output = string.Empty; + + using var process = new Process(); + + process.StartInfo = new ProcessStartInfo + { + 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(timeout)) + { + string 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/MacClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/MacClipboard.cs new file mode 100644 index 00000000..65661bef --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/MacClipboard.cs @@ -0,0 +1,122 @@ +using System; +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 MacClipboard : 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 MacClipboard() + { + _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 NotSupportedException( + "clipboard operations are not supported pbcopy and pbpaste are not available on this system."); + } + + public Task GetTextAsync() + { + nint ptr = objc_msgSend(_generalPasteboard, _stringForTypeRegister, _nsStringPboardType); + nint charArray = objc_msgSend(ptr, _utf8Register); + + return Task.FromResult(Marshal.PtrToStringAnsi(charArray)); + } + + public Task SetTextAsync(string text) + { + 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) objc_msgSend(str, sel_registerName("release")); + } + + return Task.CompletedTask; + } + + public Task ClearAsync() + { + return SetTextAsync(string.Empty); + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + 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); + } +} \ No newline at end of file diff --git a/src/Consolonia.PlatformSupport/Clipboard/WslClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/WslClipboard.cs new file mode 100644 index 00000000..d1793752 --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/WslClipboard.cs @@ -0,0 +1,83 @@ +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 readonly 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 Task GetTextAsync() + { + if (!_isSupported) return Task.FromResult(string.Empty); + + (int exitCode, string output) = + ClipboardProcessRunner.Process(_powershellPath, "-noprofile -command \"Get-Clipboard\""); + + if (exitCode == 0) return Task.FromResult(output); + + return Task.FromResult(string.Empty); + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public Task SetTextAsync(string text) + { + if (_isSupported) + { + (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"); + } + + return Task.CompletedTask; + } + } +} \ No newline at end of file diff --git a/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs new file mode 100644 index 00000000..eb10a4a1 --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/X11Clipboard.cs @@ -0,0 +1,46 @@ +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 Task ClearAsync() + { + Medo.X11.X11Clipboard.Clipboard.Clear(); + return Task.CompletedTask; + } + + public Task GetDataAsync(string format) + { + throw new NotImplementedException(); + } + + public Task GetFormatsAsync() + { + throw new NotImplementedException(); + } + + public Task GetTextAsync() + { + string text = Medo.X11.X11Clipboard.Clipboard.GetText(); + return Task.FromResult(text); + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public Task SetTextAsync(string text) + { + Medo.X11.X11Clipboard.Clipboard.SetText(text ?? string.Empty); + return Task.CompletedTask; + } + } +} \ No newline at end of file diff --git a/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs new file mode 100644 index 00000000..ff826c1c --- /dev/null +++ b/src/Consolonia.PlatformSupport/Clipboard/XClipClipboard.cs @@ -0,0 +1,87 @@ +using System; +using System.IO; +using System.Threading.Tasks; +using Avalonia.Input; +using Avalonia.Input.Platform; + +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 readonly bool _isSupported; + private readonly string _xclipPath = string.Empty; + + 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(); + string xclipargs = "-selection clipboard -o"; + + try + { + (int exitCode, string _) = + ClipboardProcessRunner.Bash($"{_xclipPath} {xclipargs} > {tempFileName}", waitForOutput: false); + + if (exitCode == 0) return await File.ReadAllTextAsync(tempFileName); + else + throw new NotSupportedException($"\"{_xclipPath} {xclipargs}\" failed. {exitCode}"); + } + finally + { + File.Delete(tempFileName); + } + } + + public Task SetDataObjectAsync(IDataObject data) + { + throw new NotImplementedException(); + } + + public Task SetTextAsync(string text) + { + if (!_isSupported) throw new NotSupportedException("xclip is not installed."); + + string xclipargs = "-selection clipboard -i"; + + (int exitCode, _) = ClipboardProcessRunner.Bash($"{_xclipPath} {xclipargs}", text); + if (exitCode != 0) throw new NotSupportedException($"\"{_xclipPath} {xclipargs} < {text}\" failed"); + + return Task.CompletedTask; + } + } +} \ No newline at end of file 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/CursesConsole.cs b/src/Consolonia.PlatformSupport/CursesConsole.cs index 6cae63dd..f75c34cd 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; @@ -206,6 +207,15 @@ private void ProcessInput() wch -= 60; k = Key.ShiftMask | Key.AltMask | MapCursesKey(wch); break; + case >= 523 and <= 570: + // Ctrl/Shift/Alt and navigation keys (arrow, home, end) + string distro = Environment.GetEnvironmentVariable("WSL_DISTRO_NAME"); + if (!string.IsNullOrEmpty(distro)) + wch -= 1; + else + wch -= 9; + k = MapCursesKey(wch); // has appropriate XxxMask internal + break; } RaiseKeyPressInternal(k); @@ -327,6 +337,21 @@ private void ProcessInput() // 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: + 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) + { + string text = bufferText[..--index]; + RaiseTextInput(text, (ulong)Stopwatch.GetTimestamp()); + } + + break; + default: k = MapCursesKey(wch2); break; diff --git a/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs b/src/Consolonia.PlatformSupport/PlatformSupportExtensions.cs index 54ca8123..c0adc639 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,9 +37,67 @@ public static AppBuilder UseAutoDetectedConsole(this AppBuilder builder) _ => new DefaultNetConsole() }; - return builder.UseConsole(console).UseAutoDetectConsoleColorMode(); + return builder.UseConsole(console) + .UseAutoDetectClipboard() + .UseAutoDetectConsoleColorMode(); } + + /// Provides cut, copy, and paste support for the OS clipboard. + /// + /// On Windows, we use the Avalonia Windows Clipboard . + /// + /// 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), we use WslClipboard class launches + /// Windows' powershell.exe via WSL interop and uses the "Set-Clipboard" and "Get-Clipboard" Powershell CmdLets. + /// + /// + /// 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. + /// + /// + public static AppBuilder UseAutoDetectClipboard(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 InprocessClipboard()); + } + + if (OperatingSystem.IsMacOS()) return builder.With(new MacClipboard()); + + if (OperatingSystem.IsLinux()) + { + if (IsWslPlatform()) + return builder.With(new WslClipboard()); + // alternatively use xclip CLI tool + //return builder.With(new XClipClipboard()); + return builder.With(new X11Clipboard()); + } + + return builder.With(new InprocessClipboard()); + } + + 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", StringComparison.OrdinalIgnoreCase) && + result.Contains("WSL", StringComparison.OrdinalIgnoreCase)) return true; + + return false; + } + + public static AppBuilder UseAutoDetectConsoleColorMode(this AppBuilder builder) { IConsoleColorMode result; diff --git a/src/Consolonia.PlatformSupport/WindowsConsole.cs b/src/Consolonia.PlatformSupport/Win32Console.cs similarity index 64% rename from src/Consolonia.PlatformSupport/WindowsConsole.cs rename to src/Consolonia.PlatformSupport/Win32Console.cs index cc99514c..4a6667e4 100644 --- a/src/Consolonia.PlatformSupport/WindowsConsole.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; @@ -111,37 +114,123 @@ public override void PauseIO(Task task) private void StartEventLoop() { - Task.Run(() => + Task.Run(async () => { while (!Disposed /*inject ThreadAbortException*/) { PauseTask?.Wait(); - var readConsoleInput = _windowsConsole.ReadConsoleInput(); - if (!readConsoleInput.Any()) - throw new NotImplementedException(); - foreach (INPUT_RECORD inputRecord in readConsoleInput) - // ReSharper disable once SwitchStatementMissingSomeEnumCasesNoDefault - switch (inputRecord.EventType) + var inputRecords = _windowsConsole.ReadConsoleInput(); + 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. + /// + /// + /// + /// + private async Task ProcessClipboardInput(IClipboard clipboard, INPUT_RECORD[] inputRecords) + { + string clipboardText = await clipboard?.GetTextAsync() ?? string.Empty; + if (clipboardText.Trim().Length == 0) + { + // no text in clipboard, just process input records + 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); + var bufferText = new StringBuilder(); + List bufferedKeyEvents = new(); + + while (inputRecords.Any()) + { + // process all input records + for (int i = 0; i < inputRecords.Length; i++) + { + INPUT_RECORD 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) { - 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; + // append the char to the buffer text + bufferText.Append(inputRecord.Event.KeyEvent.uChar); + + string currentBufferText = bufferText.ToString(); + 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); + RaiseTextInput(currentBufferText, (ulong)Stopwatch.GetTimestamp()); + + // process remaining input records + for (++i; i < inputRecords.Length; i++) + HandleInputRecord(inputRecords[i]); + return; + } + + if (!clipboardText.StartsWith(currentBufferText, StringComparison.Ordinal)) + { + // buffered text doesn't match clipboard, emit buffered key events (we already played other events live) + foreach (INPUT_RECORD 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 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 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 diff --git a/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs b/src/Tests/Consolonia.Gallery.Tests/TextBlockTests.cs index 86ecf972..e86e628b 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( - "β”‚Vivamus magna. Cras in mi at felis aliquet congue. Ut a β”‚", - "β”‚est eget ligula molestie gravida. Curabitur massa. Donecβ”‚"); + "β”‚Lorem ipsum dolor sit amet, consectetur adipiscing β”‚", + "β”‚elit. Vivamus magna. Cras in mi at felis aliquet β”‚", + "β”‚congue. Ut a est eget ligula molestie gravida. β”‚"); } [Test]