- modifiers-napi: 使用 Bun FFI 调用 macOS CGEventSourceFlagsState 检测修饰键 - image-processor-napi: 集成 sharp 库,macOS 剪贴板图像读取 (osascript) - audio-capture-napi: 基于 SoX/arecord 的跨平台音频录制 - url-handler-napi: 完善函数签名(保持 null fallback) - 修复 image-processor 类型兼容性问题 Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
95 lines
2.8 KiB
TypeScript
95 lines
2.8 KiB
TypeScript
import type { Buffer } from 'buffer'
|
|
import { isInBundledMode } from '../../utils/bundledMode.js'
|
|
|
|
export type SharpInstance = {
|
|
metadata(): Promise<{ width: number; height: number; format: string }>
|
|
resize(
|
|
width: number,
|
|
height: number,
|
|
options?: { fit?: string; withoutEnlargement?: boolean },
|
|
): SharpInstance
|
|
jpeg(options?: { quality?: number }): SharpInstance
|
|
png(options?: {
|
|
compressionLevel?: number
|
|
palette?: boolean
|
|
colors?: number
|
|
}): SharpInstance
|
|
webp(options?: { quality?: number }): SharpInstance
|
|
toBuffer(): Promise<Buffer>
|
|
}
|
|
|
|
export type SharpFunction = (input: Buffer) => SharpInstance
|
|
|
|
type SharpCreatorOptions = {
|
|
create: {
|
|
width: number
|
|
height: number
|
|
channels: 3 | 4
|
|
background: { r: number; g: number; b: number }
|
|
}
|
|
}
|
|
|
|
type SharpCreator = (options: SharpCreatorOptions) => SharpInstance
|
|
|
|
let imageProcessorModule: { default: SharpFunction } | null = null
|
|
let imageCreatorModule: { default: SharpCreator } | null = null
|
|
|
|
export async function getImageProcessor(): Promise<SharpFunction> {
|
|
if (imageProcessorModule) {
|
|
return imageProcessorModule.default
|
|
}
|
|
|
|
if (isInBundledMode()) {
|
|
// Try to load the native image processor first
|
|
try {
|
|
// Use the native image processor module
|
|
const imageProcessor = await import('image-processor-napi')
|
|
const sharpFn = (imageProcessor.sharp ?? imageProcessor.default) as SharpFunction
|
|
imageProcessorModule = { default: sharpFn }
|
|
return sharpFn
|
|
} catch {
|
|
// Fall back to sharp if native module is not available
|
|
// biome-ignore lint/suspicious/noConsole: intentional warning
|
|
console.warn(
|
|
'Native image processor not available, falling back to sharp',
|
|
)
|
|
}
|
|
}
|
|
|
|
// Use sharp for non-bundled builds or as fallback.
|
|
// Single structural cast: our SharpFunction is a subset of sharp's actual type surface.
|
|
const imported = (await import(
|
|
'sharp'
|
|
)) as unknown as MaybeDefault<SharpFunction>
|
|
const sharp = unwrapDefault(imported)
|
|
imageProcessorModule = { default: sharp }
|
|
return sharp
|
|
}
|
|
|
|
/**
|
|
* Get image creator for generating new images from scratch.
|
|
* Note: image-processor-napi doesn't support image creation,
|
|
* so this always uses sharp directly.
|
|
*/
|
|
export async function getImageCreator(): Promise<SharpCreator> {
|
|
if (imageCreatorModule) {
|
|
return imageCreatorModule.default
|
|
}
|
|
|
|
const imported = (await import(
|
|
'sharp'
|
|
)) as unknown as MaybeDefault<SharpCreator>
|
|
const sharp = unwrapDefault(imported)
|
|
imageCreatorModule = { default: sharp }
|
|
return sharp
|
|
}
|
|
|
|
// Dynamic import shape varies by module interop mode — ESM yields { default: fn }, CJS yields fn directly.
|
|
type MaybeDefault<T> = T | { default: T }
|
|
|
|
function unwrapDefault<T extends (...args: never[]) => unknown>(
|
|
mod: MaybeDefault<T>,
|
|
): T {
|
|
return typeof mod === 'function' ? mod : mod.default
|
|
}
|