支持在会话输入框直接进行文件/图片的拖拽和粘贴
This commit is contained in:
@@ -1,5 +1,6 @@
|
||||
import type {
|
||||
AdaptedContentPart,
|
||||
UserImageDisplay,
|
||||
UserResourceDisplay,
|
||||
} from "@/lib/adapters/ai-elements-adapter"
|
||||
import type { PromptDraft, PromptInputBlock } from "@/lib/types"
|
||||
@@ -10,6 +11,35 @@ function isResourceLinkBlock(
|
||||
return block.type === "resource_link"
|
||||
}
|
||||
|
||||
function isEmbeddedResourceBlock(
|
||||
block: PromptInputBlock
|
||||
): block is Extract<PromptInputBlock, { type: "resource" }> {
|
||||
return block.type === "resource"
|
||||
}
|
||||
|
||||
function isImageBlock(
|
||||
block: PromptInputBlock
|
||||
): block is Extract<PromptInputBlock, { type: "image" }> {
|
||||
return block.type === "image"
|
||||
}
|
||||
|
||||
function deriveResourceNameFromUri(uri: string): string {
|
||||
const fallback = "resource"
|
||||
const normalized = uri.trim()
|
||||
if (!normalized) return fallback
|
||||
const withoutQuery = normalized.split(/[?#]/, 1)[0]
|
||||
const candidate = withoutQuery.split(/[\\/]/).pop() ?? ""
|
||||
let decoded = ""
|
||||
if (candidate) {
|
||||
try {
|
||||
decoded = decodeURIComponent(candidate)
|
||||
} catch {
|
||||
decoded = candidate
|
||||
}
|
||||
}
|
||||
return decoded || fallback
|
||||
}
|
||||
|
||||
export function getPromptDraftDisplayText(
|
||||
draft: PromptDraft,
|
||||
attachedResourcesFallback: string
|
||||
@@ -33,9 +63,40 @@ export function buildUserMessageTextPartsFromDraft(
|
||||
export function extractUserResourcesFromDraft(
|
||||
draft: PromptDraft
|
||||
): UserResourceDisplay[] {
|
||||
return draft.blocks.filter(isResourceLinkBlock).map((resource) => ({
|
||||
const linked = draft.blocks.filter(isResourceLinkBlock).map((resource) => ({
|
||||
name: resource.name,
|
||||
uri: resource.uri,
|
||||
mime_type: resource.mime_type ?? null,
|
||||
}))
|
||||
const embedded = draft.blocks
|
||||
.filter(isEmbeddedResourceBlock)
|
||||
.map((resource) => ({
|
||||
name: deriveResourceNameFromUri(resource.uri),
|
||||
uri: resource.uri,
|
||||
mime_type: resource.mime_type ?? null,
|
||||
}))
|
||||
return [...linked, ...embedded]
|
||||
}
|
||||
|
||||
function deriveImageName(
|
||||
uri: string | null | undefined,
|
||||
mimeType: string
|
||||
): string {
|
||||
if (uri && uri.trim().length > 0) {
|
||||
const name = deriveResourceNameFromUri(uri)
|
||||
if (name !== "resource") return name
|
||||
}
|
||||
const ext = mimeType.split("/")[1]?.split("+")[0] ?? "image"
|
||||
return `image.${ext}`
|
||||
}
|
||||
|
||||
export function extractUserImagesFromDraft(
|
||||
draft: PromptDraft
|
||||
): UserImageDisplay[] {
|
||||
return draft.blocks.filter(isImageBlock).map((image) => ({
|
||||
name: deriveImageName(image.uri, image.mime_type),
|
||||
data: image.data,
|
||||
mime_type: image.mime_type,
|
||||
uri: image.uri ?? null,
|
||||
}))
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user