@nguyenphp/antigravity-marketing 1.0.16 → 1.0.19
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 +187 -74
- package/bin/index.js +4 -4
- package/package.json +4 -3
- package/templates/.agent/agents/backend-specialist.md +263 -0
- package/templates/.agent/agents/database-architect.md +226 -0
- package/templates/.agent/agents/debugger.md +225 -0
- package/templates/.agent/agents/devops-engineer.md +242 -0
- package/templates/.agent/agents/documentation-writer.md +104 -0
- package/templates/.agent/agents/explorer-agent.md +73 -0
- package/templates/.agent/agents/frontend-specialist.md +527 -0
- package/templates/.agent/agents/game-developer.md +162 -0
- package/templates/.agent/agents/mobile-developer.md +377 -0
- package/templates/.agent/agents/orchestrator.md +400 -0
- package/templates/.agent/agents/penetration-tester.md +188 -0
- package/templates/.agent/agents/performance-optimizer.md +187 -0
- package/templates/.agent/agents/project-planner.md +403 -0
- package/templates/.agent/agents/security-auditor.md +170 -0
- package/templates/.agent/agents/seo-specialist.md +111 -0
- package/templates/.agent/agents/test-engineer.md +158 -0
- package/templates/.agent/rules/GEMINI.md +248 -0
- package/templates/.agent/skills/analytics-marketing/SKILL.md +172 -324
- package/templates/.agent/skills/api-patterns/SKILL.md +81 -0
- package/templates/.agent/skills/api-patterns/api-style.md +42 -0
- package/templates/.agent/skills/api-patterns/auth.md +24 -0
- package/templates/.agent/skills/api-patterns/documentation.md +26 -0
- package/templates/.agent/skills/api-patterns/graphql.md +41 -0
- package/templates/.agent/skills/api-patterns/rate-limiting.md +31 -0
- package/templates/.agent/skills/api-patterns/response.md +37 -0
- package/templates/.agent/skills/api-patterns/rest.md +40 -0
- package/templates/.agent/skills/api-patterns/scripts/api_validator.py +211 -0
- package/templates/.agent/skills/api-patterns/security-testing.md +122 -0
- package/templates/.agent/skills/api-patterns/trpc.md +41 -0
- package/templates/.agent/skills/api-patterns/versioning.md +22 -0
- package/templates/.agent/skills/app-builder/SKILL.md +75 -0
- package/templates/.agent/skills/app-builder/agent-coordination.md +71 -0
- package/templates/.agent/skills/app-builder/feature-building.md +53 -0
- package/templates/.agent/skills/app-builder/project-detection.md +34 -0
- package/templates/.agent/skills/app-builder/scaffolding.md +118 -0
- package/templates/.agent/skills/app-builder/tech-stack.md +40 -0
- package/templates/.agent/skills/app-builder/templates/SKILL.md +39 -0
- package/templates/.agent/skills/app-builder/templates/astro-static/TEMPLATE.md +76 -0
- package/templates/.agent/skills/app-builder/templates/chrome-extension/TEMPLATE.md +92 -0
- package/templates/.agent/skills/app-builder/templates/cli-tool/TEMPLATE.md +88 -0
- package/templates/.agent/skills/app-builder/templates/electron-desktop/TEMPLATE.md +88 -0
- package/templates/.agent/skills/app-builder/templates/express-api/TEMPLATE.md +83 -0
- package/templates/.agent/skills/app-builder/templates/flutter-app/TEMPLATE.md +90 -0
- package/templates/.agent/skills/app-builder/templates/monorepo-turborepo/TEMPLATE.md +90 -0
- package/templates/.agent/skills/app-builder/templates/nextjs-fullstack/TEMPLATE.md +82 -0
- package/templates/.agent/skills/app-builder/templates/nextjs-saas/TEMPLATE.md +100 -0
- package/templates/.agent/skills/app-builder/templates/nextjs-static/TEMPLATE.md +106 -0
- package/templates/.agent/skills/app-builder/templates/nuxt-app/TEMPLATE.md +101 -0
- package/templates/.agent/skills/app-builder/templates/python-fastapi/TEMPLATE.md +83 -0
- package/templates/.agent/skills/app-builder/templates/react-native-app/TEMPLATE.md +93 -0
- package/templates/.agent/skills/architecture/SKILL.md +55 -0
- package/templates/.agent/skills/architecture/context-discovery.md +43 -0
- package/templates/.agent/skills/architecture/examples.md +94 -0
- package/templates/.agent/skills/architecture/pattern-selection.md +68 -0
- package/templates/.agent/skills/architecture/patterns-reference.md +50 -0
- package/templates/.agent/skills/architecture/trade-off-analysis.md +77 -0
- package/templates/.agent/skills/banner-design/SKILL.md +192 -0
- package/templates/.agent/skills/banner-design/references/banner-sizes-and-styles.md +118 -0
- package/templates/.agent/skills/bash-linux/SKILL.md +199 -0
- package/templates/.agent/skills/behavioral-modes/SKILL.md +242 -0
- package/templates/.agent/skills/brainstorming/SKILL.md +163 -0
- package/templates/.agent/skills/brainstorming/dynamic-questioning.md +350 -0
- package/templates/.agent/skills/brand/SKILL.md +97 -0
- package/templates/.agent/skills/brand/references/approval-checklist.md +169 -0
- package/templates/.agent/skills/brand/references/asset-organization.md +157 -0
- package/templates/.agent/skills/brand/references/brand-guideline-template.md +140 -0
- package/templates/.agent/skills/brand/references/color-palette-management.md +186 -0
- package/templates/.agent/skills/brand/references/consistency-checklist.md +94 -0
- package/templates/.agent/skills/brand/references/logo-usage-rules.md +185 -0
- package/templates/.agent/skills/brand/references/messaging-framework.md +85 -0
- package/templates/.agent/skills/brand/references/typography-specifications.md +214 -0
- package/templates/.agent/skills/brand/references/update.md +118 -0
- package/templates/.agent/skills/brand/references/visual-identity.md +96 -0
- package/templates/.agent/skills/brand/references/voice-framework.md +88 -0
- package/templates/.agent/skills/brand/scripts/extract-colors.cjs +341 -0
- package/templates/.agent/skills/brand/scripts/inject-brand-context.cjs +349 -0
- package/templates/.agent/skills/brand/scripts/sync-brand-to-tokens.cjs +266 -0
- package/templates/.agent/skills/brand/scripts/validate-asset.cjs +387 -0
- package/templates/.agent/skills/brand/templates/brand-guidelines-starter.md +275 -0
- package/templates/.agent/skills/clean-code/SKILL.md +201 -0
- package/templates/.agent/skills/code-review-checklist/SKILL.md +109 -0
- package/templates/.agent/skills/copywriting/SKILL.md +250 -0
- package/templates/.agent/skills/database-design/SKILL.md +52 -0
- package/templates/.agent/skills/database-design/database-selection.md +43 -0
- package/templates/.agent/skills/database-design/indexing.md +39 -0
- package/templates/.agent/skills/database-design/migrations.md +48 -0
- package/templates/.agent/skills/database-design/optimization.md +36 -0
- package/templates/.agent/skills/database-design/orm-selection.md +30 -0
- package/templates/.agent/skills/database-design/schema-design.md +56 -0
- package/templates/.agent/skills/database-design/scripts/schema_validator.py +172 -0
- package/templates/.agent/skills/deployment-procedures/SKILL.md +241 -0
- package/templates/.agent/skills/docker-expert/SKILL.md +409 -0
- package/templates/.agent/skills/frontend-design/animation-guide.md +331 -0
- package/templates/.agent/skills/frontend-design/color-system.md +311 -0
- package/templates/.agent/skills/frontend-design/decision-trees.md +418 -0
- package/templates/.agent/skills/frontend-design/motion-graphics.md +306 -0
- package/templates/.agent/skills/frontend-design/scripts/accessibility_checker.py +183 -0
- package/templates/.agent/skills/frontend-design/scripts/ux_audit.py +722 -0
- package/templates/.agent/skills/frontend-design/typography-system.md +345 -0
- package/templates/.agent/skills/frontend-design/ux-psychology.md +541 -0
- package/templates/.agent/skills/frontend-design/visual-effects.md +383 -0
- package/templates/.agent/skills/frontend-slides/SKILL.md +92 -0
- package/templates/.agent/skills/frontend-slides/STYLE_PRESETS.md +347 -0
- package/templates/.agent/skills/frontend-slides/animation-patterns.md +110 -0
- package/templates/.agent/skills/frontend-slides/examples/n8n-jupviec-automation.html +789 -0
- package/templates/.agent/skills/frontend-slides/examples/n8n-jupviec-automation.pptx +0 -0
- package/templates/.agent/skills/frontend-slides/html-template.md +347 -0
- package/templates/.agent/skills/frontend-slides/scripts/export-pptx.py +58 -0
- package/templates/.agent/skills/frontend-slides/scripts/extract-pptx.py +96 -0
- package/templates/.agent/skills/frontend-slides/viewport-base.css +153 -0
- package/templates/.agent/skills/game-development/2d-games/SKILL.md +119 -0
- package/templates/.agent/skills/game-development/3d-games/SKILL.md +135 -0
- package/templates/.agent/skills/game-development/SKILL.md +167 -0
- package/templates/.agent/skills/game-development/game-art/SKILL.md +185 -0
- package/templates/.agent/skills/game-development/game-audio/SKILL.md +190 -0
- package/templates/.agent/skills/game-development/game-design/SKILL.md +129 -0
- package/templates/.agent/skills/game-development/mobile-games/SKILL.md +108 -0
- package/templates/.agent/skills/game-development/multiplayer/SKILL.md +132 -0
- package/templates/.agent/skills/game-development/pc-games/SKILL.md +144 -0
- package/templates/.agent/skills/game-development/vr-ar/SKILL.md +123 -0
- package/templates/.agent/skills/game-development/web-games/SKILL.md +150 -0
- package/templates/.agent/skills/geo-fundamentals/SKILL.md +156 -0
- package/templates/.agent/skills/geo-fundamentals/scripts/geo_checker.py +289 -0
- package/templates/.agent/skills/growth-engine/SKILL.md +244 -0
- package/templates/.agent/skills/i18n-localization/SKILL.md +154 -0
- package/templates/.agent/skills/i18n-localization/scripts/i18n_checker.py +241 -0
- package/templates/.agent/skills/lint-and-validate/SKILL.md +45 -0
- package/templates/.agent/skills/lint-and-validate/scripts/lint_runner.py +172 -0
- package/templates/.agent/skills/lint-and-validate/scripts/type_coverage.py +173 -0
- package/templates/.agent/skills/marketing-report-expert/SKILL.md +70 -0
- package/templates/.agent/skills/mcp-builder/SKILL.md +176 -0
- package/templates/.agent/skills/minimax-docx/LICENSE +21 -0
- package/templates/.agent/skills/minimax-docx/SKILL.md +274 -0
- package/templates/.agent/skills/minimax-docx/assets/styles/academic_styles.xml +250 -0
- package/templates/.agent/skills/minimax-docx/assets/styles/corporate_styles.xml +284 -0
- package/templates/.agent/skills/minimax-docx/assets/styles/default_styles.xml +449 -0
- package/templates/.agent/skills/minimax-docx/assets/xsd/aesthetic-rules.xsd +470 -0
- package/templates/.agent/skills/minimax-docx/assets/xsd/business-rules.xsd +130 -0
- package/templates/.agent/skills/minimax-docx/assets/xsd/common-types.xsd +159 -0
- package/templates/.agent/skills/minimax-docx/assets/xsd/wml-subset.xsd +589 -0
- package/templates/.agent/skills/minimax-docx/references/cjk_typography.md +357 -0
- package/templates/.agent/skills/minimax-docx/references/cjk_university_template_guide.md +184 -0
- package/templates/.agent/skills/minimax-docx/references/comments_guide.md +191 -0
- package/templates/.agent/skills/minimax-docx/references/design_good_bad_examples.md +829 -0
- package/templates/.agent/skills/minimax-docx/references/design_principles.md +819 -0
- package/templates/.agent/skills/minimax-docx/references/openxml_element_order.md +308 -0
- package/templates/.agent/skills/minimax-docx/references/openxml_encyclopedia_part1.md +4061 -0
- package/templates/.agent/skills/minimax-docx/references/openxml_encyclopedia_part2.md +2820 -0
- package/templates/.agent/skills/minimax-docx/references/openxml_encyclopedia_part3.md +3381 -0
- package/templates/.agent/skills/minimax-docx/references/openxml_namespaces.md +82 -0
- package/templates/.agent/skills/minimax-docx/references/openxml_units.md +72 -0
- package/templates/.agent/skills/minimax-docx/references/scenario_a_create.md +284 -0
- package/templates/.agent/skills/minimax-docx/references/scenario_b_edit_content.md +295 -0
- package/templates/.agent/skills/minimax-docx/references/scenario_c_apply_template.md +456 -0
- package/templates/.agent/skills/minimax-docx/references/track_changes_guide.md +200 -0
- package/templates/.agent/skills/minimax-docx/references/troubleshooting.md +506 -0
- package/templates/.agent/skills/minimax-docx/references/typography_guide.md +294 -0
- package/templates/.agent/skills/minimax-docx/references/xsd_validation_guide.md +158 -0
- package/templates/.agent/skills/minimax-docx/scripts/doc_to_docx.sh +40 -0
- package/templates/.agent/skills/minimax-docx/scripts/docx_preview.sh +37 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Cli/MiniMaxAIDocx.Cli.csproj +19 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Cli/Program.cs +18 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/AnalyzeCommand.cs +147 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/ApplyTemplateCommand.cs +322 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/CreateCommand.cs +324 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/DiffCommand.cs +155 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/EditContentCommand.cs +487 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/FixOrderCommand.cs +108 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/MergeRunsCommand.cs +122 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Commands/ValidateCommand.cs +107 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/MiniMaxAIDocx.Core.csproj +15 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/OpenXml/CommentSynchronizer.cs +169 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/OpenXml/ElementOrder.cs +80 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/OpenXml/NamespaceConstants.cs +42 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/OpenXml/RunMerger.cs +81 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/OpenXml/StyleAnalyzer.cs +81 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/OpenXml/TrackChangesHelper.cs +99 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/OpenXml/UnitConverter.cs +23 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/AestheticRecipeSamples.cs +1832 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/AestheticRecipeSamples_Batch1.cs +910 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/AestheticRecipeSamples_Batch2.cs +999 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/AestheticRecipeSamples_Batch3.cs +1048 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/AestheticRecipeSamples_Batch4.cs +1038 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/CharacterFormattingSamples.cs +1020 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/DocumentCreationSamples.cs +1121 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/FieldAndTocSamples.cs +624 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/FootnoteAndCommentSamples.cs +675 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/HeaderFooterSamples.cs +838 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/ImageSamples.cs +917 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/ListAndNumberingSamples.cs +826 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/ParagraphFormattingSamples.cs +1199 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/StyleSystemSamples.cs +1487 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/TableSamples.cs +1163 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Samples/TrackChangesSamples.cs +595 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Typography/CjkHelper.cs +39 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Typography/FontDefaults.cs +24 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Typography/PageSizes.cs +20 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Validation/BusinessRuleValidator.cs +224 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Validation/GateCheckValidator.cs +148 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Validation/ValidationResult.cs +23 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.Core/Validation/XsdValidator.cs +69 -0
- package/templates/.agent/skills/minimax-docx/scripts/dotnet/MiniMaxAIDocx.slnx +4 -0
- package/templates/.agent/skills/minimax-docx/scripts/env_check.sh +196 -0
- package/templates/.agent/skills/minimax-docx/scripts/setup.ps1 +274 -0
- package/templates/.agent/skills/minimax-docx/scripts/setup.sh +504 -0
- package/templates/.agent/skills/minimax-multimodal-toolkit/SKILL.md +359 -0
- package/templates/.agent/skills/minimax-pdf/README.md +222 -0
- package/templates/.agent/skills/minimax-pdf/SKILL.md +201 -0
- package/templates/.agent/skills/minimax-pdf/design/design.md +381 -0
- package/templates/.agent/skills/minimax-pdf/scripts/cover.py +1579 -0
- package/templates/.agent/skills/minimax-pdf/scripts/fill_inspect.py +200 -0
- package/templates/.agent/skills/minimax-pdf/scripts/fill_write.py +242 -0
- package/templates/.agent/skills/minimax-pdf/scripts/make.sh +491 -0
- package/templates/.agent/skills/minimax-pdf/scripts/merge.py +112 -0
- package/templates/.agent/skills/minimax-pdf/scripts/palette.py +559 -0
- package/templates/.agent/skills/minimax-pdf/scripts/reformat_parse.py +374 -0
- package/templates/.agent/skills/minimax-pdf/scripts/render_body.py +1055 -0
- package/templates/.agent/skills/minimax-pdf/scripts/render_cover.cjs +111 -0
- package/templates/.agent/skills/minimax-xlsx/SKILL.md +138 -0
- package/templates/.agent/skills/minimax-xlsx/references/create.md +691 -0
- package/templates/.agent/skills/minimax-xlsx/references/edit.md +684 -0
- package/templates/.agent/skills/minimax-xlsx/references/fix.md +37 -0
- package/templates/.agent/skills/minimax-xlsx/references/format.md +768 -0
- package/templates/.agent/skills/minimax-xlsx/references/ooxml-cheatsheet.md +231 -0
- package/templates/.agent/skills/minimax-xlsx/references/read-analyze.md +97 -0
- package/templates/.agent/skills/minimax-xlsx/references/validate.md +772 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/formula_check.py +422 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/libreoffice_recalc.py +248 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/shared_strings_builder.py +163 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/style_audit.py +575 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/xlsx_add_column.py +395 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/xlsx_insert_row.py +274 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/xlsx_pack.py +87 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/xlsx_reader.py +362 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/xlsx_shift_rows.py +396 -0
- package/templates/.agent/skills/minimax-xlsx/scripts/xlsx_unpack.py +130 -0
- package/templates/.agent/skills/minimax-xlsx/templates/minimal_xlsx/[Content_Types].xml +9 -0
- package/templates/.agent/skills/minimax-xlsx/templates/minimal_xlsx/_rels/.rels +6 -0
- package/templates/.agent/skills/minimax-xlsx/templates/minimal_xlsx/xl/_rels/workbook.xml.rels +19 -0
- package/templates/.agent/skills/minimax-xlsx/templates/minimal_xlsx/xl/sharedStrings.xml +33 -0
- package/templates/.agent/skills/minimax-xlsx/templates/minimal_xlsx/xl/styles.xml +160 -0
- package/templates/.agent/skills/minimax-xlsx/templates/minimal_xlsx/xl/workbook.xml +30 -0
- package/templates/.agent/skills/minimax-xlsx/templates/minimal_xlsx/xl/worksheets/sheet1.xml +70 -0
- package/templates/.agent/skills/mobile-design/SKILL.md +394 -0
- package/templates/.agent/skills/mobile-design/decision-trees.md +516 -0
- package/templates/.agent/skills/mobile-design/mobile-backend.md +491 -0
- package/templates/.agent/skills/mobile-design/mobile-color-system.md +420 -0
- package/templates/.agent/skills/mobile-design/mobile-debugging.md +122 -0
- package/templates/.agent/skills/mobile-design/mobile-design-thinking.md +357 -0
- package/templates/.agent/skills/mobile-design/mobile-navigation.md +458 -0
- package/templates/.agent/skills/mobile-design/mobile-performance.md +767 -0
- package/templates/.agent/skills/mobile-design/mobile-testing.md +356 -0
- package/templates/.agent/skills/mobile-design/mobile-typography.md +433 -0
- package/templates/.agent/skills/mobile-design/platform-android.md +666 -0
- package/templates/.agent/skills/mobile-design/platform-ios.md +561 -0
- package/templates/.agent/skills/mobile-design/scripts/mobile_audit.py +670 -0
- package/templates/.agent/skills/mobile-design/touch-psychology.md +537 -0
- package/templates/.agent/skills/nestjs-expert/SKILL.md +552 -0
- package/templates/.agent/skills/nextjs-best-practices/SKILL.md +203 -0
- package/templates/.agent/skills/nodejs-best-practices/SKILL.md +333 -0
- package/templates/.agent/skills/parallel-agents/SKILL.md +175 -0
- package/templates/.agent/skills/performance-profiling/SKILL.md +143 -0
- package/templates/.agent/skills/performance-profiling/scripts/lighthouse_audit.py +76 -0
- package/templates/.agent/skills/plan-writing/SKILL.md +152 -0
- package/templates/.agent/skills/powershell-windows/SKILL.md +167 -0
- package/templates/.agent/skills/ppc-advertising/SKILL.md +183 -475
- package/templates/.agent/skills/pptx-generator/SKILL.md +249 -0
- package/templates/.agent/skills/pptx-generator/references/design-system.md +392 -0
- package/templates/.agent/skills/pptx-generator/references/editing.md +162 -0
- package/templates/.agent/skills/pptx-generator/references/pitfalls.md +112 -0
- package/templates/.agent/skills/pptx-generator/references/pptxgenjs.md +420 -0
- package/templates/.agent/skills/pptx-generator/references/slide-types.md +413 -0
- package/templates/.agent/skills/prisma-expert/SKILL.md +355 -0
- package/templates/.agent/skills/python-patterns/SKILL.md +441 -0
- package/templates/.agent/skills/react-patterns/SKILL.md +198 -0
- package/templates/.agent/skills/red-team-tactics/SKILL.md +199 -0
- package/templates/.agent/skills/remotion-best-practices/SKILL.md +45 -111
- package/templates/.agent/skills/remotion-best-practices/rules/3d.md +4 -4
- package/templates/.agent/skills/remotion-best-practices/rules/animations.md +5 -7
- package/templates/.agent/skills/remotion-best-practices/rules/assets/charts-bar-chart.tsx +173 -0
- package/templates/.agent/skills/remotion-best-practices/rules/assets/text-animations-typewriter.tsx +100 -0
- package/templates/.agent/skills/remotion-best-practices/rules/assets/text-animations-word-highlight.tsx +103 -0
- package/templates/.agent/skills/remotion-best-practices/rules/assets.md +78 -0
- package/templates/.agent/skills/remotion-best-practices/rules/audio-visualization.md +198 -0
- package/templates/.agent/skills/remotion-best-practices/rules/audio.md +1 -4
- package/templates/.agent/skills/remotion-best-practices/rules/calculate-metadata.md +47 -17
- package/templates/.agent/skills/remotion-best-practices/rules/can-decode.md +75 -0
- package/templates/.agent/skills/remotion-best-practices/rules/charts.md +80 -48
- package/templates/.agent/skills/remotion-best-practices/rules/compositions.md +22 -14
- package/templates/.agent/skills/remotion-best-practices/rules/display-captions.md +79 -21
- package/templates/.agent/skills/remotion-best-practices/rules/extract-frames.md +229 -0
- package/templates/.agent/skills/remotion-best-practices/rules/ffmpeg.md +38 -0
- package/templates/.agent/skills/remotion-best-practices/rules/fonts.md +96 -54
- package/templates/.agent/skills/remotion-best-practices/rules/get-audio-duration.md +58 -0
- package/templates/.agent/skills/remotion-best-practices/rules/get-video-dimensions.md +68 -0
- package/templates/.agent/skills/remotion-best-practices/rules/get-video-duration.md +60 -0
- package/templates/.agent/skills/remotion-best-practices/rules/gifs.md +21 -18
- package/templates/.agent/skills/remotion-best-practices/rules/images.md +6 -2
- package/templates/.agent/skills/remotion-best-practices/rules/import-srt-captions.md +69 -0
- package/templates/.agent/skills/remotion-best-practices/rules/light-leaks.md +73 -0
- package/templates/.agent/skills/remotion-best-practices/rules/lottie.md +10 -7
- package/templates/.agent/skills/remotion-best-practices/rules/maps.md +412 -0
- package/templates/.agent/skills/remotion-best-practices/rules/measuring-dom-nodes.md +34 -0
- package/templates/.agent/skills/remotion-best-practices/rules/measuring-text.md +140 -0
- package/templates/.agent/skills/remotion-best-practices/rules/parameters.md +109 -0
- package/templates/.agent/skills/remotion-best-practices/rules/sequencing.md +13 -1
- package/templates/.agent/skills/remotion-best-practices/rules/sfx.md +26 -0
- package/templates/.agent/skills/remotion-best-practices/rules/subtitles.md +36 -0
- package/templates/.agent/skills/remotion-best-practices/rules/tailwind.md +11 -0
- package/templates/.agent/skills/remotion-best-practices/rules/text-animations.md +4 -115
- package/templates/.agent/skills/remotion-best-practices/rules/timing.md +19 -19
- package/templates/.agent/skills/remotion-best-practices/rules/transcribe-captions.md +70 -0
- package/templates/.agent/skills/remotion-best-practices/rules/transitions.md +117 -42
- package/templates/.agent/skills/remotion-best-practices/rules/transparent-videos.md +106 -0
- package/templates/.agent/skills/remotion-best-practices/rules/trimming.md +51 -0
- package/templates/.agent/skills/remotion-best-practices/rules/voiceover.md +99 -0
- package/templates/.agent/skills/seo-fundamentals/SKILL.md +83 -441
- package/templates/.agent/skills/seo-fundamentals/scripts/seo_checker.py +219 -0
- package/templates/.agent/skills/server-management/SKILL.md +161 -0
- package/templates/.agent/skills/systematic-debugging/SKILL.md +109 -0
- package/templates/.agent/skills/tdd-workflow/SKILL.md +149 -0
- package/templates/.agent/skills/testing-patterns/SKILL.md +178 -0
- package/templates/.agent/skills/testing-patterns/scripts/test_runner.py +219 -0
- package/templates/.agent/skills/tutorial-video-expert/SKILL.md +88 -0
- package/templates/.agent/skills/typescript-expert/SKILL.md +429 -0
- package/templates/.agent/skills/ui-ux-pro-max/SKILL.md +1 -1
- package/templates/.agent/skills/ui-ux-pro-max/data/charts.csv +26 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/colors.csv +97 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/icons.csv +101 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/landing.csv +31 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/products.csv +97 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/prompts.csv +24 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/react-performance.csv +45 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/flutter.csv +53 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/html-tailwind.csv +56 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/nextjs.csv +53 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/nuxt-ui.csv +51 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/nuxtjs.csv +59 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/react-native.csv +52 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/react.csv +54 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/shadcn.csv +61 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/svelte.csv +54 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/swiftui.csv +51 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/stacks/vue.csv +50 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/styles.csv +59 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/typography.csv +58 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/ui-reasoning.csv +101 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/ux-guidelines.csv +100 -0
- package/templates/.agent/skills/ui-ux-pro-max/data/web-interface.csv +31 -0
- package/templates/.agent/skills/ui-ux-pro-max/scripts/core.py +257 -0
- package/templates/.agent/skills/ui-ux-pro-max/scripts/design_system.py +487 -0
- package/templates/.agent/skills/ui-ux-pro-max/scripts/search.py +76 -0
- package/templates/.agent/skills/vision-analysis/SKILL.md +174 -0
- package/templates/.agent/skills/vue-expert/SKILL.md +374 -0
- package/templates/.agent/skills/vulnerability-scanner/SKILL.md +276 -0
- package/templates/.agent/skills/vulnerability-scanner/checklists.md +121 -0
- package/templates/.agent/skills/vulnerability-scanner/scripts/security_scan.py +458 -0
- package/templates/.agent/skills/webapp-testing/SKILL.md +187 -0
- package/templates/.agent/skills/webapp-testing/scripts/playwright_runner.py +173 -0
- package/templates/.agent/workflows/analyze.md +3 -0
- package/templates/.agent/workflows/brainstorm.md +113 -0
- package/templates/.agent/workflows/brand-report.md +44 -0
- package/templates/.agent/workflows/create.md +59 -0
- package/templates/.agent/workflows/debug.md +103 -0
- package/templates/.agent/workflows/deploy.md +176 -0
- package/templates/.agent/workflows/enhance.md +63 -0
- package/templates/.agent/workflows/orchestrate.md +237 -0
- package/templates/.agent/workflows/plan.md +89 -0
- package/templates/.agent/workflows/preview.md +80 -0
- package/templates/.agent/workflows/report.md +49 -0
- package/templates/.agent/workflows/status.md +86 -0
- package/templates/.agent/workflows/test.md +144 -0
- package/templates/.agent/workflows/ui-ux-pro-max.md +231 -0
|
@@ -0,0 +1,200 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
fill_inspect.py — Inspect form fields in an existing PDF.
|
|
4
|
+
|
|
5
|
+
Usage:
|
|
6
|
+
python3 fill_inspect.py --input form.pdf
|
|
7
|
+
python3 fill_inspect.py --input form.pdf --out fields.json
|
|
8
|
+
|
|
9
|
+
Outputs a JSON summary of every fillable field: name, type, current value,
|
|
10
|
+
allowed values (for checkboxes / dropdowns), and page number.
|
|
11
|
+
|
|
12
|
+
Exit codes: 0 success, 1 bad args / file not found, 2 dep missing, 3 read error
|
|
13
|
+
"""
|
|
14
|
+
|
|
15
|
+
import argparse
|
|
16
|
+
import json
|
|
17
|
+
import sys
|
|
18
|
+
import importlib.util
|
|
19
|
+
import os
|
|
20
|
+
|
|
21
|
+
|
|
22
|
+
|
|
23
|
+
|
|
24
|
+
def ensure_deps():
|
|
25
|
+
if importlib.util.find_spec("pypdf") is None:
|
|
26
|
+
import subprocess
|
|
27
|
+
subprocess.check_call(
|
|
28
|
+
[sys.executable, "-m", "pip", "install", "--break-system-packages", "-q", "pypdf"]
|
|
29
|
+
)
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
ensure_deps()
|
|
33
|
+
from pypdf import PdfReader
|
|
34
|
+
from pypdf.generic import ArrayObject, DictionaryObject, NameObject, TextStringObject
|
|
35
|
+
|
|
36
|
+
|
|
37
|
+
# ── Field type resolution ──────────────────────────────────────────────────────
|
|
38
|
+
def _field_type(field) -> str:
|
|
39
|
+
ft = field.get("/FT")
|
|
40
|
+
if ft is None:
|
|
41
|
+
return "unknown"
|
|
42
|
+
ft = str(ft)
|
|
43
|
+
if ft == "/Tx":
|
|
44
|
+
return "text"
|
|
45
|
+
if ft == "/Btn":
|
|
46
|
+
ff = int(field.get("/Ff", 0))
|
|
47
|
+
return "radio" if ff & (1 << 15) else "checkbox"
|
|
48
|
+
if ft == "/Ch":
|
|
49
|
+
ff = int(field.get("/Ff", 0))
|
|
50
|
+
return "dropdown" if ff & (1 << 17) else "listbox"
|
|
51
|
+
if ft == "/Sig":
|
|
52
|
+
return "signature"
|
|
53
|
+
return "unknown"
|
|
54
|
+
|
|
55
|
+
|
|
56
|
+
def _field_value(field) -> str | None:
|
|
57
|
+
v = field.get("/V")
|
|
58
|
+
return str(v) if v is not None else None
|
|
59
|
+
|
|
60
|
+
|
|
61
|
+
def _field_options(field, ftype: str) -> dict:
|
|
62
|
+
extra = {}
|
|
63
|
+
if ftype in ("checkbox",):
|
|
64
|
+
ap = field.get("/AP")
|
|
65
|
+
if ap and "/N" in ap:
|
|
66
|
+
states = [str(k) for k in ap["/N"]]
|
|
67
|
+
extra["states"] = states
|
|
68
|
+
checked = next((s for s in states if s != "/Off"), None)
|
|
69
|
+
if checked:
|
|
70
|
+
extra["checked_value"] = checked
|
|
71
|
+
if ftype in ("dropdown", "listbox"):
|
|
72
|
+
opt = field.get("/Opt")
|
|
73
|
+
if opt:
|
|
74
|
+
choices = []
|
|
75
|
+
for item in opt:
|
|
76
|
+
if isinstance(item, (list, ArrayObject)) and len(item) >= 2:
|
|
77
|
+
choices.append({"value": str(item[0]), "label": str(item[1])})
|
|
78
|
+
else:
|
|
79
|
+
choices.append({"value": str(item), "label": str(item)})
|
|
80
|
+
extra["choices"] = choices
|
|
81
|
+
if ftype == "radio":
|
|
82
|
+
kids = field.get("/Kids")
|
|
83
|
+
if kids:
|
|
84
|
+
values = []
|
|
85
|
+
for kid in kids:
|
|
86
|
+
ap = kid.get("/AP")
|
|
87
|
+
if ap and "/N" in ap:
|
|
88
|
+
for k in ap["/N"]:
|
|
89
|
+
if str(k) != "/Off":
|
|
90
|
+
values.append(str(k))
|
|
91
|
+
extra["radio_values"] = values
|
|
92
|
+
return extra
|
|
93
|
+
|
|
94
|
+
|
|
95
|
+
def _walk_fields(fields, page_map: dict, parent_name: str = "") -> list:
|
|
96
|
+
"""Recursively collect all leaf fields."""
|
|
97
|
+
result = []
|
|
98
|
+
for field in fields:
|
|
99
|
+
name = str(field.get("/T", ""))
|
|
100
|
+
full = f"{parent_name}.{name}" if parent_name else name
|
|
101
|
+
|
|
102
|
+
kids = field.get("/Kids")
|
|
103
|
+
# Kids that have /T are sub-fields (groups), not widget annotations
|
|
104
|
+
if kids:
|
|
105
|
+
named_kids = [k for k in kids if "/T" in k]
|
|
106
|
+
if named_kids:
|
|
107
|
+
result.extend(_walk_fields(named_kids, page_map, full))
|
|
108
|
+
continue
|
|
109
|
+
|
|
110
|
+
ftype = _field_type(field)
|
|
111
|
+
if ftype == "unknown":
|
|
112
|
+
continue
|
|
113
|
+
|
|
114
|
+
entry = {
|
|
115
|
+
"name": full,
|
|
116
|
+
"type": ftype,
|
|
117
|
+
"value": _field_value(field),
|
|
118
|
+
}
|
|
119
|
+
entry.update(_field_options(field, ftype))
|
|
120
|
+
|
|
121
|
+
# Page lookup via /P indirect reference
|
|
122
|
+
p_ref = field.get("/P")
|
|
123
|
+
if p_ref and hasattr(p_ref, "idnum"):
|
|
124
|
+
entry["page"] = page_map.get(p_ref.idnum, "?")
|
|
125
|
+
|
|
126
|
+
result.append(entry)
|
|
127
|
+
return result
|
|
128
|
+
|
|
129
|
+
|
|
130
|
+
def inspect(pdf_path: str) -> dict:
|
|
131
|
+
try:
|
|
132
|
+
reader = PdfReader(pdf_path)
|
|
133
|
+
except Exception as e:
|
|
134
|
+
return {"status": "error", "error": str(e)}
|
|
135
|
+
|
|
136
|
+
# Build page-number lookup: {object_id: 1-based page number}
|
|
137
|
+
page_map = {}
|
|
138
|
+
for i, page in enumerate(reader.pages):
|
|
139
|
+
if hasattr(page, "indirect_reference") and page.indirect_reference:
|
|
140
|
+
page_map[page.indirect_reference.idnum] = i + 1
|
|
141
|
+
|
|
142
|
+
acroform = reader.trailer.get("/Root", {}).get("/AcroForm")
|
|
143
|
+
if acroform is None or "/Fields" not in acroform:
|
|
144
|
+
return {
|
|
145
|
+
"status": "ok",
|
|
146
|
+
"has_fields": False,
|
|
147
|
+
"field_count": 0,
|
|
148
|
+
"fields": [],
|
|
149
|
+
"note": "This PDF has no fillable form fields.",
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
fields = _walk_fields(list(acroform["/Fields"]), page_map)
|
|
153
|
+
|
|
154
|
+
return {
|
|
155
|
+
"status": "ok",
|
|
156
|
+
"has_fields": bool(fields),
|
|
157
|
+
"field_count": len(fields),
|
|
158
|
+
"fields": fields,
|
|
159
|
+
}
|
|
160
|
+
|
|
161
|
+
|
|
162
|
+
def main():
|
|
163
|
+
parser = argparse.ArgumentParser(description="Inspect PDF form fields")
|
|
164
|
+
parser.add_argument("--input", required=True, help="PDF file to inspect")
|
|
165
|
+
parser.add_argument("--out", default="", help="Write JSON to file (optional)")
|
|
166
|
+
args = parser.parse_args()
|
|
167
|
+
|
|
168
|
+
if not os.path.exists(args.input):
|
|
169
|
+
print(json.dumps({"status": "error", "error": f"File not found: {args.input}"}),
|
|
170
|
+
file=sys.stderr)
|
|
171
|
+
sys.exit(1)
|
|
172
|
+
|
|
173
|
+
result = inspect(args.input)
|
|
174
|
+
|
|
175
|
+
output = json.dumps(result, indent=2, ensure_ascii=False)
|
|
176
|
+
|
|
177
|
+
if args.out:
|
|
178
|
+
with open(args.out, "w") as f:
|
|
179
|
+
f.write(output)
|
|
180
|
+
|
|
181
|
+
print(output)
|
|
182
|
+
|
|
183
|
+
# Human-readable summary
|
|
184
|
+
if result["status"] == "ok" and result["has_fields"]:
|
|
185
|
+
print(f"\n── Fields in {args.input} ──────────────────────────────",
|
|
186
|
+
file=sys.stderr)
|
|
187
|
+
for f in result["fields"]:
|
|
188
|
+
pg = f" p.{f['page']}" if "page" in f else ""
|
|
189
|
+
val = f" = {f['value']}" if f.get("value") else ""
|
|
190
|
+
extra = ""
|
|
191
|
+
if "choices" in f:
|
|
192
|
+
extra = f" [{', '.join(c['value'] for c in f['choices'][:4])}{'…' if len(f['choices'])>4 else ''}]"
|
|
193
|
+
elif "states" in f:
|
|
194
|
+
extra = f" {f['states']}"
|
|
195
|
+
print(f" {f['type']:12} {f['name']}{pg}{val}{extra}", file=sys.stderr)
|
|
196
|
+
print("", file=sys.stderr)
|
|
197
|
+
|
|
198
|
+
|
|
199
|
+
if __name__ == "__main__":
|
|
200
|
+
main()
|
|
@@ -0,0 +1,242 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
fill_write.py — Write values into PDF form fields.
|
|
4
|
+
|
|
5
|
+
Usage:
|
|
6
|
+
# From a JSON data file
|
|
7
|
+
python3 fill_write.py --input form.pdf --data values.json --out filled.pdf
|
|
8
|
+
|
|
9
|
+
# Inline JSON
|
|
10
|
+
python3 fill_write.py --input form.pdf --out filled.pdf \
|
|
11
|
+
--values '{"FirstName": "Jane", "Agree": "true"}'
|
|
12
|
+
|
|
13
|
+
values format:
|
|
14
|
+
{
|
|
15
|
+
"FieldName": "text value", # text field
|
|
16
|
+
"CheckBox1": "true", # checkbox (true / false)
|
|
17
|
+
"Dropdown1": "OptionValue", # dropdown (must match an existing choice value)
|
|
18
|
+
"Radio1": "/Choice2" # radio (must match a radio value)
|
|
19
|
+
}
|
|
20
|
+
|
|
21
|
+
Exit codes: 0 success, 1 bad args, 2 dep missing, 3 read/write error, 4 validation error
|
|
22
|
+
"""
|
|
23
|
+
|
|
24
|
+
import argparse
|
|
25
|
+
import json
|
|
26
|
+
import os
|
|
27
|
+
import sys
|
|
28
|
+
import importlib.util
|
|
29
|
+
|
|
30
|
+
|
|
31
|
+
|
|
32
|
+
|
|
33
|
+
def ensure_deps():
|
|
34
|
+
if importlib.util.find_spec("pypdf") is None:
|
|
35
|
+
import subprocess
|
|
36
|
+
subprocess.check_call(
|
|
37
|
+
[sys.executable, "-m", "pip", "install", "--break-system-packages", "-q", "pypdf"]
|
|
38
|
+
)
|
|
39
|
+
|
|
40
|
+
|
|
41
|
+
ensure_deps()
|
|
42
|
+
from pypdf import PdfReader, PdfWriter
|
|
43
|
+
from pypdf.generic import NameObject, TextStringObject, BooleanObject
|
|
44
|
+
|
|
45
|
+
|
|
46
|
+
# ── Field helpers ─────────────────────────────────────────────────────────────
|
|
47
|
+
def _field_type(field) -> str:
|
|
48
|
+
ft = str(field.get("/FT", ""))
|
|
49
|
+
if ft == "/Tx": return "text"
|
|
50
|
+
if ft == "/Btn":
|
|
51
|
+
ff = int(field.get("/Ff", 0))
|
|
52
|
+
return "radio" if ff & (1 << 15) else "checkbox"
|
|
53
|
+
if ft == "/Ch":
|
|
54
|
+
ff = int(field.get("/Ff", 0))
|
|
55
|
+
return "dropdown" if ff & (1 << 17) else "listbox"
|
|
56
|
+
return "unknown"
|
|
57
|
+
|
|
58
|
+
|
|
59
|
+
def _get_checkbox_on_value(field) -> str:
|
|
60
|
+
"""Return the /AP /N key that means 'checked' (anything except /Off)."""
|
|
61
|
+
ap = field.get("/AP")
|
|
62
|
+
if ap and "/N" in ap:
|
|
63
|
+
for k in ap["/N"]:
|
|
64
|
+
if str(k) != "/Off":
|
|
65
|
+
return str(k)
|
|
66
|
+
return "/Yes"
|
|
67
|
+
|
|
68
|
+
|
|
69
|
+
def _get_dropdown_values(field) -> list[str]:
|
|
70
|
+
opt = field.get("/Opt")
|
|
71
|
+
if not opt:
|
|
72
|
+
return []
|
|
73
|
+
values = []
|
|
74
|
+
for item in opt:
|
|
75
|
+
try:
|
|
76
|
+
from pypdf.generic import ArrayObject
|
|
77
|
+
if isinstance(item, (list, ArrayObject)) and len(item) >= 1:
|
|
78
|
+
values.append(str(item[0]))
|
|
79
|
+
else:
|
|
80
|
+
values.append(str(item))
|
|
81
|
+
except Exception:
|
|
82
|
+
values.append(str(item))
|
|
83
|
+
return values
|
|
84
|
+
|
|
85
|
+
|
|
86
|
+
# ── Walk + fill ───────────────────────────────────────────────────────────────
|
|
87
|
+
def _walk_and_fill(fields, data: dict, filled: list, errors: list, parent: str = ""):
|
|
88
|
+
for field in fields:
|
|
89
|
+
name = str(field.get("/T", ""))
|
|
90
|
+
full = f"{parent}.{name}" if parent else name
|
|
91
|
+
|
|
92
|
+
# Recurse into named groups
|
|
93
|
+
kids = field.get("/Kids")
|
|
94
|
+
if kids:
|
|
95
|
+
named = [k for k in kids if "/T" in k]
|
|
96
|
+
if named:
|
|
97
|
+
_walk_and_fill(named, data, filled, errors, full)
|
|
98
|
+
continue
|
|
99
|
+
|
|
100
|
+
if full not in data:
|
|
101
|
+
continue
|
|
102
|
+
|
|
103
|
+
value = data[full]
|
|
104
|
+
ftype = _field_type(field)
|
|
105
|
+
|
|
106
|
+
if ftype == "text":
|
|
107
|
+
field.update({
|
|
108
|
+
NameObject("/V"): TextStringObject(str(value)),
|
|
109
|
+
NameObject("/DV"): TextStringObject(str(value)),
|
|
110
|
+
})
|
|
111
|
+
filled.append(full)
|
|
112
|
+
|
|
113
|
+
elif ftype == "checkbox":
|
|
114
|
+
truthy = str(value).lower() in ("true", "1", "yes", "on")
|
|
115
|
+
on_val = _get_checkbox_on_value(field)
|
|
116
|
+
pdf_val = on_val if truthy else "/Off"
|
|
117
|
+
field.update({
|
|
118
|
+
NameObject("/V"): NameObject(pdf_val),
|
|
119
|
+
NameObject("/AS"): NameObject(pdf_val),
|
|
120
|
+
})
|
|
121
|
+
filled.append(full)
|
|
122
|
+
|
|
123
|
+
elif ftype in ("dropdown", "listbox"):
|
|
124
|
+
allowed = _get_dropdown_values(field)
|
|
125
|
+
if allowed and str(value) not in allowed:
|
|
126
|
+
errors.append({
|
|
127
|
+
"field": full,
|
|
128
|
+
"error": f"Value '{value}' not in allowed choices: {allowed}"
|
|
129
|
+
})
|
|
130
|
+
continue
|
|
131
|
+
field.update({NameObject("/V"): TextStringObject(str(value))})
|
|
132
|
+
filled.append(full)
|
|
133
|
+
|
|
134
|
+
elif ftype == "radio":
|
|
135
|
+
# Radio value must start with /
|
|
136
|
+
pdf_val = str(value) if str(value).startswith("/") else f"/{value}"
|
|
137
|
+
field.update({
|
|
138
|
+
NameObject("/V"): NameObject(pdf_val),
|
|
139
|
+
NameObject("/AS"): NameObject(pdf_val),
|
|
140
|
+
})
|
|
141
|
+
filled.append(full)
|
|
142
|
+
|
|
143
|
+
else:
|
|
144
|
+
errors.append({"field": full, "error": f"Unsupported field type: {ftype}"})
|
|
145
|
+
|
|
146
|
+
|
|
147
|
+
def fill(pdf_path: str, out_path: str, data: dict) -> dict:
|
|
148
|
+
try:
|
|
149
|
+
reader = PdfReader(pdf_path)
|
|
150
|
+
except Exception as e:
|
|
151
|
+
return {"status": "error", "error": str(e)}
|
|
152
|
+
|
|
153
|
+
writer = PdfWriter()
|
|
154
|
+
writer.clone_document_from_reader(reader)
|
|
155
|
+
|
|
156
|
+
acroform = writer._root_object.get("/AcroForm") # type: ignore[attr-defined]
|
|
157
|
+
if acroform is None or "/Fields" not in acroform:
|
|
158
|
+
return {
|
|
159
|
+
"status": "error",
|
|
160
|
+
"error": "This PDF has no fillable form fields.",
|
|
161
|
+
"hint": "Run fill_inspect.py first to confirm the PDF has fields.",
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
# Enable appearance regeneration so viewers show the new values
|
|
165
|
+
acroform.update({NameObject("/NeedAppearances"): BooleanObject(True)})
|
|
166
|
+
|
|
167
|
+
filled: list[str] = []
|
|
168
|
+
errors: list[dict] = []
|
|
169
|
+
_walk_and_fill(list(acroform["/Fields"]), data, filled, errors)
|
|
170
|
+
|
|
171
|
+
# Warn about requested fields that were never found
|
|
172
|
+
not_found = [k for k in data if k not in filled and not any(e["field"] == k for e in errors)]
|
|
173
|
+
|
|
174
|
+
try:
|
|
175
|
+
os.makedirs(os.path.dirname(os.path.abspath(out_path)), exist_ok=True)
|
|
176
|
+
with open(out_path, "wb") as f:
|
|
177
|
+
writer.write(f)
|
|
178
|
+
except Exception as e:
|
|
179
|
+
return {"status": "error", "error": f"Write failed: {e}"}
|
|
180
|
+
|
|
181
|
+
result = {
|
|
182
|
+
"status": "ok",
|
|
183
|
+
"out": out_path,
|
|
184
|
+
"filled_count": len(filled),
|
|
185
|
+
"filled_fields": filled,
|
|
186
|
+
"size_kb": os.path.getsize(out_path) // 1024,
|
|
187
|
+
}
|
|
188
|
+
if errors:
|
|
189
|
+
result["validation_errors"] = errors
|
|
190
|
+
if not_found:
|
|
191
|
+
result["not_found"] = not_found
|
|
192
|
+
result["hint"] = "Run fill_inspect.py to see all available field names."
|
|
193
|
+
return result
|
|
194
|
+
|
|
195
|
+
|
|
196
|
+
def main():
|
|
197
|
+
parser = argparse.ArgumentParser(description="Fill PDF form fields")
|
|
198
|
+
parser.add_argument("--input", required=True, help="Input PDF with form fields")
|
|
199
|
+
parser.add_argument("--out", required=True, help="Output PDF path")
|
|
200
|
+
group = parser.add_mutually_exclusive_group(required=True)
|
|
201
|
+
group.add_argument("--data", help="Path to JSON file with field values")
|
|
202
|
+
group.add_argument("--values", help="Inline JSON string with field values")
|
|
203
|
+
args = parser.parse_args()
|
|
204
|
+
|
|
205
|
+
if not os.path.exists(args.input):
|
|
206
|
+
print(json.dumps({"status": "error", "error": f"File not found: {args.input}"}),
|
|
207
|
+
file=sys.stderr)
|
|
208
|
+
sys.exit(1)
|
|
209
|
+
|
|
210
|
+
# Load data
|
|
211
|
+
try:
|
|
212
|
+
if args.data:
|
|
213
|
+
with open(args.data) as f:
|
|
214
|
+
data = json.load(f)
|
|
215
|
+
else:
|
|
216
|
+
data = json.loads(args.values)
|
|
217
|
+
except Exception as e:
|
|
218
|
+
print(json.dumps({"status": "error", "error": f"JSON parse error: {e}"}),
|
|
219
|
+
file=sys.stderr)
|
|
220
|
+
sys.exit(1)
|
|
221
|
+
|
|
222
|
+
result = fill(args.input, args.out, data)
|
|
223
|
+
print(json.dumps(result, indent=2, ensure_ascii=False))
|
|
224
|
+
|
|
225
|
+
if result["status"] == "ok":
|
|
226
|
+
print(f"\n── Fill complete ───────────────────────────────────────",
|
|
227
|
+
file=sys.stderr)
|
|
228
|
+
print(f" Output : {result['out']}", file=sys.stderr)
|
|
229
|
+
print(f" Filled : {result['filled_count']} field(s)", file=sys.stderr)
|
|
230
|
+
if result.get("validation_errors"):
|
|
231
|
+
print(f" Errors :", file=sys.stderr)
|
|
232
|
+
for e in result["validation_errors"]:
|
|
233
|
+
print(f" • {e['field']}: {e['error']}", file=sys.stderr)
|
|
234
|
+
if result.get("not_found"):
|
|
235
|
+
print(f" Not found: {result['not_found']}", file=sys.stderr)
|
|
236
|
+
print("", file=sys.stderr)
|
|
237
|
+
else:
|
|
238
|
+
sys.exit(3)
|
|
239
|
+
|
|
240
|
+
|
|
241
|
+
if __name__ == "__main__":
|
|
242
|
+
main()
|