cmdpal: Re-re-enable the clipboard history (#40471)

_⚠️ targets #40445_

This time, for real

This really really re-enables the clipboard history command. With the
foreground fixes from #40445, we can properly dismiss ourself to give FG
to the next app window. This actually lets us paste correctly.


I took the liberty of localizing the strings and fixing up the icons
while I was at it.

Closes #38344
This commit is contained in:
Mike Griese
2025-07-09 15:42:46 -05:00
committed by GitHub
parent 100fca4468
commit ee764d5f56
11 changed files with 346 additions and 17 deletions

View File

@@ -8,6 +8,7 @@ using Microsoft.CmdPal.Common.Services;
using Microsoft.CmdPal.Ext.Apps; using Microsoft.CmdPal.Ext.Apps;
using Microsoft.CmdPal.Ext.Bookmarks; using Microsoft.CmdPal.Ext.Bookmarks;
using Microsoft.CmdPal.Ext.Calc; using Microsoft.CmdPal.Ext.Calc;
using Microsoft.CmdPal.Ext.ClipboardHistory;
using Microsoft.CmdPal.Ext.Indexer; using Microsoft.CmdPal.Ext.Indexer;
using Microsoft.CmdPal.Ext.Registry; using Microsoft.CmdPal.Ext.Registry;
using Microsoft.CmdPal.Ext.Shell; using Microsoft.CmdPal.Ext.Shell;
@@ -104,6 +105,7 @@ public partial class App : Application
services.AddSingleton<ICommandProvider, WindowWalkerCommandsProvider>(); services.AddSingleton<ICommandProvider, WindowWalkerCommandsProvider>();
services.AddSingleton<ICommandProvider, WebSearchCommandsProvider>(); services.AddSingleton<ICommandProvider, WebSearchCommandsProvider>();
services.AddSingleton<ICommandProvider, ClipboardHistoryCommandsProvider>();
// GH #38440: Users might not have WinGet installed! Or they might have // GH #38440: Users might not have WinGet installed! Or they might have
// a ridiculously old version. Or might be running as admin. // a ridiculously old version. Or might be running as admin.

View File

@@ -107,6 +107,7 @@
<ItemGroup> <ItemGroup>
<ProjectReference Include="..\..\..\common\ManagedTelemetry\Telemetry\ManagedTelemetry.csproj" /> <ProjectReference Include="..\..\..\common\ManagedTelemetry\Telemetry\ManagedTelemetry.csproj" />
<ProjectReference Include="..\ext\Microsoft.CmdPal.Ext.ClipboardHistory\Microsoft.CmdPal.Ext.ClipboardHistory.csproj" />
<ProjectReference Include="..\ext\Microsoft.CmdPal.Ext.System\Microsoft.CmdPal.Ext.System.csproj" /> <ProjectReference Include="..\ext\Microsoft.CmdPal.Ext.System\Microsoft.CmdPal.Ext.System.csproj" />
<ProjectReference Include="..\ext\Microsoft.CmdPal.Ext.WebSearch\Microsoft.CmdPal.Ext.WebSearch.csproj" /> <ProjectReference Include="..\ext\Microsoft.CmdPal.Ext.WebSearch\Microsoft.CmdPal.Ext.WebSearch.csproj" />
<ProjectReference Include="..\ext\Microsoft.CmdPal.Ext.Indexer\Microsoft.CmdPal.Ext.Indexer.csproj" /> <ProjectReference Include="..\ext\Microsoft.CmdPal.Ext.Indexer\Microsoft.CmdPal.Ext.Indexer.csproj" />

View File

@@ -16,12 +16,13 @@ public partial class ClipboardHistoryCommandsProvider : CommandProvider
{ {
_clipboardHistoryListItem = new ListItem(new ClipboardHistoryListPage()) _clipboardHistoryListItem = new ListItem(new ClipboardHistoryListPage())
{ {
Title = "Search Clipboard History", Title = Properties.Resources.list_item_title,
Icon = new IconInfo("\xE8C8"), // Copy icon Subtitle = Properties.Resources.list_item_subtitle,
Icon = Icons.ClipboardList,
}; };
DisplayName = $"Clipboard History"; DisplayName = Properties.Resources.provider_display_name;
Icon = new IconInfo("\xE8C8"); // Copy icon Icon = Icons.ClipboardList;
Id = "Windows.ClipboardHistory"; Id = "Windows.ClipboardHistory";
} }

View File

@@ -16,20 +16,20 @@ internal sealed partial class CopyCommand : InvokableCommand
{ {
_clipboardItem = clipboardItem; _clipboardItem = clipboardItem;
_clipboardFormat = clipboardFormat; _clipboardFormat = clipboardFormat;
Name = "Copy"; Name = Properties.Resources.copy_command_name;
if (clipboardFormat == ClipboardFormat.Text) if (clipboardFormat == ClipboardFormat.Text)
{ {
Icon = new("\xE8C8"); // Copy icon Icon = Icons.Copy;
} }
else else
{ {
Icon = new("\xE8B9"); // Picture icon Icon = Icons.Picture;
} }
} }
public override CommandResult Invoke() public override CommandResult Invoke()
{ {
ClipboardHelper.SetClipboardContent(_clipboardItem, _clipboardFormat); ClipboardHelper.SetClipboardContent(_clipboardItem, _clipboardFormat);
return CommandResult.ShowToast("Copied to clipboard"); return CommandResult.ShowToast(Properties.Resources.copied_toast_text);
} }
} }

