* feat: add publisher link to extension detail

* fix: improve IconMultiplexer and StoreExtDetail component rendering

* feat: add published date to extension details view

* chore: add moment.js and clean up imports in StoreExtDetail

* fix: support cloudflare worker

Otherwise cloudflare worker gets html instead of json

* refactor: move AppsCmds component to desktop app

* bump: version to 0.1.1

* fix: package.json fetching cors error

* fix: improve files field validation in verify command
This commit is contained in:
Huakun Shen 2025-02-07 01:26:56 -05:00 committed by GitHub
parent e49c0f5da5
commit 490368428e
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
14 changed files with 53 additions and 18 deletions

View File

@ -1,7 +1,7 @@
{
"name": "kksh",
"module": "dist/cli.js",
"version": "0.0.32",
"version": "0.1.0",
"type": "module",
"bin": {
"kksh": "./dist/cli.js",

View File

@ -52,7 +52,8 @@ export function verifySingleProject(projectPath: string): boolean {
logger.info(`name`, pkg.name)
logger.info(`version`, pkg.version)
logger.info(`identifier`, pkg.kunkun.identifier)
if (pkg.files?.length ?? 0 === 0) {
if ((pkg.files?.length ?? 0) === 0) {
logger.warn(
`"files" field is empty, it is recommended to include only the necessary files, e.g. dist`
)

View File

@ -1,16 +1,15 @@
<script lang="ts">
import { IconEnum, type AppInfo } from "@kksh/api/models"
import { Command } from "@kksh/svelte5"
import { IconMultiplexer } from "@kksh/ui"
import { DraggableCommandGroup } from "@kksh/ui/custom"
import { convertFileSrc } from "@tauri-apps/api/core"
import * as os from "@tauri-apps/plugin-os"
import { toast } from "svelte-sonner"
import { executeBashScript, open } from "tauri-plugin-shellx-api"
import IconMultiplexer from "../common/IconMultiplexer.svelte"
import { DraggableCommandGroup } from "../custom"
import { open } from "tauri-plugin-shellx-api"
const platform = os.platform()
let { apps }: { apps: AppInfo[] } = $props()
// let appsDisplay = $derived(apps.length > 20 ? apps.slice(0, 20) : apps)
</script>
<DraggableCommandGroup heading="Apps">

View File

@ -3,6 +3,7 @@
import { commandLaunchers } from "@/cmds"
import { builtinCmds } from "@/cmds/builtin"
import { systemCommands } from "@/cmds/system"
import AppsCmds from "@/components/main/AppsCmds.svelte"
import { i18n } from "@/i18n"
import * as m from "@/paraglide/messages"
import {
@ -21,10 +22,9 @@
import { cmdQueries } from "@/stores/cmdQuery"
import { isKeyboardEventFromInputElement } from "@/utils/dom"
import Icon from "@iconify/svelte"
import { db, toggleDevTools } from "@kksh/api/commands"
import { toggleDevTools } from "@kksh/api/commands"
import { Button, Command, DropdownMenu } from "@kksh/svelte5"
import {
AppsCmds,
BuiltinCmds,
CustomCommandInput,
ExtCmdsGroup,

View File

@ -21,7 +21,7 @@ export const breakingChangesVersionCheckpoints = [
const checkpointVersions = breakingChangesVersionCheckpoints.map((c) => c.version)
const sortedCheckpointVersions = sort(checkpointVersions)
export const version = "0.1.0"
export const version = "0.1.1"
export function isVersionBetween(v: string, start: string, end: string) {
const vCleaned = clean(v)

View File

@ -119,9 +119,7 @@ export function getJsrPackageSrcFile(
file: string
): Promise<string | undefined> {
const url = `https://jsr.io/@${scope}/${name}/${version}/${file}`
return fetch(url)
.then((res) => res.text())
.catch(() => undefined)
return fetch(url).then((res) => res.text())
}
/**

View File

@ -1,3 +1,4 @@
## Permission Table
<table>
@ -6,6 +7,7 @@
<th>Description</th>
</tr>
<tr>
<td>

View File

@ -73,6 +73,7 @@
"@std/semver": "npm:@jsr/std__semver@^1.0.3",
"dompurify": "^3.2.3",
"gsap": "^3.12.7",
"moment": "^2.30.1",
"pretty-bytes": "^6.1.1",
"shiki-magic-move": "^0.5.2",
"svelte-markdown": "^0.4.1",

View File

@ -40,6 +40,7 @@
onMount(() => {
if (icon.type === IconEnum.Svg) {
cleanedSvg = DOMPurify.sanitize(icon.value)
cleanedSvg = cleanedSvg.replace("<svg", `<svg class="${className}"`)
}
})
</script>
@ -96,7 +97,7 @@
{:else if icon.type === IconEnum.Svg}
<span
{...restProps}
class={cn(className, { invert: icon.invert, "dark:invert": icon.darkInvert })}
class={cn({ invert: icon.invert, "dark:invert": icon.darkInvert }, className)}
{style}
>
<!-- eslint-disable svelte/no-at-html-tags -->

View File

@ -8,8 +8,10 @@
import { Constants, IconMultiplexer } from "@kksh/ui"
import { cn } from "@kksh/ui/utils"
import { CircleCheckBigIcon, MoveRightIcon, Trash2Icon } from "lucide-svelte"
import moment from "moment"
import prettyBytes from "pretty-bytes"
import * as v from "valibot"
import { isInTauri } from "../../utils/tauri"
import DialogImageCarousel from "../common/DialogImageCarousel.svelte"
import PlatformsIcons from "../common/PlatformsIcons.svelte"
import TauriLink from "../common/TauriLink.svelte"
@ -20,6 +22,7 @@
let {
extPublish,
ext,
author,
installedExt,
manifest,
demoImages,
@ -35,6 +38,10 @@
}: {
extPublish: Tables<"ext_publish">
ext: Tables<"extensions">
author?: {
id: string
name: string
} | null
installedExt?: ExtPackageJson
packageJson: ExtPackageJson | null
manifest: KunkunExtManifest
@ -150,7 +157,7 @@
<ScrollArea class={cn("w-full pb-12", className)}>
<div class="flex flex-col items-center justify-between gap-4 sm:flex-row">
<div class="flex items-center gap-4">
<span class="h-12 w-12">
<span class="h-20 w-20">
<IconMultiplexer
icon={manifest.icon}
class={cn(Constants.CLASSNAMES.EXT_LOGO, "h-full w-full")}
@ -167,9 +174,12 @@
<pre class="text-muted-foreground text-xs">{extPublish.identifier}</pre>
<pre class="text-muted-foreground text-xs">Version: {extPublish.version}</pre>
<pre class="text-muted-foreground text-xs">Downloads: {ext.downloads}</pre>
<pre class="text-muted-foreground text-xs">
Size: {prettyBytes(extPublish.tarball_size)}
</pre>
<pre class="text-muted-foreground text-xs">Size: {prettyBytes(
extPublish.tarball_size
)}</pre>
<pre class="text-muted-foreground text-xs">Published At: {moment(
new Date(extPublish.created_at)
).format("YYYY-MM-DD HH:mm")}</pre>
</div>
</div>
<div class="flex items-center space-x-2">
@ -266,6 +276,18 @@
</ul>
</div>
<div>
<h2 class="text-lg font-bold">Publisher Profile</h2>
{#if !isInTauri && author}
<ul class="list-disc pl-5">
<li>
<a class="text-blue-500 hover:text-blue-600" href={`/user/${author.id}`}
>{author.name}</a
>
</li>
</ul>
{:else}
<TauriLink href={`https://kunkun.sh/user/${ext.author_id}`}>{ext.author_id}</TauriLink>
{/if}
<h2 class="text-lg font-bold">Author</h2>
{#if packageJson?.author}
<ul class="list-disc pl-5">

View File

@ -4,5 +4,4 @@ export { default as GlobalCommandPaletteFooter } from "./GlobalCommandPaletteFoo
export { default as ExtCmdsGroup } from "./ExtCmdsGroup.svelte"
export { default as SystemCmds } from "./SystemCmds.svelte"
export { default as QuickLinks } from "./QuickLinks.svelte"
export { default as AppsCmds } from "./AppsCmds.svelte"
export * from "./types"

View File

@ -0,0 +1,4 @@
import { browser } from "$app/environment"
// @ts-expect-error window.__TAURI_INTERNALS__ is not defined in the browser
export const isInTauri = browser ? !!window.__TAURI_INTERNALS__ : false

8
pnpm-lock.yaml generated
View File

@ -1198,6 +1198,9 @@ importers:
gsap:
specifier: ^3.12.7
version: 3.12.7
moment:
specifier: ^2.30.1
version: 2.30.1
pretty-bytes:
specifier: ^6.1.1
version: 6.1.1
@ -8829,6 +8832,9 @@ packages:
engines: {node: '>=18'}
hasBin: true
moment@2.30.1:
resolution: {integrity: sha512-uEmtNhbDOrWPFS+hdjFCBfy9f2YoyzRpwcl+DqpC6taX21FzsTLQVbMV/W7PzNSX6x/bhC1zA3c2UQ5NzH6how==}
mri@1.2.0:
resolution: {integrity: sha512-tzzskb3bG8LvYGFF/mDTpq3jpI6Q9wc3LEmBaghu+DdCssd1FakN7Bc0hVNmEyGq1bq3RgfkCb3cmQLpNPOroA==}
engines: {node: '>=4'}
@ -21098,6 +21104,8 @@ snapshots:
requirejs: 2.3.7
requirejs-config-file: 4.0.0
moment@2.30.1: {}
mri@1.2.0: {}
mrmime@2.0.0: {}