pi-ui-extend 0.1.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +307 -0
- package/bin/pix.mjs +219 -0
- package/dist/app/app.d.ts +96 -0
- package/dist/app/app.js +871 -0
- package/dist/app/blink-controller.d.ts +23 -0
- package/dist/app/blink-controller.js +82 -0
- package/dist/app/cli.d.ts +8 -0
- package/dist/app/cli.js +83 -0
- package/dist/app/clipboard.d.ts +1 -0
- package/dist/app/clipboard.js +24 -0
- package/dist/app/command-controller.d.ts +18 -0
- package/dist/app/command-controller.js +58 -0
- package/dist/app/command-host.d.ts +44 -0
- package/dist/app/command-host.js +1 -0
- package/dist/app/command-model-actions.d.ts +12 -0
- package/dist/app/command-model-actions.js +176 -0
- package/dist/app/command-navigation-actions.d.ts +19 -0
- package/dist/app/command-navigation-actions.js +267 -0
- package/dist/app/command-registry.d.ts +32 -0
- package/dist/app/command-registry.js +225 -0
- package/dist/app/command-runtime.d.ts +5 -0
- package/dist/app/command-runtime.js +32 -0
- package/dist/app/command-session-actions.d.ts +20 -0
- package/dist/app/command-session-actions.js +295 -0
- package/dist/app/constants.d.ts +52 -0
- package/dist/app/constants.js +103 -0
- package/dist/app/conversation-entry-renderer.d.ts +21 -0
- package/dist/app/conversation-entry-renderer.js +81 -0
- package/dist/app/conversation-shell-renderer.d.ts +5 -0
- package/dist/app/conversation-shell-renderer.js +43 -0
- package/dist/app/conversation-tool-renderer.d.ts +16 -0
- package/dist/app/conversation-tool-renderer.js +216 -0
- package/dist/app/conversation-viewport.d.ts +55 -0
- package/dist/app/conversation-viewport.js +252 -0
- package/dist/app/dcp-stats.d.ts +2 -0
- package/dist/app/dcp-stats.js +116 -0
- package/dist/app/editor-layout-renderer.d.ts +31 -0
- package/dist/app/editor-layout-renderer.js +211 -0
- package/dist/app/editor-panels.d.ts +4 -0
- package/dist/app/editor-panels.js +130 -0
- package/dist/app/extension-actions-controller.d.ts +22 -0
- package/dist/app/extension-actions-controller.js +60 -0
- package/dist/app/extension-event-bus.d.ts +3 -0
- package/dist/app/extension-event-bus.js +23 -0
- package/dist/app/extension-ui-controller.d.ts +57 -0
- package/dist/app/extension-ui-controller.js +532 -0
- package/dist/app/file-link-opener.d.ts +2 -0
- package/dist/app/file-link-opener.js +66 -0
- package/dist/app/file-links.d.ts +10 -0
- package/dist/app/file-links.js +117 -0
- package/dist/app/guards.d.ts +3 -0
- package/dist/app/guards.js +9 -0
- package/dist/app/icons.d.ts +37 -0
- package/dist/app/icons.js +97 -0
- package/dist/app/id.d.ts +1 -0
- package/dist/app/id.js +4 -0
- package/dist/app/image-click-targets.d.ts +5 -0
- package/dist/app/image-click-targets.js +32 -0
- package/dist/app/image-opener.d.ts +2 -0
- package/dist/app/image-opener.js +64 -0
- package/dist/app/input-action-controller.d.ts +47 -0
- package/dist/app/input-action-controller.js +209 -0
- package/dist/app/input-controller.d.ts +60 -0
- package/dist/app/input-controller.js +425 -0
- package/dist/app/input-paste-handler.d.ts +27 -0
- package/dist/app/input-paste-handler.js +146 -0
- package/dist/app/menu-items-controller.d.ts +32 -0
- package/dist/app/menu-items-controller.js +182 -0
- package/dist/app/message-content.d.ts +8 -0
- package/dist/app/message-content.js +115 -0
- package/dist/app/model-ref.d.ts +13 -0
- package/dist/app/model-ref.js +50 -0
- package/dist/app/model-usage-controller.d.ts +35 -0
- package/dist/app/model-usage-controller.js +99 -0
- package/dist/app/model-usage-status.d.ts +125 -0
- package/dist/app/model-usage-status.js +749 -0
- package/dist/app/mouse-controller.d.ts +182 -0
- package/dist/app/mouse-controller.js +968 -0
- package/dist/app/native-modifiers.d.ts +3 -0
- package/dist/app/native-modifiers.js +60 -0
- package/dist/app/nerd-font-controller.d.ts +11 -0
- package/dist/app/nerd-font-controller.js +90 -0
- package/dist/app/popup-action-controller.d.ts +44 -0
- package/dist/app/popup-action-controller.js +278 -0
- package/dist/app/popup-menu-controller.d.ts +183 -0
- package/dist/app/popup-menu-controller.js +1070 -0
- package/dist/app/prompt-enhancer-controller.d.ts +40 -0
- package/dist/app/prompt-enhancer-controller.js +215 -0
- package/dist/app/queued-message-controller.d.ts +62 -0
- package/dist/app/queued-message-controller.js +377 -0
- package/dist/app/render-controller.d.ts +41 -0
- package/dist/app/render-controller.js +378 -0
- package/dist/app/render-text.d.ts +19 -0
- package/dist/app/render-text.js +67 -0
- package/dist/app/request-history.d.ts +23 -0
- package/dist/app/request-history.js +131 -0
- package/dist/app/runtime.d.ts +39 -0
- package/dist/app/runtime.js +195 -0
- package/dist/app/screen-selection.d.ts +6 -0
- package/dist/app/screen-selection.js +9 -0
- package/dist/app/screen-styler.d.ts +34 -0
- package/dist/app/screen-styler.js +168 -0
- package/dist/app/scroll-controller.d.ts +51 -0
- package/dist/app/scroll-controller.js +207 -0
- package/dist/app/session-event-controller.d.ts +69 -0
- package/dist/app/session-event-controller.js +338 -0
- package/dist/app/session-history.d.ts +23 -0
- package/dist/app/session-history.js +164 -0
- package/dist/app/session-lifecycle-controller.d.ts +55 -0
- package/dist/app/session-lifecycle-controller.js +116 -0
- package/dist/app/session-search.d.ts +24 -0
- package/dist/app/session-search.js +215 -0
- package/dist/app/shell-command.d.ts +27 -0
- package/dist/app/shell-command.js +176 -0
- package/dist/app/shell-controller.d.ts +28 -0
- package/dist/app/shell-controller.js +124 -0
- package/dist/app/slash-commands.d.ts +6 -0
- package/dist/app/slash-commands.js +75 -0
- package/dist/app/startup-checks.d.ts +8 -0
- package/dist/app/startup-checks.js +59 -0
- package/dist/app/startup-info.d.ts +3 -0
- package/dist/app/startup-info.js +176 -0
- package/dist/app/status-controller.d.ts +35 -0
- package/dist/app/status-controller.js +105 -0
- package/dist/app/status-line-renderer.d.ts +68 -0
- package/dist/app/status-line-renderer.js +453 -0
- package/dist/app/subagents-files.d.ts +10 -0
- package/dist/app/subagents-files.js +193 -0
- package/dist/app/subagents-model.d.ts +23 -0
- package/dist/app/subagents-model.js +224 -0
- package/dist/app/subagents-widget-controller.d.ts +43 -0
- package/dist/app/subagents-widget-controller.js +311 -0
- package/dist/app/tab-line-renderer.d.ts +26 -0
- package/dist/app/tab-line-renderer.js +222 -0
- package/dist/app/tabs-controller.d.ts +100 -0
- package/dist/app/tabs-controller.js +885 -0
- package/dist/app/terminal-controller.d.ts +40 -0
- package/dist/app/terminal-controller.js +135 -0
- package/dist/app/terminal-edit-shortcuts.d.ts +23 -0
- package/dist/app/terminal-edit-shortcuts.js +138 -0
- package/dist/app/terminal-output-buffer.d.ts +20 -0
- package/dist/app/terminal-output-buffer.js +52 -0
- package/dist/app/toast-controller.d.ts +13 -0
- package/dist/app/toast-controller.js +40 -0
- package/dist/app/toast-renderer.d.ts +9 -0
- package/dist/app/toast-renderer.js +79 -0
- package/dist/app/todo-model.d.ts +22 -0
- package/dist/app/todo-model.js +179 -0
- package/dist/app/todo-widget-controller.d.ts +21 -0
- package/dist/app/todo-widget-controller.js +59 -0
- package/dist/app/tool-block-renderer.d.ts +26 -0
- package/dist/app/tool-block-renderer.js +439 -0
- package/dist/app/types.d.ts +550 -0
- package/dist/app/types.js +1 -0
- package/dist/app/update.d.ts +36 -0
- package/dist/app/update.js +315 -0
- package/dist/app/voice-controller.d.ts +52 -0
- package/dist/app/voice-controller.js +600 -0
- package/dist/app/workspace-actions-controller.d.ts +40 -0
- package/dist/app/workspace-actions-controller.js +215 -0
- package/dist/app/workspace-undo.d.ts +44 -0
- package/dist/app/workspace-undo.js +215 -0
- package/dist/config.d.ts +62 -0
- package/dist/config.js +527 -0
- package/dist/context-progress-bar.d.ts +17 -0
- package/dist/context-progress-bar.js +48 -0
- package/dist/default-pix-config.d.ts +1 -0
- package/dist/default-pix-config.js +375 -0
- package/dist/fuzzy.d.ts +23 -0
- package/dist/fuzzy.js +165 -0
- package/dist/input-editor-files.d.ts +15 -0
- package/dist/input-editor-files.js +62 -0
- package/dist/input-editor.d.ts +186 -0
- package/dist/input-editor.js +835 -0
- package/dist/main.d.ts +1 -0
- package/dist/main.js +12 -0
- package/dist/markdown-format.d.ts +22 -0
- package/dist/markdown-format.js +542 -0
- package/dist/sdk.d.ts +3 -0
- package/dist/sdk.js +1 -0
- package/dist/syntax-highlight.d.ts +22 -0
- package/dist/syntax-highlight.js +528 -0
- package/dist/terminal-width.d.ts +6 -0
- package/dist/terminal-width.js +245 -0
- package/dist/theme.d.ts +56 -0
- package/dist/theme.js +118 -0
- package/dist/tool-renderers/apply-patch.d.ts +2 -0
- package/dist/tool-renderers/apply-patch.js +36 -0
- package/dist/tool-renderers/ast.d.ts +2 -0
- package/dist/tool-renderers/ast.js +5 -0
- package/dist/tool-renderers/compress.d.ts +2 -0
- package/dist/tool-renderers/compress.js +126 -0
- package/dist/tool-renderers/index.d.ts +3 -0
- package/dist/tool-renderers/index.js +44 -0
- package/dist/tool-renderers/question.d.ts +2 -0
- package/dist/tool-renderers/question.js +88 -0
- package/dist/tool-renderers/read.d.ts +2 -0
- package/dist/tool-renderers/read.js +36 -0
- package/dist/tool-renderers/repo.d.ts +2 -0
- package/dist/tool-renderers/repo.js +13 -0
- package/dist/tool-renderers/shell.d.ts +3 -0
- package/dist/tool-renderers/shell.js +27 -0
- package/dist/tool-renderers/skill.d.ts +2 -0
- package/dist/tool-renderers/skill.js +132 -0
- package/dist/tool-renderers/subagents.d.ts +2 -0
- package/dist/tool-renderers/subagents.js +51 -0
- package/dist/tool-renderers/todo.d.ts +2 -0
- package/dist/tool-renderers/todo.js +9 -0
- package/dist/tool-renderers/types.d.ts +42 -0
- package/dist/tool-renderers/types.js +1 -0
- package/dist/tool-renderers/utils.d.ts +31 -0
- package/dist/tool-renderers/utils.js +230 -0
- package/dist/tool-renderers/web.d.ts +3 -0
- package/dist/tool-renderers/web.js +9 -0
- package/dist/tool-renderers/write.d.ts +2 -0
- package/dist/tool-renderers/write.js +42 -0
- package/dist/ui.d.ts +56 -0
- package/dist/ui.js +94 -0
- package/docs/release.md +81 -0
- package/extensions/question/contract.ts +100 -0
- package/extensions/question/index.ts +34 -0
- package/extensions/question/render.ts +28 -0
- package/extensions/question/result.ts +86 -0
- package/extensions/question/tool-description.ts +11 -0
- package/extensions/question/tui.ts +629 -0
- package/extensions/question/types.ts +123 -0
- package/extensions/session-title/config.ts +169 -0
- package/extensions/session-title/index.ts +459 -0
- package/extensions/terminal-bell/index.ts +315 -0
- package/external/pi-tools-suite/README.md +242 -0
- package/external/pi-tools-suite/index.ts +1 -0
- package/external/pi-tools-suite/licenses/ollama-pi-web-search.MIT +21 -0
- package/external/pi-tools-suite/licenses/opencode-mystatus-MIT.txt +21 -0
- package/external/pi-tools-suite/package.json +53 -0
- package/external/pi-tools-suite/src/antigravity-auth/auth-store.ts +194 -0
- package/external/pi-tools-suite/src/antigravity-auth/commands.ts +80 -0
- package/external/pi-tools-suite/src/antigravity-auth/constants.ts +26 -0
- package/external/pi-tools-suite/src/antigravity-auth/headers.ts +20 -0
- package/external/pi-tools-suite/src/antigravity-auth/index.ts +104 -0
- package/external/pi-tools-suite/src/antigravity-auth/models.ts +86 -0
- package/external/pi-tools-suite/src/antigravity-auth/oauth.ts +305 -0
- package/external/pi-tools-suite/src/antigravity-auth/payload.ts +423 -0
- package/external/pi-tools-suite/src/antigravity-auth/status.ts +78 -0
- package/external/pi-tools-suite/src/antigravity-auth/stream.ts +302 -0
- package/external/pi-tools-suite/src/antigravity-auth/types.ts +113 -0
- package/external/pi-tools-suite/src/ast-grep/index.ts +6 -0
- package/external/pi-tools-suite/src/ast-grep/render.ts +70 -0
- package/external/pi-tools-suite/src/ast-grep/schema.ts +109 -0
- package/external/pi-tools-suite/src/ast-grep/tool.ts +345 -0
- package/external/pi-tools-suite/src/ast-grep/types.ts +55 -0
- package/external/pi-tools-suite/src/ast-grep/utils.ts +65 -0
- package/external/pi-tools-suite/src/async-subagents/async-subagents.sample.jsonc +222 -0
- package/external/pi-tools-suite/src/async-subagents/commands.ts +518 -0
- package/external/pi-tools-suite/src/async-subagents/constants.ts +11 -0
- package/external/pi-tools-suite/src/async-subagents/core/agent-strategy.ts +74 -0
- package/external/pi-tools-suite/src/async-subagents/core/attachment-bridge.ts +133 -0
- package/external/pi-tools-suite/src/async-subagents/core/cleanup.ts +66 -0
- package/external/pi-tools-suite/src/async-subagents/core/concurrency.ts +90 -0
- package/external/pi-tools-suite/src/async-subagents/core/config.ts +819 -0
- package/external/pi-tools-suite/src/async-subagents/core/log-limits.ts +166 -0
- package/external/pi-tools-suite/src/async-subagents/core/model-fallback.ts +133 -0
- package/external/pi-tools-suite/src/async-subagents/core/paths.ts +47 -0
- package/external/pi-tools-suite/src/async-subagents/core/pi-invocation.ts +35 -0
- package/external/pi-tools-suite/src/async-subagents/core/presets.ts +67 -0
- package/external/pi-tools-suite/src/async-subagents/core/process.ts +15 -0
- package/external/pi-tools-suite/src/async-subagents/core/prompt.ts +66 -0
- package/external/pi-tools-suite/src/async-subagents/core/registry.ts +224 -0
- package/external/pi-tools-suite/src/async-subagents/core/retry.ts +191 -0
- package/external/pi-tools-suite/src/async-subagents/core/routing.ts +259 -0
- package/external/pi-tools-suite/src/async-subagents/core/sessions.ts +138 -0
- package/external/pi-tools-suite/src/async-subagents/core/spawn.ts +688 -0
- package/external/pi-tools-suite/src/async-subagents/core/state.ts +281 -0
- package/external/pi-tools-suite/src/async-subagents/core/stop.ts +131 -0
- package/external/pi-tools-suite/src/async-subagents/core/structured-result.ts +237 -0
- package/external/pi-tools-suite/src/async-subagents/core/tool-guard.ts +34 -0
- package/external/pi-tools-suite/src/async-subagents/core/types.ts +150 -0
- package/external/pi-tools-suite/src/async-subagents/core/ultrawork-auto.ts +184 -0
- package/external/pi-tools-suite/src/async-subagents/core/utils.ts +11 -0
- package/external/pi-tools-suite/src/async-subagents/format.ts +41 -0
- package/external/pi-tools-suite/src/async-subagents/index.ts +422 -0
- package/external/pi-tools-suite/src/async-subagents/lib.ts +88 -0
- package/external/pi-tools-suite/src/async-subagents/live.ts +10 -0
- package/external/pi-tools-suite/src/async-subagents/polling.ts +83 -0
- package/external/pi-tools-suite/src/async-subagents/render.ts +230 -0
- package/external/pi-tools-suite/src/async-subagents/subagent-overlay.ts +77 -0
- package/external/pi-tools-suite/src/async-subagents/tasks.ts +120 -0
- package/external/pi-tools-suite/src/async-subagents/tools/cleanup.ts +99 -0
- package/external/pi-tools-suite/src/async-subagents/tools/result.ts +179 -0
- package/external/pi-tools-suite/src/async-subagents/tools/spawn.ts +372 -0
- package/external/pi-tools-suite/src/async-subagents/tools/status.ts +60 -0
- package/external/pi-tools-suite/src/async-subagents/tools/stop.ts +79 -0
- package/external/pi-tools-suite/src/async-subagents/tools/subagents.ts +152 -0
- package/external/pi-tools-suite/src/async-subagents/tools/wait.ts +67 -0
- package/external/pi-tools-suite/src/async-subagents/types.ts +45 -0
- package/external/pi-tools-suite/src/async-subagents/ui.ts +5 -0
- package/external/pi-tools-suite/src/compress/commands.ts +440 -0
- package/external/pi-tools-suite/src/compress/compress-tool.ts +368 -0
- package/external/pi-tools-suite/src/compress/compression-blocks.ts +524 -0
- package/external/pi-tools-suite/src/compress/config.ts +310 -0
- package/external/pi-tools-suite/src/compress/dcp-tui-filter.ts +498 -0
- package/external/pi-tools-suite/src/compress/index.ts +397 -0
- package/external/pi-tools-suite/src/compress/prompts.ts +269 -0
- package/external/pi-tools-suite/src/compress/pruner-candidates.ts +176 -0
- package/external/pi-tools-suite/src/compress/pruner-compression-blocks.ts +260 -0
- package/external/pi-tools-suite/src/compress/pruner-message-ids.ts +147 -0
- package/external/pi-tools-suite/src/compress/pruner-metadata.ts +268 -0
- package/external/pi-tools-suite/src/compress/pruner-nudge.ts +315 -0
- package/external/pi-tools-suite/src/compress/pruner-tools.ts +263 -0
- package/external/pi-tools-suite/src/compress/pruner-types.ts +25 -0
- package/external/pi-tools-suite/src/compress/pruner.ts +92 -0
- package/external/pi-tools-suite/src/compress/state.ts +486 -0
- package/external/pi-tools-suite/src/compress/ui.ts +308 -0
- package/external/pi-tools-suite/src/config.ts +176 -0
- package/external/pi-tools-suite/src/context-usage.ts +31 -0
- package/external/pi-tools-suite/src/default-pi-tools-suite-config.ts +355 -0
- package/external/pi-tools-suite/src/index.ts +46 -0
- package/external/pi-tools-suite/src/lib/lsp.ts +62 -0
- package/external/pi-tools-suite/src/lib/project.ts +42 -0
- package/external/pi-tools-suite/src/lib/tool-args.ts +137 -0
- package/external/pi-tools-suite/src/lsp/_shared/config.ts +156 -0
- package/external/pi-tools-suite/src/lsp/_shared/glob.ts +60 -0
- package/external/pi-tools-suite/src/lsp/_shared/output.ts +102 -0
- package/external/pi-tools-suite/src/lsp/_shared/paths.ts +138 -0
- package/external/pi-tools-suite/src/lsp/_shared/runner.ts +64 -0
- package/external/pi-tools-suite/src/lsp/_shared/template.ts +23 -0
- package/external/pi-tools-suite/src/lsp/_shared/trust.ts +116 -0
- package/external/pi-tools-suite/src/lsp/_shared/types.ts +98 -0
- package/external/pi-tools-suite/src/lsp/async.ts +29 -0
- package/external/pi-tools-suite/src/lsp/child-process.ts +81 -0
- package/external/pi-tools-suite/src/lsp/client.ts +340 -0
- package/external/pi-tools-suite/src/lsp/constants.ts +9 -0
- package/external/pi-tools-suite/src/lsp/diagnostics-store.ts +64 -0
- package/external/pi-tools-suite/src/lsp/documents.ts +24 -0
- package/external/pi-tools-suite/src/lsp/index.ts +31 -0
- package/external/pi-tools-suite/src/lsp/lsp-utils.ts +37 -0
- package/external/pi-tools-suite/src/lsp/manager.ts +190 -0
- package/external/pi-tools-suite/src/lsp/mutation-events.ts +78 -0
- package/external/pi-tools-suite/src/lsp/renderer.ts +1 -0
- package/external/pi-tools-suite/src/lsp/tool-result.ts +6 -0
- package/external/pi-tools-suite/src/lsp/tsserver.ts +107 -0
- package/external/pi-tools-suite/src/lsp/types.ts +15 -0
- package/external/pi-tools-suite/src/model-tools/apply-patch.ts +590 -0
- package/external/pi-tools-suite/src/model-tools/index.ts +430 -0
- package/external/pi-tools-suite/src/model-tools/path-utils.ts +6 -0
- package/external/pi-tools-suite/src/model-tools/tool-args.ts +11 -0
- package/external/pi-tools-suite/src/prompt-commands/index.ts +349 -0
- package/external/pi-tools-suite/src/repo-discovery/index.ts +384 -0
- package/external/pi-tools-suite/src/repo-discovery/project.ts +7 -0
- package/external/pi-tools-suite/src/startup-section.ts +13 -0
- package/external/pi-tools-suite/src/terminal-bell/index.ts +309 -0
- package/external/pi-tools-suite/src/todo/index.ts +201 -0
- package/external/pi-tools-suite/src/todo/state/auto-clear.ts +13 -0
- package/external/pi-tools-suite/src/todo/state/invariants.ts +21 -0
- package/external/pi-tools-suite/src/todo/state/persistence.ts +94 -0
- package/external/pi-tools-suite/src/todo/state/replay.ts +38 -0
- package/external/pi-tools-suite/src/todo/state/selectors.ts +49 -0
- package/external/pi-tools-suite/src/todo/state/state-reducer.ts +362 -0
- package/external/pi-tools-suite/src/todo/state/state.ts +18 -0
- package/external/pi-tools-suite/src/todo/state/store.ts +52 -0
- package/external/pi-tools-suite/src/todo/state/task-graph.ts +57 -0
- package/external/pi-tools-suite/src/todo/todo.ts +487 -0
- package/external/pi-tools-suite/src/todo/tool/response-envelope.ts +143 -0
- package/external/pi-tools-suite/src/todo/tool/types.ts +188 -0
- package/external/pi-tools-suite/src/todo/view/format.ts +18 -0
- package/external/pi-tools-suite/src/todo/view/labels.ts +13 -0
- package/external/pi-tools-suite/src/tool-descriptions.ts +369 -0
- package/external/pi-tools-suite/src/usage/index.ts +152 -0
- package/external/pi-tools-suite/src/usage/lib/copilot.ts +535 -0
- package/external/pi-tools-suite/src/usage/lib/google.ts +478 -0
- package/external/pi-tools-suite/src/usage/lib/openai.ts +268 -0
- package/external/pi-tools-suite/src/usage/lib/types.ts +114 -0
- package/external/pi-tools-suite/src/usage/lib/utils.ts +134 -0
- package/external/pi-tools-suite/src/usage/lib/zhipu.ts +228 -0
- package/external/pi-tools-suite/src/web-search/index.ts +397 -0
- package/package.json +89 -0
- package/skills/context7/SKILL.md +69 -0
- package/skills/context7/scripts/context7.sh +73 -0
- package/skills/handoff/SKILL.md +15 -0
- package/skills/pdf/LICENSE.txt +30 -0
- package/skills/pdf/SKILL.md +314 -0
- package/skills/pdf/forms.md +294 -0
- package/skills/pdf/reference.md +612 -0
- package/skills/pdf/scripts/check_bounding_boxes.py +65 -0
- package/skills/pdf/scripts/check_fillable_fields.py +11 -0
- package/skills/pdf/scripts/convert_pdf_to_images.py +33 -0
- package/skills/pdf/scripts/create_validation_image.py +37 -0
- package/skills/pdf/scripts/extract_form_field_info.py +122 -0
- package/skills/pdf/scripts/extract_form_structure.py +115 -0
- package/skills/pdf/scripts/fill_fillable_fields.py +98 -0
- package/skills/pdf/scripts/fill_pdf_form_with_annotations.py +107 -0
- package/skills/playwright-cli/SKILL.md +388 -0
- package/skills/playwright-cli/references/element-attributes.md +23 -0
- package/skills/playwright-cli/references/playwright-tests.md +39 -0
- package/skills/playwright-cli/references/request-mocking.md +87 -0
- package/skills/playwright-cli/references/running-code.md +241 -0
- package/skills/playwright-cli/references/session-management.md +225 -0
- package/skills/playwright-cli/references/spec-driven-testing.md +305 -0
- package/skills/playwright-cli/references/storage-state.md +275 -0
- package/skills/playwright-cli/references/test-generation.md +134 -0
- package/skills/playwright-cli/references/tracing.md +139 -0
- package/skills/playwright-cli/references/video-recording.md +143 -0
- package/skills/simplify/SKILL.md +51 -0
- package/skills/skill-creator/LICENSE.txt +202 -0
- package/skills/skill-creator/SKILL.md +485 -0
- package/skills/skill-creator/agents/analyzer.md +274 -0
- package/skills/skill-creator/agents/comparator.md +202 -0
- package/skills/skill-creator/agents/grader.md +223 -0
- package/skills/skill-creator/assets/eval_review.html +146 -0
- package/skills/skill-creator/eval-viewer/generate_review.py +471 -0
- package/skills/skill-creator/eval-viewer/viewer.html +1325 -0
- package/skills/skill-creator/references/schemas.md +430 -0
- package/skills/skill-creator/scripts/__init__.py +0 -0
- package/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
- package/skills/skill-creator/scripts/generate_report.py +326 -0
- package/skills/skill-creator/scripts/improve_description.py +247 -0
- package/skills/skill-creator/scripts/package_skill.py +136 -0
- package/skills/skill-creator/scripts/quick_validate.py +103 -0
- package/skills/skill-creator/scripts/run_eval.py +310 -0
- package/skills/skill-creator/scripts/run_loop.py +328 -0
- package/skills/skill-creator/scripts/utils.py +47 -0
|
@@ -0,0 +1,33 @@
|
|
|
1
|
+
import os
|
|
2
|
+
import sys
|
|
3
|
+
|
|
4
|
+
from pdf2image import convert_from_path
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
def convert(pdf_path, output_dir, max_dim=1000):
|
|
10
|
+
images = convert_from_path(pdf_path, dpi=200)
|
|
11
|
+
|
|
12
|
+
for i, image in enumerate(images):
|
|
13
|
+
width, height = image.size
|
|
14
|
+
if width > max_dim or height > max_dim:
|
|
15
|
+
scale_factor = min(max_dim / width, max_dim / height)
|
|
16
|
+
new_width = int(width * scale_factor)
|
|
17
|
+
new_height = int(height * scale_factor)
|
|
18
|
+
image = image.resize((new_width, new_height))
|
|
19
|
+
|
|
20
|
+
image_path = os.path.join(output_dir, f"page_{i+1}.png")
|
|
21
|
+
image.save(image_path)
|
|
22
|
+
print(f"Saved page {i+1} as {image_path} (size: {image.size})")
|
|
23
|
+
|
|
24
|
+
print(f"Converted {len(images)} pages to PNG images")
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
if __name__ == "__main__":
|
|
28
|
+
if len(sys.argv) != 3:
|
|
29
|
+
print("Usage: convert_pdf_to_images.py [input pdf] [output directory]")
|
|
30
|
+
sys.exit(1)
|
|
31
|
+
pdf_path = sys.argv[1]
|
|
32
|
+
output_directory = sys.argv[2]
|
|
33
|
+
convert(pdf_path, output_directory)
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import json
|
|
2
|
+
import sys
|
|
3
|
+
|
|
4
|
+
from PIL import Image, ImageDraw
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
def create_validation_image(page_number, fields_json_path, input_path, output_path):
|
|
10
|
+
with open(fields_json_path, 'r') as f:
|
|
11
|
+
data = json.load(f)
|
|
12
|
+
|
|
13
|
+
img = Image.open(input_path)
|
|
14
|
+
draw = ImageDraw.Draw(img)
|
|
15
|
+
num_boxes = 0
|
|
16
|
+
|
|
17
|
+
for field in data["form_fields"]:
|
|
18
|
+
if field["page_number"] == page_number:
|
|
19
|
+
entry_box = field['entry_bounding_box']
|
|
20
|
+
label_box = field['label_bounding_box']
|
|
21
|
+
draw.rectangle(entry_box, outline='red', width=2)
|
|
22
|
+
draw.rectangle(label_box, outline='blue', width=2)
|
|
23
|
+
num_boxes += 2
|
|
24
|
+
|
|
25
|
+
img.save(output_path)
|
|
26
|
+
print(f"Created validation image at {output_path} with {num_boxes} bounding boxes")
|
|
27
|
+
|
|
28
|
+
|
|
29
|
+
if __name__ == "__main__":
|
|
30
|
+
if len(sys.argv) != 5:
|
|
31
|
+
print("Usage: create_validation_image.py [page number] [fields.json file] [input image path] [output image path]")
|
|
32
|
+
sys.exit(1)
|
|
33
|
+
page_number = int(sys.argv[1])
|
|
34
|
+
fields_json_path = sys.argv[2]
|
|
35
|
+
input_image_path = sys.argv[3]
|
|
36
|
+
output_image_path = sys.argv[4]
|
|
37
|
+
create_validation_image(page_number, fields_json_path, input_image_path, output_image_path)
|
|
@@ -0,0 +1,122 @@
|
|
|
1
|
+
import json
|
|
2
|
+
import sys
|
|
3
|
+
|
|
4
|
+
from pypdf import PdfReader
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
def get_full_annotation_field_id(annotation):
|
|
10
|
+
components = []
|
|
11
|
+
while annotation:
|
|
12
|
+
field_name = annotation.get('/T')
|
|
13
|
+
if field_name:
|
|
14
|
+
components.append(field_name)
|
|
15
|
+
annotation = annotation.get('/Parent')
|
|
16
|
+
return ".".join(reversed(components)) if components else None
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
def make_field_dict(field, field_id):
|
|
20
|
+
field_dict = {"field_id": field_id}
|
|
21
|
+
ft = field.get('/FT')
|
|
22
|
+
if ft == "/Tx":
|
|
23
|
+
field_dict["type"] = "text"
|
|
24
|
+
elif ft == "/Btn":
|
|
25
|
+
field_dict["type"] = "checkbox"
|
|
26
|
+
states = field.get("/_States_", [])
|
|
27
|
+
if len(states) == 2:
|
|
28
|
+
if "/Off" in states:
|
|
29
|
+
field_dict["checked_value"] = states[0] if states[0] != "/Off" else states[1]
|
|
30
|
+
field_dict["unchecked_value"] = "/Off"
|
|
31
|
+
else:
|
|
32
|
+
print(f"Unexpected state values for checkbox `${field_id}`. Its checked and unchecked values may not be correct; if you're trying to check it, visually verify the results.")
|
|
33
|
+
field_dict["checked_value"] = states[0]
|
|
34
|
+
field_dict["unchecked_value"] = states[1]
|
|
35
|
+
elif ft == "/Ch":
|
|
36
|
+
field_dict["type"] = "choice"
|
|
37
|
+
states = field.get("/_States_", [])
|
|
38
|
+
field_dict["choice_options"] = [{
|
|
39
|
+
"value": state[0],
|
|
40
|
+
"text": state[1],
|
|
41
|
+
} for state in states]
|
|
42
|
+
else:
|
|
43
|
+
field_dict["type"] = f"unknown ({ft})"
|
|
44
|
+
return field_dict
|
|
45
|
+
|
|
46
|
+
|
|
47
|
+
def get_field_info(reader: PdfReader):
|
|
48
|
+
fields = reader.get_fields()
|
|
49
|
+
|
|
50
|
+
field_info_by_id = {}
|
|
51
|
+
possible_radio_names = set()
|
|
52
|
+
|
|
53
|
+
for field_id, field in fields.items():
|
|
54
|
+
if field.get("/Kids"):
|
|
55
|
+
if field.get("/FT") == "/Btn":
|
|
56
|
+
possible_radio_names.add(field_id)
|
|
57
|
+
continue
|
|
58
|
+
field_info_by_id[field_id] = make_field_dict(field, field_id)
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
radio_fields_by_id = {}
|
|
62
|
+
|
|
63
|
+
for page_index, page in enumerate(reader.pages):
|
|
64
|
+
annotations = page.get('/Annots', [])
|
|
65
|
+
for ann in annotations:
|
|
66
|
+
field_id = get_full_annotation_field_id(ann)
|
|
67
|
+
if field_id in field_info_by_id:
|
|
68
|
+
field_info_by_id[field_id]["page"] = page_index + 1
|
|
69
|
+
field_info_by_id[field_id]["rect"] = ann.get('/Rect')
|
|
70
|
+
elif field_id in possible_radio_names:
|
|
71
|
+
try:
|
|
72
|
+
on_values = [v for v in ann["/AP"]["/N"] if v != "/Off"]
|
|
73
|
+
except KeyError:
|
|
74
|
+
continue
|
|
75
|
+
if len(on_values) == 1:
|
|
76
|
+
rect = ann.get("/Rect")
|
|
77
|
+
if field_id not in radio_fields_by_id:
|
|
78
|
+
radio_fields_by_id[field_id] = {
|
|
79
|
+
"field_id": field_id,
|
|
80
|
+
"type": "radio_group",
|
|
81
|
+
"page": page_index + 1,
|
|
82
|
+
"radio_options": [],
|
|
83
|
+
}
|
|
84
|
+
radio_fields_by_id[field_id]["radio_options"].append({
|
|
85
|
+
"value": on_values[0],
|
|
86
|
+
"rect": rect,
|
|
87
|
+
})
|
|
88
|
+
|
|
89
|
+
fields_with_location = []
|
|
90
|
+
for field_info in field_info_by_id.values():
|
|
91
|
+
if "page" in field_info:
|
|
92
|
+
fields_with_location.append(field_info)
|
|
93
|
+
else:
|
|
94
|
+
print(f"Unable to determine location for field id: {field_info.get('field_id')}, ignoring")
|
|
95
|
+
|
|
96
|
+
def sort_key(f):
|
|
97
|
+
if "radio_options" in f:
|
|
98
|
+
rect = f["radio_options"][0]["rect"] or [0, 0, 0, 0]
|
|
99
|
+
else:
|
|
100
|
+
rect = f.get("rect") or [0, 0, 0, 0]
|
|
101
|
+
adjusted_position = [-rect[1], rect[0]]
|
|
102
|
+
return [f.get("page"), adjusted_position]
|
|
103
|
+
|
|
104
|
+
sorted_fields = fields_with_location + list(radio_fields_by_id.values())
|
|
105
|
+
sorted_fields.sort(key=sort_key)
|
|
106
|
+
|
|
107
|
+
return sorted_fields
|
|
108
|
+
|
|
109
|
+
|
|
110
|
+
def write_field_info(pdf_path: str, json_output_path: str):
|
|
111
|
+
reader = PdfReader(pdf_path)
|
|
112
|
+
field_info = get_field_info(reader)
|
|
113
|
+
with open(json_output_path, "w") as f:
|
|
114
|
+
json.dump(field_info, f, indent=2)
|
|
115
|
+
print(f"Wrote {len(field_info)} fields to {json_output_path}")
|
|
116
|
+
|
|
117
|
+
|
|
118
|
+
if __name__ == "__main__":
|
|
119
|
+
if len(sys.argv) != 3:
|
|
120
|
+
print("Usage: extract_form_field_info.py [input pdf] [output json]")
|
|
121
|
+
sys.exit(1)
|
|
122
|
+
write_field_info(sys.argv[1], sys.argv[2])
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
"""
|
|
2
|
+
Extract form structure from a non-fillable PDF.
|
|
3
|
+
|
|
4
|
+
This script analyzes the PDF to find:
|
|
5
|
+
- Text labels with their exact coordinates
|
|
6
|
+
- Horizontal lines (row boundaries)
|
|
7
|
+
- Checkboxes (small rectangles)
|
|
8
|
+
|
|
9
|
+
Output: A JSON file with the form structure that can be used to generate
|
|
10
|
+
accurate field coordinates for filling.
|
|
11
|
+
|
|
12
|
+
Usage: python extract_form_structure.py <input.pdf> <output.json>
|
|
13
|
+
"""
|
|
14
|
+
|
|
15
|
+
import json
|
|
16
|
+
import sys
|
|
17
|
+
import pdfplumber
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
def extract_form_structure(pdf_path):
|
|
21
|
+
structure = {
|
|
22
|
+
"pages": [],
|
|
23
|
+
"labels": [],
|
|
24
|
+
"lines": [],
|
|
25
|
+
"checkboxes": [],
|
|
26
|
+
"row_boundaries": []
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
with pdfplumber.open(pdf_path) as pdf:
|
|
30
|
+
for page_num, page in enumerate(pdf.pages, 1):
|
|
31
|
+
structure["pages"].append({
|
|
32
|
+
"page_number": page_num,
|
|
33
|
+
"width": float(page.width),
|
|
34
|
+
"height": float(page.height)
|
|
35
|
+
})
|
|
36
|
+
|
|
37
|
+
words = page.extract_words()
|
|
38
|
+
for word in words:
|
|
39
|
+
structure["labels"].append({
|
|
40
|
+
"page": page_num,
|
|
41
|
+
"text": word["text"],
|
|
42
|
+
"x0": round(float(word["x0"]), 1),
|
|
43
|
+
"top": round(float(word["top"]), 1),
|
|
44
|
+
"x1": round(float(word["x1"]), 1),
|
|
45
|
+
"bottom": round(float(word["bottom"]), 1)
|
|
46
|
+
})
|
|
47
|
+
|
|
48
|
+
for line in page.lines:
|
|
49
|
+
if abs(float(line["x1"]) - float(line["x0"])) > page.width * 0.5:
|
|
50
|
+
structure["lines"].append({
|
|
51
|
+
"page": page_num,
|
|
52
|
+
"y": round(float(line["top"]), 1),
|
|
53
|
+
"x0": round(float(line["x0"]), 1),
|
|
54
|
+
"x1": round(float(line["x1"]), 1)
|
|
55
|
+
})
|
|
56
|
+
|
|
57
|
+
for rect in page.rects:
|
|
58
|
+
width = float(rect["x1"]) - float(rect["x0"])
|
|
59
|
+
height = float(rect["bottom"]) - float(rect["top"])
|
|
60
|
+
if 5 <= width <= 15 and 5 <= height <= 15 and abs(width - height) < 2:
|
|
61
|
+
structure["checkboxes"].append({
|
|
62
|
+
"page": page_num,
|
|
63
|
+
"x0": round(float(rect["x0"]), 1),
|
|
64
|
+
"top": round(float(rect["top"]), 1),
|
|
65
|
+
"x1": round(float(rect["x1"]), 1),
|
|
66
|
+
"bottom": round(float(rect["bottom"]), 1),
|
|
67
|
+
"center_x": round((float(rect["x0"]) + float(rect["x1"])) / 2, 1),
|
|
68
|
+
"center_y": round((float(rect["top"]) + float(rect["bottom"])) / 2, 1)
|
|
69
|
+
})
|
|
70
|
+
|
|
71
|
+
lines_by_page = {}
|
|
72
|
+
for line in structure["lines"]:
|
|
73
|
+
page = line["page"]
|
|
74
|
+
if page not in lines_by_page:
|
|
75
|
+
lines_by_page[page] = []
|
|
76
|
+
lines_by_page[page].append(line["y"])
|
|
77
|
+
|
|
78
|
+
for page, y_coords in lines_by_page.items():
|
|
79
|
+
y_coords = sorted(set(y_coords))
|
|
80
|
+
for i in range(len(y_coords) - 1):
|
|
81
|
+
structure["row_boundaries"].append({
|
|
82
|
+
"page": page,
|
|
83
|
+
"row_top": y_coords[i],
|
|
84
|
+
"row_bottom": y_coords[i + 1],
|
|
85
|
+
"row_height": round(y_coords[i + 1] - y_coords[i], 1)
|
|
86
|
+
})
|
|
87
|
+
|
|
88
|
+
return structure
|
|
89
|
+
|
|
90
|
+
|
|
91
|
+
def main():
|
|
92
|
+
if len(sys.argv) != 3:
|
|
93
|
+
print("Usage: extract_form_structure.py <input.pdf> <output.json>")
|
|
94
|
+
sys.exit(1)
|
|
95
|
+
|
|
96
|
+
pdf_path = sys.argv[1]
|
|
97
|
+
output_path = sys.argv[2]
|
|
98
|
+
|
|
99
|
+
print(f"Extracting structure from {pdf_path}...")
|
|
100
|
+
structure = extract_form_structure(pdf_path)
|
|
101
|
+
|
|
102
|
+
with open(output_path, "w") as f:
|
|
103
|
+
json.dump(structure, f, indent=2)
|
|
104
|
+
|
|
105
|
+
print(f"Found:")
|
|
106
|
+
print(f" - {len(structure['pages'])} pages")
|
|
107
|
+
print(f" - {len(structure['labels'])} text labels")
|
|
108
|
+
print(f" - {len(structure['lines'])} horizontal lines")
|
|
109
|
+
print(f" - {len(structure['checkboxes'])} checkboxes")
|
|
110
|
+
print(f" - {len(structure['row_boundaries'])} row boundaries")
|
|
111
|
+
print(f"Saved to {output_path}")
|
|
112
|
+
|
|
113
|
+
|
|
114
|
+
if __name__ == "__main__":
|
|
115
|
+
main()
|
|
@@ -0,0 +1,98 @@
|
|
|
1
|
+
import json
|
|
2
|
+
import sys
|
|
3
|
+
|
|
4
|
+
from pypdf import PdfReader, PdfWriter
|
|
5
|
+
|
|
6
|
+
from extract_form_field_info import get_field_info
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
def fill_pdf_fields(input_pdf_path: str, fields_json_path: str, output_pdf_path: str):
|
|
12
|
+
with open(fields_json_path) as f:
|
|
13
|
+
fields = json.load(f)
|
|
14
|
+
fields_by_page = {}
|
|
15
|
+
for field in fields:
|
|
16
|
+
if "value" in field:
|
|
17
|
+
field_id = field["field_id"]
|
|
18
|
+
page = field["page"]
|
|
19
|
+
if page not in fields_by_page:
|
|
20
|
+
fields_by_page[page] = {}
|
|
21
|
+
fields_by_page[page][field_id] = field["value"]
|
|
22
|
+
|
|
23
|
+
reader = PdfReader(input_pdf_path)
|
|
24
|
+
|
|
25
|
+
has_error = False
|
|
26
|
+
field_info = get_field_info(reader)
|
|
27
|
+
fields_by_ids = {f["field_id"]: f for f in field_info}
|
|
28
|
+
for field in fields:
|
|
29
|
+
existing_field = fields_by_ids.get(field["field_id"])
|
|
30
|
+
if not existing_field:
|
|
31
|
+
has_error = True
|
|
32
|
+
print(f"ERROR: `{field['field_id']}` is not a valid field ID")
|
|
33
|
+
elif field["page"] != existing_field["page"]:
|
|
34
|
+
has_error = True
|
|
35
|
+
print(f"ERROR: Incorrect page number for `{field['field_id']}` (got {field['page']}, expected {existing_field['page']})")
|
|
36
|
+
else:
|
|
37
|
+
if "value" in field:
|
|
38
|
+
err = validation_error_for_field_value(existing_field, field["value"])
|
|
39
|
+
if err:
|
|
40
|
+
print(err)
|
|
41
|
+
has_error = True
|
|
42
|
+
if has_error:
|
|
43
|
+
sys.exit(1)
|
|
44
|
+
|
|
45
|
+
writer = PdfWriter(clone_from=reader)
|
|
46
|
+
for page, field_values in fields_by_page.items():
|
|
47
|
+
writer.update_page_form_field_values(writer.pages[page - 1], field_values, auto_regenerate=False)
|
|
48
|
+
|
|
49
|
+
writer.set_need_appearances_writer(True)
|
|
50
|
+
|
|
51
|
+
with open(output_pdf_path, "wb") as f:
|
|
52
|
+
writer.write(f)
|
|
53
|
+
|
|
54
|
+
|
|
55
|
+
def validation_error_for_field_value(field_info, field_value):
|
|
56
|
+
field_type = field_info["type"]
|
|
57
|
+
field_id = field_info["field_id"]
|
|
58
|
+
if field_type == "checkbox":
|
|
59
|
+
checked_val = field_info["checked_value"]
|
|
60
|
+
unchecked_val = field_info["unchecked_value"]
|
|
61
|
+
if field_value != checked_val and field_value != unchecked_val:
|
|
62
|
+
return f'ERROR: Invalid value "{field_value}" for checkbox field "{field_id}". The checked value is "{checked_val}" and the unchecked value is "{unchecked_val}"'
|
|
63
|
+
elif field_type == "radio_group":
|
|
64
|
+
option_values = [opt["value"] for opt in field_info["radio_options"]]
|
|
65
|
+
if field_value not in option_values:
|
|
66
|
+
return f'ERROR: Invalid value "{field_value}" for radio group field "{field_id}". Valid values are: {option_values}'
|
|
67
|
+
elif field_type == "choice":
|
|
68
|
+
choice_values = [opt["value"] for opt in field_info["choice_options"]]
|
|
69
|
+
if field_value not in choice_values:
|
|
70
|
+
return f'ERROR: Invalid value "{field_value}" for choice field "{field_id}". Valid values are: {choice_values}'
|
|
71
|
+
return None
|
|
72
|
+
|
|
73
|
+
|
|
74
|
+
def monkeypatch_pydpf_method():
|
|
75
|
+
from pypdf.generic import DictionaryObject
|
|
76
|
+
from pypdf.constants import FieldDictionaryAttributes
|
|
77
|
+
|
|
78
|
+
original_get_inherited = DictionaryObject.get_inherited
|
|
79
|
+
|
|
80
|
+
def patched_get_inherited(self, key: str, default = None):
|
|
81
|
+
result = original_get_inherited(self, key, default)
|
|
82
|
+
if key == FieldDictionaryAttributes.Opt:
|
|
83
|
+
if isinstance(result, list) and all(isinstance(v, list) and len(v) == 2 for v in result):
|
|
84
|
+
result = [r[0] for r in result]
|
|
85
|
+
return result
|
|
86
|
+
|
|
87
|
+
DictionaryObject.get_inherited = patched_get_inherited
|
|
88
|
+
|
|
89
|
+
|
|
90
|
+
if __name__ == "__main__":
|
|
91
|
+
if len(sys.argv) != 4:
|
|
92
|
+
print("Usage: fill_fillable_fields.py [input pdf] [field_values.json] [output pdf]")
|
|
93
|
+
sys.exit(1)
|
|
94
|
+
monkeypatch_pydpf_method()
|
|
95
|
+
input_pdf = sys.argv[1]
|
|
96
|
+
fields_json = sys.argv[2]
|
|
97
|
+
output_pdf = sys.argv[3]
|
|
98
|
+
fill_pdf_fields(input_pdf, fields_json, output_pdf)
|
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
import json
|
|
2
|
+
import sys
|
|
3
|
+
|
|
4
|
+
from pypdf import PdfReader, PdfWriter
|
|
5
|
+
from pypdf.annotations import FreeText
|
|
6
|
+
|
|
7
|
+
|
|
8
|
+
|
|
9
|
+
|
|
10
|
+
def transform_from_image_coords(bbox, image_width, image_height, pdf_width, pdf_height):
|
|
11
|
+
x_scale = pdf_width / image_width
|
|
12
|
+
y_scale = pdf_height / image_height
|
|
13
|
+
|
|
14
|
+
left = bbox[0] * x_scale
|
|
15
|
+
right = bbox[2] * x_scale
|
|
16
|
+
|
|
17
|
+
top = pdf_height - (bbox[1] * y_scale)
|
|
18
|
+
bottom = pdf_height - (bbox[3] * y_scale)
|
|
19
|
+
|
|
20
|
+
return left, bottom, right, top
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
def transform_from_pdf_coords(bbox, pdf_height):
|
|
24
|
+
left = bbox[0]
|
|
25
|
+
right = bbox[2]
|
|
26
|
+
|
|
27
|
+
pypdf_top = pdf_height - bbox[1]
|
|
28
|
+
pypdf_bottom = pdf_height - bbox[3]
|
|
29
|
+
|
|
30
|
+
return left, pypdf_bottom, right, pypdf_top
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
def fill_pdf_form(input_pdf_path, fields_json_path, output_pdf_path):
|
|
34
|
+
|
|
35
|
+
with open(fields_json_path, "r") as f:
|
|
36
|
+
fields_data = json.load(f)
|
|
37
|
+
|
|
38
|
+
reader = PdfReader(input_pdf_path)
|
|
39
|
+
writer = PdfWriter()
|
|
40
|
+
|
|
41
|
+
writer.append(reader)
|
|
42
|
+
|
|
43
|
+
pdf_dimensions = {}
|
|
44
|
+
for i, page in enumerate(reader.pages):
|
|
45
|
+
mediabox = page.mediabox
|
|
46
|
+
pdf_dimensions[i + 1] = [mediabox.width, mediabox.height]
|
|
47
|
+
|
|
48
|
+
annotations = []
|
|
49
|
+
for field in fields_data["form_fields"]:
|
|
50
|
+
page_num = field["page_number"]
|
|
51
|
+
|
|
52
|
+
page_info = next(p for p in fields_data["pages"] if p["page_number"] == page_num)
|
|
53
|
+
pdf_width, pdf_height = pdf_dimensions[page_num]
|
|
54
|
+
|
|
55
|
+
if "pdf_width" in page_info:
|
|
56
|
+
transformed_entry_box = transform_from_pdf_coords(
|
|
57
|
+
field["entry_bounding_box"],
|
|
58
|
+
float(pdf_height)
|
|
59
|
+
)
|
|
60
|
+
else:
|
|
61
|
+
image_width = page_info["image_width"]
|
|
62
|
+
image_height = page_info["image_height"]
|
|
63
|
+
transformed_entry_box = transform_from_image_coords(
|
|
64
|
+
field["entry_bounding_box"],
|
|
65
|
+
image_width, image_height,
|
|
66
|
+
float(pdf_width), float(pdf_height)
|
|
67
|
+
)
|
|
68
|
+
|
|
69
|
+
if "entry_text" not in field or "text" not in field["entry_text"]:
|
|
70
|
+
continue
|
|
71
|
+
entry_text = field["entry_text"]
|
|
72
|
+
text = entry_text["text"]
|
|
73
|
+
if not text:
|
|
74
|
+
continue
|
|
75
|
+
|
|
76
|
+
font_name = entry_text.get("font", "Arial")
|
|
77
|
+
font_size = str(entry_text.get("font_size", 14)) + "pt"
|
|
78
|
+
font_color = entry_text.get("font_color", "000000")
|
|
79
|
+
|
|
80
|
+
annotation = FreeText(
|
|
81
|
+
text=text,
|
|
82
|
+
rect=transformed_entry_box,
|
|
83
|
+
font=font_name,
|
|
84
|
+
font_size=font_size,
|
|
85
|
+
font_color=font_color,
|
|
86
|
+
border_color=None,
|
|
87
|
+
background_color=None,
|
|
88
|
+
)
|
|
89
|
+
annotations.append(annotation)
|
|
90
|
+
writer.add_annotation(page_number=page_num - 1, annotation=annotation)
|
|
91
|
+
|
|
92
|
+
with open(output_pdf_path, "wb") as output:
|
|
93
|
+
writer.write(output)
|
|
94
|
+
|
|
95
|
+
print(f"Successfully filled PDF form and saved to {output_pdf_path}")
|
|
96
|
+
print(f"Added {len(annotations)} text annotations")
|
|
97
|
+
|
|
98
|
+
|
|
99
|
+
if __name__ == "__main__":
|
|
100
|
+
if len(sys.argv) != 4:
|
|
101
|
+
print("Usage: fill_pdf_form_with_annotations.py [input pdf] [fields.json] [output pdf]")
|
|
102
|
+
sys.exit(1)
|
|
103
|
+
input_pdf = sys.argv[1]
|
|
104
|
+
fields_json = sys.argv[2]
|
|
105
|
+
output_pdf = sys.argv[3]
|
|
106
|
+
|
|
107
|
+
fill_pdf_form(input_pdf, fields_json, output_pdf)
|