View File

@@ -6,7 +6,6 @@ using CommunityToolkit.Mvvm.Messaging;
using Microsoft.CmdPal.Common.Messages; using Microsoft.CmdPal.Common.Messages;
using Microsoft.CmdPal.Ext.ClipboardHistory.Models; using Microsoft.CmdPal.Ext.ClipboardHistory.Models;
using Microsoft.CommandPalette.Extensions.Toolkit; using Microsoft.CommandPalette.Extensions.Toolkit;
using Windows.ApplicationModel.DataTransfer; using Windows.ApplicationModel.DataTransfer;
namespace Microsoft.CmdPal.Ext.ClipboardHistory.Commands; namespace Microsoft.CmdPal.Ext.ClipboardHistory.Commands;
@@ -20,8 +19,8 @@ internal sealed partial class PasteCommand : InvokableCommand
{ {
_clipboardItem = clipboardItem; _clipboardItem = clipboardItem;
_clipboardFormat = clipboardFormat; _clipboardFormat = clipboardFormat;
Name = "Paste"; Name = Properties.Resources.paste_command_name;
Icon = new("\xE8C8"); // Copy icon Icon = Icons.Paste;
} }
private void HideWindow() private void HideWindow()
@@ -37,8 +36,10 @@ internal sealed partial class PasteCommand : InvokableCommand
{ {
ClipboardHelper.SetClipboardContent(_clipboardItem, _clipboardFormat); ClipboardHelper.SetClipboardContent(_clipboardItem, _clipboardFormat);
HideWindow(); HideWindow();
ClipboardHelper.SendPasteKeyCombination(); ClipboardHelper.SendPasteKeyCombination();
Clipboard.DeleteItemFromHistory(_clipboardItem.Item); Clipboard.DeleteItemFromHistory(_clipboardItem.Item);
return CommandResult.ShowToast("Pasting"); return CommandResult.ShowToast(Properties.Resources.paste_toast_text);
} }
} }

View File

