Route terminal clipboard through tauri-plugin-clipboard-manager; bump to 0.2.3

navigator.clipboard.readText() triggers WebView2's "Allow clipboard access?"
permission prompt on every paste. The plugin goes through IPC + the OS
clipboard directly, so the prompt never fires.

Wired the Rust plugin, granted clipboard-manager:allow-{read,write}-text in
the capabilities manifest, swapped XtermPane's copy/paste handler to use
the plugin's readText/writeText.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
This commit is contained in:
megaproxy 2026-05-22 23:27:43 +01:00
parent e94d2499d1
commit 29b15f19c1
7 changed files with 29 additions and 9 deletions

View file

@ -1,7 +1,7 @@
{
"name": "tiletopia",
"private": true,
"version": "0.2.2",
"version": "0.2.3",
"type": "module",
"scripts": {
"dev": "vite",
@ -14,6 +14,7 @@
},
"dependencies": {
"@tauri-apps/api": "^2.0.0",
"@tauri-apps/plugin-clipboard-manager": "^2.0.0",
"@xterm/addon-fit": "^0.10.0",
"@xterm/xterm": "^5.5.0",
"react": "^18.3.0",

10
pnpm-lock.yaml generated
View file

@ -11,6 +11,9 @@ importers:
'@tauri-apps/api':
specifier: ^2.0.0
version: 2.11.0
'@tauri-apps/plugin-clipboard-manager':
specifier: ^2.0.0
version: 2.3.2
'@xterm/addon-fit':
specifier: ^0.10.0
version: 0.10.0(@xterm/xterm@5.5.0)
@ -505,6 +508,9 @@ packages:
engines: {node: '>= 10'}
hasBin: true
'@tauri-apps/plugin-clipboard-manager@2.3.2':
resolution: {integrity: sha512-CUlb5Hqi2oZbcZf4VUyUH53XWPPdtpw43EUpCza5HWZJwxEoDowFzNUDt1tRUXA8Uq+XPn17Ysfptip33sG4eQ==}
'@types/babel__core@7.20.5':
resolution: {integrity: sha512-qoQprZvz5wQFJwMDqeseRXWv3rqMvhgpbXFfVyWhbx9X47POIA6i/+dXefEmZKoAgOaTdaIgNSMqMIU61yRyzA==}
@ -1172,6 +1178,10 @@ snapshots:
'@tauri-apps/cli-win32-ia32-msvc': 2.11.2
'@tauri-apps/cli-win32-x64-msvc': 2.11.2
'@tauri-apps/plugin-clipboard-manager@2.3.2':
dependencies:
'@tauri-apps/api': 2.11.0
'@types/babel__core@7.20.5':
dependencies:
'@babel/parser': 7.29.3

View file

@ -1,6 +1,6 @@
[package]
name = "tiletopia"
version = "0.2.2"
version = "0.2.3"
description = "Tiling multi-terminal manager for WSL"
authors = ["megaproxy"]
edition = "2021"

View file

@ -6,6 +6,8 @@
"permissions": [
"core:default",
"core:event:default",
"core:window:default"
"core:window:default",
"clipboard-manager:allow-read-text",
"clipboard-manager:allow-write-text"
]
}

View file

@ -15,6 +15,7 @@ pub fn run() {
.try_init();
tauri::Builder::default()
.plugin(tauri_plugin_clipboard_manager::init())
.manage(PtyManager::new())
.invoke_handler(tauri::generate_handler![
commands::list_distros,

View file

@ -1,7 +1,7 @@
{
"$schema": "https://schema.tauri.app/config/2",
"productName": "tiletopia",
"version": "0.2.2",
"version": "0.2.3",
"identifier": "com.megaproxy.tiletopia",
"build": {
"beforeDevCommand": "pnpm dev",

View file

@ -2,6 +2,10 @@ import { useRef, useEffect } from "react";
import { Terminal } from "@xterm/xterm";
import { FitAddon } from "@xterm/addon-fit";
import type { UnlistenFn } from "@tauri-apps/api/event";
import {
readText as clipboardReadText,
writeText as clipboardWriteText,
} from "@tauri-apps/plugin-clipboard-manager";
import {
spawnPane,
writeToPane,
@ -174,23 +178,25 @@ export default function XtermPane({
// Ctrl+V (which would otherwise inject ^V into the PTY). term.paste()
// routes through onData → writeToPane, so broadcasting and bracketed
// paste both keep working for free.
//
// Uses tauri-plugin-clipboard-manager instead of navigator.clipboard so
// WebView2 doesn't surface its native "Allow clipboard access?" prompt.
term?.attachCustomKeyEventHandler((e) => {
if (e.type !== "keydown") return true;
if (!e.ctrlKey || !e.shiftKey || e.altKey) return true;
if (e.code === "KeyC") {
const sel = term?.getSelection();
if (sel) {
void navigator.clipboard
.writeText(sel)
.catch((err) => console.warn("clipboard write failed:", err));
void clipboardWriteText(sel).catch((err) =>
console.warn("clipboard write failed:", err),
);
}
e.preventDefault();
return false;
}
if (e.code === "KeyV") {
e.preventDefault();
navigator.clipboard
.readText()
clipboardReadText()
.then((text) => {
if (text && term) term.paste(text);
})