Feature: enable clipboard paste for windows and linux (#185)

* feat(shell): add hidden window style to PowerShell script execution

Modify PowerShell script execution to run with hidden window style in both TypeScript API and Rust plugin to prevent visible command windows

* feat(desktop): implement cross-platform paste functionality in clipboard extension

* feat(desktop): enhance hotkey functionality with registration and update methods

* fix(desktop): remove unnecessary border from command root styling

* feat(desktop): add cross-platform paste support for clipboard extension

- Import Tauri OS plugin to detect platform
- Implement platform-specific paste methods for macOS, Windows, and Linux
- Add error handling for unsupported platforms
- Center windows in Tauri configuration

* feat(desktop): extend global key handler to support Linux control key for settings navigation
This commit is contained in:
Huakun 2025-02-22 00:41:35 -05:00 committed by GitHub
parent ec951bfc80
commit a0bd2d8573
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
7 changed files with 52 additions and 18 deletions

View File

@ -22,12 +22,14 @@
"width": 800,
"visible": false,
"height": 600,
"decorations": true
"decorations": true,
"center": true
},
{
"url": "/splashscreen",
"visible": true,
"label": "splashscreen"
"label": "splashscreen",
"center": true
}
]
},

View File

@ -1,7 +1,10 @@
import { getAllWindows } from "@tauri-apps/api/window"
import { isRegistered, register, unregister } from "@tauri-apps/plugin-global-shortcut"
import { debug, info, warn } from "@tauri-apps/plugin-log"
import * as os from "@tauri-apps/plugin-os"
import * as userInput from "tauri-plugin-user-input-api"
import { sendNotificationWithPermission } from "./notification"
import { sleep } from "./time"
/**
* Tauri global shortcut doesn't accept 'Meta' Key. This function maps browser detected keys to Tauri-accepted keys.
@ -14,6 +17,11 @@ export function mapKeyToTauriKey(key: string): string {
return key
}
/**
* Registers a global hotkey for the application. If the hotkey is already registered, it will be unregistered first.
* When the hotkey is pressed, it toggles the visibility and focus of the main window.
* @param hotkeyStr - The hotkey string to register.
*/
export async function registerAppHotkey(hotkeyStr: string) {
if (await isRegistered(hotkeyStr)) {
warn(`Hotkey (${hotkeyStr}) already registered`)
@ -46,6 +54,11 @@ export async function registerAppHotkey(hotkeyStr: string) {
})
}
/**
* Updates the application's global hotkey. If an old hotkey is provided, it will be unregistered first.
* @param newHotkey - The new hotkey combination to register.
* @param oldHotkey - The old hotkey combination to unregister, if any.
*/
export async function updateAppHotkey(newHotkey: string[], oldHotkey?: string[] | null) {
if (oldHotkey) {
const hotkeyStr = oldHotkey.map(mapKeyToTauriKey).join("+")
@ -56,3 +69,28 @@ export async function updateAppHotkey(newHotkey: string[], oldHotkey?: string[]
const hotkeyStr = newHotkey.map(mapKeyToTauriKey).join("+")
return registerAppHotkey(hotkeyStr)
}
/**
* Simulates a key combination press and release.
* @param keys - The array of keys to press and release.
*/
export async function applyKeyComb(keys: userInput.Key[]) {
await Promise.all(keys.map((key) => userInput.key("KeyPress", key)))
await sleep(50)
await Promise.all(keys.map((key) => userInput.key("KeyRelease", key)))
}
/**
* Simulates a paste operation based on the operating system.
* On macOS, it uses Command+V. On Windows and Linux, it uses Shift+Insert.
*/
export function paste() {
const _platform = os.platform()
if (_platform === "macos") {
return applyKeyComb(["MetaLeft", "KeyV"])
} else if (_platform === "windows" || _platform === "linux") {
return applyKeyComb(["ShiftLeft", "Insert"])
} else {
console.error("Unsupported platform: " + _platform)
}
}

View File

@ -76,7 +76,10 @@ export function goHomeOrCloseOnEscapeWithInput(e: KeyboardEvent) {
export async function globalKeyDownHandler(e: KeyboardEvent) {
keys.keydown(e.key)
const _platform = platform()
if ((_platform === "macos" && e.metaKey) || (_platform === "windows" && e.ctrlKey)) {
if (
(_platform === "macos" && e.metaKey) ||
((_platform === "windows" || _platform === "linux") && e.ctrlKey)
) {
if (e.key === ",") {
e.preventDefault()
goto(i18n.resolveRoute("/app/settings"))

View File

@ -95,7 +95,7 @@
}}
/>
<Command.Root
class={cn("h-screen rounded-lg border shadow-md")}
class={cn("h-screen rounded-lg shadow-md")}
bind:value={$appState.highlightedCmd}
shouldFilter={true}
loop

View File

@ -1,6 +1,8 @@
<script lang="ts">
import { paste } from "@/utils/hotkey"
import { goBack, goHome } from "@/utils/route"
import { listenToNewClipboardItem } from "@/utils/tauri-events"
import { sleep } from "@/utils/time"
import Icon from "@iconify/svelte"
import { ClipboardContentType, db } from "@kksh/api/commands"
import { SearchModeEnum, SQLSortOrderEnum, type ExtData } from "@kksh/api/models"
@ -15,7 +17,6 @@
import { onDestroy, onMount, type Snippet } from "svelte"
import { toast } from "svelte-sonner"
import clipboard from "tauri-plugin-clipboard-api"
import * as userInput from "tauri-plugin-user-input-api"
import ContentPreview from "./content-preview.svelte"
const _platform = platform()
@ -161,18 +162,6 @@
}
}
const sleep = (ms: number) => new Promise((resolve) => setTimeout(resolve, ms))
async function paste() {
await userInput.key("KeyPress", "MetaLeft")
await sleep(20)
await userInput.key("KeyPress", "KeyV")
await sleep(100)
await userInput.key("KeyRelease", "MetaLeft")
await sleep(20)
await userInput.key("KeyRelease", "KeyV")
}
function writeToClipboard(data: ExtData) {
if (!data.data) {
toast.warning("No data found")

View File

@ -168,7 +168,7 @@ export function constructShellApi(
ShellPermissionMap.executePowershellScript,
objectPermissions,
"powershell",
["-Command", script]
["-WindowStyle", "Hidden", "-Command", script]
)
return executePowershellScript(script)
}

View File

@ -17,6 +17,8 @@ pub fn run_apple_script(script: &str) -> anyhow::Result<String> {
pub fn run_powershell(script: &str) -> anyhow::Result<String> {
let output = Command::new("powershell")
.arg("-WindowStyle")
.arg("Hidden")
.arg("-Command")
.arg(script)
.output()