mirror of
https://github.com/kunkunsh/kunkun.git
synced 2025-04-04 14:46:42 +00:00

* feat: implement a file streamer for file share Only server with hardcoded file path * bump valibot version * feat: add server-info gRPC module to serve server info * feat: add ssl cert and public key to peers state When peer is online, KK always have know its cert and pub key for future secure communication * feat: add grpc ts package * Enable "/refresh-worker-extension" rest API, grpc version isn't ready yet * update pnpm lock * ci: fix CI by moving protobuf install order * ci: fix * upgrade api due to valibot incompatibility * fix: use fs instead of bun shell to be compatible with windows * skip grpc pkg build on windows * feat: local network file transfer prototype working * fix: grpc build.ts * download next to 14 * ci: add ci env try to fix next * fix: hideRefreshBtn and a few other btns' hide API in iframe ext page * feat: disable NODE_TLS_REJECT_UNAUTHORIZED for extension HMR refresh * fix: manifest json schema with objectWithRest to allow any other fields in package.json * chore: update valibot and related dependencies to version 1.0.0-beta.9 in pnpm-lock.yaml * ci: add protobuf compiler installation to manifest-schema-upload workflow * refactor: move grpc code from jarvis to a separate grpc crate for easier testing * feat(file-transfer): POC multi file + directory file transfer * feat(file-transfer): replace file transfer recursive download in ts with rust * feat(file-transfer): implement on_progress event for file transfer * feat(file-transfer): report progress every 1MB instead of 100 iterations * feat(file-transfer): add progress bar * feat(file-transfer): UI * feat(file-transfer): add file transfer bucket info preview Show total size and number of files * feat(file-transfer): improve UX Show bucket info during confirm; improve progress bar UI, prevent inconsistent width * feat(grpc): skip build in Cloudflare Pages due to missing protoc * refactor: with cargo fix, unused imports removed * ci: debug cloudflare pages env var * fix(grpc): update environment variable access for Cloudflare Pages build check * fix(grpc): add error handling for protoc command in build script * chore: update kkrpc version to 0.0.13, remove kkrpc submodule, and enhance grpc build script logging - Updated kkrpc dependency version from 0.0.12 to 0.0.13 in package.json. - Removed the kkrpc submodule from the project. - Enhanced logging in the grpc build script to include additional Cloudflare Pages environment variables for better debugging. * fix(api): typescript error, remove base.json from tsconfig * chore: update pnpm lock * fix(api): update TypeScript configuration to extend base.json and clean up unused options * refactor(api): update TypeScript configuration to extend path-alias.json and enhance compiler options * fix(api): restore KunkunManifestPermission in PermissionUnion and update valibot import in schema tests * fix: missing trait error * fix: js require replaced with import * test: fix a unit test with a more robust method --------- Co-authored-by: Huakun Shen <huaukun.shen@huakunshen.com>
110 lines
3.4 KiB
Svelte
110 lines
3.4 KiB
Svelte
<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 { 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 } 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.url(), 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 ?? false
|
|
})
|
|
</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} />
|
|
<Form.Button class="my-1">Submit</Form.Button>
|
|
</form>
|
|
</div>
|
|
{#if dev}
|
|
<div class="p-5">
|
|
<SuperDebug data={$formData} />
|
|
</div>
|
|
{/if}
|