From 84b82f47a4cf771d103fdfa1b52d653aac396a23 Mon Sep 17 00:00:00 2001 From: Huakun Shen Date: Thu, 21 Nov 2024 07:04:48 -0500 Subject: [PATCH] Improve Icon Multiplexer (#31) * feat: improve IconMultiplexer * feat: add built-in command "reload extensions" * bump package version * feat: replace @gcornut/valibot-json-schema with @valibot/to-json-schema * fix: some icon bug * fix: import path * ci: debug --- .changeset/config.json | 25 ++-- apps/cli/CHANGELOG.md | 7 + apps/cli/package.json | 2 +- apps/create-kunkun/CHANGELOG.md | 7 + apps/create-kunkun/build.ts | 1 + apps/create-kunkun/package.json | 2 +- apps/desktop/src/lib/cmds/builtin.ts | 129 ++++++++++++++--- .../standalone/settings/AddDevExtForm.svelte | 8 +- apps/desktop/src/routes/+layout.svelte | 3 - package.json | 2 +- packages/api/CHANGELOG.md | 7 + packages/api/package.json | 2 +- packages/api/src/commands/index.ts | 3 +- packages/api/src/commands/system.ts | 134 ++++++++++++++---- packages/api/src/models/icon.ts | 51 ++++--- packages/api/src/ui/server/deno.ts | 44 +++--- packages/api/src/ui/worker/components/icon.ts | 14 +- packages/api/src/version.ts | 2 +- packages/schema/package.json | 4 +- packages/schema/src/index.ts | 4 +- .../components/common/IconMultiplexer.svelte | 54 ++++++- .../extension/templates/list-view.svelte | 4 - .../ui/src/components/main/BuiltinCmds.svelte | 9 +- .../ui/src/components/main/SystemCmds.svelte | 10 +- packages/ui/src/components/main/types.ts | 2 +- packages/ui/src/utils/style.ts | 5 + pnpm-lock.yaml | 45 ++---- 27 files changed, 412 insertions(+), 168 deletions(-) create mode 100644 packages/api/CHANGELOG.md create mode 100644 packages/ui/src/utils/style.ts diff --git a/.changeset/config.json b/.changeset/config.json index edef2eb..9bdc114 100644 --- a/.changeset/config.json +++ b/.changeset/config.json @@ -1,11 +1,18 @@ { - "$schema": "https://unpkg.com/@changesets/config@3.0.3/schema.json", - "changelog": "@changesets/cli/changelog", - "commit": false, - "fixed": [], - "linked": [], - "access": "restricted", - "baseBranch": "main", - "updateInternalDependencies": "patch", - "ignore": [] + "$schema": "https://unpkg.com/@changesets/config@3.0.3/schema.json", + "changelog": "@changesets/cli/changelog", + "commit": false, + "fixed": [], + "linked": [], + "access": "restricted", + "baseBranch": "develop", + "updateInternalDependencies": "patch", + "ignore": [ + "@kksh/desktop", + "@kksh/supabase", + "@kksh/utils", + "@kksh/extension", + "@kksh/schema", + "@kksh/supabase" + ] } diff --git a/apps/cli/CHANGELOG.md b/apps/cli/CHANGELOG.md index 4cab5eb..bfd1e1a 100644 --- a/apps/cli/CHANGELOG.md +++ b/apps/cli/CHANGELOG.md @@ -1,5 +1,12 @@ # kksh +## 0.0.25 + +### Patch Changes + +- Updated dependencies + - @kksh/api@0.0.43 + ## 0.0.22 ### Patch Changes diff --git a/apps/cli/package.json b/apps/cli/package.json index 8e87cb9..8dd72aa 100644 --- a/apps/cli/package.json +++ b/apps/cli/package.json @@ -1,7 +1,7 @@ { "name": "kksh", "module": "dist/cli.js", - "version": "0.0.24", + "version": "0.0.25", "type": "module", "bin": { "kksh": "./dist/cli.js", diff --git a/apps/create-kunkun/CHANGELOG.md b/apps/create-kunkun/CHANGELOG.md index 89ccbf5..1326623 100644 --- a/apps/create-kunkun/CHANGELOG.md +++ b/apps/create-kunkun/CHANGELOG.md @@ -1,5 +1,12 @@ # create-kunkun +## 0.1.35 + +### Patch Changes + +- Updated dependencies + - @kksh/api@0.0.43 + ## 0.1.30 ### Patch Changes diff --git a/apps/create-kunkun/build.ts b/apps/create-kunkun/build.ts index 6535af8..370246a 100644 --- a/apps/create-kunkun/build.ts +++ b/apps/create-kunkun/build.ts @@ -31,6 +31,7 @@ fs.emptyDirSync(tmpDistTemplatesPath) console.log(getRootDir()) const templatesPath = path.join(getRootDir(), "../..", "packages/templates") +console.log(`Copy from ${templatesPath} to ${tmpDistTemplatesPath}`) fs.copySync(templatesPath, tmpDistTemplatesPath, { dereference: os.platform() === "win32" }) /* -------------------------------------------------------------------------- */ diff --git a/apps/create-kunkun/package.json b/apps/create-kunkun/package.json index dd15548..314e834 100644 --- a/apps/create-kunkun/package.json +++ b/apps/create-kunkun/package.json @@ -1,7 +1,7 @@ { "name": "create-kunkun", "type": "module", - "version": "0.1.34", + "version": "0.1.35", "bin": { "create-kunkun": "dist/index.mjs" }, diff --git a/apps/desktop/src/lib/cmds/builtin.ts b/apps/desktop/src/lib/cmds/builtin.ts index cb70ae6..888cefb 100644 --- a/apps/desktop/src/lib/cmds/builtin.ts +++ b/apps/desktop/src/lib/cmds/builtin.ts @@ -1,5 +1,6 @@ -import { appConfig, appState, auth } from "@/stores" +import { appConfig, appState, auth, extensions } from "@/stores" import { checkUpdateAndInstall } from "@/utils/updater" +import { IconEnum } from "@kksh/api/models" import type { BuiltinCmd } from "@kksh/ui/types" import { getVersion } from "@tauri-apps/api/app" import { WebviewWindow } from "@tauri-apps/api/webviewWindow" @@ -10,11 +11,15 @@ import { toast } from "svelte-sonner" import { derived } from "svelte/store" import * as clipboard from "tauri-plugin-clipboard-api" import { v4 as uuidv4 } from "uuid" +import { hexColor } from "valibot" export const rawBuiltinCmds: BuiltinCmd[] = [ { name: "Store", - iconifyIcon: "streamline:store-2-solid", + icon: { + type: IconEnum.Iconify, + value: "streamline:store-2-solid" + }, description: "Go to Extension Store", function: async () => { appState.clearSearchTerm() @@ -23,7 +28,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Sign In", - iconifyIcon: "mdi:login-variant", + icon: { + type: IconEnum.Iconify, + value: "mdi:login-variant" + }, description: "", function: async () => { goto("/auth") @@ -31,7 +39,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Sign Out", - iconifyIcon: "mdi:logout-variant", + icon: { + type: IconEnum.Iconify, + value: "mdi:logout-variant" + }, description: "", function: async () => { auth @@ -42,7 +53,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Show Draggable Area", - iconifyIcon: "mingcute:move-fill", + icon: { + type: IconEnum.Iconify, + value: "mingcute:move-fill" + }, description: "", function: async () => { // select all html elements with attribute data-tauri-drag-region @@ -59,7 +73,11 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Add Dev Extension", - iconifyIcon: "lineicons:dev", + icon: { + type: IconEnum.Iconify, + value: "lineicons:dev", + hexColor: "#0f0" + }, description: "", function: async () => { appState.clearSearchTerm() @@ -68,7 +86,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Kunkun Version", - iconifyIcon: "stash:version-solid", + icon: { + type: IconEnum.Iconify, + value: "stash:version-solid" + }, description: "", function: async () => { toast.success(`Kunkun Version: ${await getVersion()}`) @@ -76,7 +97,11 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Set Dev Extension Path", - iconifyIcon: "lineicons:dev", + icon: { + type: IconEnum.Iconify, + value: "lineicons:dev", + hexColor: "#0f0" + }, description: "", function: async () => { // const appStateStore = useAppStateStore() @@ -86,7 +111,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Extension Window Troubleshooter", - iconifyIcon: "material-symbols:window-outline", + icon: { + type: IconEnum.Iconify, + value: "material-symbols:window-outline" + }, description: "", function: async () => { appState.clearSearchTerm() @@ -102,7 +130,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Extension Permission Inspector", - iconifyIcon: "hugeicons:inspect-code", + icon: { + type: IconEnum.Iconify, + value: "hugeicons:inspect-code" + }, description: "", function: async () => { appState.clearSearchTerm() @@ -112,7 +143,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Extension Loading Troubleshooter", - iconifyIcon: "material-symbols:troubleshoot", + icon: { + type: IconEnum.Iconify, + value: "material-symbols:troubleshoot" + }, description: "", function: async () => { appState.clearSearchTerm() @@ -122,7 +156,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Create Quicklink", - iconifyIcon: "material-symbols:link", + icon: { + type: IconEnum.Iconify, + value: "material-symbols:link" + }, description: "Create a Quicklink", function: async () => { appState.clearSearchTerm() @@ -131,7 +168,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Settings", - iconifyIcon: "solar:settings-linear", + icon: { + type: IconEnum.Iconify, + value: "solar:settings-linear" + }, description: "Open Settings", function: async () => { goto("/settings") @@ -140,7 +180,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Check Update", - iconifyIcon: "material-symbols:update", + icon: { + type: IconEnum.Iconify, + value: "material-symbols:update" + }, description: "Check for updates", function: async () => { checkUpdateAndInstall() @@ -149,7 +192,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Check Beta Update", - iconifyIcon: "material-symbols:update", + icon: { + type: IconEnum.Iconify, + value: "material-symbols:update" + }, description: "Check for Beta updates", function: async () => { checkUpdateAndInstall({ beta: true }) @@ -158,15 +204,34 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Reload", - iconifyIcon: "tabler:reload", + icon: { + type: IconEnum.Iconify, + value: "tabler:reload" + }, description: "Reload this page", function: async () => { location.reload() } }, + { + name: "Reload Extensions", + icon: { + type: IconEnum.Iconify, + value: "tabler:reload" + }, + description: "Reload Extensions", + function: async () => { + extensions.init().then(() => { + appState.clearSearchTerm() + }) + } + }, { name: "Dance", - iconifyIcon: "mdi:dance-pole", + icon: { + type: IconEnum.Iconify, + value: "mdi:dance-pole" + }, description: "Dance", function: async () => { goto("/dance") @@ -174,7 +239,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Quit Kunkun", - iconifyIcon: "emojione:cross-mark-button", + icon: { + type: IconEnum.Iconify, + value: "emojione:cross-mark-button" + }, description: "Quit Kunkun", function: async () => { exit(0) @@ -182,7 +250,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Toggle Dev Extension HMR", - iconifyIcon: "ri:toggle-line", + icon: { + type: IconEnum.Iconify, + value: "ri:toggle-line" + }, description: "Load dev extensions from their dev server URLs", function: async () => { appConfig.update((config) => { @@ -197,7 +268,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Pin Current Screenshot", - iconifyIcon: "material-symbols:screenshot-monitor-outline", + icon: { + type: IconEnum.Iconify, + value: "material-symbols:screenshot-monitor-outline" + }, description: "Pin the current screenshot", function: async () => { appState.clearSearchTerm() @@ -220,7 +294,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "MDNS Debugger", - iconifyIcon: "material-symbols:wifi-find", + icon: { + type: IconEnum.Iconify, + value: "material-symbols:wifi-find" + }, description: "MDNS Debugger", function: async () => { goto("/troubleshooters/mdns-debugger") @@ -232,7 +309,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Toggle Hide On Blur", - iconifyIcon: "ri:toggle-line", + icon: { + type: IconEnum.Iconify, + value: "ri:toggle-line" + }, description: "Toggle Hide On Blur", function: async () => { appConfig.update((config) => { @@ -247,7 +327,10 @@ export const rawBuiltinCmds: BuiltinCmd[] = [ }, { name: "Toggle Developer Mode", - iconifyIcon: "hugeicons:developer", + icon: { + type: IconEnum.Iconify, + value: "hugeicons:developer" + }, description: "Toggle Developer Mode", function: async () => { appConfig.update((config) => { diff --git a/apps/desktop/src/lib/components/standalone/settings/AddDevExtForm.svelte b/apps/desktop/src/lib/components/standalone/settings/AddDevExtForm.svelte index 4ac9362..b65aa6e 100644 --- a/apps/desktop/src/lib/components/standalone/settings/AddDevExtForm.svelte +++ b/apps/desktop/src/lib/components/standalone/settings/AddDevExtForm.svelte @@ -102,7 +102,6 @@ if (!selected) { return toast.warning("No File Selected") } - console.log(selected) for (const tarballPath of selected) { await extensions.installTarball(tarballPath, $appConfig.devExtensionPath) } @@ -129,7 +128,10 @@ -
+
+
diff --git a/apps/desktop/src/routes/+layout.svelte b/apps/desktop/src/routes/+layout.svelte index fcfae36..bdfba4a 100644 --- a/apps/desktop/src/routes/+layout.svelte +++ b/apps/desktop/src/routes/+layout.svelte @@ -63,9 +63,6 @@ .fixPathEnv() .then(() => { info("fixed path env") - shellx.hasCommand("ffprobe").then((res) => { - console.log("has ffprobe:", res) - }) }) .catch(error) diff --git a/package.json b/package.json index 606a2e8..fca2fd8 100644 --- a/package.json +++ b/package.json @@ -21,7 +21,7 @@ "turbo": "^2.3.0", "typescript": "5.6.3" }, - "packageManager": "pnpm@9.14.1", + "packageManager": "pnpm@9.14.2", "engines": { "node": ">=22" }, diff --git a/packages/api/CHANGELOG.md b/packages/api/CHANGELOG.md new file mode 100644 index 0000000..e2d3bb2 --- /dev/null +++ b/packages/api/CHANGELOG.md @@ -0,0 +1,7 @@ +# @kksh/api + +## 0.0.43 + +### Patch Changes + +- More Icon Options diff --git a/packages/api/package.json b/packages/api/package.json index 5485a47..255ccd4 100644 --- a/packages/api/package.json +++ b/packages/api/package.json @@ -1,6 +1,6 @@ { "name": "@kksh/api", - "version": "0.0.41", + "version": "0.0.43", "type": "module", "exports": { ".": "./src/index.ts", diff --git a/packages/api/src/commands/index.ts b/packages/api/src/commands/index.ts index 4074dc0..a996a29 100644 --- a/packages/api/src/commands/index.ts +++ b/packages/api/src/commands/index.ts @@ -1,9 +1,9 @@ export * from "./apps" export * from "./fs" export * from "./server" -export * from "./system" export * from "./tools" export * from "./extension" +export * from "./system" export * from "./store" export * as db from "./db" export { JarvisExtDB } from "./db" @@ -12,3 +12,4 @@ export * from "./fileSearch" export * from "./utils" export * as macSecurity from "./mac-security" export * from "./mdns" +export * from "./common" diff --git a/packages/api/src/commands/system.ts b/packages/api/src/commands/system.ts index 40746be..64a4ceb 100644 --- a/packages/api/src/commands/system.ts +++ b/packages/api/src/commands/system.ts @@ -1,8 +1,8 @@ +import { generateJarvisPluginCommand } from "@kksh/api/commands" +import { AppInfo, IconEnum, SysCommand } from "@kksh/api/models" import { invoke } from "@tauri-apps/api/core" import { platform } from "@tauri-apps/plugin-os" import { parse } from "valibot" -import { AppInfo, IconEnum, SysCommand } from "../models" -import { generateJarvisPluginCommand } from "./common" export function openTrash(): Promise { return invoke(generateJarvisPluginCommand("open_trash")) @@ -119,56 +119,80 @@ export function getSelectedFilesInFileExplorer(): Promise { export const rawSystemCommands = [ { name: "Open Trash", - icon: "uil:trash", + icon: { + value: "uil:trash", + type: IconEnum.Iconify + }, confirmRequired: false, function: openTrash, platforms: ["macos", "linux", "windows"] }, { name: "Empty Trash", - icon: "uil:trash", + icon: { + value: "uil:trash", + type: IconEnum.Iconify + }, confirmRequired: true, function: emptyTrash, platforms: ["macos", "linux", "windows"] }, { name: "Shutdown", - icon: "mdi:shutdown", + icon: { + value: "mdi:shutdown", + type: IconEnum.Iconify + }, confirmRequired: true, function: shutdown, platforms: ["macos", "linux", "windows"] }, { name: "Reboot", - icon: "mdi:restart", + icon: { + value: "mdi:restart", + type: IconEnum.Iconify + }, confirmRequired: true, function: reboot, platforms: ["macos", "linux", "windows"] }, { name: "Sleep", - icon: "carbon:asleep", + icon: { + value: "carbon:asleep", + type: IconEnum.Iconify + }, confirmRequired: false, function: sleep, platforms: ["macos", "linux", "windows"] }, { name: "Toggle System Appearance", - icon: "line-md:light-dark", + icon: { + value: "line-md:light-dark", + type: IconEnum.Iconify + }, confirmRequired: false, function: toggleSystemAppearance, platforms: ["macos"] }, { name: "Show Desktop", - icon: "bi:window-desktop", + icon: { + value: "bi:window-desktop", + type: IconEnum.Iconify + }, confirmRequired: false, function: showDesktop, platforms: ["macos"] }, { name: "Quit App", - icon: "charm:cross", + icon: { + value: "charm:cross", + type: IconEnum.Iconify + }, confirmRequired: false, function: quitAllApps, platforms: [] @@ -176,119 +200,170 @@ export const rawSystemCommands = [ }, { name: "Sleep Displays", - icon: "solar:display-broken", + icon: { + value: "solar:display-broken", + type: IconEnum.Iconify + }, confirmRequired: false, function: sleepDisplays, platforms: ["macos"] }, { name: "Set Volume to 0%", - icon: "flowbite:volume-mute-outline", + icon: { + value: "flowbite:volume-mute-outline", + type: IconEnum.Iconify + }, confirmRequired: false, function: setVolumeTo0, platforms: ["macos", "linux", "windows"] }, { name: "Set Volume to 25%", - icon: "flowbite:volume-down-solid", + icon: { + value: "flowbite:volume-down-solid", + type: IconEnum.Iconify + }, confirmRequired: false, function: setVolumeTo25, platforms: ["macos", "linux", "windows"] }, { name: "Set Volume to 50%", - icon: "flowbite:volume-down-solid", + icon: { + value: "flowbite:volume-down-solid", + type: IconEnum.Iconify + }, confirmRequired: false, function: setVolumeTo50, platforms: ["macos", "linux", "windows"] }, { name: "Set Volume to 75%", - icon: "flowbite:volume-down-solid", + icon: { + value: "flowbite:volume-down-solid", + type: IconEnum.Iconify + }, confirmRequired: false, function: setVolumeTo75, platforms: ["macos", "linux", "windows"] }, { name: "Set Volume to 100%", - icon: "flowbite:volume-up-solid", + icon: { + value: "flowbite:volume-up-solid", + type: IconEnum.Iconify + }, confirmRequired: false, function: setVolumeTo100, platforms: ["macos", "linux", "windows"] }, { name: "Turn Volume Up", - icon: "flowbite:volume-down-solid", + icon: { + value: "flowbite:volume-up-solid", + type: IconEnum.Iconify + }, confirmRequired: false, function: turnVolumeUp, platforms: ["macos", "linux", "windows"] }, { name: "Turn Volume Down", - icon: "flowbite:volume-down-outline", + icon: { + value: "flowbite:volume-down-outline", + type: IconEnum.Iconify + }, confirmRequired: false, function: turnVolumeDown, platforms: ["macos", "linux", "windows"] }, { name: "Toggle Mute", - icon: "flowbite:volume-down-outline", + icon: { + value: "flowbite:volume-down-outline", + type: IconEnum.Iconify + }, confirmRequired: false, function: toggleMute, platforms: ["macos", "linux", "windows"] }, { name: "Mute", - icon: "flowbite:volume-mute-solid", + icon: { + value: "flowbite:volume-mute-solid", + type: IconEnum.Iconify + }, confirmRequired: false, function: mute, platforms: ["macos", "linux"] }, { name: "Unmute", - icon: "flowbite:volume-mute-solid", + icon: { + value: "flowbite:volume-mute-solid", + type: IconEnum.Iconify + }, confirmRequired: false, function: unmute, platforms: ["macos", "linux"] }, { name: "Toggle Stage Manager", - icon: "material-symbols:dashboard", + icon: { + value: "material-symbols:dashboard", + type: IconEnum.Iconify + }, confirmRequired: false, function: toggleStageManager, platforms: [] }, { name: "Toggle Bluetooth", - icon: "material-symbols:bluetooth", + icon: { + value: "material-symbols:bluetooth", + type: IconEnum.Iconify + }, confirmRequired: false, function: toggleBluetooth, platforms: [] }, { name: "Toggle Hidden Files", - icon: "mdi:hide", + icon: { + value: "mdi:hide", + type: IconEnum.Iconify + }, confirmRequired: false, function: toggleHiddenFiles, platforms: [] }, { name: "Eject All Disks", - icon: "ph:eject-fill", + icon: { + value: "ph:eject-fill", + type: IconEnum.Iconify + }, confirmRequired: true, function: ejectAllDisks, platforms: ["macos"] }, { name: "Log Out User", - icon: "ic:baseline-logout", + icon: { + value: "ic:baseline-logout", + type: IconEnum.Iconify + }, confirmRequired: false, function: logoutUser, platforms: ["macos", "linux", "windows"] }, { name: "Hide All Apps Except Frontmost", - icon: "mdi:hide", + icon: { + value: "mdi:hide", + type: IconEnum.Iconify + }, confirmRequired: false, function: hideAllAppsExceptFrontmost, platforms: [] @@ -301,10 +376,7 @@ export function getSystemCommands(): SysCommand[] { .map((cmd) => ({ name: cmd.name, value: "system-cmd" + cmd.name.split(" ").join("-").toLowerCase(), - icon: { - value: cmd.icon, - type: IconEnum.Iconify - }, + icon: cmd.icon, keywords: cmd.name.split(" "), function: cmd.function, confirmRequired: cmd.confirmRequired diff --git a/packages/api/src/models/icon.ts b/packages/api/src/models/icon.ts index 26872e9..7403299 100644 --- a/packages/api/src/models/icon.ts +++ b/packages/api/src/models/icon.ts @@ -1,13 +1,4 @@ -import { - boolean, - enum_, - literal, - nullable, - object, - optional, - string, - type InferOutput -} from "valibot" +import * as v from "valibot" import { NodeName, NodeNameEnum } from "./constants" /* -------------------------------------------------------------------------- */ @@ -20,17 +11,39 @@ export enum IconEnum { Base64PNG = "base64-png", Text = "text" } -export const IconType = enum_(IconEnum) -export type IconType = InferOutput +export const IconType = v.enum_(IconEnum) +export type IconType = v.InferOutput -export const Icon = object({ +export type Icon = { + type: IconType + value: string + invert?: boolean + darkInvert?: boolean + hexColor?: string + bgColor?: string + fallback?: Icon +} + +export const BaseIcon = v.object({ type: IconType, - value: string(), - invert: optional(boolean()) + value: v.string(), + invert: v.optional(v.boolean()), + darkInvert: v.optional(v.boolean()), + hexColor: v.optional(v.string()), + bgColor: v.optional(v.string()) + // hexColor: v.optional(v.pipe(v.string(), v.hexColor("The hex color is badly formatted."))), + // bgColor: v.optional(v.pipe(v.string(), v.hexColor("The hex color is badly formatted."))) }) -export type Icon = InferOutput -export const IconNode = object({ + +export const Icon: v.GenericSchema = v.object({ + ...BaseIcon.entries, + fallback: v.optional(v.lazy(() => Icon)) +}) + +export const IconNode = v.object({ + ...BaseIcon.entries, nodeName: NodeName, - ...Icon.entries + fallback: v.optional(v.lazy(() => Icon)) }) -export type IconNode = InferOutput + +export type IconNode = v.InferOutput diff --git a/packages/api/src/ui/server/deno.ts b/packages/api/src/ui/server/deno.ts index beb276c..df874f9 100644 --- a/packages/api/src/ui/server/deno.ts +++ b/packages/api/src/ui/server/deno.ts @@ -255,8 +255,6 @@ export async function verifyDenoCmdPermission( // now we have command requested permissions, we need to compare with permissions defined in manifest /* ----------------------- Check Allow All Permissions ---------------------- */ - console.log("config: ", config) - console.log("allowAllEnv: ", allowAllEnv) if (config.allowAllEnv && !allowAllEnv) { throw new Error("allowAllEnv is not allowed") @@ -280,26 +278,40 @@ export async function verifyDenoCmdPermission( throw new Error("allowAllSys is not allowed") } - if (!allowAllEnv && difference(config.allowEnv, allowEnv).length > 0) { - throw new Error(`allowEnv is not allowed: ${difference(config.allowEnv, allowEnv)}`) + function isSubsetOf(subset: T[] | undefined, superset: T[]): boolean { + if (!subset) { + // if subset is undefined, this means extension didn't request this permission + return true + } + return subset.every((item) => superset.includes(item)) } - if (!allowAllNet && difference(config.allowNet, allowNet).length > 0) { - throw new Error(`allowNet is not allowed: ${difference(config.allowNet, allowNet)}`) + + function getDisallowed(allowEnv: string[], userAllow: string[]): string[] { + return userAllow.filter((env) => !allowEnv?.includes(env)) } - if (!allowAllRead && difference(config.allowRead, allowRead).length > 0) { - throw new Error(`allowRead is not allowed: ${difference(config.allowRead, allowRead)}`) + + if (!allowAllEnv && !isSubsetOf(config.allowEnv, allowEnv)) { + throw new Error(`allowEnv is not allowed: ${getDisallowed(allowEnv, config.allowEnv ?? [])}`) } - if (!allowAllWrite && difference(config.allowWrite, allowWrite).length > 0) { - throw new Error(`allowWrite is not allowed: ${difference(config.allowWrite, allowWrite)}`) + if (!allowAllNet && !isSubsetOf(config.allowNet, allowNet)) { + throw new Error(`allowNet is not allowed: ${getDisallowed(allowNet, config.allowNet ?? [])}`) } - if (!allowAllRun && difference(config.allowRun, allowRun).length > 0) { - throw new Error(`allowRun is not allowed: ${difference(config.allowRun, allowRun)}`) + if (!allowAllRead && !isSubsetOf(config.allowRead, allowRead)) { + throw new Error(`allowRead is not allowed: ${getDisallowed(allowRead, config.allowRead ?? [])}`) } - if (!allowAllFfi && difference(config.allowFfi, allowFfi).length > 0) { - throw new Error(`allowFfi is not allowed: ${difference(config.allowFfi, allowFfi)}`) + if (!allowAllWrite && !isSubsetOf(config.allowWrite, allowWrite)) { + throw new Error( + `allowWrite is not allowed: ${getDisallowed(allowWrite, config.allowWrite ?? [])}` + ) } - if (!allowAllSys && difference(config.allowSys, allowSys).length > 0) { - throw new Error(`allowSys is not allowed: ${difference(config.allowSys, allowSys)}`) + if (!allowAllRun && !isSubsetOf(config.allowRun, allowRun)) { + throw new Error(`allowRun is not allowed: ${getDisallowed(allowRun, config.allowRun ?? [])}`) + } + if (!allowAllFfi && !isSubsetOf(config.allowFfi, allowFfi)) { + throw new Error(`allowFfi is not allowed: ${getDisallowed(allowFfi, config.allowFfi ?? [])}`) + } + if (!allowAllSys && !isSubsetOf(config.allowSys, allowSys)) { + throw new Error(`allowSys is not allowed: ${getDisallowed(allowSys, config.allowSys ?? [])}`) } } diff --git a/packages/api/src/ui/worker/components/icon.ts b/packages/api/src/ui/worker/components/icon.ts index 205a3ae..6ed2174 100644 --- a/packages/api/src/ui/worker/components/icon.ts +++ b/packages/api/src/ui/worker/components/icon.ts @@ -6,17 +6,29 @@ export class Icon implements IconNode, IComponent { nodeName: NodeName = NodeNameEnum.Icon type: IconType value: string + invert?: boolean + darkInvert?: boolean + hexColor?: string + bgColor?: string constructor(model: TIcon) { this.type = model.type this.value = model.value + this.invert = model.invert + this.darkInvert = model.darkInvert + this.hexColor = model.hexColor + this.bgColor = model.bgColor } toModel(): IconNode { return { nodeName: this.nodeName, type: this.type, - value: this.value + value: this.value, + invert: this.invert, + darkInvert: this.darkInvert, + hexColor: this.hexColor, + bgColor: this.bgColor } } } diff --git a/packages/api/src/version.ts b/packages/api/src/version.ts index 920bf83..4bd3670 100644 --- a/packages/api/src/version.ts +++ b/packages/api/src/version.ts @@ -17,7 +17,7 @@ export const breakingChangesVersionCheckpoints = [ const checkpointVersions = breakingChangesVersionCheckpoints.map((c) => c.version) const sortedCheckpointVersions = sort(checkpointVersions) -export const version = "0.0.41" +export const version = "0.0.43" export function isVersionBetween(v: string, start: string, end: string) { const vCleaned = clean(v) diff --git a/packages/schema/package.json b/packages/schema/package.json index 0b33136..9df05ef 100644 --- a/packages/schema/package.json +++ b/packages/schema/package.json @@ -14,8 +14,8 @@ "print-schema": "bun scripts/print-schema.ts" }, "devDependencies": { - "@gcornut/valibot-json-schema": "^0.42.0", - "@types/bun": "latest" + "@types/bun": "latest", + "@valibot/to-json-schema": "1.0.0-beta.3" }, "peerDependencies": { "@kksh/supabase": "workspace:*", diff --git a/packages/schema/src/index.ts b/packages/schema/src/index.ts index 99a3fb2..af21cdc 100644 --- a/packages/schema/src/index.ts +++ b/packages/schema/src/index.ts @@ -1,5 +1,5 @@ -import { toJSONSchema } from "@gcornut/valibot-json-schema" +import { toJsonSchema } from "@valibot/to-json-schema" export function getJsonSchema(schema: any) { - return JSON.stringify(toJSONSchema({ schema }), null, 2) + return JSON.stringify(toJsonSchema(schema), null, 2) } diff --git a/packages/ui/src/components/common/IconMultiplexer.svelte b/packages/ui/src/components/common/IconMultiplexer.svelte index bc8b21f..9c68df9 100644 --- a/packages/ui/src/components/common/IconMultiplexer.svelte +++ b/packages/ui/src/components/common/IconMultiplexer.svelte @@ -3,6 +3,10 @@ import { IconEnum, IconType, Icon as TIcon } from "@kksh/api/models" import { Button } from "@kksh/svelte5" import { cn } from "@kksh/ui/utils" + import * as v from "valibot" + import { styleObjectToString } from "../../utils/style" + + const hexColorValidator = v.pipe(v.string(), v.hexColor("The hex color is badly formatted.")) const { icon, @@ -11,15 +15,35 @@ }: { icon: TIcon; class?: string; [key: string]: any } = $props() let remoteIconError = $state(false) + + function fillHexColor(style: Record, key: string, value?: string) { + if (!value) return + const parseRes = v.safeParse(hexColorValidator, value) + if (!parseRes.success) { + console.error(v.flatten(parseRes.issues)) + return + } + if (parseRes.output) style[key] = parseRes.output + } + + let customStyle = $derived.by(() => { + const style: Record = {} + fillHexColor(style, "color", icon.hexColor) + fillHexColor(style, "background-color", icon.bgColor) + return style + }) + + let style = $derived(styleObjectToString(customStyle)) {#if icon.type === IconEnum.RemoteUrl} {#if !remoteIconError} { remoteIconError = true }} @@ -28,34 +52,50 @@ {:else} {/if} {:else if icon.type === IconEnum.Iconify} - + {:else if icon.type === IconEnum.Base64PNG} {:else if icon.type === IconEnum.Text} {:else if icon.type === IconEnum.Svg} - {@html icon.value} + {@html icon.value} {:else} {/if} diff --git a/packages/ui/src/components/extension/templates/list-view.svelte b/packages/ui/src/components/extension/templates/list-view.svelte index 9c684d2..433d934 100644 --- a/packages/ui/src/components/extension/templates/list-view.svelte +++ b/packages/ui/src/components/extension/templates/list-view.svelte @@ -59,10 +59,6 @@ highlightedValue = value } - $effect(() => { - console.log("onListItemSelected", onListItemSelected) - }) - $effect(() => { if (highlightedValue.startsWith("{")) { onHighlightedItemChanged?.(JSON.parse(highlightedValue).value) diff --git a/packages/ui/src/components/main/BuiltinCmds.svelte b/packages/ui/src/components/main/BuiltinCmds.svelte index 0c4c3a2..e68f4a6 100644 --- a/packages/ui/src/components/main/BuiltinCmds.svelte +++ b/packages/ui/src/components/main/BuiltinCmds.svelte @@ -22,15 +22,8 @@ keywords={cmd.keywords} > - + {cmd.name} - {/each} diff --git a/packages/ui/src/components/main/SystemCmds.svelte b/packages/ui/src/components/main/SystemCmds.svelte index afa59dc..66d8dda 100644 --- a/packages/ui/src/components/main/SystemCmds.svelte +++ b/packages/ui/src/components/main/SystemCmds.svelte @@ -2,6 +2,7 @@ import { CmdTypeEnum, IconEnum, SysCommand } from "@kksh/api/models" import { Command } from "@kksh/svelte5" import { IconMultiplexer } from "@kksh/ui" + import { confirm } from "@tauri-apps/plugin-dialog" import { DraggableCommandGroup } from "../custom" import { CmdValue } from "./types" @@ -12,8 +13,13 @@ {#each systemCommands as cmd} { - cmd.function() + onSelect={async () => { + if (cmd.confirmRequired) { + const confirmed = await confirm(`Are you sure you want to run ${cmd.name}?`) + if (confirmed) { + cmd.function() + } + } }} value={JSON.stringify({ cmdName: cmd.name, diff --git a/packages/ui/src/components/main/types.ts b/packages/ui/src/components/main/types.ts index a16d685..f4ba69d 100644 --- a/packages/ui/src/components/main/types.ts +++ b/packages/ui/src/components/main/types.ts @@ -11,7 +11,7 @@ export type BuiltinCmd = { id: string name: string description: string - iconifyIcon: string + icon: Icon keywords?: string[] function: () => Promise flags?: { diff --git a/packages/ui/src/utils/style.ts b/packages/ui/src/utils/style.ts new file mode 100644 index 0000000..a46ab44 --- /dev/null +++ b/packages/ui/src/utils/style.ts @@ -0,0 +1,5 @@ +export function styleObjectToString(style: Record) { + return Object.entries(style) + .map(([key, value]) => `${key}: ${value};`) + .join("") +} diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml index 2c6b23e..2fd8c2d 100644 --- a/pnpm-lock.yaml +++ b/pnpm-lock.yaml @@ -629,17 +629,17 @@ importers: version: link:../supabase typescript: specifier: ^5.0.0 - version: 5.5.4 + version: 5.6.3 valibot: specifier: ^0.40.0 - version: 0.40.0(typescript@5.5.4) + version: 0.40.0(typescript@5.6.3) devDependencies: - '@gcornut/valibot-json-schema': - specifier: ^0.42.0 - version: 0.42.0(esbuild@0.24.0)(typescript@5.5.4) '@types/bun': specifier: latest version: 1.1.13 + '@valibot/to-json-schema': + specifier: 1.0.0-beta.3 + version: 1.0.0-beta.3(valibot@0.40.0(typescript@5.6.3)) packages/supabase: dependencies: @@ -1960,10 +1960,6 @@ packages: resolution: {integrity: sha512-3xGptCurm23e7nuPQkdrE5rEs1FeTPHhAUsBuwwqG4/YeZLwJOoYZv+fmsppUEfo5y9lzUwNQrNqLS/q7HMc7g==} hasBin: true - '@gcornut/valibot-json-schema@0.42.0': - resolution: {integrity: sha512-4Et4AN6wmqeA0PfU5Clkv/IS27wiefsWf6TemAZrb75uzkClYEFavim7SboeKwbll9Nbsn2Iv0LT/HS5H7orZg==} - hasBin: true - '@hapi/hoek@9.3.0': resolution: {integrity: sha512-/c6rf4UJlmHlC9b5BaNvzAcFv7HZ2QHaV0D4/HNlBdvFnvQq8RI4kYdhyPCl7Xj+oWvTWQ8ujhqS53LIgAe6KQ==} @@ -4396,6 +4392,11 @@ packages: '@unovis/ts': 1.4.4 vue: ^3 + '@valibot/to-json-schema@1.0.0-beta.3': + resolution: {integrity: sha512-20XQh1u5sOLwS3NOB7oHCo3clQ9h4GlavXgLKMux2PYpHowb7P97cND0dg8T3+fE1WoKVACcLppvzAPpSx0F+Q==} + peerDependencies: + valibot: ^1.0.0 || ^1.0.0-beta.5 || ^1.0.0-rc + '@vee-validate/zod@4.14.7': resolution: {integrity: sha512-UD0Tfyz1cKKd7BinnUztqKL+oeMjg/T4ZEguN/uZV4DsR9z7gdrD0lOuOU7aVl9UpVK6NM7MhDka35Lj7b/DTw==} @@ -9486,14 +9487,6 @@ packages: typescript: optional: true - valibot@0.42.1: - resolution: {integrity: sha512-3keXV29Ar5b//Hqi4MbSdV7lfVp6zuYLZuA9V1PvQUsXqogr+u5lvLPLk3A4f74VUXDnf/JfWMN6sB+koJ/FFw==} - peerDependencies: - typescript: '>=5' - peerDependenciesMeta: - typescript: - optional: true - valibot@1.0.0-beta.7: resolution: {integrity: sha512-8CsDu3tqyg7quEHMzCOYdQ/d9NlmVQKtd4AlFje6oJpvqo70EIZjSakKIeWltJyNAiUtdtLe0LAk4625gavoeQ==} peerDependencies: @@ -11129,16 +11122,6 @@ snapshots: esbuild-runner: 2.2.2(esbuild@0.24.0) optional: true - '@gcornut/valibot-json-schema@0.42.0(esbuild@0.24.0)(typescript@5.5.4)': - dependencies: - valibot: 0.42.1(typescript@5.5.4) - optionalDependencies: - '@types/json-schema': 7.0.15 - esbuild-runner: 2.2.2(esbuild@0.24.0) - transitivePeerDependencies: - - esbuild - - typescript - '@hapi/hoek@9.3.0': optional: true @@ -14312,6 +14295,10 @@ snapshots: '@unovis/ts': 1.4.4 vue: 3.5.13(typescript@5.6.3) + '@valibot/to-json-schema@1.0.0-beta.3(valibot@0.40.0(typescript@5.6.3))': + dependencies: + valibot: 0.40.0(typescript@5.6.3) + '@vee-validate/zod@4.14.7(vue@3.5.13(typescript@5.6.3))': dependencies: type-fest: 4.27.0 @@ -20363,10 +20350,6 @@ snapshots: optionalDependencies: typescript: 5.6.3 - valibot@0.42.1(typescript@5.5.4): - optionalDependencies: - typescript: 5.5.4 - valibot@1.0.0-beta.7(typescript@5.6.3): optionalDependencies: typescript: 5.6.3