mirror of
https://github.com/kunkunsh/kunkun.git
synced 2025-04-04 14:46:42 +00:00
Feature: Quick Link (#17)
* feat: add page for add quick link (not working yet) * upgrade @kksh/svelte5 * fix: infinite recursive footer * dep: add @kksh/svelte5 to ui package * dep: add supabase-js * dep: add @iconify/svelte * style: modify StoreExtDetail width control * fixed: UI for extension store detail * feat: add page to create quick link * feat: display quick links in cmd palette * snapshot * show queries in command input * feat: quick link fully implemented * refactor: format all with prettier * feat: add icon picker for quick link adder * fix: make invert for icon optional, caused many types to crash
This commit is contained in:
parent
f043d7afe0
commit
ce42409a39
@ -115,16 +115,15 @@ export const builtinCmds: BuiltinCmd[] = [
|
|||||||
goto("/troubleshooters/extension-loading")
|
goto("/troubleshooters/extension-loading")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
// {
|
{
|
||||||
// name: "Create Quicklink",
|
name: "Create Quicklink",
|
||||||
// iconifyIcon: "material-symbols:link",
|
iconifyIcon: "material-symbols:link",
|
||||||
// description: "Create a Quicklink",
|
description: "Create a Quicklink",
|
||||||
// function: async () => {
|
function: async () => {
|
||||||
// const appStateStore = useAppStateStore()
|
appState.clearSearchTerm()
|
||||||
// appStateStore.setSearchTermSync("")
|
goto("/extension/create-quick-link")
|
||||||
// goto("/create-quicklink")
|
}
|
||||||
// }
|
},
|
||||||
// },
|
|
||||||
// {
|
// {
|
||||||
// name: "Settings",
|
// name: "Settings",
|
||||||
// iconifyIcon: "solar:settings-linear",
|
// iconifyIcon: "solar:settings-linear",
|
||||||
|
@ -2,6 +2,7 @@ import { CmdTypeEnum, CustomUiCmd, ExtPackageJsonExtra, TemplateUiCmd } from "@k
|
|||||||
import type { CommandLaunchers, OnExtCmdSelect } from "@kksh/ui/types"
|
import type { CommandLaunchers, OnExtCmdSelect } from "@kksh/ui/types"
|
||||||
import * as v from "valibot"
|
import * as v from "valibot"
|
||||||
import { onCustomUiCmdSelect, onTemplateUiCmdSelect } from "./ext"
|
import { onCustomUiCmdSelect, onTemplateUiCmdSelect } from "./ext"
|
||||||
|
import { onQuickLinkSelect } from "./quick-links"
|
||||||
|
|
||||||
const onExtCmdSelect: OnExtCmdSelect = (
|
const onExtCmdSelect: OnExtCmdSelect = (
|
||||||
ext: ExtPackageJsonExtra,
|
ext: ExtPackageJsonExtra,
|
||||||
@ -20,4 +21,4 @@ const onExtCmdSelect: OnExtCmdSelect = (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export const commandLaunchers = { onExtCmdSelect } satisfies CommandLaunchers
|
export const commandLaunchers = { onExtCmdSelect, onQuickLinkSelect } satisfies CommandLaunchers
|
||||||
|
25
apps/desktop/src/lib/cmds/quick-links.ts
Normal file
25
apps/desktop/src/lib/cmds/quick-links.ts
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
import { appState } from "@/stores"
|
||||||
|
import type { CmdQuery, CmdValue } from "@kksh/ui/types"
|
||||||
|
import { open } from "tauri-plugin-shellx-api"
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Given some link like https://google.com/search?q={argument}&query={query}
|
||||||
|
* Find {argument} and {query}
|
||||||
|
*/
|
||||||
|
export function findAllArgsInLink(link: string): string[] {
|
||||||
|
const regex = /\{([^}]+)\}/g
|
||||||
|
const matches = [...link.matchAll(regex)]
|
||||||
|
return matches.map((match) => match[1])
|
||||||
|
}
|
||||||
|
|
||||||
|
export function onQuickLinkSelect(quickLink: CmdValue, queries: CmdQuery[]) {
|
||||||
|
console.log(quickLink, queries)
|
||||||
|
let qlink = quickLink.data
|
||||||
|
for (const arg of queries) {
|
||||||
|
console.log(`replace all {${arg.name}} with ${arg.value}`)
|
||||||
|
qlink = qlink.replaceAll(`{${arg.name}}`, arg.value)
|
||||||
|
}
|
||||||
|
appState.clearSearchTerm()
|
||||||
|
console.log(qlink)
|
||||||
|
open(qlink)
|
||||||
|
}
|
@ -1,89 +0,0 @@
|
|||||||
<!-- This file renders the main command palette, a list of commands -->
|
|
||||||
<!-- This is not placed in @kksh/ui because it depends on the app config and is very complex,
|
|
||||||
passing everything through props will be very complicated and hard to maintain.
|
|
||||||
-->
|
|
||||||
<script lang="ts">
|
|
||||||
import { systemCommands } from "@/cmds/system"
|
|
||||||
import { devStoreExts, installedStoreExts } from "@/stores"
|
|
||||||
import { getActiveElementNodeName } from "@/utils/dom"
|
|
||||||
import type { ExtPackageJsonExtra } from "@kksh/api/models"
|
|
||||||
import { isExtPathInDev } from "@kksh/extension/utils"
|
|
||||||
import { Command } from "@kksh/svelte5"
|
|
||||||
import type { AppConfig, AppState } from "@kksh/types"
|
|
||||||
import {
|
|
||||||
BuiltinCmds,
|
|
||||||
CustomCommandInput,
|
|
||||||
ExtCmdsGroup,
|
|
||||||
GlobalCommandPaletteFooter,
|
|
||||||
SystemCmds
|
|
||||||
} from "@kksh/ui/main"
|
|
||||||
import type { BuiltinCmd, CommandLaunchers } from "@kksh/ui/types"
|
|
||||||
import { cn } from "@kksh/ui/utils"
|
|
||||||
import type { Writable } from "svelte/store"
|
|
||||||
|
|
||||||
const {
|
|
||||||
extensions,
|
|
||||||
appConfig,
|
|
||||||
class: className,
|
|
||||||
commandLaunchers,
|
|
||||||
appState,
|
|
||||||
builtinCmds
|
|
||||||
}: {
|
|
||||||
extensions: ExtPackageJsonExtra[]
|
|
||||||
appConfig: Writable<AppConfig>
|
|
||||||
class?: string
|
|
||||||
commandLaunchers: CommandLaunchers
|
|
||||||
appState: Writable<AppState>
|
|
||||||
builtinCmds: BuiltinCmd[]
|
|
||||||
} = $props()
|
|
||||||
|
|
||||||
function onKeyDown(event: KeyboardEvent) {
|
|
||||||
if (event.key === "Escape") {
|
|
||||||
if (getActiveElementNodeName() === "INPUT") {
|
|
||||||
;(event.target as HTMLInputElement).value = ""
|
|
||||||
if ((event.target as HTMLInputElement | undefined)?.id === "main-command-input") {
|
|
||||||
$appState.searchTerm = ""
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<svelte:window on:keydown={onKeyDown} />
|
|
||||||
<Command.Root
|
|
||||||
class={cn("rounded-lg border shadow-md", className)}
|
|
||||||
bind:value={$appState.highlightedCmd}
|
|
||||||
loop
|
|
||||||
>
|
|
||||||
<CustomCommandInput
|
|
||||||
autofocus
|
|
||||||
id="main-command-input"
|
|
||||||
placeholder="Type a command or search..."
|
|
||||||
bind:value={$appState.searchTerm}
|
|
||||||
/>
|
|
||||||
<Command.List class="max-h-screen grow">
|
|
||||||
<Command.Empty data-tauri-drag-region>No results found.</Command.Empty>
|
|
||||||
{#if $appConfig.extensionsInstallDir && $devStoreExts.length > 0}
|
|
||||||
<ExtCmdsGroup
|
|
||||||
extensions={$devStoreExts}
|
|
||||||
heading="Dev Extensions"
|
|
||||||
isDev={true}
|
|
||||||
onExtCmdSelect={commandLaunchers.onExtCmdSelect}
|
|
||||||
hmr={$appConfig.hmr}
|
|
||||||
/>
|
|
||||||
{/if}
|
|
||||||
{#if $appConfig.extensionsInstallDir && $installedStoreExts.length > 0}
|
|
||||||
<ExtCmdsGroup
|
|
||||||
extensions={$installedStoreExts}
|
|
||||||
heading="Extensions"
|
|
||||||
isDev={false}
|
|
||||||
hmr={false}
|
|
||||||
onExtCmdSelect={commandLaunchers.onExtCmdSelect}
|
|
||||||
/>
|
|
||||||
{/if}
|
|
||||||
<BuiltinCmds {builtinCmds} />
|
|
||||||
<SystemCmds {systemCommands} />
|
|
||||||
<Command.Separator />
|
|
||||||
</Command.List>
|
|
||||||
<GlobalCommandPaletteFooter />
|
|
||||||
</Command.Root>
|
|
@ -1,5 +1,8 @@
|
|||||||
|
import { findAllArgsInLink } from "@/cmds/quick-links"
|
||||||
|
import { CmdTypeEnum } from "@kksh/api/models"
|
||||||
import type { AppState } from "@kksh/types"
|
import type { AppState } from "@kksh/types"
|
||||||
import { get, writable, type Writable } from "svelte/store"
|
import type { CmdValue } from "@kksh/ui/types"
|
||||||
|
import { derived, get, writable, type Writable } from "svelte/store"
|
||||||
|
|
||||||
export const defaultAppState: AppState = {
|
export const defaultAppState: AppState = {
|
||||||
searchTerm: "",
|
searchTerm: "",
|
||||||
@ -24,3 +27,13 @@ function createAppState(): Writable<AppState> & AppStateAPI {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export const appState = createAppState()
|
export const appState = createAppState()
|
||||||
|
|
||||||
|
// export const cmdQueries = derived(appState, ($appState) => {
|
||||||
|
// if ($appState.highlightedCmd.startsWith("{")) {
|
||||||
|
// const parsedCmd = JSON.parse($appState.highlightedCmd) as CmdValue
|
||||||
|
// if (parsedCmd.cmdType === CmdTypeEnum.QuickLink && parsedCmd.data) {
|
||||||
|
// return findAllArgsInLink(parsedCmd.data).map((arg) => ({ name: arg, value: "" }))
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// return []
|
||||||
|
// })
|
||||||
|
23
apps/desktop/src/lib/stores/cmdQuery.ts
Normal file
23
apps/desktop/src/lib/stores/cmdQuery.ts
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
import { findAllArgsInLink } from "@/cmds/quick-links"
|
||||||
|
import { CmdTypeEnum } from "@kksh/api/models"
|
||||||
|
import type { CmdQuery, CmdValue } from "@kksh/ui/main"
|
||||||
|
import { derived, get, writable, type Writable } from "svelte/store"
|
||||||
|
import { appState } from "./appState"
|
||||||
|
|
||||||
|
function createCmdQueryStore(): Writable<CmdQuery[]> {
|
||||||
|
const store = writable<CmdQuery[]>([])
|
||||||
|
appState.subscribe(($appState) => {
|
||||||
|
if ($appState.highlightedCmd.startsWith("{")) {
|
||||||
|
const parsedCmd = JSON.parse($appState.highlightedCmd) as CmdValue
|
||||||
|
if (parsedCmd.cmdType === CmdTypeEnum.QuickLink && parsedCmd.data) {
|
||||||
|
return store.set(findAllArgsInLink(parsedCmd.data).map((arg) => ({ name: arg, value: "" })))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
store.set([])
|
||||||
|
})
|
||||||
|
return {
|
||||||
|
...store
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const cmdQueries = createCmdQueryStore()
|
@ -3,3 +3,4 @@ export * from "./appState"
|
|||||||
export * from "./winExtMap"
|
export * from "./winExtMap"
|
||||||
export * from "./extensions"
|
export * from "./extensions"
|
||||||
export * from "./auth"
|
export * from "./auth"
|
||||||
|
export * from "./quick-links"
|
||||||
|
41
apps/desktop/src/lib/stores/quick-links.ts
Normal file
41
apps/desktop/src/lib/stores/quick-links.ts
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
import type { Icon } from "@kksh/api/models"
|
||||||
|
import { createQuickLinkCommand, getAllQuickLinkCommands } from "@kksh/extension/db"
|
||||||
|
import type { CmdQuery, QuickLink } from "@kksh/ui/types"
|
||||||
|
import { get, writable, type Writable } from "svelte/store"
|
||||||
|
|
||||||
|
export interface QuickLinkAPI {
|
||||||
|
get: () => QuickLink[]
|
||||||
|
init: () => Promise<void>
|
||||||
|
refresh: () => Promise<void>
|
||||||
|
createQuickLink: (name: string, link: string, icon: Icon) => Promise<void>
|
||||||
|
}
|
||||||
|
|
||||||
|
function createQuickLinksStore(): Writable<QuickLink[]> & QuickLinkAPI {
|
||||||
|
const store = writable<QuickLink[]>([])
|
||||||
|
|
||||||
|
async function init() {
|
||||||
|
refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
async function refresh() {
|
||||||
|
const cmds = await getAllQuickLinkCommands()
|
||||||
|
console.log(cmds)
|
||||||
|
|
||||||
|
store.set(cmds.map((cmd) => ({ link: cmd.data.link, name: cmd.name, icon: cmd.data.icon })))
|
||||||
|
}
|
||||||
|
|
||||||
|
async function createQuickLink(name: string, link: string, icon: Icon) {
|
||||||
|
await createQuickLinkCommand(name, link, icon)
|
||||||
|
await refresh()
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...store,
|
||||||
|
get: () => get(store),
|
||||||
|
init,
|
||||||
|
refresh,
|
||||||
|
createQuickLink
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export const quickLinks = createQuickLinksStore()
|
175
apps/desktop/src/lib/utils/command-score.ts
Normal file
175
apps/desktop/src/lib/utils/command-score.ts
Normal file
@ -0,0 +1,175 @@
|
|||||||
|
// This file is taken from https://github.com/huntabyte/bits-ui/blob/7f7bf6f6b736cf34e57a0d87aab01074c33efd46/packages/bits-ui/src/lib/bits/command/command.svelte.ts#L1
|
||||||
|
|
||||||
|
// eslint-disable-next-line ts/ban-ts-comment
|
||||||
|
// @ts-nocheck
|
||||||
|
// The scores are arranged so that a continuous match of characters will
|
||||||
|
// result in a total score of 1.
|
||||||
|
//
|
||||||
|
// The best case, this character is a match, and either this is the start
|
||||||
|
// of the string, or the previous character was also a match.
|
||||||
|
const SCORE_CONTINUE_MATCH = 1
|
||||||
|
// A new match at the start of a word scores better than a new match
|
||||||
|
// elsewhere as it's more likely that the user will type the starts
|
||||||
|
// of fragments.
|
||||||
|
// NOTE: We score word jumps between spaces slightly higher than slashes, brackets
|
||||||
|
// hyphens, etc.
|
||||||
|
const SCORE_SPACE_WORD_JUMP = 0.9
|
||||||
|
const SCORE_NON_SPACE_WORD_JUMP = 0.8
|
||||||
|
// Any other match isn't ideal, but we include it for completeness.
|
||||||
|
const SCORE_CHARACTER_JUMP = 0.17
|
||||||
|
// If the user transposed two letters, it should be significantly penalized.
|
||||||
|
//
|
||||||
|
// i.e. "ouch" is more likely than "curtain" when "uc" is typed.
|
||||||
|
const SCORE_TRANSPOSITION = 0.1
|
||||||
|
// The goodness of a match should decay slightly with each missing
|
||||||
|
// character.
|
||||||
|
//
|
||||||
|
// i.e. "bad" is more likely than "bard" when "bd" is typed.
|
||||||
|
//
|
||||||
|
// This will not change the order of suggestions based on SCORE_* until
|
||||||
|
// 100 characters are inserted between matches.
|
||||||
|
const PENALTY_SKIPPED = 0.999
|
||||||
|
// The goodness of an exact-case match should be higher than a
|
||||||
|
// case-insensitive match by a small amount.
|
||||||
|
//
|
||||||
|
// i.e. "HTML" is more likely than "haml" when "HM" is typed.
|
||||||
|
//
|
||||||
|
// This will not change the order of suggestions based on SCORE_* until
|
||||||
|
// 1000 characters are inserted between matches.
|
||||||
|
const PENALTY_CASE_MISMATCH = 0.9999
|
||||||
|
// Match higher for letters closer to the beginning of the word
|
||||||
|
const PENALTY_DISTANCE_FROM_START = 0.9
|
||||||
|
// If the word has more characters than the user typed, it should
|
||||||
|
// be penalised slightly.
|
||||||
|
//
|
||||||
|
// i.e. "html" is more likely than "html5" if I type "html".
|
||||||
|
//
|
||||||
|
// However, it may well be the case that there's a sensible secondary
|
||||||
|
// ordering (like alphabetical) that it makes sense to rely on when
|
||||||
|
// there are many prefix matches, so we don't make the penalty increase
|
||||||
|
// with the number of tokens.
|
||||||
|
const PENALTY_NOT_COMPLETE = 0.99
|
||||||
|
|
||||||
|
const IS_GAP_REGEXP = /[\\/_+.#"@[({&]/
|
||||||
|
const COUNT_GAPS_REGEXP = /[\\/_+.#"@[({&]/g
|
||||||
|
const IS_SPACE_REGEXP = /[\s-]/
|
||||||
|
const COUNT_SPACE_REGEXP = /[\s-]/g
|
||||||
|
|
||||||
|
function commandScoreInner(
|
||||||
|
string,
|
||||||
|
abbreviation,
|
||||||
|
lowerString,
|
||||||
|
lowerAbbreviation,
|
||||||
|
stringIndex,
|
||||||
|
abbreviationIndex,
|
||||||
|
memoizedResults
|
||||||
|
) {
|
||||||
|
if (abbreviationIndex === abbreviation.length) {
|
||||||
|
if (stringIndex === string.length) {
|
||||||
|
return SCORE_CONTINUE_MATCH
|
||||||
|
}
|
||||||
|
return PENALTY_NOT_COMPLETE
|
||||||
|
}
|
||||||
|
|
||||||
|
const memoizeKey = `${stringIndex},${abbreviationIndex}`
|
||||||
|
if (memoizedResults[memoizeKey] !== undefined) {
|
||||||
|
return memoizedResults[memoizeKey]
|
||||||
|
}
|
||||||
|
|
||||||
|
const abbreviationChar = lowerAbbreviation.charAt(abbreviationIndex)
|
||||||
|
let index = lowerString.indexOf(abbreviationChar, stringIndex)
|
||||||
|
let highScore = 0
|
||||||
|
|
||||||
|
let score, transposedScore, wordBreaks, spaceBreaks
|
||||||
|
|
||||||
|
while (index >= 0) {
|
||||||
|
score = commandScoreInner(
|
||||||
|
string,
|
||||||
|
abbreviation,
|
||||||
|
lowerString,
|
||||||
|
lowerAbbreviation,
|
||||||
|
index + 1,
|
||||||
|
abbreviationIndex + 1,
|
||||||
|
memoizedResults
|
||||||
|
)
|
||||||
|
if (score > highScore) {
|
||||||
|
if (index === stringIndex) {
|
||||||
|
score *= SCORE_CONTINUE_MATCH
|
||||||
|
} else if (IS_GAP_REGEXP.test(string.charAt(index - 1))) {
|
||||||
|
score *= SCORE_NON_SPACE_WORD_JUMP
|
||||||
|
wordBreaks = string.slice(stringIndex, index - 1).match(COUNT_GAPS_REGEXP)
|
||||||
|
if (wordBreaks && stringIndex > 0) {
|
||||||
|
score *= PENALTY_SKIPPED ** wordBreaks.length
|
||||||
|
}
|
||||||
|
} else if (IS_SPACE_REGEXP.test(string.charAt(index - 1))) {
|
||||||
|
score *= SCORE_SPACE_WORD_JUMP
|
||||||
|
spaceBreaks = string.slice(stringIndex, index - 1).match(COUNT_SPACE_REGEXP)
|
||||||
|
if (spaceBreaks && stringIndex > 0) {
|
||||||
|
score *= PENALTY_SKIPPED ** spaceBreaks.length
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
score *= SCORE_CHARACTER_JUMP
|
||||||
|
if (stringIndex > 0) {
|
||||||
|
score *= PENALTY_SKIPPED ** (index - stringIndex)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (string.charAt(index) !== abbreviation.charAt(abbreviationIndex)) {
|
||||||
|
score *= PENALTY_CASE_MISMATCH
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
(score < SCORE_TRANSPOSITION &&
|
||||||
|
lowerString.charAt(index - 1) === lowerAbbreviation.charAt(abbreviationIndex + 1)) ||
|
||||||
|
(lowerAbbreviation.charAt(abbreviationIndex + 1) ===
|
||||||
|
lowerAbbreviation.charAt(abbreviationIndex) && // allow duplicate letters. Ref #7428
|
||||||
|
lowerString.charAt(index - 1) !== lowerAbbreviation.charAt(abbreviationIndex))
|
||||||
|
) {
|
||||||
|
transposedScore = commandScoreInner(
|
||||||
|
string,
|
||||||
|
abbreviation,
|
||||||
|
lowerString,
|
||||||
|
lowerAbbreviation,
|
||||||
|
index + 1,
|
||||||
|
abbreviationIndex + 2,
|
||||||
|
memoizedResults
|
||||||
|
)
|
||||||
|
|
||||||
|
if (transposedScore * SCORE_TRANSPOSITION > score) {
|
||||||
|
score = transposedScore * SCORE_TRANSPOSITION
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (score > highScore) {
|
||||||
|
highScore = score
|
||||||
|
}
|
||||||
|
|
||||||
|
index = lowerString.indexOf(abbreviationChar, index + 1)
|
||||||
|
}
|
||||||
|
|
||||||
|
memoizedResults[memoizeKey] = highScore
|
||||||
|
return highScore
|
||||||
|
}
|
||||||
|
|
||||||
|
function formatInput(string) {
|
||||||
|
// convert all valid space characters to space so they match each other
|
||||||
|
return string.toLowerCase().replace(COUNT_SPACE_REGEXP, " ")
|
||||||
|
}
|
||||||
|
|
||||||
|
export function commandScore(string: string, abbreviation: string, aliases?: string[]): number {
|
||||||
|
/* NOTE:
|
||||||
|
* in the original, we used to do the lower-casing on each recursive call, but this meant that toLowerCase()
|
||||||
|
* was the dominating cost in the algorithm, passing both is a little ugly, but considerably faster.
|
||||||
|
*/
|
||||||
|
string = aliases && aliases.length > 0 ? `${`${string} ${aliases?.join(" ")}`}` : string
|
||||||
|
return commandScoreInner(
|
||||||
|
string,
|
||||||
|
abbreviation,
|
||||||
|
formatInput(string),
|
||||||
|
formatInput(abbreviation),
|
||||||
|
0,
|
||||||
|
0,
|
||||||
|
{}
|
||||||
|
)
|
||||||
|
}
|
@ -1,7 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import AppContext from "@/components/context/AppContext.svelte"
|
import AppContext from "@/components/context/AppContext.svelte"
|
||||||
import "../app.css"
|
import "../app.css"
|
||||||
import { appConfig, appState, extensions } from "@/stores"
|
import { appConfig, appState, extensions, quickLinks } from "@/stores"
|
||||||
import { initDeeplink } from "@/utils/deeplink"
|
import { initDeeplink } from "@/utils/deeplink"
|
||||||
import { isInMainWindow } from "@/utils/window"
|
import { isInMainWindow } from "@/utils/window"
|
||||||
import {
|
import {
|
||||||
@ -21,8 +21,9 @@
|
|||||||
const unlisteners: UnlistenFn[] = []
|
const unlisteners: UnlistenFn[] = []
|
||||||
|
|
||||||
onMount(async () => {
|
onMount(async () => {
|
||||||
unlisteners.push(await attachConsole())
|
attachConsole().then((unlistener) => unlisteners.push(unlistener))
|
||||||
unlisteners.push(await initDeeplink())
|
initDeeplink().then((unlistener) => unlisteners.push(unlistener))
|
||||||
|
quickLinks.init()
|
||||||
appConfig.init()
|
appConfig.init()
|
||||||
if (isInMainWindow()) {
|
if (isInMainWindow()) {
|
||||||
extensions.init()
|
extensions.init()
|
||||||
|
@ -1,18 +1,135 @@
|
|||||||
|
<!-- This file renders the main command palette, a list of commands -->
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { commandLaunchers } from "@/cmds"
|
import { commandLaunchers } from "@/cmds"
|
||||||
import { builtinCmds } from "@/cmds/builtin"
|
import { builtinCmds } from "@/cmds/builtin"
|
||||||
import CommandPalette from "@/components/main/CommandPalette.svelte"
|
import { systemCommands } from "@/cmds/system"
|
||||||
import { appState } from "@/stores"
|
import { appConfig, appState, devStoreExts, installedStoreExts, quickLinks } from "@/stores"
|
||||||
import { appConfig } from "@/stores/appConfig"
|
import { cmdQueries } from "@/stores/cmdQuery"
|
||||||
import { extensions } from "@/stores/extensions"
|
import { commandScore } from "@/utils/command-score"
|
||||||
import "@kksh/ui"
|
import { getActiveElementNodeName } from "@/utils/dom"
|
||||||
|
import { openDevTools } from "@kksh/api/commands"
|
||||||
|
import type { ExtPackageJsonExtra } from "@kksh/api/models"
|
||||||
|
import { isExtPathInDev } from "@kksh/extension/utils"
|
||||||
|
import { Button, Command, DropdownMenu } from "@kksh/svelte5"
|
||||||
|
import type { AppConfig, AppState } from "@kksh/types"
|
||||||
|
import {
|
||||||
|
BuiltinCmds,
|
||||||
|
CustomCommandInput,
|
||||||
|
ExtCmdsGroup,
|
||||||
|
GlobalCommandPaletteFooter,
|
||||||
|
QuickLinks,
|
||||||
|
SystemCmds
|
||||||
|
} from "@kksh/ui/main"
|
||||||
|
import type { BuiltinCmd, CmdValue, CommandLaunchers } from "@kksh/ui/types"
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import { getCurrentWebviewWindow } from "@tauri-apps/api/webviewWindow"
|
||||||
|
import { exit } from "@tauri-apps/plugin-process"
|
||||||
|
import { EllipsisVerticalIcon } from "lucide-svelte"
|
||||||
|
import type { Writable } from "svelte/store"
|
||||||
|
|
||||||
|
function onKeyDown(event: KeyboardEvent) {
|
||||||
|
if (event.key === "Escape") {
|
||||||
|
if (getActiveElementNodeName() === "INPUT") {
|
||||||
|
;(event.target as HTMLInputElement).value = ""
|
||||||
|
if ((event.target as HTMLInputElement | undefined)?.id === "main-command-input") {
|
||||||
|
$appState.searchTerm = ""
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<CommandPalette
|
<svelte:window on:keydown={onKeyDown} />
|
||||||
class="h-screen"
|
<Command.Root
|
||||||
extensions={$extensions}
|
class={cn("h-screen rounded-lg border shadow-md")}
|
||||||
{appState}
|
bind:value={$appState.highlightedCmd}
|
||||||
{appConfig}
|
filter={(value, search, keywords) => {
|
||||||
{commandLaunchers}
|
return commandScore(
|
||||||
{builtinCmds}
|
value.startsWith("{") ? (JSON.parse(value) as CmdValue).cmdName : value,
|
||||||
/>
|
search,
|
||||||
|
keywords
|
||||||
|
)
|
||||||
|
}}
|
||||||
|
loop
|
||||||
|
>
|
||||||
|
<CustomCommandInput
|
||||||
|
autofocus
|
||||||
|
id="main-command-input"
|
||||||
|
placeholder={$cmdQueries.length === 0 ? "Type a command or search..." : undefined}
|
||||||
|
bind:value={$appState.searchTerm}
|
||||||
|
>
|
||||||
|
{#snippet rightSlot()}
|
||||||
|
<span
|
||||||
|
class={cn("absolute flex space-x-2")}
|
||||||
|
style={`left: ${$appState.searchTerm.length + 3}ch`}
|
||||||
|
>
|
||||||
|
{#each $cmdQueries as cmdQuery}
|
||||||
|
{@const queryWidth = Math.max(cmdQuery.name.length, cmdQuery.value.length) + 2}
|
||||||
|
<input
|
||||||
|
class="bg-muted rounded-md border border-gray-300 pl-2 font-mono focus:outline-none dark:border-gray-600"
|
||||||
|
type="text"
|
||||||
|
placeholder={cmdQuery.name}
|
||||||
|
style={`width: ${queryWidth}ch`}
|
||||||
|
onkeydown={(evt) => {
|
||||||
|
if (evt.key === "Enter") {
|
||||||
|
evt.preventDefault()
|
||||||
|
evt.stopPropagation()
|
||||||
|
commandLaunchers.onQuickLinkSelect(
|
||||||
|
JSON.parse($appState.highlightedCmd),
|
||||||
|
$cmdQueries
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
bind:value={cmdQuery.value}
|
||||||
|
/>
|
||||||
|
{/each}
|
||||||
|
</span>
|
||||||
|
|
||||||
|
<DropdownMenu.Root>
|
||||||
|
<DropdownMenu.Trigger>
|
||||||
|
<Button variant="outline" size="icon"><EllipsisVerticalIcon /></Button>
|
||||||
|
</DropdownMenu.Trigger>
|
||||||
|
<DropdownMenu.Content>
|
||||||
|
<DropdownMenu.Group>
|
||||||
|
<DropdownMenu.GroupHeading>Settings</DropdownMenu.GroupHeading>
|
||||||
|
<DropdownMenu.Separator />
|
||||||
|
<DropdownMenu.Item onclick={() => exit()}>Quit</DropdownMenu.Item>
|
||||||
|
<DropdownMenu.Item onclick={() => openDevTools()}>Open Dev Tools</DropdownMenu.Item>
|
||||||
|
<DropdownMenu.Item onclick={() => getCurrentWebviewWindow().hide()}
|
||||||
|
>Close Window</DropdownMenu.Item
|
||||||
|
>
|
||||||
|
</DropdownMenu.Group>
|
||||||
|
</DropdownMenu.Content>
|
||||||
|
</DropdownMenu.Root>
|
||||||
|
{/snippet}
|
||||||
|
</CustomCommandInput>
|
||||||
|
<Command.List class="max-h-screen grow">
|
||||||
|
<Command.Empty data-tauri-drag-region>No results found.</Command.Empty>
|
||||||
|
<Command.Separator />
|
||||||
|
{#if $appConfig.extensionsInstallDir && $devStoreExts.length > 0}
|
||||||
|
<ExtCmdsGroup
|
||||||
|
extensions={$devStoreExts}
|
||||||
|
heading="Dev Extensions"
|
||||||
|
isDev={true}
|
||||||
|
onExtCmdSelect={commandLaunchers.onExtCmdSelect}
|
||||||
|
hmr={$appConfig.hmr}
|
||||||
|
/>
|
||||||
|
<Command.Separator />
|
||||||
|
{/if}
|
||||||
|
{#if $appConfig.extensionsInstallDir && $installedStoreExts.length > 0}
|
||||||
|
<ExtCmdsGroup
|
||||||
|
extensions={$installedStoreExts}
|
||||||
|
heading="Extensions"
|
||||||
|
isDev={false}
|
||||||
|
hmr={false}
|
||||||
|
onExtCmdSelect={commandLaunchers.onExtCmdSelect}
|
||||||
|
/>
|
||||||
|
<Command.Separator />
|
||||||
|
{/if}
|
||||||
|
<QuickLinks quickLinks={$quickLinks} />
|
||||||
|
<BuiltinCmds {builtinCmds} />
|
||||||
|
<Command.Separator />
|
||||||
|
<SystemCmds {systemCommands} />
|
||||||
|
</Command.List>
|
||||||
|
<GlobalCommandPaletteFooter />
|
||||||
|
</Command.Root>
|
||||||
|
110
apps/desktop/src/routes/extension/create-quick-link/+page.svelte
Normal file
110
apps/desktop/src/routes/extension/create-quick-link/+page.svelte
Normal file
@ -0,0 +1,110 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { quickLinks } from "@/stores/quick-links"
|
||||||
|
import { goBackOnEscape } from "@/utils/key"
|
||||||
|
import { goBack } from "@/utils/route"
|
||||||
|
import { Icon, IconEnum, IconType } from "@kksh/api/models"
|
||||||
|
import { createQuickLinkCommand } from "@kksh/extension/db"
|
||||||
|
import { Button, Input } from "@kksh/svelte5"
|
||||||
|
import { Form, IconSelector } from "@kksh/ui"
|
||||||
|
import { dev } from "$app/environment"
|
||||||
|
import { ArrowLeftIcon } from "lucide-svelte"
|
||||||
|
import { toast } from "svelte-sonner"
|
||||||
|
import SuperDebug, { defaults, superForm } from "sveltekit-superforms"
|
||||||
|
import { valibot, valibotClient, zod, zodClient } from "sveltekit-superforms/adapters"
|
||||||
|
import * as v from "valibot"
|
||||||
|
|
||||||
|
const formSchema = v.object({
|
||||||
|
name: v.pipe(v.string(), v.minLength(1), v.maxLength(100)),
|
||||||
|
link: v.pipe(v.string(), v.minLength(5), v.maxLength(1000)),
|
||||||
|
iconType: IconType,
|
||||||
|
iconValue: v.string(),
|
||||||
|
invertIcon: v.boolean()
|
||||||
|
})
|
||||||
|
let icon = $state<Icon>({
|
||||||
|
type: IconEnum.Iconify,
|
||||||
|
value: "material-symbols:link",
|
||||||
|
invert: false
|
||||||
|
})
|
||||||
|
const form = superForm(defaults(valibot(formSchema)), {
|
||||||
|
validators: valibotClient(formSchema),
|
||||||
|
SPA: true,
|
||||||
|
onUpdate({ form, cancel }) {
|
||||||
|
cancel()
|
||||||
|
if (!form.valid) return
|
||||||
|
const { name, link, iconType, iconValue } = form.data
|
||||||
|
quickLinks
|
||||||
|
.createQuickLink(name, link, icon)
|
||||||
|
.then(() => {
|
||||||
|
toast.success("Quicklink created successfully")
|
||||||
|
goBack()
|
||||||
|
})
|
||||||
|
.catch((err) => {
|
||||||
|
toast.error("Failed to create quicklink", { description: err })
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const { form: formData, enhance, errors } = form
|
||||||
|
const placeholders = {
|
||||||
|
name: "Quick Link Name",
|
||||||
|
link: "https://google.com/search?q={argument}"
|
||||||
|
}
|
||||||
|
|
||||||
|
const defaultFaviconUrl = $derived(
|
||||||
|
$formData.link ? new URL($formData.link).origin + "/favicon.ico" : undefined
|
||||||
|
)
|
||||||
|
$effect(() => {
|
||||||
|
if (defaultFaviconUrl && defaultFaviconUrl.length > 0) {
|
||||||
|
icon.type = IconEnum.RemoteUrl
|
||||||
|
icon.value = defaultFaviconUrl
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
$formData.iconType = icon.type
|
||||||
|
$formData.iconValue = icon.value
|
||||||
|
$formData.invertIcon = icon.invert
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<svelte:window on:keydown={goBackOnEscape} />
|
||||||
|
<Button variant="outline" size="icon" class="fixed left-2 top-2 z-50" onclick={goBack}>
|
||||||
|
<ArrowLeftIcon class="h-4 w-4" />
|
||||||
|
</Button>
|
||||||
|
<div class="h-12" data-tauri-drag-region></div>
|
||||||
|
<div class="container">
|
||||||
|
<h1 class="text-2xl font-bold">Create Quick Link</h1>
|
||||||
|
<form method="POST" use:enhance>
|
||||||
|
<Form.Field {form} name="name">
|
||||||
|
<Form.Control>
|
||||||
|
{#snippet children({ props })}
|
||||||
|
<Form.Label>Name</Form.Label>
|
||||||
|
<Input {...props} bind:value={$formData.name} placeholder={placeholders.name} />
|
||||||
|
{/snippet}
|
||||||
|
</Form.Control>
|
||||||
|
<Form.Description>Quick Link Display Name</Form.Description>
|
||||||
|
<Form.FieldErrors />
|
||||||
|
</Form.Field>
|
||||||
|
<Form.Field {form} name="link">
|
||||||
|
<Form.Control>
|
||||||
|
{#snippet children({ props })}
|
||||||
|
<Form.Label>Link</Form.Label>
|
||||||
|
<Input {...props} bind:value={$formData.link} placeholder={placeholders.link} />
|
||||||
|
{/snippet}
|
||||||
|
</Form.Control>
|
||||||
|
<Form.Description>Quick Link URL</Form.Description>
|
||||||
|
<Form.FieldErrors />
|
||||||
|
</Form.Field>
|
||||||
|
<IconSelector class="border" bind:icon />
|
||||||
|
<input name="iconType" hidden type="text" bind:value={$formData.iconType} />
|
||||||
|
<input name="iconValue" hidden type="text" bind:value={$formData.iconValue} />
|
||||||
|
<input name="invertIcon" hidden type="text" bind:value={$formData.invertIcon} />
|
||||||
|
<br />
|
||||||
|
<Form.Button>Submit</Form.Button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
{#if dev}
|
||||||
|
<div class="p-3">
|
||||||
|
<SuperDebug data={$formData} />
|
||||||
|
</div>
|
||||||
|
{/if}
|
@ -0,0 +1,7 @@
|
|||||||
|
import { z } from "zod"
|
||||||
|
|
||||||
|
export const formSchema = z.object({
|
||||||
|
username: z.string().min(2).max(50)
|
||||||
|
})
|
||||||
|
|
||||||
|
export type FormSchema = typeof formSchema
|
@ -76,7 +76,7 @@
|
|||||||
<ArrowLeft class="size-4" />
|
<ArrowLeft class="size-4" />
|
||||||
</Button>
|
</Button>
|
||||||
{/snippet}
|
{/snippet}
|
||||||
<Command.Root class="h-screen rounded-lg border shadow-md">
|
<Command.Root class="h-screen rounded-lg border shadow-md" loop>
|
||||||
<CustomCommandInput
|
<CustomCommandInput
|
||||||
autofocus
|
autofocus
|
||||||
placeholder="Type a command or search..."
|
placeholder="Type a command or search..."
|
||||||
|
@ -16,7 +16,6 @@
|
|||||||
import { get, derived as storeDerived } from "svelte/store"
|
import { get, derived as storeDerived } from "svelte/store"
|
||||||
|
|
||||||
const { data } = $props()
|
const { data } = $props()
|
||||||
// let { ext, manifest } = data
|
|
||||||
const ext = $derived(data.ext)
|
const ext = $derived(data.ext)
|
||||||
const manifest = $derived(data.manifest)
|
const manifest = $derived(data.manifest)
|
||||||
const installedExt = storeDerived(installedStoreExts, ($e) => {
|
const installedExt = storeDerived(installedStoreExts, ($e) => {
|
||||||
@ -117,20 +116,21 @@
|
|||||||
.uninstallStoreExtensionByIdentifier(ext.identifier)
|
.uninstallStoreExtensionByIdentifier(ext.identifier)
|
||||||
.then((uninstalledExt) => {
|
.then((uninstalledExt) => {
|
||||||
toast.success(`${uninstalledExt.name} Uninstalled`)
|
toast.success(`${uninstalledExt.name} Uninstalled`)
|
||||||
|
loading.uninstall = false
|
||||||
|
showBtn.uninstall = false
|
||||||
|
showBtn.install = true
|
||||||
})
|
})
|
||||||
.catch((err) => {
|
.catch((err) => {
|
||||||
toast.error("Fail to uninstall extension", { description: err })
|
toast.error("Fail to uninstall extension", { description: err })
|
||||||
error(`Fail to uninstall store extension (${ext.identifier}): ${err}`)
|
error(`Fail to uninstall store extension (${ext.identifier}): ${err}`)
|
||||||
})
|
})
|
||||||
.finally(() => {
|
.finally(() => {})
|
||||||
loading.uninstall = false
|
|
||||||
showBtn.uninstall = false
|
|
||||||
showBtn.install = true
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function onEnterPressed() {
|
function onEnterPressed() {
|
||||||
return onInstallSelected()
|
if (showBtn.install) {
|
||||||
|
return onInstallSelected()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleKeydown(e: KeyboardEvent) {
|
function handleKeydown(e: KeyboardEvent) {
|
||||||
@ -153,6 +153,7 @@
|
|||||||
<ArrowLeftIcon />
|
<ArrowLeftIcon />
|
||||||
</Button>
|
</Button>
|
||||||
<StoreExtDetail
|
<StoreExtDetail
|
||||||
|
class="px-5"
|
||||||
{ext}
|
{ext}
|
||||||
{manifest}
|
{manifest}
|
||||||
installedExt={$installedExt}
|
installedExt={$installedExt}
|
||||||
|
@ -1,19 +1,20 @@
|
|||||||
{
|
{
|
||||||
"extends": "./.svelte-kit/tsconfig.json",
|
"extends": "./.svelte-kit/tsconfig.json",
|
||||||
"compilerOptions": {
|
"compilerOptions": {
|
||||||
"allowJs": true,
|
"allowJs": true,
|
||||||
"checkJs": true,
|
"checkJs": true,
|
||||||
"esModuleInterop": true,
|
"esModuleInterop": true,
|
||||||
"forceConsistentCasingInFileNames": true,
|
"forceConsistentCasingInFileNames": true,
|
||||||
"resolveJsonModule": true,
|
"resolveJsonModule": true,
|
||||||
"skipLibCheck": true,
|
"skipLibCheck": true,
|
||||||
"sourceMap": true,
|
"sourceMap": true,
|
||||||
"strict": true,
|
"strict": true,
|
||||||
"moduleResolution": "bundler"
|
"moduleResolution": "bundler"
|
||||||
}
|
},
|
||||||
// Path aliases are handled by https://kit.svelte.dev/docs/configuration#alias
|
// Path aliases are handled by https://kit.svelte.dev/docs/configuration#alias
|
||||||
// except $lib which is handled by https://kit.svelte.dev/docs/configuration#files
|
// except $lib which is handled by https://kit.svelte.dev/docs/configuration#files
|
||||||
//
|
//
|
||||||
// If you want to overwrite includes/excludes, make sure to copy over the relevant includes/excludes
|
// If you want to overwrite includes/excludes, make sure to copy over the relevant includes/excludes
|
||||||
// from the referenced tsconfig.json - TypeScript does not merge them in
|
// from the referenced tsconfig.json - TypeScript does not merge them in
|
||||||
|
"include": ["src/**/*"]
|
||||||
}
|
}
|
||||||
|
@ -12,7 +12,7 @@
|
|||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@ianvs/prettier-plugin-sort-imports": "^4.3.1",
|
"@ianvs/prettier-plugin-sort-imports": "^4.3.1",
|
||||||
"@kksh/api": "workspace:*",
|
"@kksh/api": "workspace:*",
|
||||||
"@kksh/svelte5": "0.1.2-beta.4",
|
"@kksh/svelte5": "0.1.2-beta.8",
|
||||||
"prettier": "^3.2.5",
|
"prettier": "^3.2.5",
|
||||||
"prettier-plugin-svelte": "^3.2.7",
|
"prettier-plugin-svelte": "^3.2.7",
|
||||||
"prettier-plugin-tailwindcss": "^0.6.8",
|
"prettier-plugin-tailwindcss": "^0.6.8",
|
||||||
|
@ -41,6 +41,8 @@ export type Ext = InferOutput<typeof Ext>
|
|||||||
|
|
||||||
export enum CmdTypeEnum {
|
export enum CmdTypeEnum {
|
||||||
HeadlessWorker = "headless_worker",
|
HeadlessWorker = "headless_worker",
|
||||||
|
Builtin = "builtin",
|
||||||
|
System = "system",
|
||||||
UiWorker = "ui_worker",
|
UiWorker = "ui_worker",
|
||||||
UiIframe = "ui_iframe",
|
UiIframe = "ui_iframe",
|
||||||
QuickLink = "quick_link",
|
QuickLink = "quick_link",
|
||||||
@ -55,12 +57,18 @@ export const ExtCmd = object({
|
|||||||
name: string(),
|
name: string(),
|
||||||
type: CmdType,
|
type: CmdType,
|
||||||
data: string(),
|
data: string(),
|
||||||
alias: optional(string()),
|
alias: nullable(optional(string())),
|
||||||
hotkey: optional(string()),
|
hotkey: nullable(optional(string())),
|
||||||
enabled: boolean()
|
enabled: boolean()
|
||||||
})
|
})
|
||||||
export type ExtCmd = InferOutput<typeof ExtCmd>
|
export type ExtCmd = InferOutput<typeof ExtCmd>
|
||||||
|
|
||||||
|
export const QuickLinkCmd = object({
|
||||||
|
...ExtCmd.entries,
|
||||||
|
data: object({ link: string(), icon: Icon })
|
||||||
|
})
|
||||||
|
export type QuickLinkCmd = InferOutput<typeof QuickLinkCmd>
|
||||||
|
|
||||||
export const ExtData = object({
|
export const ExtData = object({
|
||||||
dataId: number(),
|
dataId: number(),
|
||||||
extId: number(),
|
extId: number(),
|
||||||
|
@ -1,4 +1,13 @@
|
|||||||
import { enum_, literal, object, string, type InferOutput } from "valibot"
|
import {
|
||||||
|
boolean,
|
||||||
|
enum_,
|
||||||
|
literal,
|
||||||
|
nullable,
|
||||||
|
object,
|
||||||
|
optional,
|
||||||
|
string,
|
||||||
|
type InferOutput
|
||||||
|
} from "valibot"
|
||||||
import { NodeName, NodeNameEnum } from "./constants"
|
import { NodeName, NodeNameEnum } from "./constants"
|
||||||
|
|
||||||
/* -------------------------------------------------------------------------- */
|
/* -------------------------------------------------------------------------- */
|
||||||
@ -16,7 +25,8 @@ export type IconType = InferOutput<typeof IconType>
|
|||||||
|
|
||||||
export const Icon = object({
|
export const Icon = object({
|
||||||
type: IconType,
|
type: IconType,
|
||||||
value: string()
|
value: string(),
|
||||||
|
invert: optional(boolean())
|
||||||
})
|
})
|
||||||
export type Icon = InferOutput<typeof Icon>
|
export type Icon = InferOutput<typeof Icon>
|
||||||
export const IconNode = object({
|
export const IconNode = object({
|
||||||
|
@ -1,5 +1,13 @@
|
|||||||
import { db } from "@kksh/api/commands"
|
import { db } from "@kksh/api/commands"
|
||||||
import { ExtPackageJson, ExtPackageJsonExtra } from "@kksh/api/models"
|
import {
|
||||||
|
CmdTypeEnum,
|
||||||
|
ExtCmd,
|
||||||
|
ExtPackageJson,
|
||||||
|
ExtPackageJsonExtra,
|
||||||
|
Icon,
|
||||||
|
QuickLinkCmd
|
||||||
|
} from "@kksh/api/models"
|
||||||
|
import * as v from "valibot"
|
||||||
|
|
||||||
export async function upsertExtension(extPkgJson: ExtPackageJson, extFullPath: string) {
|
export async function upsertExtension(extPkgJson: ExtPackageJson, extFullPath: string) {
|
||||||
const extInDb = await db.getUniqueExtensionByIdentifier(extPkgJson.kunkun.identifier)
|
const extInDb = await db.getUniqueExtensionByIdentifier(extPkgJson.kunkun.identifier)
|
||||||
@ -12,3 +20,38 @@ export async function upsertExtension(extPkgJson: ExtPackageJson, extFullPath: s
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export async function createQuickLinkCommand(name: string, link: string, icon: Icon) {
|
||||||
|
const extension = await db.getExtQuickLinks()
|
||||||
|
return db.createCommand({
|
||||||
|
extId: extension.extId,
|
||||||
|
name,
|
||||||
|
cmdType: CmdTypeEnum.QuickLink,
|
||||||
|
data: JSON.stringify({
|
||||||
|
link,
|
||||||
|
icon
|
||||||
|
}),
|
||||||
|
enabled: true
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function getAllQuickLinkCommands(): Promise<QuickLinkCmd[]> {
|
||||||
|
const extension = await db.getExtQuickLinks()
|
||||||
|
const cmds = await db.getCommandsByExtId(extension.extId)
|
||||||
|
return cmds
|
||||||
|
.map((cmd) => {
|
||||||
|
try {
|
||||||
|
cmd.data = JSON.parse(cmd.data)
|
||||||
|
const parsedData = v.safeParse(QuickLinkCmd, cmd)
|
||||||
|
if (!parsedData.success) {
|
||||||
|
console.warn("Fail to parse quick link command", cmd)
|
||||||
|
console.error(v.flatten(parsedData.issues))
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
return parsedData.output
|
||||||
|
} catch (error) {
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.filter((cmd) => cmd !== null)
|
||||||
|
}
|
||||||
|
@ -8,7 +8,8 @@
|
|||||||
".": "./src/index.ts"
|
".": "./src/index.ts"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@kksh/api": "workspace:*"
|
"@kksh/api": "workspace:*",
|
||||||
|
"@supabase/supabase-js": "^2.46.1"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/bun": "latest"
|
"@types/bun": "latest"
|
||||||
|
@ -8,10 +8,10 @@
|
|||||||
},
|
},
|
||||||
"aliases": {
|
"aliases": {
|
||||||
"components": "@kksh/ui/src/components",
|
"components": "@kksh/ui/src/components",
|
||||||
"utils": "@kksh/ui/src/utils",
|
"utils": "@kksh/ui/utils",
|
||||||
"ui": "@kksh/ui/src/components/ui",
|
"ui": "@kksh/ui/src/components/ui",
|
||||||
"hooks": "@kksh/ui/src/hooks"
|
"hooks": "@kksh/ui/src/hooks"
|
||||||
},
|
},
|
||||||
"typescript": true,
|
"typescript": true,
|
||||||
"registry": "https://next.shadcn-svelte.com/registry"
|
"registry": "https://next.shadcn-svelte.com/registry"
|
||||||
}
|
}
|
@ -35,19 +35,24 @@
|
|||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@kksh/api": "workspace:*",
|
"@kksh/api": "workspace:*",
|
||||||
|
"@kksh/svelte5": "^0.1.2-beta.8",
|
||||||
"@types/bun": "latest",
|
"@types/bun": "latest",
|
||||||
"bits-ui": "1.0.0-next.36",
|
"bits-ui": "1.0.0-next.45",
|
||||||
|
"@iconify/svelte": "^4.0.2",
|
||||||
"clsx": "^2.1.1",
|
"clsx": "^2.1.1",
|
||||||
|
"formsnap": "2.0.0-next.1",
|
||||||
"lucide-svelte": "^0.454.0",
|
"lucide-svelte": "^0.454.0",
|
||||||
"mode-watcher": "^0.4.1",
|
"mode-watcher": "^0.4.1",
|
||||||
"paneforge": "1.0.0-next.1",
|
"paneforge": "1.0.0-next.1",
|
||||||
"shiki": "^1.22.2",
|
"shiki": "^1.22.2",
|
||||||
"svelte-radix": "^2.0.1",
|
"svelte-radix": "^2.0.1",
|
||||||
"svelte-sonner": "^0.3.28",
|
"svelte-sonner": "^0.3.28",
|
||||||
|
"sveltekit-superforms": "^2.20.0",
|
||||||
"tailwind-merge": "^2.5.4",
|
"tailwind-merge": "^2.5.4",
|
||||||
"tailwind-variants": "^0.2.1",
|
"tailwind-variants": "^0.2.1",
|
||||||
"tailwindcss": "^3.4.14",
|
"tailwindcss": "^3.4.14",
|
||||||
"tailwindcss-animate": "^1.0.7"
|
"tailwindcss-animate": "^1.0.7",
|
||||||
|
"zod": "^3.23.8"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@formkit/auto-animate": "^0.8.2",
|
"@formkit/auto-animate": "^0.8.2",
|
||||||
|
@ -12,17 +12,37 @@
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if icon.type === IconEnum.RemoteUrl}
|
{#if icon.type === IconEnum.RemoteUrl}
|
||||||
<img loading="lazy" class={cn("", className)} src={icon.value} alt="" {...restProps} />
|
<img
|
||||||
|
loading="lazy"
|
||||||
|
class={cn("", className, { invert: icon.invert })}
|
||||||
|
src={icon.value}
|
||||||
|
alt=""
|
||||||
|
{...restProps}
|
||||||
|
/>
|
||||||
{:else if icon.type === IconEnum.Iconify}
|
{:else if icon.type === IconEnum.Iconify}
|
||||||
<Icon icon={icon.value} class={cn("", className)} {...restProps} />
|
<Icon icon={icon.value} class={cn("", className, { invert: icon.invert })} {...restProps} />
|
||||||
{:else if icon.type === IconEnum.Base64PNG}
|
{:else if icon.type === IconEnum.Base64PNG}
|
||||||
<img loading="lazy" src="data:image/png;base64, {icon.value}" alt="" {...restProps} />
|
<img
|
||||||
|
class={cn(className, { invert: icon.invert })}
|
||||||
|
loading="lazy"
|
||||||
|
src="data:image/png;base64, {icon.value}"
|
||||||
|
alt=""
|
||||||
|
{...restProps}
|
||||||
|
/>
|
||||||
{:else if icon.type === IconEnum.Text}
|
{:else if icon.type === IconEnum.Text}
|
||||||
<Button class={cn("shrink-0 text-center", className)} size="icon" {...restProps}>
|
<Button
|
||||||
|
class={cn("shrink-0 text-center", className, { invert: icon.invert })}
|
||||||
|
size="icon"
|
||||||
|
{...restProps}
|
||||||
|
>
|
||||||
{icon.value}
|
{icon.value}
|
||||||
</Button>
|
</Button>
|
||||||
{:else if icon.type === IconEnum.Svg}
|
{:else if icon.type === IconEnum.Svg}
|
||||||
<span {...restProps}>{@html icon.value}</span>
|
<span {...restProps} class={cn(className, { invert: icon.invert })}>{@html icon.value}</span>
|
||||||
{:else}
|
{:else}
|
||||||
<Icon icon="mingcute:appstore-fill" class={cn("", className)} {...restProps} />
|
<Icon
|
||||||
|
icon="mingcute:appstore-fill"
|
||||||
|
class={cn("", className, { invert: icon.invert })}
|
||||||
|
{...restProps}
|
||||||
|
/>
|
||||||
{/if}
|
{/if}
|
||||||
|
56
packages/ui/src/components/common/IconSelector.svelte
Normal file
56
packages/ui/src/components/common/IconSelector.svelte
Normal file
@ -0,0 +1,56 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Icon, IconEnum, IconType } from "@kksh/api/models"
|
||||||
|
import { Button, Checkbox, Label, Select, Textarea } from "@kksh/svelte5"
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import { open } from "tauri-plugin-shellx-api"
|
||||||
|
import IconMultiplexer from "./IconMultiplexer.svelte"
|
||||||
|
|
||||||
|
let { icon = $bindable<Icon>(), class: className }: { icon?: Icon; class?: string } = $props()
|
||||||
|
const iconOptions: Record<string, IconType> = {
|
||||||
|
"Remote Url": IconEnum.RemoteUrl,
|
||||||
|
Iconify: IconEnum.Iconify,
|
||||||
|
Svg: IconEnum.Svg,
|
||||||
|
"Base64 PNG": IconEnum.Base64PNG
|
||||||
|
}
|
||||||
|
const iconOptionsArray = $derived(Object.entries(iconOptions))
|
||||||
|
const triggerContent = $derived(
|
||||||
|
iconOptionsArray.find(([_, value]) => value === icon.type)?.[0] ?? "Select a fruit"
|
||||||
|
)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<div class="flex flex-col gap-2">
|
||||||
|
<Select.Root type="single" name="icontype" bind:value={icon.type}>
|
||||||
|
<Select.Trigger class="w-[180px]">
|
||||||
|
{triggerContent}
|
||||||
|
</Select.Trigger>
|
||||||
|
<Select.Content>
|
||||||
|
<Select.Group>
|
||||||
|
<Select.GroupHeading>Icon Type</Select.GroupHeading>
|
||||||
|
{#each iconOptionsArray as [label, value]}
|
||||||
|
<Select.Item {value}>{label}</Select.Item>
|
||||||
|
{/each}
|
||||||
|
</Select.Group>
|
||||||
|
</Select.Content>
|
||||||
|
</Select.Root>
|
||||||
|
<Textarea bind:value={icon.value} placeholder="Icon Value" />
|
||||||
|
{#if icon.type === IconEnum.Iconify}
|
||||||
|
<Button onclick={() => open("https://icon-sets.iconify.design/")} size="sm" variant="secondary">
|
||||||
|
Pick Iconify icon name
|
||||||
|
</Button>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
<div class="flex items-center space-x-2">
|
||||||
|
<Checkbox id="terms" bind:checked={icon.invert} aria-labelledby="terms-label" />
|
||||||
|
<Label
|
||||||
|
id="terms-label"
|
||||||
|
for="terms"
|
||||||
|
class="text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"
|
||||||
|
>
|
||||||
|
Invert Icon Color
|
||||||
|
</Label>
|
||||||
|
</div>
|
||||||
|
<h2 class="font-semibold">Icon Preview</h2>
|
||||||
|
{#if icon.type && icon.value && icon.value.length > 0}
|
||||||
|
<IconMultiplexer class="h-12 w-12" {icon} />
|
||||||
|
{/if}
|
||||||
|
</div>
|
77
packages/ui/src/components/common/IconSelectorDialog.svelte
Normal file
77
packages/ui/src/components/common/IconSelectorDialog.svelte
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Icon, IconEnum, IconType } from "@kksh/api/models"
|
||||||
|
import { Button, ButtonModule, Dialog, Input, Label, Select } from "@kksh/svelte5"
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import { ImageIcon } from "lucide-svelte"
|
||||||
|
import IconMultiplexer from "./IconMultiplexer.svelte"
|
||||||
|
|
||||||
|
const { icon, class: className }: { icon?: Icon; class?: string } = $props()
|
||||||
|
function onClick(e: MouseEvent) {
|
||||||
|
e.preventDefault()
|
||||||
|
e.stopPropagation()
|
||||||
|
console.log("clicked")
|
||||||
|
}
|
||||||
|
|
||||||
|
let iconType = $state<string>(icon?.type ?? IconEnum.Iconify)
|
||||||
|
const iconOptions: Record<string, IconType> = {
|
||||||
|
Iconify: IconEnum.Iconify,
|
||||||
|
"Remote Url": IconEnum.RemoteUrl,
|
||||||
|
Svg: IconEnum.Svg,
|
||||||
|
"Base64 PNG": IconEnum.Base64PNG,
|
||||||
|
Text: IconEnum.Text
|
||||||
|
}
|
||||||
|
const iconOptionsArray = $derived(Object.entries(iconOptions))
|
||||||
|
const triggerContent = $derived(
|
||||||
|
iconOptionsArray.find(([_, value]) => value === iconType)?.[0] ?? "Select a fruit"
|
||||||
|
)
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<button class={cn("block h-12 w-12", className)} onclick={onClick}>
|
||||||
|
{#if icon}
|
||||||
|
<IconMultiplexer {icon} />
|
||||||
|
{:else}
|
||||||
|
<ImageIcon class="h-full w-full" />
|
||||||
|
{/if}
|
||||||
|
</button>
|
||||||
|
|
||||||
|
<Dialog.Root open={true}>
|
||||||
|
<Dialog.Trigger class={ButtonModule.buttonVariants({ variant: "outline" })}>
|
||||||
|
Select Icon
|
||||||
|
</Dialog.Trigger>
|
||||||
|
<Dialog.Content class="sm:max-w-[425px]">
|
||||||
|
<Dialog.Header>
|
||||||
|
<Dialog.Title>Select Icon</Dialog.Title>
|
||||||
|
<!-- <Dialog.Description></Dialog.Description> -->
|
||||||
|
</Dialog.Header>
|
||||||
|
|
||||||
|
<Select.Root type="single" name="icontype" bind:value={iconType}>
|
||||||
|
<Select.Trigger class="w-[180px]">
|
||||||
|
{triggerContent}
|
||||||
|
</Select.Trigger>
|
||||||
|
<Select.Content>
|
||||||
|
<Select.Group>
|
||||||
|
<Select.GroupHeading>Fruits</Select.GroupHeading>
|
||||||
|
{#each iconOptionsArray as [label, value]}
|
||||||
|
<Select.Item {value}>{label}</Select.Item>
|
||||||
|
{/each}
|
||||||
|
</Select.Group>
|
||||||
|
</Select.Content>
|
||||||
|
</Select.Root>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<!-- <div class="grid gap-4 py-4">
|
||||||
|
<div class="grid grid-cols-4 items-center gap-4">
|
||||||
|
<Label for="name" class="text-right">Name</Label>
|
||||||
|
<Input id="name" value="Pedro Duarte" class="col-span-3" />
|
||||||
|
</div>
|
||||||
|
<div class="grid grid-cols-4 items-center gap-4">
|
||||||
|
<Label for="username" class="text-right">Username</Label>
|
||||||
|
<Input id="username" value="@peduarte" class="col-span-3" />
|
||||||
|
</div>
|
||||||
|
</div> -->
|
||||||
|
<Dialog.Footer>
|
||||||
|
<Button type="submit">Save</Button>
|
||||||
|
</Dialog.Footer>
|
||||||
|
</Dialog.Content>
|
||||||
|
</Dialog.Root>
|
@ -1 +1,2 @@
|
|||||||
export { default as IconMultiplexer } from "./IconMultiplexer.svelte"
|
export { default as IconMultiplexer } from "./IconMultiplexer.svelte"
|
||||||
|
export { default as IconSelector } from "./IconSelector.svelte"
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import autoAnimate from "@formkit/auto-animate"
|
import autoAnimate from "@formkit/auto-animate"
|
||||||
import Icon from "@iconify/svelte"
|
import Icon from "@iconify/svelte"
|
||||||
import { Button, buttonVariants, Collapsible, ScrollArea } from "@kksh/svelte5"
|
import { Button, ButtonModule, Collapsible, ScrollArea } from "@kksh/svelte5"
|
||||||
import { Error, Layouts, Shiki } from "@kksh/ui"
|
import { Error, Layouts, Shiki } from "@kksh/ui"
|
||||||
import { ChevronsUpDown } from "lucide-svelte"
|
import { ChevronsUpDown } from "lucide-svelte"
|
||||||
import { type Snippet } from "svelte"
|
import { type Snippet } from "svelte"
|
||||||
@ -13,7 +13,7 @@
|
|||||||
class: className,
|
class: className,
|
||||||
rawJsonError,
|
rawJsonError,
|
||||||
onGoBack,
|
onGoBack,
|
||||||
footer
|
footer: footer2
|
||||||
}: {
|
}: {
|
||||||
title: string
|
title: string
|
||||||
message: string
|
message: string
|
||||||
@ -45,7 +45,7 @@
|
|||||||
<div class="flex items-center justify-between space-x-4 px-4">
|
<div class="flex items-center justify-between space-x-4 px-4">
|
||||||
<h4 class="text-sm font-semibold">Raw Error JSON</h4>
|
<h4 class="text-sm font-semibold">Raw Error JSON</h4>
|
||||||
<Collapsible.Trigger
|
<Collapsible.Trigger
|
||||||
class={buttonVariants({ variant: "ghost", size: "sm", class: "w-9 p-0" })}
|
class={ButtonModule.buttonVariants({ variant: "ghost", size: "sm", class: "w-9 p-0" })}
|
||||||
>
|
>
|
||||||
<ChevronsUpDown class="size-4" />
|
<ChevronsUpDown class="size-4" />
|
||||||
</Collapsible.Trigger>
|
</Collapsible.Trigger>
|
||||||
@ -58,8 +58,8 @@
|
|||||||
</Collapsible.Root>
|
</Collapsible.Root>
|
||||||
<br />
|
<br />
|
||||||
{#snippet footer()}
|
{#snippet footer()}
|
||||||
{#if footer}
|
{#if footer2}
|
||||||
{@render footer()}
|
{@render footer2()}
|
||||||
{:else}
|
{:else}
|
||||||
<Button variant="default" class="w-full" onclick={onGoBack} disabled={enterDown}>
|
<Button variant="default" class="w-full" onclick={onGoBack} disabled={enterDown}>
|
||||||
Go Back
|
Go Back
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import autoAnimate from "@formkit/auto-animate"
|
import autoAnimate from "@formkit/auto-animate"
|
||||||
import Icon from "@iconify/svelte"
|
import Icon from "@iconify/svelte"
|
||||||
import { ExtPackageJsonExtra, IconEnum, KunkunExtManifest } from "@kksh/api/models"
|
import { ExtPackageJson, IconEnum, KunkunExtManifest } from "@kksh/api/models"
|
||||||
import { type Tables } from "@kksh/api/supabase/types"
|
import { type Tables } from "@kksh/api/supabase/types"
|
||||||
import { Button, ScrollArea, Separator } from "@kksh/svelte5"
|
import { Button, ScrollArea, Separator } from "@kksh/svelte5"
|
||||||
import { Constants, IconMultiplexer } from "@kksh/ui"
|
import { Constants, IconMultiplexer } from "@kksh/ui"
|
||||||
@ -26,7 +26,7 @@
|
|||||||
imageDialogOpen = $bindable(false)
|
imageDialogOpen = $bindable(false)
|
||||||
}: {
|
}: {
|
||||||
ext: Tables<"ext_publish">
|
ext: Tables<"ext_publish">
|
||||||
installedExt?: ExtPackageJsonExtra
|
installedExt?: ExtPackageJson
|
||||||
manifest: KunkunExtManifest
|
manifest: KunkunExtManifest
|
||||||
demoImages: string[]
|
demoImages: string[]
|
||||||
class?: string
|
class?: string
|
||||||
@ -116,7 +116,7 @@
|
|||||||
{/snippet}
|
{/snippet}
|
||||||
|
|
||||||
<div data-tauri-drag-region class="h-14"></div>
|
<div data-tauri-drag-region class="h-14"></div>
|
||||||
<ScrollArea class="container pb-12">
|
<ScrollArea class={cn("w-full pb-12", className)}>
|
||||||
<div class="flex items-center gap-4">
|
<div class="flex items-center gap-4">
|
||||||
<span class="h-12 w-12">
|
<span class="h-12 w-12">
|
||||||
<IconMultiplexer
|
<IconMultiplexer
|
||||||
|
@ -1,9 +1,9 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { IconEnum } from "@kksh/api/models"
|
import { CmdTypeEnum, IconEnum } from "@kksh/api/models"
|
||||||
import { Command } from "@kksh/svelte5"
|
import { Command } from "@kksh/svelte5"
|
||||||
import { IconMultiplexer } from "@kksh/ui"
|
import { IconMultiplexer } from "@kksh/ui"
|
||||||
import { DraggableCommandGroup } from "@kksh/ui/custom"
|
import { DraggableCommandGroup } from "../custom"
|
||||||
import type { BuiltinCmd } from "./types"
|
import type { BuiltinCmd, CmdValue } from "./types"
|
||||||
|
|
||||||
const { builtinCmds }: { builtinCmds: BuiltinCmd[] } = $props()
|
const { builtinCmds }: { builtinCmds: BuiltinCmd[] } = $props()
|
||||||
</script>
|
</script>
|
||||||
@ -15,6 +15,10 @@
|
|||||||
onSelect={() => {
|
onSelect={() => {
|
||||||
cmd.function()
|
cmd.function()
|
||||||
}}
|
}}
|
||||||
|
value={JSON.stringify({
|
||||||
|
cmdName: cmd.name,
|
||||||
|
cmdType: CmdTypeEnum.Builtin
|
||||||
|
} satisfies CmdValue)}
|
||||||
>
|
>
|
||||||
<span class="flex gap-2">
|
<span class="flex gap-2">
|
||||||
<IconMultiplexer
|
<IconMultiplexer
|
||||||
|
@ -8,9 +8,11 @@
|
|||||||
class: className,
|
class: className,
|
||||||
value = $bindable(""),
|
value = $bindable(""),
|
||||||
leftSlot,
|
leftSlot,
|
||||||
|
rightSlot,
|
||||||
...restProps
|
...restProps
|
||||||
}: CommandPrimitive.InputProps & {
|
}: CommandPrimitive.InputProps & {
|
||||||
leftSlot?: Snippet
|
leftSlot?: Snippet
|
||||||
|
rightSlot?: Snippet
|
||||||
} = $props()
|
} = $props()
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@ -25,4 +27,5 @@
|
|||||||
bind:value
|
bind:value
|
||||||
{...restProps}
|
{...restProps}
|
||||||
/>
|
/>
|
||||||
|
{@render rightSlot?.()}
|
||||||
</div>
|
</div>
|
||||||
|
@ -1,11 +1,11 @@
|
|||||||
<!-- This file renders a group of extension commands -->
|
<!-- This file renders a group of extension commands -->
|
||||||
<!-- Input props to this component is an array of ExtPackageJsonExtra[] -->
|
<!-- Input props to this component is an array of ExtPackageJsonExtra[] -->
|
||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { CustomUiCmd, ExtPackageJsonExtra, TemplateUiCmd } from "@kksh/api/models"
|
import { CmdTypeEnum, CustomUiCmd, ExtPackageJsonExtra, TemplateUiCmd } from "@kksh/api/models"
|
||||||
import { Badge, Command } from "@kksh/svelte5"
|
import { Badge, Command } from "@kksh/svelte5"
|
||||||
import { IconMultiplexer } from "@kksh/ui"
|
import { IconMultiplexer } from "@kksh/ui"
|
||||||
import { DraggableCommandGroup } from "@kksh/ui/custom"
|
import { DraggableCommandGroup } from "../custom"
|
||||||
import type { OnExtCmdSelect } from "./types"
|
import type { CmdValue, OnExtCmdSelect } from "./types"
|
||||||
|
|
||||||
const {
|
const {
|
||||||
extensions,
|
extensions,
|
||||||
@ -28,6 +28,10 @@
|
|||||||
onSelect={() => {
|
onSelect={() => {
|
||||||
onExtCmdSelect(ext, cmd, { isDev, hmr })
|
onExtCmdSelect(ext, cmd, { isDev, hmr })
|
||||||
}}
|
}}
|
||||||
|
value={JSON.stringify({
|
||||||
|
cmdName: cmd.name,
|
||||||
|
cmdType: cmd.type
|
||||||
|
} satisfies CmdValue)}
|
||||||
>
|
>
|
||||||
<span class="flex gap-2">
|
<span class="flex gap-2">
|
||||||
<IconMultiplexer icon={cmd.icon ?? ext.kunkun.icon} class="!h-5 !w-5 shrink-0" />
|
<IconMultiplexer icon={cmd.icon ?? ext.kunkun.icon} class="!h-5 !w-5 shrink-0" />
|
||||||
|
31
packages/ui/src/components/main/QuickLinks.svelte
Normal file
31
packages/ui/src/components/main/QuickLinks.svelte
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { CmdTypeEnum, IconEnum } from "@kksh/api/models"
|
||||||
|
import { Command } from "@kksh/svelte5"
|
||||||
|
import { IconMultiplexer } from "@kksh/ui"
|
||||||
|
import { DraggableCommandGroup } from "../custom"
|
||||||
|
import { CmdValue, type CmdQuery, type QuickLink } from "./types"
|
||||||
|
|
||||||
|
const { quickLinks }: { quickLinks: QuickLink[] } = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<DraggableCommandGroup heading="Quick Links">
|
||||||
|
{#each quickLinks as cmd}
|
||||||
|
<Command.Item
|
||||||
|
class="flex justify-between"
|
||||||
|
onSelect={() => {
|
||||||
|
console.log(cmd)
|
||||||
|
}}
|
||||||
|
keywords={["quick", "link"]}
|
||||||
|
value={JSON.stringify({
|
||||||
|
cmdName: cmd.name,
|
||||||
|
cmdType: CmdTypeEnum.QuickLink,
|
||||||
|
data: cmd.link
|
||||||
|
} satisfies CmdValue)}
|
||||||
|
>
|
||||||
|
<span class="flex gap-2">
|
||||||
|
<IconMultiplexer icon={cmd.icon} class="!h-5 !w-5 shrink-0" />
|
||||||
|
<span>{cmd.name}</span>
|
||||||
|
</span>
|
||||||
|
</Command.Item>
|
||||||
|
{/each}
|
||||||
|
</DraggableCommandGroup>
|
@ -1,8 +1,9 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { IconEnum, SysCommand } from "@kksh/api/models"
|
import { CmdTypeEnum, IconEnum, SysCommand } from "@kksh/api/models"
|
||||||
import { Command } from "@kksh/svelte5"
|
import { Command } from "@kksh/svelte5"
|
||||||
import { IconMultiplexer } from "@kksh/ui"
|
import { IconMultiplexer } from "@kksh/ui"
|
||||||
import { DraggableCommandGroup } from "@kksh/ui/custom"
|
import { DraggableCommandGroup } from "../custom"
|
||||||
|
import { CmdValue } from "./types"
|
||||||
|
|
||||||
const { systemCommands }: { systemCommands: SysCommand[] } = $props()
|
const { systemCommands }: { systemCommands: SysCommand[] } = $props()
|
||||||
</script>
|
</script>
|
||||||
@ -14,6 +15,10 @@
|
|||||||
onSelect={() => {
|
onSelect={() => {
|
||||||
cmd.function()
|
cmd.function()
|
||||||
}}
|
}}
|
||||||
|
value={JSON.stringify({
|
||||||
|
cmdName: cmd.name,
|
||||||
|
cmdType: CmdTypeEnum.System
|
||||||
|
} satisfies CmdValue)}
|
||||||
>
|
>
|
||||||
<span class="flex gap-2">
|
<span class="flex gap-2">
|
||||||
{#if cmd.icon}
|
{#if cmd.icon}
|
||||||
|
@ -3,4 +3,5 @@ export { default as CustomCommandInput } from "./CustomCommandInput.svelte"
|
|||||||
export { default as GlobalCommandPaletteFooter } from "./GlobalCommandPaletteFooter.svelte"
|
export { default as GlobalCommandPaletteFooter } from "./GlobalCommandPaletteFooter.svelte"
|
||||||
export { default as ExtCmdsGroup } from "./ExtCmdsGroup.svelte"
|
export { default as ExtCmdsGroup } from "./ExtCmdsGroup.svelte"
|
||||||
export { default as SystemCmds } from "./SystemCmds.svelte"
|
export { default as SystemCmds } from "./SystemCmds.svelte"
|
||||||
|
export { default as QuickLinks } from "./QuickLinks.svelte"
|
||||||
export * from "./types"
|
export * from "./types"
|
||||||
|
@ -1,4 +1,11 @@
|
|||||||
import type { CustomUiCmd, ExtPackageJsonExtra, TemplateUiCmd } from "@kksh/api/models"
|
import {
|
||||||
|
CmdType,
|
||||||
|
Icon,
|
||||||
|
type CustomUiCmd,
|
||||||
|
type ExtPackageJsonExtra,
|
||||||
|
type TemplateUiCmd
|
||||||
|
} from "@kksh/api/models"
|
||||||
|
import * as v from "valibot"
|
||||||
|
|
||||||
export type BuiltinCmd = {
|
export type BuiltinCmd = {
|
||||||
name: string
|
name: string
|
||||||
@ -15,4 +22,18 @@ export type OnExtCmdSelect = (
|
|||||||
|
|
||||||
export type CommandLaunchers = {
|
export type CommandLaunchers = {
|
||||||
onExtCmdSelect: OnExtCmdSelect
|
onExtCmdSelect: OnExtCmdSelect
|
||||||
|
onQuickLinkSelect: (quickLink: CmdValue, queries: CmdQuery[]) => void
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Command Value used in the command search
|
||||||
|
*/
|
||||||
|
export const CmdValue = v.object({
|
||||||
|
cmdName: v.string(),
|
||||||
|
cmdType: CmdType,
|
||||||
|
data: v.optional(v.any())
|
||||||
|
})
|
||||||
|
export type CmdValue = v.InferOutput<typeof CmdValue>
|
||||||
|
|
||||||
|
export type CmdQuery = { name: string; value: string }
|
||||||
|
export type QuickLink = { name: string; link: string; icon: Icon }
|
||||||
|
16
packages/ui/src/components/theme/mode-toggle.svelte
Normal file
16
packages/ui/src/components/theme/mode-toggle.svelte
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Button } from "@kksh/svelte5"
|
||||||
|
import Moon from "lucide-svelte/icons/moon"
|
||||||
|
import Sun from "lucide-svelte/icons/sun"
|
||||||
|
import { toggleMode } from "mode-watcher"
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<Button onclick={toggleMode} variant="outline" size="icon">
|
||||||
|
<Sun
|
||||||
|
class="h-[1.2rem] w-[1.2rem] rotate-0 scale-100 transition-all dark:-rotate-90 dark:scale-0"
|
||||||
|
/>
|
||||||
|
<Moon
|
||||||
|
class="absolute h-[1.2rem] w-[1.2rem] rotate-90 scale-0 transition-all dark:rotate-0 dark:scale-100"
|
||||||
|
/>
|
||||||
|
<span class="sr-only">Toggle theme</span>
|
||||||
|
</Button>
|
7
packages/ui/src/components/ui/form/form-button.svelte
Normal file
7
packages/ui/src/components/ui/form/form-button.svelte
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { ButtonModule } from "@kksh/svelte5"
|
||||||
|
|
||||||
|
let { ref = $bindable(null), ...restProps }: ButtonModule.Props = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<ButtonModule.Root type="submit" bind:ref {...restProps} />
|
17
packages/ui/src/components/ui/form/form-description.svelte
Normal file
17
packages/ui/src/components/ui/form/form-description.svelte
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import type { WithoutChild } from "bits-ui"
|
||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
|
||||||
|
let {
|
||||||
|
ref = $bindable(null),
|
||||||
|
class: className,
|
||||||
|
...restProps
|
||||||
|
}: WithoutChild<FormPrimitive.DescriptionProps> = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormPrimitive.Description
|
||||||
|
bind:ref
|
||||||
|
class={cn("text-muted-foreground text-[0.8rem]", className)}
|
||||||
|
{...restProps}
|
||||||
|
/>
|
31
packages/ui/src/components/ui/form/form-element-field.svelte
Normal file
31
packages/ui/src/components/ui/form/form-element-field.svelte
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
<script lang="ts" module>
|
||||||
|
import type { FormPathLeaves as _FormPathLeaves } from "sveltekit-superforms"
|
||||||
|
|
||||||
|
type T = Record<string, unknown>
|
||||||
|
type U = _FormPathLeaves<T>
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts" generics="T extends Record<string, unknown>, U extends _FormPathLeaves<T>">
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import type { WithElementRef } from "bits-ui"
|
||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
import type { HTMLAttributes } from "svelte/elements"
|
||||||
|
|
||||||
|
let {
|
||||||
|
ref = $bindable(null),
|
||||||
|
class: className,
|
||||||
|
form,
|
||||||
|
name,
|
||||||
|
children: childrenProp,
|
||||||
|
...restProps
|
||||||
|
}: WithElementRef<HTMLAttributes<HTMLDivElement>> &
|
||||||
|
FormPrimitive.ElementFieldProps<T, U> = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormPrimitive.ElementField {form} {name}>
|
||||||
|
{#snippet children({ constraints, errors, tainted, value })}
|
||||||
|
<div bind:this={ref} class={cn("space-y-2", className)} {...restProps}>
|
||||||
|
{@render childrenProp?.({ constraints, errors, tainted, value: value as T[U] })}
|
||||||
|
</div>
|
||||||
|
{/snippet}
|
||||||
|
</FormPrimitive.ElementField>
|
30
packages/ui/src/components/ui/form/form-field-errors.svelte
Normal file
30
packages/ui/src/components/ui/form/form-field-errors.svelte
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import type { WithoutChild } from "bits-ui"
|
||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
|
||||||
|
let {
|
||||||
|
ref = $bindable(null),
|
||||||
|
class: className,
|
||||||
|
errorClasses,
|
||||||
|
children: childrenProp,
|
||||||
|
...restProps
|
||||||
|
}: WithoutChild<FormPrimitive.FieldErrorsProps> & {
|
||||||
|
errorClasses?: string | undefined | null
|
||||||
|
} = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormPrimitive.FieldErrors
|
||||||
|
class={cn("text-destructive text-[0.8rem] font-medium", className)}
|
||||||
|
{...restProps}
|
||||||
|
>
|
||||||
|
{#snippet children({ errors, errorProps })}
|
||||||
|
{#if childrenProp}
|
||||||
|
{@render childrenProp({ errors, errorProps })}
|
||||||
|
{:else}
|
||||||
|
{#each errors as error}
|
||||||
|
<div {...errorProps} class={cn(errorClasses)}>{error}</div>
|
||||||
|
{/each}
|
||||||
|
{/if}
|
||||||
|
{/snippet}
|
||||||
|
</FormPrimitive.FieldErrors>
|
31
packages/ui/src/components/ui/form/form-field.svelte
Normal file
31
packages/ui/src/components/ui/form/form-field.svelte
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
<script lang="ts" module>
|
||||||
|
import type { FormPath as _FormPath } from "sveltekit-superforms"
|
||||||
|
|
||||||
|
type T = Record<string, unknown>
|
||||||
|
type U = _FormPath<T>
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts" generics="T extends Record<string, unknown>, U extends _FormPath<T>">
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import type { WithElementRef, WithoutChildren } from "bits-ui"
|
||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
import type { HTMLAttributes } from "svelte/elements"
|
||||||
|
|
||||||
|
let {
|
||||||
|
ref = $bindable(null),
|
||||||
|
class: className,
|
||||||
|
form,
|
||||||
|
name,
|
||||||
|
children: childrenProp,
|
||||||
|
...restProps
|
||||||
|
}: FormPrimitive.FieldProps<T, U> &
|
||||||
|
WithoutChildren<WithElementRef<HTMLAttributes<HTMLDivElement>>> = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormPrimitive.Field {form} {name}>
|
||||||
|
{#snippet children({ constraints, errors, tainted, value })}
|
||||||
|
<div bind:this={ref} class={cn("space-y-2", className)} {...restProps}>
|
||||||
|
{@render childrenProp?.({ constraints, errors, tainted, value: value as T[U] })}
|
||||||
|
</div>
|
||||||
|
{/snippet}
|
||||||
|
</FormPrimitive.Field>
|
22
packages/ui/src/components/ui/form/form-fieldset.svelte
Normal file
22
packages/ui/src/components/ui/form/form-fieldset.svelte
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
<script lang="ts" module>
|
||||||
|
import type { FormPath as _FormPath } from "sveltekit-superforms"
|
||||||
|
|
||||||
|
type T = Record<string, unknown>
|
||||||
|
type U = _FormPath<T>
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts" generics="T extends Record<string, unknown>, U extends _FormPath<T>">
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import type { WithoutChild } from "bits-ui"
|
||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
|
||||||
|
let {
|
||||||
|
ref = $bindable(null),
|
||||||
|
class: className,
|
||||||
|
form,
|
||||||
|
name,
|
||||||
|
...restProps
|
||||||
|
}: WithoutChild<FormPrimitive.FieldsetProps<T, U>> = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormPrimitive.Fieldset bind:ref {form} {name} class={cn("space-y-2", className)} {...restProps} />
|
21
packages/ui/src/components/ui/form/form-label.svelte
Normal file
21
packages/ui/src/components/ui/form/form-label.svelte
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { Label } from "@kksh/svelte5"
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import type { WithoutChild } from "bits-ui"
|
||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
|
||||||
|
let {
|
||||||
|
ref = $bindable(null),
|
||||||
|
children,
|
||||||
|
class: className,
|
||||||
|
...restProps
|
||||||
|
}: WithoutChild<FormPrimitive.LabelProps> = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormPrimitive.Label {...restProps} bind:ref>
|
||||||
|
{#snippet child({ props })}
|
||||||
|
<Label {...props} class={cn("data-[fs-error]:text-destructive", className)}>
|
||||||
|
{@render children?.()}
|
||||||
|
</Label>
|
||||||
|
{/snippet}
|
||||||
|
</FormPrimitive.Label>
|
17
packages/ui/src/components/ui/form/form-legend.svelte
Normal file
17
packages/ui/src/components/ui/form/form-legend.svelte
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import type { WithoutChild } from "bits-ui"
|
||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
|
||||||
|
let {
|
||||||
|
ref = $bindable(null),
|
||||||
|
class: className,
|
||||||
|
...restProps
|
||||||
|
}: WithoutChild<FormPrimitive.LegendProps> = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<FormPrimitive.Legend
|
||||||
|
bind:ref
|
||||||
|
{...restProps}
|
||||||
|
class={cn("data-[fs-error]:text-destructive text-sm font-medium leading-none", className)}
|
||||||
|
/>
|
33
packages/ui/src/components/ui/form/index.ts
Normal file
33
packages/ui/src/components/ui/form/index.ts
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
import * as FormPrimitive from "formsnap"
|
||||||
|
import Button from "./form-button.svelte"
|
||||||
|
import Description from "./form-description.svelte"
|
||||||
|
import ElementField from "./form-element-field.svelte"
|
||||||
|
import FieldErrors from "./form-field-errors.svelte"
|
||||||
|
import Field from "./form-field.svelte"
|
||||||
|
import Fieldset from "./form-fieldset.svelte"
|
||||||
|
import Label from "./form-label.svelte"
|
||||||
|
import Legend from "./form-legend.svelte"
|
||||||
|
|
||||||
|
const Control = FormPrimitive.Control as typeof FormPrimitive.Control
|
||||||
|
|
||||||
|
export {
|
||||||
|
Field,
|
||||||
|
Control,
|
||||||
|
Label,
|
||||||
|
FieldErrors,
|
||||||
|
Description,
|
||||||
|
Fieldset,
|
||||||
|
Legend,
|
||||||
|
ElementField,
|
||||||
|
Button,
|
||||||
|
//
|
||||||
|
Field as FormField,
|
||||||
|
Control as FormControl,
|
||||||
|
Description as FormDescription,
|
||||||
|
Label as FormLabel,
|
||||||
|
FieldErrors as FormFieldErrors,
|
||||||
|
Fieldset as FormFieldset,
|
||||||
|
Legend as FormLegend,
|
||||||
|
ElementField as FormElementField,
|
||||||
|
Button as FormButton
|
||||||
|
}
|
7
packages/ui/src/components/ui/label/index.ts
Normal file
7
packages/ui/src/components/ui/label/index.ts
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
import Root from "./label.svelte"
|
||||||
|
|
||||||
|
export {
|
||||||
|
Root,
|
||||||
|
//
|
||||||
|
Root as Label
|
||||||
|
}
|
15
packages/ui/src/components/ui/label/label.svelte
Normal file
15
packages/ui/src/components/ui/label/label.svelte
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
<script lang="ts">
|
||||||
|
import { cn } from "@kksh/ui/utils"
|
||||||
|
import { Label as LabelPrimitive } from "bits-ui"
|
||||||
|
|
||||||
|
let { ref = $bindable(null), class: className, ...restProps }: LabelPrimitive.RootProps = $props()
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<LabelPrimitive.Root
|
||||||
|
bind:ref
|
||||||
|
class={cn(
|
||||||
|
"text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70",
|
||||||
|
className
|
||||||
|
)}
|
||||||
|
{...restProps}
|
||||||
|
/>
|
@ -1,5 +1,5 @@
|
|||||||
export { default as Shiki } from "./components/code/shiki.svelte"
|
export { default as Shiki } from "./components/code/shiki.svelte"
|
||||||
export { IconMultiplexer } from "./components/common"
|
export * from "./components/common"
|
||||||
export * as Layouts from "./components/layouts/index"
|
export * as Layouts from "./components/layouts/index"
|
||||||
export * as Error from "./components/error/index"
|
export * as Error from "./components/error/index"
|
||||||
export * as Common from "./components/common/index"
|
export * as Common from "./components/common/index"
|
||||||
@ -9,3 +9,5 @@ export * as Extension from "./components/extension/index"
|
|||||||
export { default as GridAnimation } from "./components/animation/grid-animation.svelte"
|
export { default as GridAnimation } from "./components/animation/grid-animation.svelte"
|
||||||
export { default as ViewTransition } from "./components/transition/view-transition.svelte"
|
export { default as ViewTransition } from "./components/transition/view-transition.svelte"
|
||||||
export * as Constants from "./constants"
|
export * as Constants from "./constants"
|
||||||
|
export * as Form from "./components/ui/form"
|
||||||
|
export { default as ModeToggle } from "./components/theme/mode-toggle.svelte"
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
{
|
{
|
||||||
"extends": "../typescript-config/base.json"
|
"extends": "../typescript-config/base.json",
|
||||||
|
"include": ["src/**/*"]
|
||||||
}
|
}
|
||||||
|
77
pnpm-lock.yaml
generated
77
pnpm-lock.yaml
generated
@ -88,8 +88,8 @@ importers:
|
|||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:packages/api
|
version: link:packages/api
|
||||||
'@kksh/svelte5':
|
'@kksh/svelte5':
|
||||||
specifier: 0.1.2-beta.4
|
specifier: 0.1.2-beta.8
|
||||||
version: 0.1.2-beta.4(lucide-svelte@0.454.0(svelte@5.1.9))(svelte-sonner@0.3.28(svelte@5.1.9))(svelte@5.1.9)
|
version: 0.1.2-beta.8(lucide-svelte@0.454.0(svelte@5.1.9))(svelte-sonner@0.3.28(svelte@5.1.9))(svelte@5.1.9)
|
||||||
prettier:
|
prettier:
|
||||||
specifier: ^3.2.5
|
specifier: ^3.2.5
|
||||||
version: 3.3.3
|
version: 3.3.3
|
||||||
@ -434,6 +434,9 @@ importers:
|
|||||||
'@kksh/api':
|
'@kksh/api':
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../api
|
version: link:../api
|
||||||
|
'@supabase/supabase-js':
|
||||||
|
specifier: ^2.46.1
|
||||||
|
version: 2.46.1
|
||||||
typescript:
|
typescript:
|
||||||
specifier: ^5.0.0
|
specifier: ^5.0.0
|
||||||
version: 5.5.4
|
version: 5.5.4
|
||||||
@ -466,18 +469,27 @@ importers:
|
|||||||
specifier: ^3.12.5
|
specifier: ^3.12.5
|
||||||
version: 3.12.5
|
version: 3.12.5
|
||||||
devDependencies:
|
devDependencies:
|
||||||
|
'@iconify/svelte':
|
||||||
|
specifier: ^4.0.2
|
||||||
|
version: 4.0.2(svelte@5.1.9)
|
||||||
'@kksh/api':
|
'@kksh/api':
|
||||||
specifier: workspace:*
|
specifier: workspace:*
|
||||||
version: link:../api
|
version: link:../api
|
||||||
|
'@kksh/svelte5':
|
||||||
|
specifier: ^0.1.2-beta.8
|
||||||
|
version: 0.1.2-beta.8(lucide-svelte@0.454.0(svelte@5.1.9))(svelte-sonner@0.3.28(svelte@5.1.9))(svelte@5.1.9)
|
||||||
'@types/bun':
|
'@types/bun':
|
||||||
specifier: latest
|
specifier: latest
|
||||||
version: 1.1.13
|
version: 1.1.13
|
||||||
bits-ui:
|
bits-ui:
|
||||||
specifier: 1.0.0-next.36
|
specifier: 1.0.0-next.45
|
||||||
version: 1.0.0-next.36(svelte@5.1.9)
|
version: 1.0.0-next.45(svelte@5.1.9)
|
||||||
clsx:
|
clsx:
|
||||||
specifier: ^2.1.1
|
specifier: ^2.1.1
|
||||||
version: 2.1.1
|
version: 2.1.1
|
||||||
|
formsnap:
|
||||||
|
specifier: 2.0.0-next.1
|
||||||
|
version: 2.0.0-next.1(svelte@5.1.9)(sveltekit-superforms@2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3))
|
||||||
lucide-svelte:
|
lucide-svelte:
|
||||||
specifier: ^0.454.0
|
specifier: ^0.454.0
|
||||||
version: 0.454.0(svelte@5.1.9)
|
version: 0.454.0(svelte@5.1.9)
|
||||||
@ -496,6 +508,9 @@ importers:
|
|||||||
svelte-sonner:
|
svelte-sonner:
|
||||||
specifier: ^0.3.28
|
specifier: ^0.3.28
|
||||||
version: 0.3.28(svelte@5.1.9)
|
version: 0.3.28(svelte@5.1.9)
|
||||||
|
sveltekit-superforms:
|
||||||
|
specifier: ^2.20.0
|
||||||
|
version: 2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)
|
||||||
tailwind-merge:
|
tailwind-merge:
|
||||||
specifier: ^2.5.4
|
specifier: ^2.5.4
|
||||||
version: 2.5.4
|
version: 2.5.4
|
||||||
@ -508,6 +523,9 @@ importers:
|
|||||||
tailwindcss-animate:
|
tailwindcss-animate:
|
||||||
specifier: ^1.0.7
|
specifier: ^1.0.7
|
||||||
version: 1.0.7(tailwindcss@3.4.14)
|
version: 1.0.7(tailwindcss@3.4.14)
|
||||||
|
zod:
|
||||||
|
specifier: ^3.23.8
|
||||||
|
version: 3.23.8
|
||||||
|
|
||||||
packages/utils:
|
packages/utils:
|
||||||
dependencies:
|
dependencies:
|
||||||
@ -1279,12 +1297,12 @@ packages:
|
|||||||
'@jsr/std__semver@1.0.3':
|
'@jsr/std__semver@1.0.3':
|
||||||
resolution: {integrity: sha512-d1uBT0Muxhd3yBIw9ZE1Q/4N1Y0td0EJe1AqwM3hP05IMwaWQV/miksQOPR3rup3bVovuIvqBm7WJcoUripdQA==, tarball: https://npm.jsr.io/~/11/@jsr/std__semver/1.0.3.tgz}
|
resolution: {integrity: sha512-d1uBT0Muxhd3yBIw9ZE1Q/4N1Y0td0EJe1AqwM3hP05IMwaWQV/miksQOPR3rup3bVovuIvqBm7WJcoUripdQA==, tarball: https://npm.jsr.io/~/11/@jsr/std__semver/1.0.3.tgz}
|
||||||
|
|
||||||
'@kksh/svelte5@0.1.2-beta.4':
|
'@kksh/svelte5@0.1.2-beta.8':
|
||||||
resolution: {integrity: sha512-QUA3wl4aOUcnfTo37/l37fWlhwf0lbtnNugDiwBJgrjaLeNF6nEzlYFKxxsA7erR6hY6VZCeVlf01S3ZgPva4w==}
|
resolution: {integrity: sha512-cXReqYbZ/KXICucNREz6T/1DguTlGjvkq2qlUriYH/lgfgsUjbq8ZPhVsCIb0qlDT9IYIu5Pc/8X+azGqFirRA==}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
lucide-svelte: ^0.416.0
|
lucide-svelte: ^0.454.0
|
||||||
svelte: ^5.0.0
|
svelte: ^5.1.10
|
||||||
svelte-sonner: ^0.3.27
|
svelte-sonner: ^0.3.28
|
||||||
|
|
||||||
'@manypkg/find-root@1.1.0':
|
'@manypkg/find-root@1.1.0':
|
||||||
resolution: {integrity: sha512-mki5uBvhHzO8kYYix/WRy2WX8S3B5wdVSc9D6KcU5lQNglP2yt58/VfLuAK49glRXChosY8ap2oJ1qgma3GUVA==}
|
resolution: {integrity: sha512-mki5uBvhHzO8kYYix/WRy2WX8S3B5wdVSc9D6KcU5lQNglP2yt58/VfLuAK49glRXChosY8ap2oJ1qgma3GUVA==}
|
||||||
@ -2138,6 +2156,12 @@ packages:
|
|||||||
peerDependencies:
|
peerDependencies:
|
||||||
svelte: ^5.0.0-next.1
|
svelte: ^5.0.0-next.1
|
||||||
|
|
||||||
|
bits-ui@1.0.0-next.45:
|
||||||
|
resolution: {integrity: sha512-kt7gYIirEo2Rg1hMudcGEzSHogQTA22d/j1x8v+wIshsIqqcCN6DXJZpTojSCQWxny8IEa9CRnLwAzY4B2qf1Q==}
|
||||||
|
engines: {node: '>=18', pnpm: '>=8.7.0'}
|
||||||
|
peerDependencies:
|
||||||
|
svelte: ^5.0.0-next.1
|
||||||
|
|
||||||
bl@4.1.0:
|
bl@4.1.0:
|
||||||
resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==}
|
resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==}
|
||||||
|
|
||||||
@ -2647,6 +2671,13 @@ packages:
|
|||||||
svelte: ^4.0.0 || ^5.0.0-next.1
|
svelte: ^4.0.0 || ^5.0.0-next.1
|
||||||
sveltekit-superforms: ^2.3.0
|
sveltekit-superforms: ^2.3.0
|
||||||
|
|
||||||
|
formsnap@2.0.0-next.1:
|
||||||
|
resolution: {integrity: sha512-ha8r9eMmsGEGMY+ljV3FEyTtB72E7dt95y9HHUbCcaDnjbz3Q6n00BHLz7dfBZ9rqyaMeIO200EmP1IcYMExeg==}
|
||||||
|
engines: {node: '>=18', pnpm: '>=8.7.0'}
|
||||||
|
peerDependencies:
|
||||||
|
svelte: ^5.0.0
|
||||||
|
sveltekit-superforms: ^2.19.0
|
||||||
|
|
||||||
fraction.js@4.3.7:
|
fraction.js@4.3.7:
|
||||||
resolution: {integrity: sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew==}
|
resolution: {integrity: sha512-ZsDfxO51wGAXREY55a7la9LScWpwv9RxIrYABrlvOFBlH/ShPnrtsXeuUIfXKKOVicNxQ+o8JTbJvjS4M89yew==}
|
||||||
|
|
||||||
@ -3760,11 +3791,11 @@ packages:
|
|||||||
svelte:
|
svelte:
|
||||||
optional: true
|
optional: true
|
||||||
|
|
||||||
svelte-persisted-store@0.11.0:
|
svelte-persisted-store@0.12.0:
|
||||||
resolution: {integrity: sha512-9RgJ5DrawGyyfK22A80cfu8Jose3CV8YjEZKz9Tn94rQ0tWyEmYr+XI+wrVF6wjRbW99JMDSVcFRiM3XzVJj/w==}
|
resolution: {integrity: sha512-BdBQr2SGSJ+rDWH8/aEV5GthBJDapVP0GP3fuUCA7TjYG5ctcB+O9Mj9ZC0+Jo1oJMfZUd1y9H68NFRR5MyIJA==}
|
||||||
engines: {node: '>=0.14'}
|
engines: {node: '>=0.14'}
|
||||||
peerDependencies:
|
peerDependencies:
|
||||||
svelte: ^3.48.0 || ^4.0.0 || ^5.0.0-next.0
|
svelte: ^3.48.0 || ^4 || ^5
|
||||||
|
|
||||||
svelte-radix@2.0.1:
|
svelte-radix@2.0.1:
|
||||||
resolution: {integrity: sha512-YrX44Dj+Rp6YZuPSjdmyd6P8QTkb2NXwySUCZYzjwkP6Cl3dZaTBPPeaSOutP3v3ycQ2XwyNOpyn4p0QcN+uYQ==}
|
resolution: {integrity: sha512-YrX44Dj+Rp6YZuPSjdmyd6P8QTkb2NXwySUCZYzjwkP6Cl3dZaTBPPeaSOutP3v3ycQ2XwyNOpyn4p0QcN+uYQ==}
|
||||||
@ -5274,11 +5305,11 @@ snapshots:
|
|||||||
|
|
||||||
'@jsr/std__semver@1.0.3': {}
|
'@jsr/std__semver@1.0.3': {}
|
||||||
|
|
||||||
'@kksh/svelte5@0.1.2-beta.4(lucide-svelte@0.454.0(svelte@5.1.9))(svelte-sonner@0.3.28(svelte@5.1.9))(svelte@5.1.9)':
|
'@kksh/svelte5@0.1.2-beta.8(lucide-svelte@0.454.0(svelte@5.1.9))(svelte-sonner@0.3.28(svelte@5.1.9))(svelte@5.1.9)':
|
||||||
dependencies:
|
dependencies:
|
||||||
lucide-svelte: 0.454.0(svelte@5.1.9)
|
lucide-svelte: 0.454.0(svelte@5.1.9)
|
||||||
svelte: 5.1.9
|
svelte: 5.1.9
|
||||||
svelte-persisted-store: 0.11.0(svelte@5.1.9)
|
svelte-persisted-store: 0.12.0(svelte@5.1.9)
|
||||||
svelte-sonner: 0.3.28(svelte@5.1.9)
|
svelte-sonner: 0.3.28(svelte@5.1.9)
|
||||||
|
|
||||||
'@manypkg/find-root@1.1.0':
|
'@manypkg/find-root@1.1.0':
|
||||||
@ -6310,6 +6341,16 @@ snapshots:
|
|||||||
svelte: 5.1.9
|
svelte: 5.1.9
|
||||||
svelte-toolbelt: 0.4.6(svelte@5.1.9)
|
svelte-toolbelt: 0.4.6(svelte@5.1.9)
|
||||||
|
|
||||||
|
bits-ui@1.0.0-next.45(svelte@5.1.9):
|
||||||
|
dependencies:
|
||||||
|
'@floating-ui/core': 1.6.8
|
||||||
|
'@floating-ui/dom': 1.6.12
|
||||||
|
'@internationalized/date': 3.5.6
|
||||||
|
esm-env: 1.1.4
|
||||||
|
runed: 0.15.3(svelte@5.1.9)
|
||||||
|
svelte: 5.1.9
|
||||||
|
svelte-toolbelt: 0.4.6(svelte@5.1.9)
|
||||||
|
|
||||||
bl@4.1.0:
|
bl@4.1.0:
|
||||||
dependencies:
|
dependencies:
|
||||||
buffer: 5.7.1
|
buffer: 5.7.1
|
||||||
@ -6897,6 +6938,12 @@ snapshots:
|
|||||||
svelte: 5.1.9
|
svelte: 5.1.9
|
||||||
sveltekit-superforms: 2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)
|
sveltekit-superforms: 2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)
|
||||||
|
|
||||||
|
formsnap@2.0.0-next.1(svelte@5.1.9)(sveltekit-superforms@2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)):
|
||||||
|
dependencies:
|
||||||
|
svelte: 5.1.9
|
||||||
|
svelte-toolbelt: 0.4.6(svelte@5.1.9)
|
||||||
|
sveltekit-superforms: 2.20.0(@sveltejs/kit@2.7.4(@sveltejs/vite-plugin-svelte@4.0.0(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(svelte@5.1.9)(vite@5.4.10(@types/node@22.8.7)(terser@5.36.0)))(@types/json-schema@7.0.15)(svelte@5.1.9)(typescript@5.6.3)
|
||||||
|
|
||||||
fraction.js@4.3.7: {}
|
fraction.js@4.3.7: {}
|
||||||
|
|
||||||
fs-extra@11.2.0:
|
fs-extra@11.2.0:
|
||||||
@ -7918,7 +7965,7 @@ snapshots:
|
|||||||
optionalDependencies:
|
optionalDependencies:
|
||||||
svelte: 5.1.9
|
svelte: 5.1.9
|
||||||
|
|
||||||
svelte-persisted-store@0.11.0(svelte@5.1.9):
|
svelte-persisted-store@0.12.0(svelte@5.1.9):
|
||||||
dependencies:
|
dependencies:
|
||||||
svelte: 5.1.9
|
svelte: 5.1.9
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user