From 742cf3af097702a0ee8a5e172fa1c6d4c7464ef7 Mon Sep 17 00:00:00 2001 From: Huakun Shen Date: Tue, 18 Feb 2025 07:38:43 -0500 Subject: [PATCH 01/22] docs(readme): add extension request section to README --- README.md | 17 ++++++++++++++++- 1 file changed, 16 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index c1f05f6..d6e820a 100644 --- a/README.md +++ b/README.md @@ -46,7 +46,22 @@ -## Star History + + + + + + + + +
Extension Request
+ + You can initiate a poll in the Extension Requests discussion section to gauge interest in your request. If there is significant demand, the extension may be considered for implementation. +
From 60f442dafbe5583d1a91b511b25986e6e04b8cd2 Mon Sep 17 00:00:00 2001 From: Huakun Shen Date: Wed, 19 Feb 2025 03:06:07 -0500 Subject: [PATCH 02/22] feat: add cache table to supabase --- deno.lock | 8 ++++++++ packages/supabase/src/database.types.ts | 24 ++++++++++++++++++++++++ 2 files changed, 32 insertions(+) diff --git a/deno.lock b/deno.lock index ea2e1a4..b32769c 100644 --- a/deno.lock +++ b/deno.lock @@ -51,6 +51,7 @@ "npm:@tauri-apps/api@^2.2.0": "2.2.0", "npm:@tauri-apps/cli@^2.2.2": "2.2.7", "npm:@tauri-apps/cli@^2.2.7": "2.2.7", + "npm:@tauri-apps/plugin-autostart@^2.2.0": "2.2.0", "npm:@tauri-apps/plugin-deep-link@^2.2.0": "2.2.0", "npm:@tauri-apps/plugin-dialog@^2.2.0": "2.2.0", "npm:@tauri-apps/plugin-fs@^2.2.0": "2.2.0", @@ -5025,6 +5026,12 @@ "@tauri-apps/cli-win32-x64-msvc" ] }, + "@tauri-apps/plugin-autostart@2.2.0": { + "integrity": "sha512-TzVcDZdOvdot0avkpstUWJKKEl4cyxLpFB9DZZRW5zH8k+Bv8IVJmO0zyYuw+7oKlGdHOINbD/7Je7GHMViw5w==", + "dependencies": [ + "@tauri-apps/api@2.2.0" + ] + }, "@tauri-apps/plugin-deep-link@2.2.0": { "integrity": "sha512-H6mkxr2KZ3XJcKL44tiq6cOjCw9DL8OgU1xjn3j26Qsn+H/roPFiyhR7CHuB8Ar+sQFj4YVlfmJwtBajK2FETQ==", "dependencies": [ @@ -15068,6 +15075,7 @@ "npm:@tanstack/table-core@^8.20.5", "npm:@tauri-apps/api@^2.1.1", "npm:@tauri-apps/cli@^2.2.7", + "npm:@tauri-apps/plugin-autostart@^2.2.0", "npm:@tauri-apps/plugin-shell@^2.2.0", "npm:@tauri-apps/plugin-stronghold@^2.2.0", "npm:@types/bun@latest", diff --git a/packages/supabase/src/database.types.ts b/packages/supabase/src/database.types.ts index b80cc87..8cd7968 100644 --- a/packages/supabase/src/database.types.ts +++ b/packages/supabase/src/database.types.ts @@ -3,6 +3,30 @@ export type Json = string | number | boolean | null | { [key: string]: Json | un export type Database = { public: { Tables: { + cache: { + Row: { + created_at: string + data: Json | null + expiry_date: string | null + id: number + key: string + } + Insert: { + created_at?: string + data?: Json | null + expiry_date?: string | null + id?: number + key: string + } + Update: { + created_at?: string + data?: Json | null + expiry_date?: string | null + id?: number + key?: string + } + Relationships: [] + } events: { Row: { created_at: string From dfd84db783aa8c06f617ff777ab42ab231ada798 Mon Sep 17 00:00:00 2001 From: Huakun Shen Date: Wed, 19 Feb 2025 03:36:18 -0500 Subject: [PATCH 03/22] docs(readme): update extension request discussion link and formatting --- README.md | 10 ++++------ .../jarvis/permissions/autogenerated/reference.md | 2 ++ 2 files changed, 6 insertions(+), 6 deletions(-) diff --git a/README.md b/README.md index d6e820a..69afd1e 100644 --- a/README.md +++ b/README.md @@ -52,12 +52,10 @@ - - You can initiate a poll in the Extension Requests discussion section to gauge interest in your request. If there is significant demand, the extension may be considered for implementation. + You can initiate an extension request in the Extension Requests discussion + section to gauge interest in your request. +
+ If there is significant demand, the extension may be considered for implementation. diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md b/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md index 7020b75..19f9805 100644 --- a/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md @@ -1,3 +1,4 @@ + ## Permission Table @@ -6,6 +7,7 @@ + + + + + + + + + + + diff --git a/packages/tauri-plugins/jarvis/permissions/schemas/schema.json b/packages/tauri-plugins/jarvis/permissions/schemas/schema.json index 49bd95c..32302ff 100644 --- a/packages/tauri-plugins/jarvis/permissions/schemas/schema.json +++ b/packages/tauri-plugins/jarvis/permissions/schemas/schema.json @@ -349,6 +349,16 @@ "type": "string", "const": "deny-compress-tarball" }, + { + "description": "Enables the copy_dir_all command without any pre-configured scope.", + "type": "string", + "const": "allow-copy-dir-all" + }, + { + "description": "Denies the copy_dir_all command without any pre-configured scope.", + "type": "string", + "const": "deny-copy-dir-all" + }, { "description": "Enables the create_command command without any pre-configured scope.", "type": "string", diff --git a/packages/tauri-plugins/jarvis/src/commands/fs.rs b/packages/tauri-plugins/jarvis/src/commands/fs.rs index cbbeaca..a0c63a7 100644 --- a/packages/tauri-plugins/jarvis/src/commands/fs.rs +++ b/packages/tauri-plugins/jarvis/src/commands/fs.rs @@ -35,3 +35,11 @@ pub async fn unzip( ) -> Result<(), String> { utils::fs::unzip(&path, &destination_folder, overwrite).map_err(|err| err.to_string()) } + +#[tauri::command] +/// copy one dir to another (or location) +/// +/// uses the [`dircpy`](https://crates.io/crates/dircpy) crate for recursive directory copying. +pub async fn copy_dir_all(from: String, to: String) -> Result<(), String> { + dircpy::copy_dir(from, to).map_err(|e| e.to_string()) +} diff --git a/packages/tauri-plugins/jarvis/src/lib.rs b/packages/tauri-plugins/jarvis/src/lib.rs index 16aabde..58da91d 100644 --- a/packages/tauri-plugins/jarvis/src/lib.rs +++ b/packages/tauri-plugins/jarvis/src/lib.rs @@ -115,6 +115,7 @@ pub fn init() -> TauriPlugin { commands::fs::decompress_tarball, commands::fs::compress_tarball, commands::fs::unzip, + commands::fs::copy_dir_all, /* ------------------------------- file search ------------------------------ */ commands::file_search::file_search, /* ------------------------------- extensions ------------------------------- */ From b5ea128aca540eae6435e20fb03318947e4d623b Mon Sep 17 00:00:00 2001 From: Huakun Date: Sat, 22 Feb 2025 03:00:00 -0500 Subject: [PATCH 16/22] feat(shell): add hidden window style to PowerShell script execution (#182) Modify PowerShell script execution to run with hidden window style in both TypeScript API and Rust plugin to prevent visible command windows From ed20f9a142a5d2419817e3e1e08aa5ab23b9995b Mon Sep 17 00:00:00 2001 From: Huakun Date: Sat, 22 Feb 2025 04:36:44 -0500 Subject: [PATCH 17/22] Fix: linux extension loading (#188) * fix: change escape key behavior to navigate home instead of back * feat: load custom ui extension with http server on Linux Linux now uses the same loading approach as Windows due to a bug https://github.com/tauri-apps/tauri/issues/12767 * feat: add tauri-plugin-system-info-api dependency to deno.lock --- apps/desktop/package.json | 2 +- apps/desktop/src/lib/cmds/ext.ts | 3 ++- apps/desktop/src/routes/app/extension/store/+page.svelte | 2 +- deno.lock | 1 + 4 files changed, 5 insertions(+), 3 deletions(-) diff --git a/apps/desktop/package.json b/apps/desktop/package.json index 0ef9fc8..daaa5f9 100644 --- a/apps/desktop/package.json +++ b/apps/desktop/package.json @@ -1,6 +1,6 @@ { "name": "@kksh/desktop", - "version": "0.1.24", + "version": "0.1.25", "description": "", "type": "module", "scripts": { diff --git a/apps/desktop/src/lib/cmds/ext.ts b/apps/desktop/src/lib/cmds/ext.ts index fe49456..8eea4bf 100644 --- a/apps/desktop/src/lib/cmds/ext.ts +++ b/apps/desktop/src/lib/cmds/ext.ts @@ -163,7 +163,8 @@ export async function onCustomUiCmdSelect( extPath: ext.extPath, dist: cmd.dist }) - if (platform() === "windows" && !useDevMain) { + const _platform = platform() + if ((_platform === "windows" || _platform === "linux") && !useDevMain) { const addr = await spawnExtensionFileServer(winLabel) // addr has format "127.0.0.1:" console.log("Extension file server address: ", addr) const newUrl = `http://${addr}` diff --git a/apps/desktop/src/routes/app/extension/store/+page.svelte b/apps/desktop/src/routes/app/extension/store/+page.svelte index 7e8918b..ca08cf1 100644 --- a/apps/desktop/src/routes/app/extension/store/+page.svelte +++ b/apps/desktop/src/routes/app/extension/store/+page.svelte @@ -111,7 +111,7 @@ function onkeydown(e: KeyboardEvent) { if (e.key === "Escape") { if (document.activeElement === listviewInputRef) { - goBack() + goHome() } } } diff --git a/deno.lock b/deno.lock index b32769c..5180460 100644 --- a/deno.lock +++ b/deno.lock @@ -181,6 +181,7 @@ "npm:tauri-api-adapter@~0.3.20": "0.3.20_typescript@5.6.3", "npm:tauri-plugin-clipboard-api@^2.1.11": "2.1.11_typescript@5.6.3", "npm:tauri-plugin-shellx-api@^2.0.14": "2.0.14", + "npm:tauri-plugin-system-info-api@2.0.8": "2.0.8_typescript@5.6.3", "npm:ts-proto@^2.3.0": "2.6.1", "npm:tslib@^2.8.1": "2.8.1", "npm:turbo@^2.3.4": "2.4.0", From 52919b8d2f0b784417c92b0d008f621259a29ee1 Mon Sep 17 00:00:00 2001 From: Huakun Date: Sat, 22 Feb 2025 05:14:04 -0500 Subject: [PATCH 18/22] fix(desktop): set macOS accessory activation policy to Accessory (#189) --- apps/desktop/src-tauri/src/lib.rs | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/apps/desktop/src-tauri/src/lib.rs b/apps/desktop/src-tauri/src/lib.rs index d02f488..7eacd17 100644 --- a/apps/desktop/src-tauri/src/lib.rs +++ b/apps/desktop/src-tauri/src/lib.rs @@ -230,7 +230,8 @@ pub fn run() { } // setup::deeplink::setup_deeplink(app); // #[cfg(all(target_os = "macos", debug_assertions))] - // app.set_activation_policy(ActivationPolicy::Accessory); + #[cfg(target_os = "macos")] + app.set_activation_policy(ActivationPolicy::Accessory); // let mut store = StoreBuilder::new("appConfig.bin").build(app.handle().clone()); // let store = app.handle().store_builder("appConfig.json").build()?; From 330678cb45c37329d091b8992cc530d5652ea4e7 Mon Sep 17 00:00:00 2001 From: Huakun Date: Sat, 22 Feb 2025 05:36:53 -0500 Subject: [PATCH 19/22] Fix: linux app launch (#190) * fix(desktop): update version to 0.1.26 and enhance app execution handling for Linux * fix(desktop): remove redundant app path display in AppsCmds component --- apps/desktop/package.json | 2 +- apps/desktop/src/lib/components/main/AppsCmds.svelte | 12 ++++++++++-- 2 files changed, 11 insertions(+), 3 deletions(-) diff --git a/apps/desktop/package.json b/apps/desktop/package.json index daaa5f9..3bb6b3f 100644 --- a/apps/desktop/package.json +++ b/apps/desktop/package.json @@ -1,6 +1,6 @@ { "name": "@kksh/desktop", - "version": "0.1.25", + "version": "0.1.26", "description": "", "type": "module", "scripts": { diff --git a/apps/desktop/src/lib/components/main/AppsCmds.svelte b/apps/desktop/src/lib/components/main/AppsCmds.svelte index 4b8d56c..80445a2 100644 --- a/apps/desktop/src/lib/components/main/AppsCmds.svelte +++ b/apps/desktop/src/lib/components/main/AppsCmds.svelte @@ -6,7 +6,7 @@ import { convertFileSrc } from "@tauri-apps/api/core" import * as os from "@tauri-apps/plugin-os" import { toast } from "svelte-sonner" - import { open } from "tauri-plugin-shellx-api" + import { executeBashScript, open } from "tauri-plugin-shellx-api" const platform = os.platform() let { apps }: { apps: AppInfo[] } = $props() @@ -23,8 +23,16 @@ } else { toast.error("No executable path found for this app") } - } else { + } else if (platform === "macos") { open(app.app_desktop_path) + } else if (platform === "linux") { + if (app.app_path_exe) { + executeBashScript(app.app_path_exe) + } else { + toast.error("No executable path found for this app") + } + } else { + toast.error("Unsupported platform") } }} value={app.name} From 441abbcdae8d90354f7497ed81bcab40ae23f3cf Mon Sep 17 00:00:00 2001 From: Huakun Date: Sat, 22 Feb 2025 07:21:15 -0500 Subject: [PATCH 20/22] Fix: clipboard paste timing (#191) * fix(desktop): improve hotkey and paste functionality with refined key press timing * fix(clipboard): streamline clipboard handling for macOS and add TODO for Windows/Linux support * removed a comment --- apps/desktop/package.json | 2 +- apps/desktop/src/lib/utils/hotkey.ts | 15 +++++++++++---- .../routes/app/extension/clipboard/+page.svelte | 13 +++++-------- 3 files changed, 17 insertions(+), 13 deletions(-) diff --git a/apps/desktop/package.json b/apps/desktop/package.json index 3bb6b3f..86cf587 100644 --- a/apps/desktop/package.json +++ b/apps/desktop/package.json @@ -1,6 +1,6 @@ { "name": "@kksh/desktop", - "version": "0.1.26", + "version": "0.1.27", "description": "", "type": "module", "scripts": { diff --git a/apps/desktop/src/lib/utils/hotkey.ts b/apps/desktop/src/lib/utils/hotkey.ts index dcebd2f..468b415 100644 --- a/apps/desktop/src/lib/utils/hotkey.ts +++ b/apps/desktop/src/lib/utils/hotkey.ts @@ -75,16 +75,23 @@ export async function updateAppHotkey(newHotkey: string[], oldHotkey?: string[] * @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))) + // await Promise.all(keys.map((key) => userInput.key("KeyPress", key))) + for (const key of keys) { + await userInput.key("KeyPress", key) + await sleep(20) + } + await sleep(100) + for (const key of keys) { + await userInput.key("KeyRelease", key) + await sleep(20) + } } /** * 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() { +export async function paste() { const _platform = os.platform() if (_platform === "macos") { return applyKeyComb(["MetaLeft", "KeyV"]) diff --git a/apps/desktop/src/routes/app/extension/clipboard/+page.svelte b/apps/desktop/src/routes/app/extension/clipboard/+page.svelte index c364f1b..bac90d4 100644 --- a/apps/desktop/src/routes/app/extension/clipboard/+page.svelte +++ b/apps/desktop/src/routes/app/extension/clipboard/+page.svelte @@ -191,14 +191,11 @@ return Promise.reject(new Error("No data found")) } return writeToClipboard(data).then(async () => { - if (_platform === "macos") { - // TODO: add support for Windows and Linux - return app - .hide() - .then(() => sleep(100)) - .then(() => curWin.hide()) - .then(() => paste()) - } + return app + .hide() + .then(() => sleep(100)) + .then(() => curWin.hide()) + .then(() => paste()) }) }) .then(() => toast.success("Copied to clipboard")) From 8d49f504957a00c0f68c7f936b546123aebeb8a8 Mon Sep 17 00:00:00 2001 From: Luca Giannini <68999840+structwafel@users.noreply.github.com> Date: Sat, 22 Feb 2025 15:12:47 +0100 Subject: [PATCH 21/22] fix: handle failing icon loading linux gracefully (#193) --- packages/tauri-plugins/jarvis/src/utils/icns.rs | 11 +++++++---- 1 file changed, 7 insertions(+), 4 deletions(-) diff --git a/packages/tauri-plugins/jarvis/src/utils/icns.rs b/packages/tauri-plugins/jarvis/src/utils/icns.rs index cef887d..1845d30 100644 --- a/packages/tauri-plugins/jarvis/src/utils/icns.rs +++ b/packages/tauri-plugins/jarvis/src/utils/icns.rs @@ -52,10 +52,13 @@ pub fn load_icns(icns_path: &PathBuf) -> anyhow::Result { #[cfg(target_os = "linux")] pub fn load_icon(path: PathBuf) -> tauri::http::Response> { match path.exists() { - true => { - let bytes = std::fs::read(&path).expect("Error reading file"); - tauri::http::Response::builder().body(bytes).unwrap() - } + true => match std::fs::read(&path) { + Ok(bytes) => tauri::http::Response::builder().body(bytes).unwrap(), + Err(err) => tauri::http::Response::builder() + .status(tauri::http::StatusCode::NOT_FOUND) + .body(format!("error loading icon: {:?}", err).as_bytes().to_vec()) + .unwrap(), + }, false => { let res = tauri::http::Response::builder() .status(tauri::http::StatusCode::NOT_FOUND) From 8940d25245c746210bccbc308c6f592b21df4ea3 Mon Sep 17 00:00:00 2001 From: Luca Giannini <68999840+structwafel@users.noreply.github.com> Date: Sat, 22 Feb 2025 15:54:06 +0100 Subject: [PATCH 22/22] fix: only show apps with name (#194) * only show apps with name * format * update pnpm lock --------- Co-authored-by: Huakun Shen --- .../src/lib/components/main/AppsCmds.svelte | 2 +- pnpm-lock.yaml | 17 ++++++++++++++++- 2 files changed, 17 insertions(+), 2 deletions(-) diff --git a/apps/desktop/src/lib/components/main/AppsCmds.svelte b/apps/desktop/src/lib/components/main/AppsCmds.svelte index 80445a2..e54ba3c 100644 --- a/apps/desktop/src/lib/components/main/AppsCmds.svelte +++ b/apps/desktop/src/lib/components/main/AppsCmds.svelte @@ -13,7 +13,7 @@ - {#each apps as app} + {#each apps.filter((app) => app.name) as app} { diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 074f3cc..b47e89f 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -815,7 +815,7 @@ importers: devDependencies: '@types/bun': specifier: latest - version: 1.2.2 + version: 1.2.3 packages/templates/template-ext-next: dependencies: @@ -5130,6 +5130,9 @@ packages: '@types/bun@1.2.2': resolution: {integrity: sha512-tr74gdku+AEDN5ergNiBnplr7hpDp3V1h7fqI2GcR/rsUaM39jpSeKH0TFibRvU0KwniRx5POgaYnaXbk0hU+w==} + '@types/bun@1.2.3': + resolution: {integrity: sha512-054h79ipETRfjtsCW9qJK8Ipof67Pw9bodFWmkfkaUaRiIQ1dIV2VTlheshlBx3mpKr0KeK8VqnMMCtgN9rQtw==} + '@types/cookie@0.6.0': resolution: {integrity: sha512-4Kh9a6B2bQciAhf7FSuMRRkUWecJgJu9nPnx3yzpsfXX/c50REIqpHY4C82bXP90qrLtXtkDxTZosYO3UpOwlA==} @@ -6180,6 +6183,9 @@ packages: bun-types@1.2.2: resolution: {integrity: sha512-RCbMH5elr9gjgDGDhkTTugA21XtJAy/9jkKe/G3WR2q17VPGhcquf9Sir6uay9iW+7P/BV0CAHA1XlHXMAVKHg==} + bun-types@1.2.3: + resolution: {integrity: sha512-P7AeyTseLKAvgaZqQrvp3RqFM3yN9PlcLuSTe7SoJOfZkER73mLdT2vEQi8U64S1YvM/ldcNiQjn0Sn7H9lGgg==} + bundle-name@4.1.0: resolution: {integrity: sha512-tjwM5exMg6BGRI+kNmTntNsvdZS1X8BFYS6tnJ2hdH0kVxM6/eVZ2xy+FqStSWvYmtfFMDLIxurorHwDKfDz5Q==} engines: {node: '>=18'} @@ -16633,6 +16639,10 @@ snapshots: dependencies: bun-types: 1.2.2 + '@types/bun@1.2.3': + dependencies: + bun-types: 1.2.3 + '@types/cookie@0.6.0': {} '@types/d3-array@3.2.1': {} @@ -18183,6 +18193,11 @@ snapshots: '@types/node': 22.13.1 '@types/ws': 8.5.14 + bun-types@1.2.3: + dependencies: + '@types/node': 22.13.1 + '@types/ws': 8.5.14 + bundle-name@4.1.0: dependencies: run-applescript: 7.0.0
Description
From 0bd65db3e5b2dafa97c9f8a7cade92ba694f85f5 Mon Sep 17 00:00:00 2001 From: Huakun Shen Date: Wed, 19 Feb 2025 03:43:51 -0500 Subject: [PATCH 04/22] docs(readme): improve extension request section with direct submission link --- README.md | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/README.md b/README.md index 69afd1e..f62d21d 100644 --- a/README.md +++ b/README.md @@ -52,7 +52,9 @@
- You can initiate an extension request in the Extension Requests discussion + You can Submit Extension Request + request in the + Extension Requests discussion section to gauge interest in your request.
If there is significant demand, the extension may be considered for implementation. From d3215d386d59664a892fa939f99d94af11662991 Mon Sep 17 00:00:00 2001 From: Huakun Shen Date: Wed, 19 Feb 2025 03:57:31 -0500 Subject: [PATCH 05/22] docs(readme): refine extension request discussion link --- README.md | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/README.md b/README.md index f62d21d..f6a8686 100644 --- a/README.md +++ b/README.md @@ -54,7 +54,7 @@
You can Submit Extension Request request in the - Extension Requests discussion + Extension Requests discussion section to gauge interest in your request.
If there is significant demand, the extension may be considered for implementation. From 9cfb59e7e49848579e1269e21149ce8597872d54 Mon Sep 17 00:00:00 2001 From: Huakun Date: Wed, 19 Feb 2025 07:56:36 -0500 Subject: [PATCH 06/22] fix: system command not triggerred (#165) --- packages/ui/src/components/main/SystemCmds.svelte | 2 ++ 1 file changed, 2 insertions(+) diff --git a/packages/ui/src/components/main/SystemCmds.svelte b/packages/ui/src/components/main/SystemCmds.svelte index ed755fc..11c4532 100644 --- a/packages/ui/src/components/main/SystemCmds.svelte +++ b/packages/ui/src/components/main/SystemCmds.svelte @@ -21,6 +21,8 @@ if (confirmed) { cmd.function() } + } else { + cmd.function() } }} value={cmd.name} From 369a9719fdd807b1dc84b83087f315647242e1da Mon Sep 17 00:00:00 2001 From: Huakun Date: Wed, 19 Feb 2025 08:47:43 -0500 Subject: [PATCH 07/22] feat(desktop): implement quick install hotkey for store (#164) --- .../routes/app/extension/store/+page.svelte | 99 ++++++++++++++++++- packages/api/src/ui/index.ts | 1 + .../components/extension/ExtListItem.svelte | 9 +- 3 files changed, 102 insertions(+), 7 deletions(-) diff --git a/apps/desktop/src/routes/app/extension/store/+page.svelte b/apps/desktop/src/routes/app/extension/store/+page.svelte index ae6f15d..7e8918b 100644 --- a/apps/desktop/src/routes/app/extension/store/+page.svelte +++ b/apps/desktop/src/routes/app/extension/store/+page.svelte @@ -1,9 +1,12 @@ - + {#snippet leftSlot()} {/snippet} - + { + if (e.key === "Enter") { + const modifier = _platform === "macos" ? e.metaKey : e.ctrlKey + if (modifier) { + if (highlightedCmd) { + onExtItemInstall(highlightedCmd) + } + } else { + if (highlightedCmd) { + onExtItemSelected(highlightedCmd) + } + } + } + }} /> No results found. @@ -105,11 +167,38 @@ {ext} installedVersion={$installedExtsMap[ext.identifier]} isUpgradable={!!$upgradableExpsMap[ext.identifier]} - onSelect={() => onExtItemSelected(ext)} + onSelect={() => {}} onUpgrade={() => onExtItemUpgrade(ext)} onInstall={() => onExtItemInstall(ext)} /> {/each} - + { + if (value === "install") { + console.log("install") + if (highlightedCmd) { + onExtItemInstall(highlightedCmd) + } + } + }} + onDefaultActionSelected={() => { + console.log("default install") + if (highlightedCmd) { + onExtItemInstall(highlightedCmd) + } + }} + /> diff --git a/packages/api/src/ui/index.ts b/packages/api/src/ui/index.ts index 2360e9d..07a3397 100644 --- a/packages/api/src/ui/index.ts +++ b/packages/api/src/ui/index.ts @@ -21,3 +21,4 @@ export type { export * from "../api/client" // all client types export type { IUiTemplate } from "./template" export type { IShell } from "../api/shell" +export * from "./template/components" diff --git a/packages/ui/src/components/extension/ExtListItem.svelte b/packages/ui/src/components/extension/ExtListItem.svelte index 0496beb..20387c0 100644 --- a/packages/ui/src/components/extension/ExtListItem.svelte +++ b/packages/ui/src/components/extension/ExtListItem.svelte @@ -1,6 +1,6 @@ - + Date: Wed, 19 Feb 2025 08:58:29 -0500 Subject: [PATCH 08/22] feat(desktop): improve app config loading, Merge loaded config with default config --- apps/desktop/src-tauri/src/lib.rs | 1 + apps/desktop/src/lib/stores/appConfig.ts | 5 ++++- 2 files changed, 5 insertions(+), 1 deletion(-) diff --git a/apps/desktop/src-tauri/src/lib.rs b/apps/desktop/src-tauri/src/lib.rs index 76ead7c..8ccf218 100644 --- a/apps/desktop/src-tauri/src/lib.rs +++ b/apps/desktop/src-tauri/src/lib.rs @@ -66,6 +66,7 @@ pub fn run() { // ); // } if KUNKUN_PUBLISH == "true" { + // only include updater with KUNKUN_PUBLISH = true, this is used to avoid using updater in beta build CI, which requires secrets for the updater endpoint if updater is included println!("KUNKUN_PUBLISH: {}", KUNKUN_PUBLISH); builder = builder.plugin(tauri_plugin_updater::Builder::new().build()); } diff --git a/apps/desktop/src/lib/stores/appConfig.ts b/apps/desktop/src/lib/stores/appConfig.ts index 0feb422..eccaad0 100644 --- a/apps/desktop/src/lib/stores/appConfig.ts +++ b/apps/desktop/src/lib/stores/appConfig.ts @@ -48,7 +48,10 @@ function createAppConfig(): WithSyncStore & Ap async function init() { debug("Initializing app config") const persistStore = await load("kk-config.json", { autoSave: true }) - const loadedConfig = await persistStore.get("config") + let loadedConfig = await persistStore.get("config") + if (typeof loadedConfig === "object") { + loadedConfig = { ...defaultAppConfig, ...loadedConfig } + } const parseRes = v.safeParse(PersistedAppConfig, loadedConfig) if (parseRes.success) { console.log("Parse Persisted App Config Success", parseRes.output) From ac6e2c3f7857df44b1fb580b2fd3023ff794870c Mon Sep 17 00:00:00 2001 From: Huakun Shen Date: Wed, 19 Feb 2025 09:00:19 -0500 Subject: [PATCH 09/22] chore(desktop): bump version to 0.1.24 --- apps/desktop/package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/apps/desktop/package.json b/apps/desktop/package.json index 84f0cba..0ef9fc8 100644 --- a/apps/desktop/package.json +++ b/apps/desktop/package.json @@ -1,6 +1,6 @@ { "name": "@kksh/desktop", - "version": "0.1.23", + "version": "0.1.24", "description": "", "type": "module", "scripts": { From 07c62e236ce3eee9c2bf5e2ccfbdef60c2ec88bd Mon Sep 17 00:00:00 2001 From: Huakun Date: Thu, 20 Feb 2025 04:02:50 -0500 Subject: [PATCH 10/22] feat(desktop): enhance clipboard extension with auto-paste functionality (#171) - Add writeToClipboard utility function to handle different clipboard content types - Implement paste() method to simulate keyboard paste action - Update onItemSelected to hide app and auto-paste selected clipboard item - Add core:app:allow-app-hide capability to support app hiding --- .../src-tauri/capabilities/default.json | 1 + .../app/extension/clipboard/+page.svelte | 64 ++++++++++++++----- 2 files changed, 49 insertions(+), 16 deletions(-) diff --git a/apps/desktop/src-tauri/capabilities/default.json b/apps/desktop/src-tauri/capabilities/default.json index c5aae8d..11e7006 100644 --- a/apps/desktop/src-tauri/capabilities/default.json +++ b/apps/desktop/src-tauri/capabilities/default.json @@ -39,6 +39,7 @@ "core:webview:allow-create-webview", "core:webview:allow-create-webview-window", "core:app:default", + "core:app:allow-app-hide", "core:resources:default", "core:menu:default", "core:tray:default", diff --git a/apps/desktop/src/routes/app/extension/clipboard/+page.svelte b/apps/desktop/src/routes/app/extension/clipboard/+page.svelte index d3781b5..b013ee2 100644 --- a/apps/desktop/src/routes/app/extension/clipboard/+page.svelte +++ b/apps/desktop/src/routes/app/extension/clipboard/+page.svelte @@ -7,13 +7,17 @@ import { Button, Command, Resizable } from "@kksh/svelte5" import { Constants } from "@kksh/ui" import { CustomCommandInput, GlobalCommandPaletteFooter } from "@kksh/ui/main" + import { app } from "@tauri-apps/api" import type { UnlistenFn } from "@tauri-apps/api/event" + import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow" import { ArrowLeft, FileQuestionIcon, ImageIcon, LetterTextIcon } from "lucide-svelte" 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 curWin = getCurrentWebviewWindow() let searchTerm = $state("") let clipboardHistoryList = $state([]) let highlightedItemValue = $state("") @@ -155,29 +159,57 @@ } } + 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") + return Promise.reject(new Error("No data found")) + } + const dataType = data?.dataType as ClipboardContentType + switch (dataType) { + case "Text": + return clipboard.writeText(data.data) + case "Image": + return clipboard.writeImageBase64(data.data) + case "Html": + return clipboard.writeHtmlAndText(data.data, data.searchText ?? data.data) + case "Rtf": + return clipboard.writeRtf(data.data) + default: + return Promise.reject(new Error("Unsupported data type: " + dataType)) + } + } + function onItemSelected(dataId: number) { - // fetch data from db db.getExtensionDataById(dataId) .then((data) => { - if (!data || !data.data) { - return - } - const dataType = data?.dataType as ClipboardContentType - switch (dataType) { - case "Text": - return clipboard.writeText(data.data) - case "Image": - return clipboard.writeImageBase64(data.data) - case "Html": - return clipboard.writeHtml(data.data) - case "Rtf": - return clipboard.writeRtf(data.data) - default: - return Promise.reject(new Error("Unsupported data type: " + dataType)) + console.log("data", data) + if (!data) { + toast.warning("No data found") + return Promise.reject(new Error("No data found")) } + return writeToClipboard(data).then(async () => + app + .hide() + .then(() => sleep(100)) + .then(() => curWin.hide()) + .then(() => paste()) + ) }) .then(() => toast.success("Copied to clipboard")) .catch((err) => { + console.error(err) toast.error("Failed to fetch data from db", { description: err.message }) From 8a9f6bcb09a5239effbadf41ec9789077f3c12c4 Mon Sep 17 00:00:00 2001 From: Huakun Date: Fri, 21 Feb 2025 05:59:09 -0500 Subject: [PATCH 11/22] UI Update (#179) * feat: add publisher link to extension detail * fix: improve IconMultiplexer and StoreExtDetail component rendering * feat: add published date to extension details view * chore: add moment.js and clean up imports in StoreExtDetail * fix: support cloudflare worker Otherwise cloudflare worker gets html instead of json * refactor: move AppsCmds component to desktop app * bump: version to 0.1.1 * fix: package.json fetching cors error * fix: improve files field validation in verify command * feat: make dropdown width in main search input dynamic * feat(ui): add install button for web extension store * update submodules * format * feat(desktop): disable clipboard auto paste for windows and linux They are not implemented yet, current code only works for mac --- Cargo.lock | 2 +- apps/desktop/src/routes/app/+page.svelte | 2 +- .../app/extension/clipboard/+page.svelte | 19 +++++--- .../app/extension/ui-worker/+page.svelte | 5 +- .../extension/StoreExtDetail.svelte | 47 ++++++++++--------- vendors/tauri-plugin-system-info | 2 +- 6 files changed, 41 insertions(+), 36 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index a0a409a..a7a5475 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -7776,7 +7776,7 @@ dependencies = [ [[package]] name = "tauri-plugin-system-info" -version = "2.0.8" +version = "2.0.9" dependencies = [ "serde", "starship-battery", diff --git a/apps/desktop/src/routes/app/+page.svelte b/apps/desktop/src/routes/app/+page.svelte index ca0a2f7..bd9718c 100644 --- a/apps/desktop/src/routes/app/+page.svelte +++ b/apps/desktop/src/routes/app/+page.svelte @@ -139,7 +139,7 @@ - + exit()}> diff --git a/apps/desktop/src/routes/app/extension/clipboard/+page.svelte b/apps/desktop/src/routes/app/extension/clipboard/+page.svelte index b013ee2..b702cfd 100644 --- a/apps/desktop/src/routes/app/extension/clipboard/+page.svelte +++ b/apps/desktop/src/routes/app/extension/clipboard/+page.svelte @@ -10,6 +10,7 @@ import { app } from "@tauri-apps/api" import type { UnlistenFn } from "@tauri-apps/api/event" import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow" + import { platform } from "@tauri-apps/plugin-os" import { ArrowLeft, FileQuestionIcon, ImageIcon, LetterTextIcon } from "lucide-svelte" import { onDestroy, onMount, type Snippet } from "svelte" import { toast } from "svelte-sonner" @@ -17,6 +18,7 @@ import * as userInput from "tauri-plugin-user-input-api" import ContentPreview from "./content-preview.svelte" + const _platform = platform() const curWin = getCurrentWebviewWindow() let searchTerm = $state("") let clipboardHistoryList = $state([]) @@ -199,13 +201,16 @@ toast.warning("No data found") return Promise.reject(new Error("No data found")) } - return writeToClipboard(data).then(async () => - app - .hide() - .then(() => sleep(100)) - .then(() => curWin.hide()) - .then(() => paste()) - ) + return writeToClipboard(data).then(async () => { + if (_platform === "macos") { + // TODO: add support for Windows and Linux + return app + .hide() + .then(() => sleep(100)) + .then(() => curWin.hide()) + .then(() => paste()) + } + }) }) .then(() => toast.success("Copied to clipboard")) .catch((err) => { diff --git a/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte b/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte index 8fef2f9..cb6810f 100644 --- a/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte +++ b/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte @@ -281,7 +281,6 @@ ) { // actionPanelOpen = true setTimeout(() => { - console.log("toggle action panel") actionPanelOpen = !actionPanelOpen if (!actionPanelOpen) { onActionPanelBlur() @@ -298,8 +297,6 @@ function onkeydown(e: KeyboardEvent) { if (e.key === "Escape") { - console.log(document.activeElement) - console.log(document.activeElement?.nodeName) if (document.activeElement?.nodeName === "INPUT") { console.log("input") } @@ -376,7 +373,7 @@ {pbar} onGoBack={goBack} onSubmit={(formData: Record) => { - console.log("formData", formData) + console.log("Submit formData", formData) workerAPI?.onFormSubmit(formData) }} /> diff --git a/packages/ui/src/components/extension/StoreExtDetail.svelte b/packages/ui/src/components/extension/StoreExtDetail.svelte index b9b9f45..a59edbb 100644 --- a/packages/ui/src/components/extension/StoreExtDetail.svelte +++ b/packages/ui/src/components/extension/StoreExtDetail.svelte @@ -182,28 +182,31 @@ ).format("YYYY-MM-DD HH:mm")} -
- {#if metadata && metadata.sourceType === ExtPublishSourceTypeEnum.jsr} - - - - {:else if metadata && metadata.sourceType === ExtPublishSourceTypeEnum.npm} - - - - {/if} - {#if metadata && metadata?.git?.commit && metadata?.rekorLogIndex && metadata?.git?.owner && metadata?.git?.repo} - - - - {metadata.git.owner}/{metadata.git.repo} - - - {/if} -
+ {#if !isInTauri} + + {/if} + +
+ {#if metadata && metadata.sourceType === ExtPublishSourceTypeEnum.jsr} + + + + {:else if metadata && metadata.sourceType === ExtPublishSourceTypeEnum.npm} + + + + {/if} + {#if metadata && metadata?.git?.commit && metadata?.rekorLogIndex && metadata?.git?.owner && metadata?.git?.repo} + + + + {metadata.git.owner}/{metadata.git.repo} + + + {/if}
{#if metadata && metadata?.git?.commit && metadata?.rekorLogIndex && metadata?.git?.owner && metadata?.git?.repo} diff --git a/vendors/tauri-plugin-system-info b/vendors/tauri-plugin-system-info index 282b6df..5d0424d 160000 --- a/vendors/tauri-plugin-system-info +++ b/vendors/tauri-plugin-system-info @@ -1 +1 @@ -Subproject commit 282b6df90c6e4fc596e1be33e923e13270174ee1 +Subproject commit 5d0424ddb61e217a03a4d3d75ef9a761d9901c41 From 4d90e2cf298b0719ad8e5e54787fa4ac676d33e0 Mon Sep 17 00:00:00 2001 From: Huakun Date: Fri, 21 Feb 2025 05:59:25 -0500 Subject: [PATCH 12/22] Feature: watch manifest and reload for dev extension (#181) * feat(desktop): add file system watch support for extensions - Enable file system watching for package.json and test files - Update Cargo.toml to include file system watch feature - Add console logging for file system events * feat(desktop): add extension auto reload mechanism for dev extensions - Implement `reloadExtension` method in extensions store - Add Tauri event for reloading a single extension - Update UI worker to watch package.json and trigger extension reload - Add Linux dependencies for building from source in CONTRIBUTING.md --- CONTRIBUTING.md | 11 + Cargo.lock | 293 ++++++++++++++---- apps/desktop/src-tauri/Cargo.toml | 14 +- apps/desktop/src/lib/stores/extensions.ts | 18 ++ apps/desktop/src/lib/types/fs.ts | 12 + apps/desktop/src/lib/utils/init.ts | 7 +- apps/desktop/src/lib/utils/tauri-events.ts | 10 + .../app/extension/ui-worker/+page.svelte | 30 +- 8 files changed, 321 insertions(+), 74 deletions(-) create mode 100644 apps/desktop/src/lib/types/fs.ts diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index ec27d75..25a1ecd 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -26,6 +26,8 @@ If you are interested in contributing to the project, please read the following - [cmake](https://cmake.org/) - MacOS: `brew install cmake` - Linux: `sudo apt install -y cmake` +- Other Linux Dependencies + - `sudo apt-get install -y protobuf-compiler libwebkit2gtk-4.1-dev libappindicator3-dev librsvg2-dev patchelf libxdo-dev` ### Setup @@ -44,6 +46,15 @@ cd apps/desktop pnpm tauri dev ``` +### Build from Source + +If you have problem running the app, consider building from source to see if it works. + +```bash +cd apps/desktop +pnpm tauri build +``` + ## i188n If you are willing to help with the translation, please use translations in json files in `apps/desktop/messages`. diff --git a/Cargo.lock b/Cargo.lock index a7a5475..b39534d 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -837,7 +837,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "bf4918709cc4dd777ad2b6303ed03cb37f3ca0ccede8c1b0d28ac6db8f4710e0" dependencies = [ "once_cell", - "proc-macro-crate 2.0.2", + "proc-macro-crate 2.0.0", "proc-macro2", "quote", "syn 2.0.87", @@ -1013,26 +1013,26 @@ dependencies = [ [[package]] name = "cargo_metadata" -version = "0.18.1" +version = "0.19.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "2d886547e41f740c616ae73108f6eb70afe6d940c7bc697cb30f13daec073037" +checksum = "8769706aad5d996120af43197bf46ef6ad0fda35216b4505f926a365a232d924" dependencies = [ "camino", "cargo-platform", "semver", "serde", "serde_json", - "thiserror 1.0.66", + "thiserror 2.0.3", ] [[package]] name = "cargo_toml" -version = "0.17.2" +version = "0.21.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "8a969e13a7589e9e3e4207e153bae624ade2b5622fb4684a4923b23ec3d57719" +checksum = "5fbd1fe9db3ebf71b89060adaf7b0504c2d6a425cf061313099547e382c2e472" dependencies = [ "serde", - "toml 0.8.2", + "toml 0.8.20", ] [[package]] @@ -1882,6 +1882,15 @@ dependencies = [ "dirs-sys 0.4.1", ] +[[package]] +name = "dirs" +version = "6.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c3e8aa94d75141228480295a7d0e7feb620b1a5ad9f12bc40be62411e38cce4e" +dependencies = [ + "dirs-sys 0.5.0", +] + [[package]] name = "dirs-next" version = "2.0.0" @@ -1899,7 +1908,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "1b1d1d91c932ef41c0f2663aa8b0ca0342d444d842c06914aa0a7e352d0bada6" dependencies = [ "libc", - "redox_users", + "redox_users 0.4.6", "winapi", ] @@ -1911,10 +1920,22 @@ checksum = "520f05a5cbd335fae5a99ff7a6ab8627577660ee5cfd6a94a6a929b52ff0321c" dependencies = [ "libc", "option-ext", - "redox_users", + "redox_users 0.4.6", "windows-sys 0.48.0", ] +[[package]] +name = "dirs-sys" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "e01a3366d27ee9890022452ee61b2b63a67e6f13f58900b651ff5665f0bb1fab" +dependencies = [ + "libc", + "option-ext", + "redox_users 0.5.0", + "windows-sys 0.59.0", +] + [[package]] name = "dirs-sys-next" version = "0.1.2" @@ -1922,7 +1943,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "4ebda144c4fe02d1f7ea1a7d9641b6fc6b580adcfa024ae48797ecdeb6825b4d" dependencies = [ "libc", - "redox_users", + "redox_users 0.4.6", "winapi", ] @@ -2098,7 +2119,7 @@ dependencies = [ "cc", "memchr", "rustc_version", - "toml 0.8.2", + "toml 0.8.20", "vswhom", "winreg 0.52.0", ] @@ -2336,6 +2357,15 @@ dependencies = [ "rustc_version", ] +[[package]] +name = "file-id" +version = "0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "6bc904b9bbefcadbd8e3a9fb0d464a9b979de6324c03b3c663e8994f46a5be36" +dependencies = [ + "windows-sys 0.52.0", +] + [[package]] name = "filetime" version = "0.2.25" @@ -2439,6 +2469,15 @@ version = "1.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "42703706b716c37f96a77aea830392ad231f44c9e9a67872fa5548707e11b11c" +[[package]] +name = "fsevent-sys" +version = "4.1.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "76ee7a02da4d231650c7cea31349b889be2f45ddb3ef3032d2ec8185f6313fd2" +dependencies = [ + "libc", +] + [[package]] name = "funty" version = "2.0.0" @@ -2720,6 +2759,18 @@ dependencies = [ "wasm-bindgen", ] +[[package]] +name = "getrandom" +version = "0.3.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "43a49c392881ce6d5c3b8cb70f98717b7c07aabbdff06687b9030dbfbe2725f8" +dependencies = [ + "cfg-if", + "libc", + "wasi 0.13.3+wasi-0.2.2", + "windows-targets 0.52.6", +] + [[package]] name = "ghash" version = "0.5.1" @@ -2808,7 +2859,7 @@ source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0bb0228f477c0900c880fd78c8759b95c7636dbd7842707f49e132378aa2acdc" dependencies = [ "heck 0.4.1", - "proc-macro-crate 2.0.2", + "proc-macro-crate 2.0.0", "proc-macro-error", "proc-macro2", "quote", @@ -3769,6 +3820,26 @@ dependencies = [ "windows-sys 0.59.0", ] +[[package]] +name = "kqueue" +version = "1.0.8" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "7447f1ca1b7b563588a205fe93dea8df60fd981423a768bc1c0ded35ed147d0c" +dependencies = [ + "kqueue-sys", + "libc", +] + +[[package]] +name = "kqueue-sys" +version = "1.0.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "ed9625ffda8729b85e45cf04090035ac368927b8cebc34898e7c120f52e4838b" +dependencies = [ + "bitflags 1.3.2", + "libc", +] + [[package]] name = "kuchikiki" version = "0.8.2" @@ -4263,6 +4334,7 @@ checksum = "80e04d1dcff3aae0704555fe5fee3bcfaf3d1fdf8a7e521d5b9d2b42acb52cec" dependencies = [ "hermit-abi 0.3.9", "libc", + "log", "wasi 0.11.0+wasi-snapshot-preview1", "windows-sys 0.52.0", ] @@ -4448,6 +4520,38 @@ version = "0.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "0676bb32a98c1a483ce53e500a81ad9c3d5b3f7c920c28c24e9cb0980d0b5bc8" +[[package]] +name = "notify" +version = "7.0.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "c533b4c39709f9ba5005d8002048266593c1cfaf3c5f0739d5b8ab0c6c504009" +dependencies = [ + "bitflags 2.6.0", + "filetime", + "fsevent-sys", + "inotify", + "kqueue", + "libc", + "log", + "mio 1.0.2", + "notify-types", + "walkdir", + "windows-sys 0.52.0", +] + +[[package]] +name = "notify-debouncer-full" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "9dcf855483228259b2353f89e99df35fc639b2b2510d1166e4858e3f67ec1afb" +dependencies = [ + "file-id", + "log", + "notify", + "notify-types", + "walkdir", +] + [[package]] name = "notify-rust" version = "4.11.3" @@ -4461,6 +4565,16 @@ dependencies = [ "zbus", ] +[[package]] +name = "notify-types" +version = "1.0.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "585d3cb5e12e01aed9e8a1f70d5c6b5e86fe2a6e48fc8cd0b3e0b8df6f6eb174" +dependencies = [ + "instant", + "serde", +] + [[package]] name = "ntapi" version = "0.4.1" @@ -4596,7 +4710,7 @@ version = "0.7.3" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "af1844ef2428cc3e1cb900be36181049ef3d3193c63e43026cfe202983b27a56" dependencies = [ - "proc-macro-crate 2.0.2", + "proc-macro-crate 1.3.1", "proc-macro2", "quote", "syn 2.0.87", @@ -5419,11 +5533,10 @@ dependencies = [ [[package]] name = "proc-macro-crate" -version = "2.0.2" +version = "2.0.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "b00f26d3400549137f92511a46ac1cd8ce37cb5598a96d382381458b992a5d24" +checksum = "7e8366a6159044a37876a2b9817124296703c586a5c92e2c53751fa06d8d43e8" dependencies = [ - "toml_datetime", "toml_edit 0.20.2", ] @@ -5892,6 +6005,17 @@ dependencies = [ "thiserror 1.0.66", ] +[[package]] +name = "redox_users" +version = "0.5.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dd6f9d3d47bdd2ad6945c5015a226ec6155d0bcdfd8f7cd29f86b71f8de99d2b" +dependencies = [ + "getrandom 0.2.15", + "libredox", + "thiserror 2.0.3", +] + [[package]] name = "regex" version = "1.11.1" @@ -7161,15 +7285,15 @@ dependencies = [ "cfg-expr", "heck 0.5.0", "pkg-config", - "toml 0.8.2", + "toml 0.8.20", "version-compare", ] [[package]] name = "tao" -version = "0.30.8" +version = "0.31.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6682a07cf5bab0b8a2bd20d0a542917ab928b5edb75ebd4eda6b05cbaab872da" +checksum = "3731d04d4ac210cd5f344087733943b9bfb1a32654387dad4d1c70de21aee2c9" dependencies = [ "bitflags 2.6.0", "cocoa 0.26.0", @@ -7182,7 +7306,6 @@ dependencies = [ "gdkwayland-sys", "gdkx11-sys", "gtk", - "instant", "jni", "lazy_static", "libc", @@ -7240,13 +7363,13 @@ checksum = "61c41af27dd6d1e27b1b16b489db798443478cef1f06a660c96db617ba5de3b1" [[package]] name = "tauri" -version = "2.1.1" +version = "2.2.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e545de0a2dfe296fa67db208266cd397c5a55ae782da77973ef4c4fac90e9f2c" +checksum = "58a998b6be84104ca05c7e9a21f2180ddec020c8b84ea59a8fc8530a2a19588d" dependencies = [ "anyhow", "bytes", - "dirs 5.0.1", + "dirs 6.0.0", "dunce", "embed_plist", "futures-util", @@ -7291,9 +7414,9 @@ dependencies = [ [[package]] name = "tauri-build" -version = "2.0.3" +version = "2.0.5" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7bd2a4bcfaf5fb9f4be72520eefcb61ae565038f8ccba2a497d8c28f463b8c01" +checksum = "8e950124f6779c6cf98e3260c7a6c8488a74aa6350dd54c6950fdaa349bca2df" dependencies = [ "anyhow", "cargo_toml", @@ -7307,15 +7430,15 @@ dependencies = [ "serde_json", "tauri-utils", "tauri-winres", - "toml 0.8.2", + "toml 0.8.20", "walkdir", ] [[package]] name = "tauri-codegen" -version = "2.0.3" +version = "2.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "bf79faeecf301d3e969b1fae977039edb77a4c1f25cc0a961be298b54bff97cf" +checksum = "f77894f9ddb5cb6c04fcfe8c8869ebe0aded4dabf19917118d48be4a95599ab5" dependencies = [ "base64 0.22.1", "brotli", @@ -7350,9 +7473,9 @@ dependencies = [ [[package]] name = "tauri-macros" -version = "2.0.3" +version = "2.0.4" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "c52027c8c5afb83166dacddc092ee8fff50772f9646d461d8c33ee887e447a03" +checksum = "3240a5caed760a532e8f687be6f05b2c7d11a1d791fb53ccc08cfeb3e5308736" dependencies = [ "heck 0.5.0", "proc-macro2", @@ -7375,7 +7498,7 @@ dependencies = [ "serde", "serde_json", "tauri-utils", - "toml 0.8.2", + "toml 0.8.20", "walkdir", ] @@ -7470,6 +7593,8 @@ dependencies = [ "anyhow", "dunce", "glob", + "notify", + "notify-debouncer-full", "percent-encoding", "schemars", "serde", @@ -7479,7 +7604,7 @@ dependencies = [ "tauri-plugin", "tauri-utils", "thiserror 2.0.3", - "toml 0.8.2", + "toml 0.8.20", "url", "uuid", ] @@ -7501,9 +7626,9 @@ dependencies = [ [[package]] name = "tauri-plugin-http" -version = "2.2.0" +version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "e62a9bde54d6a0218b63f5a248f02056ad4316ba6ad81dfb9e4f73715df5deb1" +checksum = "3a8137a106e0741fdd357366178fc6e0597abe7d20796f53f44171a1bcec1683" dependencies = [ "data-url", "http 1.1.0", @@ -7594,16 +7719,16 @@ dependencies = [ [[package]] name = "tauri-plugin-log" -version = "2.2.0" +version = "2.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "eddd784c138c08a43954bc3e735402e6b2b2ee8d8c254a7391f4e77c01273dd5" +checksum = "367a28a5e0ca39eac98005699466e8906edc4a2a8f8e13a5f1a71dc0bea6c677" dependencies = [ "android_logger", "byte-unit", - "cocoa 0.26.0", "fern", "log", - "objc", + "objc2", + "objc2-foundation", "serde", "serde_json", "serde_repr", @@ -7634,9 +7759,9 @@ dependencies = [ [[package]] name = "tauri-plugin-notification" -version = "2.2.0" +version = "2.2.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "46ab803095f14ac6521fdb6477210a49e86fed6623c3c97d8e4b2b35e045e922" +checksum = "0f8d3ee5207d3359ca2b714545664f24f70374d795bf91f7c1935a494003a57d" dependencies = [ "log", "notify-rust", @@ -7851,9 +7976,9 @@ dependencies = [ [[package]] name = "tauri-runtime" -version = "2.2.0" +version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "cce18d43f80d4aba3aa8a0c953bbe835f3d0f2370aca75e8dbb14bd4bab27958" +checksum = "2274ef891ccc0a8d318deffa9d70053f947664d12d58b9c0d1ae5e89237e01f7" dependencies = [ "dpi", "gtk", @@ -7870,9 +7995,9 @@ dependencies = [ [[package]] name = "tauri-runtime-wry" -version = "2.2.0" +version = "2.3.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9f442a38863e10129ffe2cec7bd09c2dcf8a098a3a27801a476a304d5bb991d2" +checksum = "3707b40711d3b9f6519150869e358ffbde7c57567fb9b5a8b51150606939b2a0" dependencies = [ "gtk", "http 1.1.0", @@ -7896,9 +8021,9 @@ dependencies = [ [[package]] name = "tauri-utils" -version = "2.1.0" +version = "2.1.1" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "9271a88f99b4adea0dc71d0baca4505475a0bbd139fb135f62958721aaa8fe54" +checksum = "96fb10e7cc97456b2d5b9c03e335b5de5da982039a303a20d10006885e4523a0" dependencies = [ "brotli", "cargo_metadata", @@ -7924,7 +8049,7 @@ dependencies = [ "serde_with", "swift-rs", "thiserror 2.0.3", - "toml 0.8.2", + "toml 0.8.20", "url", "urlpattern", "uuid", @@ -8209,21 +8334,21 @@ dependencies = [ [[package]] name = "toml" -version = "0.8.2" +version = "0.8.20" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "185d8ab0dfbb35cf1399a6344d8484209c088f75f8f68230da55d48d95d43e3d" +checksum = "cd87a5cdd6ffab733b2f74bc4fd7ee5fff6634124999ac278c35fc78c6120148" dependencies = [ "serde", "serde_spanned", "toml_datetime", - "toml_edit 0.20.2", + "toml_edit 0.22.24", ] [[package]] name = "toml_datetime" -version = "0.6.3" +version = "0.6.8" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "7cda73e2f1397b1262d6dfdcef8aafae14d1de7748d66822d3bfeeb6d03e5e4b" +checksum = "0dd7358ecb8fc2f8d014bf86f6f638ce72ba252a2c3a2572f2a795f1d23efb41" dependencies = [ "serde", ] @@ -8238,7 +8363,7 @@ dependencies = [ "serde", "serde_spanned", "toml_datetime", - "winnow", + "winnow 0.5.40", ] [[package]] @@ -8246,12 +8371,23 @@ name = "toml_edit" version = "0.20.2" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "396e4d48bbb2b7554c944bde63101b5ae446cff6ec4a24227428f15eb72ef338" +dependencies = [ + "indexmap 2.6.0", + "toml_datetime", + "winnow 0.5.40", +] + +[[package]] +name = "toml_edit" +version = "0.22.24" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "17b4795ff5edd201c7cd6dca065ae59972ce77d1b80fa0a84d94950ece7d1474" dependencies = [ "indexmap 2.6.0", "serde", "serde_spanned", "toml_datetime", - "winnow", + "winnow 0.7.3", ] [[package]] @@ -8628,11 +8764,11 @@ checksum = "06abde3611657adf66d383f00b093d7faecc7fa57071cce2578660c9f1010821" [[package]] name = "uuid" -version = "1.11.0" +version = "1.14.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "f8c5f0a0af699448548ad1a2fbf920fb4bee257eae39953ba95cb84891a0446a" +checksum = "93d59ca99a559661b96bf898d8fce28ed87935fd2bea9f05983c1464dd6c71b1" dependencies = [ - "getrandom 0.2.15", + "getrandom 0.3.1", "serde", ] @@ -8742,6 +8878,15 @@ version = "0.11.0+wasi-snapshot-preview1" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" +[[package]] +name = "wasi" +version = "0.13.3+wasi-0.2.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "26816d2e1a4a36a2940b96c5296ce403917633dff8f3440e9b236ed6f6bacad2" +dependencies = [ + "wit-bindgen-rt", +] + [[package]] name = "wasm-bindgen" version = "0.2.95" @@ -8947,9 +9092,9 @@ dependencies = [ [[package]] name = "webview2-com" -version = "0.33.0" +version = "0.34.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "6f61ff3d9d0ee4efcb461b14eb3acfda2702d10dc329f339303fc3e57215ae2c" +checksum = "823e7ebcfaea51e78f72c87fc3b65a1e602c321f407a0b36dbb327d7bb7cd921" dependencies = [ "webview2-com-macros", "webview2-com-sys", @@ -8972,9 +9117,9 @@ dependencies = [ [[package]] name = "webview2-com-sys" -version = "0.33.0" +version = "0.34.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "a3a3e2eeb58f82361c93f9777014668eb3d07e7d174ee4c819575a9208011886" +checksum = "7a82bce72db6e5ee83c68b5de1e2cd6ea195b9fbff91cb37df5884cbe3222df4" dependencies = [ "thiserror 1.0.66", "windows 0.58.0", @@ -9507,6 +9652,15 @@ dependencies = [ "memchr", ] +[[package]] +name = "winnow" +version = "0.7.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e7f4ea97f6f78012141bcdb6a216b2609f0979ada50b20ca5b52dde2eac2bb1" +dependencies = [ + "memchr", +] + [[package]] name = "winreg" version = "0.10.1" @@ -9537,10 +9691,19 @@ dependencies = [ ] [[package]] -name = "wry" -version = "0.47.0" +name = "wit-bindgen-rt" +version = "0.33.0" source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "553ca1ce149982123962fac2506aa75b8b76288779a77e72b12fa2fc34938647" +checksum = "3268f3d866458b787f390cf61f4bbb563b922d091359f9608842999eaee3943c" +dependencies = [ + "bitflags 2.6.0", +] + +[[package]] +name = "wry" +version = "0.48.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a2e33c08b174442ff80d5c791020696f9f8b4e4a87b8cfc7494aad6167ec44e1" dependencies = [ "base64 0.22.1", "block2", @@ -9568,7 +9731,7 @@ dependencies = [ "sha2", "soup3", "tao-macros", - "thiserror 1.0.66", + "thiserror 2.0.3", "url", "webkit2gtk", "webkit2gtk-sys", diff --git a/apps/desktop/src-tauri/Cargo.toml b/apps/desktop/src-tauri/Cargo.toml index 7b6b186..431c82d 100644 --- a/apps/desktop/src-tauri/Cargo.toml +++ b/apps/desktop/src-tauri/Cargo.toml @@ -14,10 +14,10 @@ name = "kunkun_lib" crate-type = ["staticlib", "cdylib", "rlib"] [build-dependencies] -tauri-build = { version = "2.0.3", features = [] } +tauri-build = { version = "2.0.5", features = [] } [dependencies] -tauri = { version = "2.1.1", features = [ +tauri = { version = "2.2.5", features = [ "macos-private-api", "image-png", "image-ico", @@ -35,11 +35,11 @@ log = { workspace = true } urlencoding = "2.1.3" tauri-plugin-process = "2.2.0" tauri-plugin-shellx = "2.0.12" -tauri-plugin-fs = "2.2.0" +tauri-plugin-fs = { version = "2.2.0", features = ["watch"] } tauri-plugin-dialog = "2.2.0" -tauri-plugin-notification = "2.2.0" +tauri-plugin-notification = "2.2.1" tauri-plugin-os = "2.2.0" -tauri-plugin-http = "2.2.0" +tauri-plugin-http = "2.3.0" tauri-plugin-upload = { workspace = true } # tauri-plugin-upload = "2.2.1" tauri-plugin-jarvis = { workspace = true } @@ -50,10 +50,10 @@ tauri-plugin-user-input = { workspace = true } tauri-plugin-clipboard = { workspace = true } tauri-plugin-store = "2.2.0" tauri-plugin-deep-link = "2.2.0" -tauri-plugin-log = { version = "2.2.0", features = ["colored"] } +tauri-plugin-log = { version = "2.2.1", features = ["colored"] } crypto = { workspace = true } zip = "2.2.2" -uuid = "1.11.0" +uuid = "1.14.0" # tauri-plugin-devtools = "2.0.0" obfstr = { workspace = true } base64 = { workspace = true } diff --git a/apps/desktop/src/lib/stores/extensions.ts b/apps/desktop/src/lib/stores/extensions.ts index e3bd0c7..d1da587 100644 --- a/apps/desktop/src/lib/stores/extensions.ts +++ b/apps/desktop/src/lib/stores/extensions.ts @@ -29,6 +29,7 @@ function createExtensionsStore(): Writable & { tarballUrl: string, extras?: { overwritePackageJson?: string } ) => Promise + reloadExtension: (extPath: string) => Promise } { const store = writable([]) @@ -42,6 +43,22 @@ function createExtensionsStore(): Writable & { }) } + // if dev extension's package.json is changed, use this function to reload commands + async function reloadExtension(extPath: string) { + const ext = get(extensions).find((ext) => ext.extPath === extPath) + if (ext) { + const pkgJsonPath = await path.join(extPath, "package.json") + const ext = await extAPI.loadExtensionManifestFromDisk(pkgJsonPath) + // replace the old extension with the new one + store.update((exts) => { + // filter out the old extension + return [...exts.filter((e) => e.extPath !== extPath), ext] + }) + } else { + console.warn(`reloadExtension: Extension ${extPath} not found`) + } + } + /** * Get all extensions installed from the store (non-dev extensions) */ @@ -195,6 +212,7 @@ function createExtensionsStore(): Writable & { return { ...store, init, + reloadExtension, getExtensionsFromStore, findStoreExtensionByIdentifier, registerNewExtensionByPath, diff --git a/apps/desktop/src/lib/types/fs.ts b/apps/desktop/src/lib/types/fs.ts new file mode 100644 index 0000000..1204098 --- /dev/null +++ b/apps/desktop/src/lib/types/fs.ts @@ -0,0 +1,12 @@ +import * as v from "valibot" + +export const WatchEvent = v.object({ + type: v.object({ + modify: v.object({ + kind: v.union([v.literal("data"), v.literal("metadata")]), + mode: v.union([v.literal("any"), v.literal("content")]) + }) + }), + paths: v.array(v.string()) +}) +export type WatchEvent = v.InferOutput diff --git a/apps/desktop/src/lib/utils/init.ts b/apps/desktop/src/lib/utils/init.ts index 85480d0..bd8751c 100644 --- a/apps/desktop/src/lib/utils/init.ts +++ b/apps/desktop/src/lib/utils/init.ts @@ -1,8 +1,9 @@ -import { appConfig } from "@/stores" +import { appConfig, extensions } from "@/stores" import { getCurrentWindow } from "@tauri-apps/api/window" import { info } from "@tauri-apps/plugin-log" import { dev } from "$app/environment" import { mapKeyToTauriKey, registerAppHotkey } from "./hotkey" +import { listenToReloadOneExtension } from "./tauri-events" /** * Initialize the app @@ -11,6 +12,10 @@ export function init() { const window = getCurrentWindow() if (window.label === "main") { initMainWindow() + listenToReloadOneExtension(({ payload: { extPath } }) => { + info(`listenToReloadOneExtension in main window: ${extPath}`) + extensions.reloadExtension(extPath) + }) } if (!dev) { diff --git a/apps/desktop/src/lib/utils/tauri-events.ts b/apps/desktop/src/lib/utils/tauri-events.ts index 1102342..8350ef4 100644 --- a/apps/desktop/src/lib/utils/tauri-events.ts +++ b/apps/desktop/src/lib/utils/tauri-events.ts @@ -16,6 +16,8 @@ export const FileDragOver = "tauri://drag-over" export const NewClipboardItemAddedEvent = "new_clipboard_item_added" export const RefreshConfigEvent = "kunkun://refresh-config" export const RefreshExtEvent = "kunkun://refresh-extensions" +export const ReloadOneExtensionEvent = "kunkun://reload-one-extension" + export function listenToFileDrop(cb: EventCallback<{ paths: string[] }>) { return listen<{ paths: string[] }>(FileDragDrop, cb) } @@ -55,3 +57,11 @@ export function emitRefreshDevExt() { export function listenToRefreshDevExt(cb: EventCallback) { return listen(DEEP_LINK_PATH_REFRESH_DEV_EXTENSION, cb) } + +export function emitReloadOneExtension(extPath: string) { + return emitTo("main", ReloadOneExtensionEvent, { extPath }) +} + +export function listenToReloadOneExtension(cb: EventCallback<{ extPath: string }>) { + return listen(ReloadOneExtensionEvent, cb) +} diff --git a/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte b/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte index cb6810f..d5bbbaf 100644 --- a/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte +++ b/apps/desktop/src/routes/app/extension/ui-worker/+page.svelte @@ -3,8 +3,13 @@ import { appState } from "@/stores/appState.js" import { keys } from "@/stores/keys" import { winExtMap } from "@/stores/winExtMap.js" + import { WatchEvent } from "@/types/fs.js" import { helperAPI } from "@/utils/helper.js" - import { listenToFileDrop, listenToRefreshDevExt } from "@/utils/tauri-events.js" + import { + emitReloadOneExtension, + listenToFileDrop, + listenToRefreshDevExt + } from "@/utils/tauri-events.js" import { isInMainWindow } from "@/utils/window.js" import { db } from "@kksh/api/commands" import { @@ -28,6 +33,7 @@ import type { IKunkunFullServerAPI } from "@kunkunapi/src/api/server" import type { UnlistenFn } from "@tauri-apps/api/event" import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow" + import * as fs from "@tauri-apps/plugin-fs" import { readTextFile } from "@tauri-apps/plugin-fs" import { debug } from "@tauri-apps/plugin-log" import { platform } from "@tauri-apps/plugin-os" @@ -57,6 +63,7 @@ let loaded = $state(false) let listview: Templates.ListView | undefined = $state(undefined) const _platform = platform() + let unlistenPkgJsonWatch: UnlistenFn | undefined async function goBack() { if (isInMainWindow()) { @@ -242,6 +249,22 @@ worker?.terminate() } }) + + function onPkgJsonChange(evt: fs.WatchEvent) { + const parsed = v.safeParse(WatchEvent, evt) + if (parsed.success) { + if ( + parsed.output.type.modify.kind === "data" && + parsed.output.type.modify.mode === "content" && + parsed.output.paths.includes(data.pkgJsonPath) + ) { + console.log("pkgJson changed", parsed.output.paths) + // emit event to reload extension commands + emitReloadOneExtension(loadedExt.extPath) + } + } + } + onMount(async () => { setTimeout(() => { appState.setLoadingBar(true) @@ -261,11 +284,16 @@ appState.setLoadingBar(false) loaded = true }, 500) + console.log("watching", data.pkgJsonPath) + fs.watch(data.pkgJsonPath, onPkgJsonChange).then((unlisten) => { + unlistenPkgJsonWatch = unlisten + }) }) onDestroy(() => { unlistenRefreshWorkerExt?.() unlistenFileDrop?.() + unlistenPkgJsonWatch?.() winExtMap.unregisterExtensionFromWindow(appWin.label) extensionLoadingBar = false appState.setActionPanel(undefined) From ec951bfc808e5147db4d7b6c9037724935b24ab7 Mon Sep 17 00:00:00 2001 From: Luca Giannini <68999840+structwafel@users.noreply.github.com> Date: Fri, 21 Feb 2025 12:28:08 +0100 Subject: [PATCH 13/22] add toggle cli command to show/hide kunkun (#176) * add toggle cli command * cleaner * keep unused imports --- apps/desktop/src-tauri/src/lib.rs | 21 ++++++++++++++++----- 1 file changed, 16 insertions(+), 5 deletions(-) diff --git a/apps/desktop/src-tauri/src/lib.rs b/apps/desktop/src-tauri/src/lib.rs index 8ccf218..c30f386 100644 --- a/apps/desktop/src-tauri/src/lib.rs +++ b/apps/desktop/src-tauri/src/lib.rs @@ -72,11 +72,22 @@ pub fn run() { } let shell_unlocked = true; builder = builder - .plugin(tauri_plugin_single_instance::init(|app, args, cwd| { - let _ = app - .get_webview_window("main") - .expect("no main window") - .set_focus(); + .plugin(tauri_plugin_single_instance::init(|app, args, _cwd| { + let window = app.get_webview_window("main").expect("no main window"); + + // if toggle is passed, we want to show/hide the main window + if args.get(1).map_or(false, |arg| arg == "toggle") { + if window.is_visible().unwrap_or(false) { + log::info!("hiding main window"); + window.hide().unwrap(); + } else { + log::info!("showing main window"); + window.show().unwrap(); + window.set_focus().unwrap(); + }; + } else { + let _ = window.set_focus(); + } })) .plugin( tauri_plugin_log::Builder::new() From a0bd2d8573a804664c7bffd42c77fd4b14d1152e Mon Sep 17 00:00:00 2001 From: Huakun Date: Sat, 22 Feb 2025 00:41:35 -0500 Subject: [PATCH 14/22] 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 --- apps/desktop/src-tauri/tauri.conf.json | 6 ++- apps/desktop/src/lib/utils/hotkey.ts | 38 +++++++++++++++++++ apps/desktop/src/lib/utils/key.ts | 5 ++- apps/desktop/src/routes/app/+page.svelte | 2 +- .../app/extension/clipboard/+page.svelte | 15 +------- packages/api/src/api/server/shell.ts | 2 +- .../tauri-plugins/jarvis/src/utils/script.rs | 2 + 7 files changed, 52 insertions(+), 18 deletions(-) diff --git a/apps/desktop/src-tauri/tauri.conf.json b/apps/desktop/src-tauri/tauri.conf.json index a617fcc..0fc2bc4 100644 --- a/apps/desktop/src-tauri/tauri.conf.json +++ b/apps/desktop/src-tauri/tauri.conf.json @@ -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 } ] }, diff --git a/apps/desktop/src/lib/utils/hotkey.ts b/apps/desktop/src/lib/utils/hotkey.ts index 11a97ae..dcebd2f 100644 --- a/apps/desktop/src/lib/utils/hotkey.ts +++ b/apps/desktop/src/lib/utils/hotkey.ts @@ -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) + } +} diff --git a/apps/desktop/src/lib/utils/key.ts b/apps/desktop/src/lib/utils/key.ts index 423fbc3..3c27522 100644 --- a/apps/desktop/src/lib/utils/key.ts +++ b/apps/desktop/src/lib/utils/key.ts @@ -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")) diff --git a/apps/desktop/src/routes/app/+page.svelte b/apps/desktop/src/routes/app/+page.svelte index bd9718c..93da636 100644 --- a/apps/desktop/src/routes/app/+page.svelte +++ b/apps/desktop/src/routes/app/+page.svelte @@ -95,7 +95,7 @@ }} /> + 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") diff --git a/packages/api/src/api/server/shell.ts b/packages/api/src/api/server/shell.ts index d994f2d..676c3e1 100644 --- a/packages/api/src/api/server/shell.ts +++ b/packages/api/src/api/server/shell.ts @@ -168,7 +168,7 @@ export function constructShellApi( ShellPermissionMap.executePowershellScript, objectPermissions, "powershell", - ["-Command", script] + ["-WindowStyle", "Hidden", "-Command", script] ) return executePowershellScript(script) } diff --git a/packages/tauri-plugins/jarvis/src/utils/script.rs b/packages/tauri-plugins/jarvis/src/utils/script.rs index 03dfac0..1ea6ff9 100644 --- a/packages/tauri-plugins/jarvis/src/utils/script.rs +++ b/packages/tauri-plugins/jarvis/src/utils/script.rs @@ -17,6 +17,8 @@ pub fn run_apple_script(script: &str) -> anyhow::Result { pub fn run_powershell(script: &str) -> anyhow::Result { let output = Command::new("powershell") + .arg("-WindowStyle") + .arg("Hidden") .arg("-Command") .arg(script) .output() From 872b601338f9958af6fb59b34c7276f4d7c80000 Mon Sep 17 00:00:00 2001 From: Luca Giannini <68999840+structwafel@users.noreply.github.com> Date: Sat, 22 Feb 2025 08:58:41 +0100 Subject: [PATCH 15/22] crosslink issue on linux, and cleanup entire dir (#177) * crosslink issue on linux, and cleanup entire dir * moved copy_dir_all to rust * using dircpy instead of diy due to complications * refactor: move copy_dir_all to jarvis plugin All commands are in jarvis plugin, this is more organized. And this API will be exposed to extensions. --------- Co-authored-by: Huakun Shen --- Cargo.lock | 44 +++++++++++++++++++ apps/desktop/src-tauri/Cargo.toml | 1 + apps/desktop/src-tauri/src/lib.rs | 2 +- packages/api/src/commands/fs.ts | 12 +++++ packages/extension/src/install.ts | 12 ++++- packages/tauri-plugins/jarvis/Cargo.toml | 2 +- packages/tauri-plugins/jarvis/build.rs | 1 + .../tauri-plugins/jarvis/permissions/all.toml | 3 +- .../autogenerated/commands/copy_dir_all.toml | 13 ++++++ .../permissions/autogenerated/reference.md | 26 +++++++++++ .../jarvis/permissions/schemas/schema.json | 10 +++++ .../tauri-plugins/jarvis/src/commands/fs.rs | 8 ++++ packages/tauri-plugins/jarvis/src/lib.rs | 1 + 13 files changed, 130 insertions(+), 5 deletions(-) create mode 100644 packages/tauri-plugins/jarvis/permissions/autogenerated/commands/copy_dir_all.toml diff --git a/Cargo.lock b/Cargo.lock index b39534d..41c3392 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1554,6 +1554,19 @@ dependencies = [ "cfg-if", ] +[[package]] +name = "crossbeam" +version = "0.8.4" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "1137cd7e7fc0fb5d3c5a8678be38ec56e819125d8d7907411fe24ccb943faca8" +dependencies = [ + "crossbeam-channel", + "crossbeam-deque", + "crossbeam-epoch", + "crossbeam-queue", + "crossbeam-utils", +] + [[package]] name = "crossbeam-channel" version = "0.5.13" @@ -1582,6 +1595,15 @@ dependencies = [ "crossbeam-utils", ] +[[package]] +name = "crossbeam-queue" +version = "0.3.12" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0f58bbc28f91df819d0aa2a2c00cd19754769c2fad90579b3592b1c9ba7a3115" +dependencies = [ + "crossbeam-utils", +] + [[package]] name = "crossbeam-utils" version = "0.8.20" @@ -1864,6 +1886,17 @@ dependencies = [ "subtle", ] +[[package]] +name = "dircpy" +version = "0.3.19" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "a88521b0517f5f9d51d11925d8ab4523497dcf947073fa3231a311b63941131c" +dependencies = [ + "jwalk", + "log", + "walkdir", +] + [[package]] name = "dirs" version = "4.0.0" @@ -3782,6 +3815,16 @@ dependencies = [ "serde_json", ] +[[package]] +name = "jwalk" +version = "0.8.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "2735847566356cd2179a2a38264839308f7079fa96e6bd5a42d740460e003c56" +dependencies = [ + "crossbeam", + "rayon", +] + [[package]] name = "k256" version = "0.13.4" @@ -7660,6 +7703,7 @@ dependencies = [ "cocoa 0.24.1", "crypto", "db", + "dircpy", "flate2", "futures-util", "grpc", diff --git a/apps/desktop/src-tauri/Cargo.toml b/apps/desktop/src-tauri/Cargo.toml index 431c82d..5b9443a 100644 --- a/apps/desktop/src-tauri/Cargo.toml +++ b/apps/desktop/src-tauri/Cargo.toml @@ -59,6 +59,7 @@ obfstr = { workspace = true } base64 = { workspace = true } tauri-plugin-stronghold = "2.2.0" + [target."cfg(target_os = \"macos\")".dependencies] cocoa = "0.24.1" mac-security-rs = { workspace = true } diff --git a/apps/desktop/src-tauri/src/lib.rs b/apps/desktop/src-tauri/src/lib.rs index c30f386..d02f488 100644 --- a/apps/desktop/src-tauri/src/lib.rs +++ b/apps/desktop/src-tauri/src/lib.rs @@ -131,7 +131,7 @@ pub fn run() { .plugin(tauri_plugin_network::init()) .plugin(tauri_plugin_system_info::init()) .invoke_handler(tauri::generate_handler![ - commands::keyring::get_stronghold_key + commands::keyring::get_stronghold_key, ]); let app = builder diff --git a/packages/api/src/commands/fs.ts b/packages/api/src/commands/fs.ts index 0a01491..dec063c 100644 --- a/packages/api/src/commands/fs.ts +++ b/packages/api/src/commands/fs.ts @@ -68,3 +68,15 @@ export function unzip( overwrite: options?.overwrite ?? false }) } + +/** + * Copy directory from one location to another in tauri backend + * + * Needed as Linux cannot rename paths across different filesystems + * + * @param from Source directory + * @param to Destination directory + */ +export async function copy_dir_all(from: string, to: string) { + await invoke(generateJarvisPluginCommand("copy_dir_all"), { from, to }) +} diff --git a/packages/extension/src/install.ts b/packages/extension/src/install.ts index 4a01c3b..0b2197f 100644 --- a/packages/extension/src/install.ts +++ b/packages/extension/src/install.ts @@ -3,7 +3,7 @@ * including install, uninstall, upgrade, check app-extension compatibility, etc. */ import { isCompatible } from "@kksh/api" -import { db, decompressTarball } from "@kksh/api/commands" +import { copy_dir_all, db, decompressTarball } from "@kksh/api/commands" import type { ExtPackageJsonExtra } from "@kksh/api/models" import { SBExt } from "@kksh/supabase/models" import { greaterThan, parse as parseSemver } from "@std/semver" @@ -76,7 +76,15 @@ export async function installTarball( } } - await fs.rename(decompressDest, extInstallPath) + // copy all files from decompressDest to extInstallPat + await copy_dir_all(decompressDest, extInstallPath) + + // Clean up temp directory + // we need the actual temp dir, as decompressDest is the /tmp/uuidv4/package dir + const tempDir = await path.dirname(decompressDest) + // tempDir is "/tmp/uuidv4" + await fs.remove(tempDir, { recursive: true }) + await db.createExtension({ identifier: manifest.kunkun.identifier, version: manifest.version, diff --git a/packages/tauri-plugins/jarvis/Cargo.toml b/packages/tauri-plugins/jarvis/Cargo.toml index 27f37d6..32cd8fa 100644 --- a/packages/tauri-plugins/jarvis/Cargo.toml +++ b/packages/tauri-plugins/jarvis/Cargo.toml @@ -56,7 +56,7 @@ grpc = { workspace = true } futures-util = "0.3.31" rayon = { workspace = true } local-ip-address = "0.6.3" - +dircpy = "0.3.19" [target.'cfg(target_os = "macos")'.dependencies] tauri-icns = "0.1.0" diff --git a/packages/tauri-plugins/jarvis/build.rs b/packages/tauri-plugins/jarvis/build.rs index 2b29e07..f3bcbc4 100644 --- a/packages/tauri-plugins/jarvis/build.rs +++ b/packages/tauri-plugins/jarvis/build.rs @@ -50,6 +50,7 @@ const COMMANDS: &[&str] = &[ "decompress_tarball", "compress_tarball", "unzip", + "copy_dir_all", /* -------------------------------------------------------------------------- */ /* File Search */ /* -------------------------------------------------------------------------- */ diff --git a/packages/tauri-plugins/jarvis/permissions/all.toml b/packages/tauri-plugins/jarvis/permissions/all.toml index 011b085..3e5222d 100644 --- a/packages/tauri-plugins/jarvis/permissions/all.toml +++ b/packages/tauri-plugins/jarvis/permissions/all.toml @@ -49,9 +49,10 @@ commands.allow = [ # File System "decompress_tarball", "compress_tarball", + "copy_dir_all", + "unzip", # File Search "file_search", - "unzip", # Path "get_default_extensions_dir", "get_default_extensions_storage_dir", diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/copy_dir_all.toml b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/copy_dir_all.toml new file mode 100644 index 0000000..5199628 --- /dev/null +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/commands/copy_dir_all.toml @@ -0,0 +1,13 @@ +# Automatically generated - DO NOT EDIT! + +"$schema" = "../../schemas/schema.json" + +[[permission]] +identifier = "allow-copy-dir-all" +description = "Enables the copy_dir_all command without any pre-configured scope." +commands.allow = ["copy_dir_all"] + +[[permission]] +identifier = "deny-copy-dir-all" +description = "Denies the copy_dir_all command without any pre-configured scope." +commands.deny = ["copy_dir_all"] diff --git a/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md b/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md index 19f9805..43a78fd 100644 --- a/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md +++ b/packages/tauri-plugins/jarvis/permissions/autogenerated/reference.md @@ -154,6 +154,32 @@ Denies the compress_tarball command without any pre-configured scope.
+`jarvis:allow-copy-dir-all` + + + +Enables the copy_dir_all command without any pre-configured scope. + +
+ +`jarvis:deny-copy-dir-all` + + + +Denies the copy_dir_all command without any pre-configured scope. + +
+ `jarvis:allow-create-command`