Files
claude-code-rev/src/utils/computerUse/inputLoader.ts
2026-03-31 19:56:09 +08:00

46 lines
1.4 KiB
TypeScript

import type {
ComputerUseInput,
ComputerUseInputAPI,
} from '@ant/computer-use-input'
let cached: ComputerUseInputAPI | undefined
function unwrapDefaultExport<T>(mod: T | { default: T }): T {
return (
typeof mod === 'object' &&
mod !== null &&
'default' in mod &&
mod.default !== undefined
? mod.default
: mod
) as T
}
/**
* Package's js/index.js reads COMPUTER_USE_INPUT_NODE_PATH (baked by
* build-with-plugins.ts on darwin targets, unset otherwise — falls through to
* the node_modules prebuilds/ path).
*
* The package exports a discriminated union on `isSupported` — narrowed here
* once so callers get the bare `ComputerUseInputAPI` without re-checking.
*
* key()/keys() dispatch enigo work onto DispatchQueue.main via
* dispatch2::run_on_main, then block a tokio worker on a channel. Under
* Electron (CFRunLoop drains the main queue) this works; under libuv
* (Node/bun) the main queue never drains and the promise hangs. The executor
* calls these inside drainRunLoop().
*/
export function requireComputerUseInput(): ComputerUseInputAPI {
if (cached) return cached
// eslint-disable-next-line @typescript-eslint/no-require-imports
const input = unwrapDefaultExport(
require('@ant/computer-use-input') as ComputerUseInput | {
default: ComputerUseInput
},
)
if (!input.isSupported) {
throw new Error('@ant/computer-use-input is not supported on this platform')
}
return (cached = input)
}