@@ -59,10 +59,10 @@ internal static class ClipboardHelper
output.SetText(text); output.SetText(text);
try try
{ {
// Clipboard.SetContentWithOptions(output, null);
ClipboardThreadQueue.EnqueueTask(() => ClipboardThreadQueue.EnqueueTask(() =>
{ {
Clipboard.SetContent(output); Clipboard.SetContent(output);
Flush(); Flush();
ExtensionHost.LogMessage(new LogMessage() { Message = "Copied text to clipboard" }); ExtensionHost.LogMessage(new LogMessage() { Message = "Copied text to clipboard" });
}); });
@@ -87,7 +87,7 @@ internal static class ClipboardHelper
{ {
try try
{ {
Task.Run(Clipboard.Flush).Wait(); Clipboard.Flush();
return; return;
} }
catch (Exception ex) catch (Exception ex)

View File

@@ -0,0 +1,18 @@
// Copyright (c) Microsoft Corporation
// The Microsoft Corporation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
using Microsoft.CommandPalette.Extensions.Toolkit;
namespace Microsoft.CmdPal.Ext.ClipboardHistory;
internal sealed class Icons
{
internal static IconInfo Copy { get; } = new("\xE8C8");
internal static IconInfo Picture { get; } = new("\xE8B9");
internal static IconInfo Paste { get; } = new("\uE77F");
internal static IconInfo ClipboardList { get; } = new("\uF0E3");
}

View File

@@ -15,4 +15,19 @@
<ItemGroup> <ItemGroup>
<PackageReference Include="CommunityToolkit.Mvvm" /> <PackageReference Include="CommunityToolkit.Mvvm" />
</ItemGroup> </ItemGroup>
<!-- String resources -->
<ItemGroup>
<Compile Update="Properties\Resources.Designer.cs">
<DependentUpon>Resources.resx</DependentUpon>
<DesignTime>True</DesignTime>
<AutoGen>True</AutoGen>
</Compile>
</ItemGroup>
<ItemGroup>
<EmbeddedResource Update="Properties\Resources.resx">
<LastGenOutput>Resources.Designer.cs</LastGenOutput>
<Generator>PublicResXFileCodeGenerator</Generator>
</EmbeddedResource>
</ItemGroup>
</Project> </Project>

View File

@@ -24,8 +24,8 @@ internal sealed partial class ClipboardHistoryListPage : ListPage
{ {
clipboardHistory = []; clipboardHistory = [];
_defaultIconPath = string.Empty; _defaultIconPath = string.Empty;
Icon = new("\uF0E3"); // ClipboardList icon Icon = Icons.ClipboardList;
Name = "Clipboard History"; Name = Properties.Resources.clipboard_history_page_name;
Id = "com.microsoft.cmdpal.clipboardHistory"; Id = "com.microsoft.cmdpal.clipboardHistory";
ShowDetails = true; ShowDetails = true;
@@ -113,7 +113,7 @@ internal sealed partial class ClipboardHistoryListPage : ListPage
{ {
// TODO GH #108 We need to figure out some logging // TODO GH #108 We need to figure out some logging
// Logger.LogError("Loading clipboard history failed", ex); // Logger.LogError("Loading clipboard history failed", ex);
ExtensionHost.ShowStatus(new StatusMessage() { Message = "Loading clipboard history failed", State = MessageState.Error }, StatusContext.Page); ExtensionHost.ShowStatus(new StatusMessage() { Message = Properties.Resources.clipboard_failed_to_load, State = MessageState.Error }, StatusContext.Page);
ExtensionHost.LogMessage(ex.ToString()); ExtensionHost.LogMessage(ex.ToString());
} }
} }

View File

@@ -0,0 +1,144 @@
//------------------------------------------------------------------------------
// <auto-generated>
// This code was generated by a tool.
// Runtime Version:4.0.30319.42000
//
// Changes to this file may cause incorrect behavior and will be lost if
// the code is regenerated.
// </auto-generated>
//------------------------------------------------------------------------------
namespace Microsoft.CmdPal.Ext.ClipboardHistory.Properties {
using System;
/// <summary>
/// A strongly-typed resource class, for looking up localized strings, etc.
/// </summary>
// This class was auto-generated by the StronglyTypedResourceBuilder
// class via a tool like ResGen or Visual Studio.
// To add or remove a member, edit your .ResX file then rerun ResGen
// with the /str option, or rebuild your VS project.
[global::System.CodeDom.Compiler.GeneratedCodeAttribute("System.Resources.Tools.StronglyTypedResourceBuilder", "17.0.0.0")]
[global::System.Diagnostics.DebuggerNonUserCodeAttribute()]
[global::System.Runtime.CompilerServices.CompilerGeneratedAttribute()]
public class Resources {
private static global::System.Resources.ResourceManager resourceMan;
private static global::System.Globalization.CultureInfo resourceCulture;
[global::System.Diagnostics.CodeAnalysis.SuppressMessageAttribute("Microsoft.Performance", "CA1811:AvoidUncalledPrivateCode")]
internal Resources() {
}
/// <summary>
/// Returns the cached ResourceManager instance used by this class.
/// </summary>
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
public static global::System.Resources.ResourceManager ResourceManager {
get {
if (object.ReferenceEquals(resourceMan, null)) {
global::System.Resources.ResourceManager temp = new global::System.Resources.ResourceManager("Microsoft.CmdPal.Ext.ClipboardHistory.Properties.Resources", typeof(Resources).Assembly);
resourceMan = temp;
}
return resourceMan;
}
}
/// <summary>
/// Overrides the current thread's CurrentUICulture property for all
/// resource lookups using this strongly typed resource class.
/// </summary>
[global::System.ComponentModel.EditorBrowsableAttribute(global::System.ComponentModel.EditorBrowsableState.Advanced)]
public static global::System.Globalization.CultureInfo Culture {
get {
return resourceCulture;
}
set {
resourceCulture = value;
}
}
/// <summary>
/// Looks up a localized string similar to Loading clipboard history failed.
/// </summary>
public static string clipboard_failed_to_load {
get {
return ResourceManager.GetString("clipboard_failed_to_load", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Open.
/// </summary>
public static string clipboard_history_page_name {
get {
return ResourceManager.GetString("clipboard_history_page_name", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Copied to clipboard.
/// </summary>
public static string copied_toast_text {
get {
return ResourceManager.GetString("copied_toast_text", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Copy.
/// </summary>
public static string copy_command_name {
get {
return ResourceManager.GetString("copy_command_name", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Copy, paste, and search items on the clipboard.
/// </summary>
public static string list_item_subtitle {
get {
return ResourceManager.GetString("list_item_subtitle", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Clipboard History.
/// </summary>
public static string list_item_title {
get {
return ResourceManager.GetString("list_item_title", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Paste.
/// </summary>
public static string paste_command_name {
get {
return ResourceManager.GetString("paste_command_name", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Pasting.
/// </summary>
public static string paste_toast_text {
get {
return ResourceManager.GetString("paste_toast_text", resourceCulture);
}
}
/// <summary>
/// Looks up a localized string similar to Clipboard History.
/// </summary>
public static string provider_display_name {
get {
return ResourceManager.GetString("provider_display_name", resourceCulture);
}
}
}
}

View File

@@ -0,0 +1,147 @@
<?xml version="1.0" encoding="utf-8"?>
<root>
<!--
Microsoft ResX Schema
Version 2.0
The primary goals of this format is to allow a simple XML format
that is mostly human readable. The generation and parsing of the
various data types are done through the TypeConverter classes
associated with the data types.
Example:
... ado.net/XML headers & schema ...
<resheader name="resmimetype">text/microsoft-resx</resheader>
<resheader name="version">2.0</resheader>
<resheader name="reader">System.Resources.ResXResourceReader, System.Windows.Forms, ...</resheader>
<resheader name="writer">System.Resources.ResXResourceWriter, System.Windows.Forms, ...</resheader>
<data name="Name1"><value>this is my long string</value><comment>this is a comment</comment></data>
<data name="Color1" type="System.Drawing.Color, System.Drawing">Blue</data>
<data name="Bitmap1" mimetype="application/x-microsoft.net.object.binary.base64">
<value>[base64 mime encoded serialized .NET Framework object]</value>
</data>
<data name="Icon1" type="System.Drawing.Icon, System.Drawing" mimetype="application/x-microsoft.net.object.bytearray.base64">
<value>[base64 mime encoded string representing a byte array form of the .NET Framework object]</value>
<comment>This is a comment</comment>
</data>
There are any number of "resheader" rows that contain simple
name/value pairs.
Each data row contains a name, and value. The row also contains a
type or mimetype. Type corresponds to a .NET class that support
text/value conversion through the TypeConverter architecture.
Classes that don't support this are serialized and stored with the
mimetype set.
The mimetype is used for serialized objects, and tells the
ResXResourceReader how to depersist the object. This is currently not
extensible. For a given mimetype the value must be set accordingly:
Note - application/x-microsoft.net.object.binary.base64 is the format
that the ResXResourceWriter will generate, however the reader can
read any of the formats listed below.
mimetype: application/x-microsoft.net.object.binary.base64
value : The object must be serialized with
: System.Runtime.Serialization.Formatters.Binary.BinaryFormatter
: and then encoded with base64 encoding.
mimetype: application/x-microsoft.net.object.soap.base64
value : The object must be serialized with
: System.Runtime.Serialization.Formatters.Soap.SoapFormatter
: and then encoded with base64 encoding.
mimetype: application/x-microsoft.net.object.bytearray.base64
value : The object must be serialized into a byte array
: using a System.ComponentModel.TypeConverter
: and then encoded with base64 encoding.
-->
<xsd:schema id="root" xmlns="" xmlns:xsd="http://www.w3.org/2001/XMLSchema" xmlns:msdata="urn:schemas-microsoft-com:xml-msdata">
<xsd:import namespace="http://www.w3.org/XML/1998/namespace" />
<xsd:element name="root" msdata:IsDataSet="true">
<xsd:complexType>
<xsd:choice maxOccurs="unbounded">
<xsd:element name="metadata">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" />
</xsd:sequence>
<xsd:attribute name="name" use="required" type="xsd:string" />
<xsd:attribute name="type" type="xsd:string" />
<xsd:attribute name="mimetype" type="xsd:string" />
<xsd:attribute ref="xml:space" />
</xsd:complexType>
</xsd:element>
<xsd:element name="assembly">
<xsd:complexType>
<xsd:attribute name="alias" type="xsd:string" />
<xsd:attribute name="name" type="xsd:string" />
</xsd:complexType>
</xsd:element>
<xsd:element name="data">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
<xsd:element name="comment" type="xsd:string" minOccurs="0" msdata:Ordinal="2" />
</xsd:sequence>
<xsd:attribute name="name" type="xsd:string" use="required" msdata:Ordinal="1" />
<xsd:attribute name="type" type="xsd:string" msdata:Ordinal="3" />
<xsd:attribute name="mimetype" type="xsd:string" msdata:Ordinal="4" />
<xsd:attribute ref="xml:space" />
</xsd:complexType>
</xsd:element>
<xsd:element name="resheader">
<xsd:complexType>
<xsd:sequence>
<xsd:element name="value" type="xsd:string" minOccurs="0" msdata:Ordinal="1" />
</xsd:sequence>
<xsd:attribute name="name" type="xsd:string" use="required" />
</xsd:complexType>
</xsd:element>
</xsd:choice>
</xsd:complexType>
</xsd:element>
</xsd:schema>
<resheader name="resmimetype">
<value>text/microsoft-resx</value>
</resheader>
<resheader name="version">
<value>2.0</value>
</resheader>
<resheader name="reader">
<value>System.Resources.ResXResourceReader, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader>
<resheader name="writer">
<value>System.Resources.ResXResourceWriter, System.Windows.Forms, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089</value>
</resheader>
<data name="copy_command_name" xml:space="preserve">
<value>Copy</value>
</data>
<data name="paste_command_name" xml:space="preserve">
<value>Paste</value>
</data>
<data name="paste_toast_text" xml:space="preserve">
<value>Pasting</value>
</data>
<data name="copied_toast_text" xml:space="preserve">
<value>Copied to clipboard</value>
</data>
<data name="list_item_title" xml:space="preserve">
<value>Clipboard History</value>
</data>
<data name="list_item_subtitle" xml:space="preserve">
<value>Copy, paste, and search items on the clipboard</value>
</data>
<data name="provider_display_name" xml:space="preserve">
<value>Clipboard History</value>
</data>
<data name="clipboard_history_page_name" xml:space="preserve">
<value>Open</value>
</data>
<data name="clipboard_failed_to_load" xml:space="preserve">
<value>Loading clipboard history failed</value>
</data>
</root>