@vodailoc/kilo-kit-mcp 1.1.0 → 1.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/.mcp/kilo-kit.codex-windows.toml +5 -0
- package/LICENSE +190 -190
- package/QUICKSTART.md +265 -255
- package/README.md +290 -266
- package/mcp/README.md +29 -5
- package/mcp/dist/server.js +1 -1
- package/mcp/package.json +1 -2
- package/package.json +3 -2
- package/skills/README.md +647 -647
- package/skills/SKILLS_INDEX.md +139 -139
- package/skills/ai-media/ai-multimodal/.env.example +97 -97
- package/skills/ai-media/ai-multimodal/SKILL.md +357 -357
- package/skills/ai-media/ai-multimodal/references/audio-processing.md +373 -373
- package/skills/ai-media/ai-multimodal/references/image-generation.md +558 -558
- package/skills/ai-media/ai-multimodal/references/video-analysis.md +502 -502
- package/skills/ai-media/ai-multimodal/references/vision-understanding.md +483 -483
- package/skills/ai-media/ai-multimodal/scripts/document_converter.py +395 -395
- package/skills/ai-media/ai-multimodal/scripts/gemini_batch_process.py +480 -480
- package/skills/ai-media/ai-multimodal/scripts/media_optimizer.py +506 -506
- package/skills/ai-media/ai-multimodal/scripts/requirements.txt +26 -26
- package/skills/ai-media/ai-multimodal/scripts/tests/requirements.txt +20 -20
- package/skills/ai-media/ai-multimodal/scripts/tests/test_document_converter.py +299 -299
- package/skills/ai-media/ai-multimodal/scripts/tests/test_gemini_batch_process.py +362 -362
- package/skills/ai-media/ai-multimodal/scripts/tests/test_media_optimizer.py +373 -373
- package/skills/ai-media/media-processing/SKILL.md +358 -358
- package/skills/ai-media/media-processing/references/ffmpeg-encoding.md +358 -358
- package/skills/ai-media/media-processing/references/ffmpeg-filters.md +503 -503
- package/skills/ai-media/media-processing/references/ffmpeg-streaming.md +403 -403
- package/skills/ai-media/media-processing/references/format-compatibility.md +375 -375
- package/skills/ai-media/media-processing/references/imagemagick-batch.md +612 -612
- package/skills/ai-media/media-processing/references/imagemagick-editing.md +623 -623
- package/skills/ai-media/media-processing/scripts/batch_resize.py +342 -342
- package/skills/ai-media/media-processing/scripts/media_convert.py +311 -311
- package/skills/ai-media/media-processing/scripts/requirements.txt +24 -24
- package/skills/ai-media/media-processing/scripts/tests/requirements.txt +2 -2
- package/skills/ai-media/media-processing/scripts/tests/test_batch_resize.py +372 -372
- package/skills/ai-media/media-processing/scripts/tests/test_media_convert.py +259 -259
- package/skills/ai-media/media-processing/scripts/tests/test_video_optimize.py +397 -397
- package/skills/ai-media/media-processing/scripts/video_optimize.py +414 -414
- package/skills/ai-media/screenshot/LICENSE.txt +201 -201
- package/skills/ai-media/screenshot/SKILL.md +267 -267
- package/skills/ai-media/screenshot/agents/openai.yaml +6 -6
- package/skills/ai-media/screenshot/assets/screenshot-small.svg +5 -5
- package/skills/ai-media/screenshot/scripts/ensure_macos_permissions.sh +54 -54
- package/skills/ai-media/screenshot/scripts/macos_display_info.swift +22 -22
- package/skills/ai-media/screenshot/scripts/macos_permissions.swift +40 -40
- package/skills/ai-media/screenshot/scripts/macos_window_info.swift +126 -126
- package/skills/ai-media/screenshot/scripts/take_screenshot.ps1 +163 -163
- package/skills/ai-media/screenshot/scripts/take_screenshot.py +585 -585
- package/skills/ai-media/sora/LICENSE.txt +201 -201
- package/skills/ai-media/sora/SKILL.md +153 -153
- package/skills/ai-media/sora/agents/openai.yaml +6 -6
- package/skills/ai-media/sora/assets/sora-small.svg +4 -4
- package/skills/ai-media/sora/references/cinematic-shots.md +53 -53
- package/skills/ai-media/sora/references/cli.md +248 -248
- package/skills/ai-media/sora/references/codex-network.md +28 -28
- package/skills/ai-media/sora/references/prompting.md +137 -137
- package/skills/ai-media/sora/references/sample-prompts.md +95 -95
- package/skills/ai-media/sora/references/social-ads.md +42 -42
- package/skills/ai-media/sora/references/troubleshooting.md +58 -58
- package/skills/ai-media/sora/references/video-api.md +45 -45
- package/skills/ai-media/sora/scripts/sora.py +970 -970
- package/skills/design/aesthetic/SKILL.md +121 -121
- package/skills/design/aesthetic/assets/design-guideline-template.md +163 -163
- package/skills/design/aesthetic/assets/design-story-template.md +135 -135
- package/skills/design/aesthetic/references/design-principles.md +62 -62
- package/skills/design/aesthetic/references/design-resources.md +75 -75
- package/skills/design/aesthetic/references/micro-interactions.md +53 -53
- package/skills/design/aesthetic/references/storytelling-design.md +50 -50
- package/skills/design/figma/LICENSE.txt +202 -202
- package/skills/design/figma/SKILL.md +42 -42
- package/skills/design/figma/agents/openai.yaml +14 -14
- package/skills/design/figma/assets/figma-small.svg +3 -3
- package/skills/design/figma/assets/icon.svg +28 -28
- package/skills/design/figma/references/figma-mcp-config.md +35 -35
- package/skills/design/figma/references/figma-tools-and-prompts.md +34 -34
- package/skills/design/figma-implement-design/LICENSE.txt +202 -202
- package/skills/design/figma-implement-design/SKILL.md +264 -264
- package/skills/design/figma-implement-design/agents/openai.yaml +14 -14
- package/skills/design/figma-implement-design/assets/figma-small.svg +3 -3
- package/skills/design/figma-implement-design/assets/icon.svg +28 -28
- package/skills/design/frontend-design/SKILL.md +41 -41
- package/skills/design/frontend-design/references/animejs.md +395 -395
- package/skills/design/ui-styling/LICENSE.txt +201 -201
- package/skills/design/ui-styling/SKILL.md +321 -321
- package/skills/design/ui-styling/canvas-fonts/ArsenalSC-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/BigShoulders-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Boldonse-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/BricolageGrotesque-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/CrimsonPro-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/DMMono-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/EricaOne-OFL.txt +94 -94
- package/skills/design/ui-styling/canvas-fonts/GeistMono-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Gloock-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/IBMPlexMono-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/InstrumentSans-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Italiana-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/JetBrainsMono-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Jura-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/LibreBaskerville-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Lora-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/NationalPark-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/NothingYouCouldDo-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Outfit-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/PixelifySans-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/PoiretOne-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/RedHatMono-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Silkscreen-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/SmoochSans-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/Tektur-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/WorkSans-OFL.txt +93 -93
- package/skills/design/ui-styling/canvas-fonts/YoungSerif-OFL.txt +93 -93
- package/skills/design/ui-styling/references/canvas-design-system.md +320 -320
- package/skills/design/ui-styling/references/shadcn-accessibility.md +471 -471
- package/skills/design/ui-styling/references/shadcn-components.md +424 -424
- package/skills/design/ui-styling/references/shadcn-theming.md +373 -373
- package/skills/design/ui-styling/references/tailwind-customization.md +483 -483
- package/skills/design/ui-styling/references/tailwind-responsive.md +382 -382
- package/skills/design/ui-styling/references/tailwind-utilities.md +455 -455
- package/skills/design/ui-styling/scripts/requirements.txt +17 -17
- package/skills/design/ui-styling/scripts/shadcn_add.py +292 -292
- package/skills/design/ui-styling/scripts/tailwind_config_gen.py +456 -456
- package/skills/design/ui-styling/scripts/tests/requirements.txt +3 -3
- package/skills/design/ui-styling/scripts/tests/test_shadcn_add.py +266 -266
- package/skills/design/ui-styling/scripts/tests/test_tailwind_config_gen.py +336 -336
- package/skills/engineering/aspnet-core/LICENSE.txt +201 -201
- package/skills/engineering/aspnet-core/SKILL.md +61 -61
- package/skills/engineering/aspnet-core/agents/openai.yaml +5 -5
- package/skills/engineering/aspnet-core/references/_sections.md +40 -40
- package/skills/engineering/aspnet-core/references/apis-minimal-and-controllers.md +81 -81
- package/skills/engineering/aspnet-core/references/data-state-and-services.md +69 -69
- package/skills/engineering/aspnet-core/references/program-and-pipeline.md +103 -103
- package/skills/engineering/aspnet-core/references/realtime-grpc-and-background-work.md +58 -58
- package/skills/engineering/aspnet-core/references/security-and-identity.md +75 -75
- package/skills/engineering/aspnet-core/references/source-map.md +43 -43
- package/skills/engineering/aspnet-core/references/stack-selection.md +63 -63
- package/skills/engineering/aspnet-core/references/testing-performance-and-operations.md +92 -92
- package/skills/engineering/aspnet-core/references/ui-blazor.md +53 -53
- package/skills/engineering/aspnet-core/references/ui-mvc.md +56 -56
- package/skills/engineering/aspnet-core/references/ui-razor-pages.md +55 -55
- package/skills/engineering/aspnet-core/references/versioning-and-upgrades.md +51 -51
- package/skills/engineering/backend-development/SKILL.md +95 -95
- package/skills/engineering/backend-development/references/backend-api-design.md +495 -495
- package/skills/engineering/backend-development/references/backend-architecture.md +454 -454
- package/skills/engineering/backend-development/references/backend-authentication.md +338 -338
- package/skills/engineering/backend-development/references/backend-code-quality.md +659 -659
- package/skills/engineering/backend-development/references/backend-debugging.md +904 -904
- package/skills/engineering/backend-development/references/backend-devops.md +494 -494
- package/skills/engineering/backend-development/references/backend-mindset.md +387 -387
- package/skills/engineering/backend-development/references/backend-performance.md +397 -397
- package/skills/engineering/backend-development/references/backend-security.md +290 -290
- package/skills/engineering/backend-development/references/backend-technologies.md +256 -256
- package/skills/engineering/backend-development/references/backend-testing.md +429 -429
- package/skills/engineering/better-auth/SKILL.md +204 -204
- package/skills/engineering/better-auth/references/advanced-features.md +553 -553
- package/skills/engineering/better-auth/references/database-integration.md +577 -577
- package/skills/engineering/better-auth/references/email-password-auth.md +416 -416
- package/skills/engineering/better-auth/references/oauth-providers.md +430 -430
- package/skills/engineering/better-auth/scripts/better_auth_init.py +521 -521
- package/skills/engineering/better-auth/scripts/requirements.txt +15 -15
- package/skills/engineering/better-auth/scripts/tests/test_better_auth_init.py +421 -421
- package/skills/engineering/code-review/SKILL.md +140 -140
- package/skills/engineering/code-review/references/code-review-reception.md +208 -208
- package/skills/engineering/code-review/references/requesting-code-review.md +104 -104
- package/skills/engineering/code-review/references/verification-before-completion.md +138 -138
- package/skills/engineering/context-engineering/SKILL.md +86 -86
- package/skills/engineering/context-engineering/references/context-compression.md +84 -84
- package/skills/engineering/context-engineering/references/context-degradation.md +93 -93
- package/skills/engineering/context-engineering/references/context-fundamentals.md +75 -75
- package/skills/engineering/context-engineering/references/context-optimization.md +82 -82
- package/skills/engineering/context-engineering/references/evaluation.md +89 -89
- package/skills/engineering/context-engineering/references/memory-systems.md +88 -88
- package/skills/engineering/context-engineering/references/multi-agent-patterns.md +90 -90
- package/skills/engineering/context-engineering/references/project-development.md +97 -97
- package/skills/engineering/context-engineering/references/tool-design.md +86 -86
- package/skills/engineering/context-engineering/scripts/compression_evaluator.py +329 -329
- package/skills/engineering/context-engineering/scripts/context_analyzer.py +294 -294
- package/skills/engineering/databases/SKILL.md +232 -232
- package/skills/engineering/databases/references/mongodb-aggregation.md +447 -447
- package/skills/engineering/databases/references/mongodb-atlas.md +465 -465
- package/skills/engineering/databases/references/mongodb-crud.md +408 -408
- package/skills/engineering/databases/references/mongodb-indexing.md +442 -442
- package/skills/engineering/databases/references/postgresql-administration.md +594 -594
- package/skills/engineering/databases/references/postgresql-performance.md +527 -527
- package/skills/engineering/databases/references/postgresql-psql-cli.md +467 -467
- package/skills/engineering/databases/references/postgresql-queries.md +475 -475
- package/skills/engineering/databases/scripts/db_backup.py +502 -502
- package/skills/engineering/databases/scripts/db_migrate.py +414 -414
- package/skills/engineering/databases/scripts/db_performance_check.py +444 -444
- package/skills/engineering/databases/scripts/requirements.txt +20 -20
- package/skills/engineering/databases/scripts/tests/requirements.txt +4 -4
- package/skills/engineering/databases/scripts/tests/test_db_backup.py +340 -340
- package/skills/engineering/databases/scripts/tests/test_db_migrate.py +277 -277
- package/skills/engineering/databases/scripts/tests/test_db_performance_check.py +370 -370
- package/skills/engineering/diagnose/SKILL.md +117 -117
- package/skills/engineering/diagnose/scripts/hitl-loop.template.sh +41 -41
- package/skills/engineering/docs-seeker/SKILL.md +207 -207
- package/skills/engineering/docs-seeker/WORKFLOWS.md +505 -505
- package/skills/engineering/docs-seeker/references/best-practices.md +632 -632
- package/skills/engineering/docs-seeker/references/documentation-sources.md +461 -461
- package/skills/engineering/docs-seeker/references/error-handling.md +621 -621
- package/skills/engineering/docs-seeker/references/limitations.md +821 -821
- package/skills/engineering/docs-seeker/references/performance.md +574 -574
- package/skills/engineering/docs-seeker/references/tool-selection.md +262 -262
- package/skills/engineering/frontend-development/SKILL.md +398 -398
- package/skills/engineering/frontend-development/resources/common-patterns.md +330 -330
- package/skills/engineering/frontend-development/resources/complete-examples.md +871 -871
- package/skills/engineering/frontend-development/resources/component-patterns.md +501 -501
- package/skills/engineering/frontend-development/resources/data-fetching.md +766 -766
- package/skills/engineering/frontend-development/resources/file-organization.md +501 -501
- package/skills/engineering/frontend-development/resources/loading-and-error-states.md +500 -500
- package/skills/engineering/frontend-development/resources/performance.md +405 -405
- package/skills/engineering/frontend-development/resources/routing-guide.md +363 -363
- package/skills/engineering/frontend-development/resources/styling-guide.md +427 -427
- package/skills/engineering/frontend-development/resources/typescript-standards.md +417 -417
- package/skills/engineering/improve-codebase-architecture/DEEPENING.md +37 -37
- package/skills/engineering/improve-codebase-architecture/INTERFACE-DESIGN.md +44 -44
- package/skills/engineering/improve-codebase-architecture/LANGUAGE.md +53 -53
- package/skills/engineering/improve-codebase-architecture/SKILL.md +71 -71
- package/skills/engineering/openai-docs/LICENSE.txt +201 -201
- package/skills/engineering/openai-docs/SKILL.md +69 -69
- package/skills/engineering/openai-docs/agents/openai.yaml +14 -14
- package/skills/engineering/openai-docs/assets/openai-small.svg +3 -3
- package/skills/engineering/openai-docs/references/gpt-5p4-prompting-guide.md +433 -433
- package/skills/engineering/openai-docs/references/latest-model.md +35 -35
- package/skills/engineering/openai-docs/references/upgrading-to-gpt-5p4.md +164 -164
- package/skills/engineering/playwright/LICENSE.txt +201 -201
- package/skills/engineering/playwright/NOTICE.txt +14 -14
- package/skills/engineering/playwright/SKILL.md +147 -147
- package/skills/engineering/playwright/agents/openai.yaml +6 -6
- package/skills/engineering/playwright/assets/playwright-small.svg +3 -3
- package/skills/engineering/playwright/references/cli.md +116 -116
- package/skills/engineering/playwright/references/workflows.md +95 -95
- package/skills/engineering/playwright/scripts/playwright_cli.sh +25 -25
- package/skills/engineering/playwright-interactive/LICENSE.txt +201 -201
- package/skills/engineering/playwright-interactive/NOTICE.txt +13 -13
- package/skills/engineering/playwright-interactive/SKILL.md +689 -689
- package/skills/engineering/playwright-interactive/agents/openai.yaml +6 -6
- package/skills/engineering/playwright-interactive/assets/playwright-small.svg +3 -3
- package/skills/engineering/render-deploy/LICENSE.txt +201 -201
- package/skills/engineering/render-deploy/SKILL.md +479 -479
- package/skills/engineering/render-deploy/agents/openai.yaml +14 -14
- package/skills/engineering/render-deploy/assets/docker.yaml +62 -62
- package/skills/engineering/render-deploy/assets/go-api.yaml +35 -35
- package/skills/engineering/render-deploy/assets/nextjs-postgres.yaml +35 -35
- package/skills/engineering/render-deploy/assets/node-express.yaml +25 -25
- package/skills/engineering/render-deploy/assets/python-django.yaml +89 -89
- package/skills/engineering/render-deploy/assets/render-small.svg +3 -3
- package/skills/engineering/render-deploy/assets/static-site.yaml +54 -54
- package/skills/engineering/render-deploy/references/blueprint-spec.md +718 -718
- package/skills/engineering/render-deploy/references/codebase-analysis.md +49 -49
- package/skills/engineering/render-deploy/references/configuration-guide.md +603 -603
- package/skills/engineering/render-deploy/references/deployment-details.md +224 -224
- package/skills/engineering/render-deploy/references/direct-creation.md +113 -113
- package/skills/engineering/render-deploy/references/error-patterns.md +13 -13
- package/skills/engineering/render-deploy/references/post-deploy-checks.md +36 -36
- package/skills/engineering/render-deploy/references/runtimes.md +473 -473
- package/skills/engineering/render-deploy/references/service-types.md +450 -450
- package/skills/engineering/render-deploy/references/troubleshooting-basics.md +36 -36
- package/skills/engineering/repomix/SKILL.md +215 -215
- package/skills/engineering/repomix/references/configuration.md +211 -211
- package/skills/engineering/repomix/references/usage-patterns.md +232 -232
- package/skills/engineering/repomix/scripts/README.md +179 -179
- package/skills/engineering/repomix/scripts/repomix_batch.py +455 -455
- package/skills/engineering/repomix/scripts/repos.example.json +15 -15
- package/skills/engineering/repomix/scripts/requirements.txt +15 -15
- package/skills/engineering/repomix/scripts/tests/test_repomix_batch.py +531 -531
- package/skills/engineering/setup-matt-pocock-skills/SKILL.md +121 -121
- package/skills/engineering/setup-matt-pocock-skills/domain.md +51 -51
- package/skills/engineering/setup-matt-pocock-skills/issue-tracker-github.md +22 -22
- package/skills/engineering/setup-matt-pocock-skills/issue-tracker-gitlab.md +23 -23
- package/skills/engineering/setup-matt-pocock-skills/issue-tracker-local.md +19 -19
- package/skills/engineering/setup-matt-pocock-skills/triage-labels.md +15 -15
- package/skills/engineering/shopify/README.md +66 -66
- package/skills/engineering/shopify/SKILL.md +319 -319
- package/skills/engineering/shopify/references/app-development.md +470 -470
- package/skills/engineering/shopify/references/extensions.md +493 -493
- package/skills/engineering/shopify/references/themes.md +498 -498
- package/skills/engineering/shopify/scripts/requirements.txt +19 -19
- package/skills/engineering/shopify/scripts/shopify_init.py +423 -423
- package/skills/engineering/shopify/scripts/tests/test_shopify_init.py +385 -385
- package/skills/engineering/tdd/SKILL.md +109 -109
- package/skills/engineering/tdd/deep-modules.md +33 -33
- package/skills/engineering/tdd/interface-design.md +31 -31
- package/skills/engineering/tdd/mocking.md +59 -59
- package/skills/engineering/tdd/refactoring.md +10 -10
- package/skills/engineering/tdd/tests.md +61 -61
- package/skills/engineering/to-issues/SKILL.md +81 -81
- package/skills/engineering/to-prd/SKILL.md +74 -74
- package/skills/engineering/triage/AGENT-BRIEF.md +168 -168
- package/skills/engineering/triage/OUT-OF-SCOPE.md +101 -101
- package/skills/engineering/triage/SKILL.md +103 -103
- package/skills/engineering/web-frameworks/SKILL.md +324 -324
- package/skills/engineering/web-frameworks/references/nextjs-app-router.md +465 -465
- package/skills/engineering/web-frameworks/references/nextjs-data-fetching.md +459 -459
- package/skills/engineering/web-frameworks/references/nextjs-optimization.md +511 -511
- package/skills/engineering/web-frameworks/references/nextjs-server-components.md +495 -495
- package/skills/engineering/web-frameworks/references/remix-icon-integration.md +603 -603
- package/skills/engineering/web-frameworks/references/turborepo-caching.md +551 -551
- package/skills/engineering/web-frameworks/references/turborepo-pipelines.md +517 -517
- package/skills/engineering/web-frameworks/references/turborepo-setup.md +542 -542
- package/skills/engineering/web-frameworks/scripts/nextjs_init.py +547 -547
- package/skills/engineering/web-frameworks/scripts/requirements.txt +16 -16
- package/skills/engineering/web-frameworks/scripts/tests/requirements.txt +3 -3
- package/skills/engineering/web-frameworks/scripts/tests/test_nextjs_init.py +319 -319
- package/skills/engineering/web-frameworks/scripts/tests/test_turborepo_migrate.py +374 -374
- package/skills/engineering/web-frameworks/scripts/turborepo_migrate.py +394 -394
- package/skills/engineering/write-a-skill/SKILL.md +117 -117
- package/skills/kilo-kit/SKILL.md +346 -346
- package/skills/kilo-kit/_template/SKILL.md +185 -185
- package/skills/kilo-kit/debugging/root-cause/SKILL.md +360 -360
- package/skills/kilo-kit/debugging/systematic/SKILL.md +339 -339
- package/skills/kilo-kit/debugging/verification/SKILL.md +424 -424
- package/skills/kilo-kit/development/backend/SKILL.md +540 -540
- package/skills/kilo-kit/development/security/SKILL.md +529 -529
- package/skills/kilo-kit/quality/code-review/SKILL.md +297 -297
- package/skills/kilo-kit/quality/testing/SKILL.md +540 -540
- package/skills/kilo-kit/references/output-formats.md +204 -204
- package/skills/kilo-kit/references/patterns.md +156 -156
- package/skills/kilo-kit/references/performance-benchmarks.md +90 -90
- package/skills/operations/chrome-devtools/SKILL.md +392 -392
- package/skills/operations/chrome-devtools/references/cdp-domains.md +694 -694
- package/skills/operations/chrome-devtools/references/performance-guide.md +940 -940
- package/skills/operations/chrome-devtools/references/puppeteer-reference.md +953 -953
- package/skills/operations/chrome-devtools/scripts/PERSISTENT-BROWSER.md +107 -107
- package/skills/operations/chrome-devtools/scripts/README.md +213 -213
- package/skills/operations/chrome-devtools/scripts/__tests__/selector.test.js +210 -210
- package/skills/operations/chrome-devtools/scripts/click.js +79 -79
- package/skills/operations/chrome-devtools/scripts/close-persistent.js +36 -36
- package/skills/operations/chrome-devtools/scripts/console.js +75 -75
- package/skills/operations/chrome-devtools/scripts/evaluate.js +49 -49
- package/skills/operations/chrome-devtools/scripts/fill.js +72 -72
- package/skills/operations/chrome-devtools/scripts/install-deps.sh +181 -181
- package/skills/operations/chrome-devtools/scripts/install.sh +83 -83
- package/skills/operations/chrome-devtools/scripts/launch-persistent.js +71 -71
- package/skills/operations/chrome-devtools/scripts/lib/browser.js +144 -144
- package/skills/operations/chrome-devtools/scripts/lib/selector.js +178 -178
- package/skills/operations/chrome-devtools/scripts/navigate.js +46 -46
- package/skills/operations/chrome-devtools/scripts/network.js +102 -102
- package/skills/operations/chrome-devtools/scripts/package-lock.json +1206 -1206
- package/skills/operations/chrome-devtools/scripts/package.json +15 -15
- package/skills/operations/chrome-devtools/scripts/performance.js +145 -145
- package/skills/operations/chrome-devtools/scripts/screenshot.js +180 -180
- package/skills/operations/chrome-devtools/scripts/snapshot.js +131 -131
- package/skills/operations/devops/.env.example +76 -76
- package/skills/operations/devops/SKILL.md +285 -285
- package/skills/operations/devops/references/browser-rendering.md +305 -305
- package/skills/operations/devops/references/cloudflare-d1-kv.md +123 -123
- package/skills/operations/devops/references/cloudflare-platform.md +271 -271
- package/skills/operations/devops/references/cloudflare-r2-storage.md +280 -280
- package/skills/operations/devops/references/cloudflare-workers-advanced.md +312 -312
- package/skills/operations/devops/references/cloudflare-workers-apis.md +309 -309
- package/skills/operations/devops/references/cloudflare-workers-basics.md +418 -418
- package/skills/operations/devops/references/docker-basics.md +297 -297
- package/skills/operations/devops/references/docker-compose.md +292 -292
- package/skills/operations/devops/references/gcloud-platform.md +297 -297
- package/skills/operations/devops/references/gcloud-services.md +304 -304
- package/skills/operations/devops/scripts/cloudflare_deploy.py +269 -269
- package/skills/operations/devops/scripts/docker_optimize.py +320 -320
- package/skills/operations/devops/scripts/requirements.txt +20 -20
- package/skills/operations/devops/scripts/tests/requirements.txt +3 -3
- package/skills/operations/devops/scripts/tests/test_cloudflare_deploy.py +285 -285
- package/skills/operations/devops/scripts/tests/test_docker_optimize.py +436 -436
- package/skills/operations/mcp-builder/LICENSE.txt +201 -201
- package/skills/operations/mcp-builder/SKILL.md +328 -328
- package/skills/operations/mcp-builder/reference/evaluation.md +601 -601
- package/skills/operations/mcp-builder/reference/mcp_best_practices.md +915 -915
- package/skills/operations/mcp-builder/reference/node_mcp_server.md +915 -915
- package/skills/operations/mcp-builder/reference/python_mcp_server.md +751 -751
- package/skills/operations/mcp-builder/scripts/connections.py +151 -151
- package/skills/operations/mcp-builder/scripts/evaluation.py +373 -373
- package/skills/operations/mcp-builder/scripts/example_evaluation.xml +22 -22
- package/skills/operations/mcp-builder/scripts/requirements.txt +2 -2
- package/skills/operations/mcp-management/README.md +219 -219
- package/skills/operations/mcp-management/SKILL.md +175 -175
- package/skills/operations/mcp-management/assets/tools.json +3043 -3043
- package/skills/operations/mcp-management/references/configuration.md +114 -114
- package/skills/operations/mcp-management/references/gemini-cli-integration.md +201 -201
- package/skills/operations/mcp-management/references/mcp-protocol.md +116 -116
- package/skills/operations/mcp-management/scripts/.env.example +10 -10
- package/skills/operations/mcp-management/scripts/cli.ts +155 -155
- package/skills/operations/mcp-management/scripts/dist/analyze-tools.js +70 -70
- package/skills/operations/mcp-management/scripts/dist/cli.js +131 -131
- package/skills/operations/mcp-management/scripts/dist/mcp-client.js +115 -115
- package/skills/operations/mcp-management/scripts/mcp-client.ts +163 -163
- package/skills/operations/mcp-management/scripts/package.json +18 -18
- package/skills/operations/mcp-management/scripts/tsconfig.json +15 -15
- package/skills/problem-solving/collision-zone-thinking/SKILL.md +62 -62
- package/skills/problem-solving/defense-in-depth/SKILL.md +130 -130
- package/skills/problem-solving/inversion-exercise/SKILL.md +58 -58
- package/skills/problem-solving/meta-pattern-recognition/SKILL.md +54 -54
- package/skills/problem-solving/root-cause-tracing/SKILL.md +177 -177
- package/skills/problem-solving/root-cause-tracing/find-polluter.sh +63 -63
- package/skills/problem-solving/scale-game/SKILL.md +63 -63
- package/skills/problem-solving/sequential-thinking/README.md +118 -118
- package/skills/problem-solving/sequential-thinking/SKILL.md +93 -93
- package/skills/problem-solving/sequential-thinking/references/advanced.md +122 -122
- package/skills/problem-solving/sequential-thinking/references/examples.md +274 -274
- package/skills/problem-solving/simplification-cascades/SKILL.md +76 -76
- package/skills/problem-solving/when-stuck/SKILL.md +88 -88
- package/skills/productivity/caveman/SKILL.md +49 -49
- package/skills/productivity/grill-me/SKILL.md +10 -10
- package/skills/productivity/grill-with-docs/ADR-FORMAT.md +47 -47
- package/skills/productivity/grill-with-docs/CONTEXT-FORMAT.md +77 -77
- package/skills/productivity/grill-with-docs/SKILL.md +88 -88
- package/skills/productivity/writing-skills/graphviz-conventions.dot +171 -171
- package/skills/productivity/zoom-out/SKILL.md +7 -7
- package/skills/writing-docs/doc/LICENSE.txt +201 -201
- package/skills/writing-docs/doc/SKILL.md +80 -80
- package/skills/writing-docs/doc/agents/openai.yaml +6 -6
- package/skills/writing-docs/doc/assets/doc-small.svg +3 -3
- package/skills/writing-docs/doc/scripts/render_docx.py +296 -296
- package/skills/writing-docs/docx/LICENSE.txt +30 -30
- package/skills/writing-docs/docx/SKILL.md +196 -196
- package/skills/writing-docs/docx/docx-js.md +349 -349
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -1499
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -146
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -1085
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -11
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -3081
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -23
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -185
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -287
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -1676
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -28
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -144
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -174
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -25
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -18
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -59
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -56
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -195
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -582
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -25
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -4439
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -570
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -509
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -12
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -108
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -96
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -3646
- package/skills/writing-docs/docx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -116
- package/skills/writing-docs/docx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -42
- package/skills/writing-docs/docx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -50
- package/skills/writing-docs/docx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -49
- package/skills/writing-docs/docx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -33
- package/skills/writing-docs/docx/ooxml/schemas/mce/mc.xsd +75 -75
- package/skills/writing-docs/docx/ooxml/schemas/microsoft/wml-2010.xsd +560 -560
- package/skills/writing-docs/docx/ooxml/schemas/microsoft/wml-2012.xsd +67 -67
- package/skills/writing-docs/docx/ooxml/schemas/microsoft/wml-2018.xsd +14 -14
- package/skills/writing-docs/docx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -20
- package/skills/writing-docs/docx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -13
- package/skills/writing-docs/docx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -4
- package/skills/writing-docs/docx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -8
- package/skills/writing-docs/docx/ooxml/scripts/pack.py +159 -159
- package/skills/writing-docs/docx/ooxml/scripts/unpack.py +29 -29
- package/skills/writing-docs/docx/ooxml/scripts/validate.py +69 -69
- package/skills/writing-docs/docx/ooxml/scripts/validation/__init__.py +15 -15
- package/skills/writing-docs/docx/ooxml/scripts/validation/base.py +951 -951
- package/skills/writing-docs/docx/ooxml/scripts/validation/docx.py +274 -274
- package/skills/writing-docs/docx/ooxml/scripts/validation/pptx.py +315 -315
- package/skills/writing-docs/docx/ooxml/scripts/validation/redlining.py +279 -279
- package/skills/writing-docs/docx/ooxml.md +609 -609
- package/skills/writing-docs/docx/scripts/__init__.py +1 -1
- package/skills/writing-docs/docx/scripts/document.py +1276 -1276
- package/skills/writing-docs/docx/scripts/templates/comments.xml +2 -2
- package/skills/writing-docs/docx/scripts/templates/commentsExtended.xml +2 -2
- package/skills/writing-docs/docx/scripts/templates/commentsExtensible.xml +2 -2
- package/skills/writing-docs/docx/scripts/templates/commentsIds.xml +2 -2
- package/skills/writing-docs/docx/scripts/templates/people.xml +2 -2
- package/skills/writing-docs/docx/scripts/utilities.py +374 -374
- package/skills/writing-docs/mermaidjs-v11/SKILL.md +115 -115
- package/skills/writing-docs/mermaidjs-v11/references/cli-usage.md +228 -228
- package/skills/writing-docs/mermaidjs-v11/references/configuration.md +232 -232
- package/skills/writing-docs/mermaidjs-v11/references/diagram-types.md +315 -315
- package/skills/writing-docs/mermaidjs-v11/references/examples.md +344 -344
- package/skills/writing-docs/mermaidjs-v11/references/integration.md +310 -310
- package/skills/writing-docs/pdf/LICENSE.txt +30 -30
- package/skills/writing-docs/pdf/SKILL.md +294 -294
- package/skills/writing-docs/pdf/forms.md +205 -205
- package/skills/writing-docs/pdf/reference.md +611 -611
- package/skills/writing-docs/pdf/scripts/check_bounding_boxes.py +70 -70
- package/skills/writing-docs/pdf/scripts/check_bounding_boxes_test.py +226 -226
- package/skills/writing-docs/pdf/scripts/check_fillable_fields.py +12 -12
- package/skills/writing-docs/pdf/scripts/convert_pdf_to_images.py +35 -35
- package/skills/writing-docs/pdf/scripts/create_validation_image.py +41 -41
- package/skills/writing-docs/pdf/scripts/extract_form_field_info.py +152 -152
- package/skills/writing-docs/pdf/scripts/fill_fillable_fields.py +114 -114
- package/skills/writing-docs/pdf/scripts/fill_pdf_form_with_annotations.py +107 -107
- package/skills/writing-docs/pptx/LICENSE.txt +30 -30
- package/skills/writing-docs/pptx/SKILL.md +483 -483
- package/skills/writing-docs/pptx/html2pptx.md +624 -624
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -1499
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -146
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -1085
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -11
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -3081
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -23
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -185
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -287
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -1676
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -28
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -144
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -174
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -25
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -18
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -59
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -56
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -195
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -582
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -25
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -4439
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -570
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -509
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -12
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -108
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -96
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -3646
- package/skills/writing-docs/pptx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -116
- package/skills/writing-docs/pptx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -42
- package/skills/writing-docs/pptx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -50
- package/skills/writing-docs/pptx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -49
- package/skills/writing-docs/pptx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -33
- package/skills/writing-docs/pptx/ooxml/schemas/mce/mc.xsd +75 -75
- package/skills/writing-docs/pptx/ooxml/schemas/microsoft/wml-2010.xsd +560 -560
- package/skills/writing-docs/pptx/ooxml/schemas/microsoft/wml-2012.xsd +67 -67
- package/skills/writing-docs/pptx/ooxml/schemas/microsoft/wml-2018.xsd +14 -14
- package/skills/writing-docs/pptx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -20
- package/skills/writing-docs/pptx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -13
- package/skills/writing-docs/pptx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -4
- package/skills/writing-docs/pptx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -8
- package/skills/writing-docs/pptx/ooxml/scripts/pack.py +159 -159
- package/skills/writing-docs/pptx/ooxml/scripts/unpack.py +29 -29
- package/skills/writing-docs/pptx/ooxml/scripts/validate.py +69 -69
- package/skills/writing-docs/pptx/ooxml/scripts/validation/__init__.py +15 -15
- package/skills/writing-docs/pptx/ooxml/scripts/validation/base.py +951 -951
- package/skills/writing-docs/pptx/ooxml/scripts/validation/docx.py +274 -274
- package/skills/writing-docs/pptx/ooxml/scripts/validation/pptx.py +315 -315
- package/skills/writing-docs/pptx/ooxml/scripts/validation/redlining.py +279 -279
- package/skills/writing-docs/pptx/ooxml.md +426 -426
- package/skills/writing-docs/pptx/scripts/html2pptx.js +978 -978
- package/skills/writing-docs/pptx/scripts/inventory.py +1020 -1020
- package/skills/writing-docs/pptx/scripts/rearrange.py +231 -231
- package/skills/writing-docs/pptx/scripts/replace.py +385 -385
- package/skills/writing-docs/pptx/scripts/thumbnail.py +450 -450
- package/skills/writing-docs/slides/LICENSE.txt +201 -201
- package/skills/writing-docs/slides/SKILL.md +71 -71
- package/skills/writing-docs/slides/agents/openai.yaml +6 -6
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/code.js +104 -104
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/image.js +333 -333
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/index.js +33 -33
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/latex.js +51 -51
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/layout.js +643 -643
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/layout_builders.js +358 -358
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/svg.js +36 -36
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/text.js +789 -789
- package/skills/writing-docs/slides/assets/pptxgenjs_helpers/util.js +24 -24
- package/skills/writing-docs/slides/assets/slides-small.svg +3 -3
- package/skills/writing-docs/slides/references/pptxgenjs-helpers.md +61 -61
- package/skills/writing-docs/slides/scripts/create_montage.py +300 -300
- package/skills/writing-docs/slides/scripts/detect_font.py +873 -873
- package/skills/writing-docs/slides/scripts/ensure_raster_image.py +202 -202
- package/skills/writing-docs/slides/scripts/render_slides.py +273 -273
- package/skills/writing-docs/slides/scripts/slides_test.py +201 -201
- package/skills/writing-docs/template-skill/SKILL.md +26 -26
- package/skills/writing-docs/xlsx/LICENSE.txt +30 -30
- package/skills/writing-docs/xlsx/SKILL.md +288 -288
- package/skills/writing-docs/xlsx/recalc.py +177 -177
- package/src/core/KILO_MASTER.md +448 -448
- package/src/tools/validate-skill.js +421 -421
|
@@ -1,385 +1,385 @@
|
|
|
1
|
-
#!/usr/bin/env python3
|
|
2
|
-
"""Apply text replacements to PowerPoint presentation.
|
|
3
|
-
|
|
4
|
-
Usage:
|
|
5
|
-
python replace.py <input.pptx> <replacements.json> <output.pptx>
|
|
6
|
-
|
|
7
|
-
The replacements JSON should have the structure output by inventory.py.
|
|
8
|
-
ALL text shapes identified by inventory.py will have their text cleared
|
|
9
|
-
unless "paragraphs" is specified in the replacements for that shape.
|
|
10
|
-
"""
|
|
11
|
-
|
|
12
|
-
import json
|
|
13
|
-
import sys
|
|
14
|
-
from pathlib import Path
|
|
15
|
-
from typing import Any, Dict, List
|
|
16
|
-
|
|
17
|
-
from inventory import InventoryData, extract_text_inventory
|
|
18
|
-
from pptx import Presentation
|
|
19
|
-
from pptx.dml.color import RGBColor
|
|
20
|
-
from pptx.enum.dml import MSO_THEME_COLOR
|
|
21
|
-
from pptx.enum.text import PP_ALIGN
|
|
22
|
-
from pptx.oxml.xmlchemy import OxmlElement
|
|
23
|
-
from pptx.util import Pt
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
def clear_paragraph_bullets(paragraph):
|
|
27
|
-
"""Clear bullet formatting from a paragraph."""
|
|
28
|
-
pPr = paragraph._element.get_or_add_pPr()
|
|
29
|
-
|
|
30
|
-
# Remove existing bullet elements
|
|
31
|
-
for child in list(pPr):
|
|
32
|
-
if (
|
|
33
|
-
child.tag.endswith("buChar")
|
|
34
|
-
or child.tag.endswith("buNone")
|
|
35
|
-
or child.tag.endswith("buAutoNum")
|
|
36
|
-
or child.tag.endswith("buFont")
|
|
37
|
-
):
|
|
38
|
-
pPr.remove(child)
|
|
39
|
-
|
|
40
|
-
return pPr
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
def apply_paragraph_properties(paragraph, para_data: Dict[str, Any]):
|
|
44
|
-
"""Apply formatting properties to a paragraph."""
|
|
45
|
-
# Get the text but don't set it on paragraph directly yet
|
|
46
|
-
text = para_data.get("text", "")
|
|
47
|
-
|
|
48
|
-
# Get or create paragraph properties
|
|
49
|
-
pPr = clear_paragraph_bullets(paragraph)
|
|
50
|
-
|
|
51
|
-
# Handle bullet formatting
|
|
52
|
-
if para_data.get("bullet", False):
|
|
53
|
-
level = para_data.get("level", 0)
|
|
54
|
-
paragraph.level = level
|
|
55
|
-
|
|
56
|
-
# Calculate font-proportional indentation
|
|
57
|
-
font_size = para_data.get("font_size", 18.0)
|
|
58
|
-
level_indent_emu = int((font_size * (1.6 + level * 1.6)) * 12700)
|
|
59
|
-
hanging_indent_emu = int(-font_size * 0.8 * 12700)
|
|
60
|
-
|
|
61
|
-
# Set indentation
|
|
62
|
-
pPr.attrib["marL"] = str(level_indent_emu)
|
|
63
|
-
pPr.attrib["indent"] = str(hanging_indent_emu)
|
|
64
|
-
|
|
65
|
-
# Add bullet character
|
|
66
|
-
buChar = OxmlElement("a:buChar")
|
|
67
|
-
buChar.set("char", "•")
|
|
68
|
-
pPr.append(buChar)
|
|
69
|
-
|
|
70
|
-
# Default to left alignment for bullets if not specified
|
|
71
|
-
if "alignment" not in para_data:
|
|
72
|
-
paragraph.alignment = PP_ALIGN.LEFT
|
|
73
|
-
else:
|
|
74
|
-
# Remove indentation for non-bullet text
|
|
75
|
-
pPr.attrib["marL"] = "0"
|
|
76
|
-
pPr.attrib["indent"] = "0"
|
|
77
|
-
|
|
78
|
-
# Add buNone element
|
|
79
|
-
buNone = OxmlElement("a:buNone")
|
|
80
|
-
pPr.insert(0, buNone)
|
|
81
|
-
|
|
82
|
-
# Apply alignment
|
|
83
|
-
if "alignment" in para_data:
|
|
84
|
-
alignment_map = {
|
|
85
|
-
"LEFT": PP_ALIGN.LEFT,
|
|
86
|
-
"CENTER": PP_ALIGN.CENTER,
|
|
87
|
-
"RIGHT": PP_ALIGN.RIGHT,
|
|
88
|
-
"JUSTIFY": PP_ALIGN.JUSTIFY,
|
|
89
|
-
}
|
|
90
|
-
if para_data["alignment"] in alignment_map:
|
|
91
|
-
paragraph.alignment = alignment_map[para_data["alignment"]]
|
|
92
|
-
|
|
93
|
-
# Apply spacing
|
|
94
|
-
if "space_before" in para_data:
|
|
95
|
-
paragraph.space_before = Pt(para_data["space_before"])
|
|
96
|
-
if "space_after" in para_data:
|
|
97
|
-
paragraph.space_after = Pt(para_data["space_after"])
|
|
98
|
-
if "line_spacing" in para_data:
|
|
99
|
-
paragraph.line_spacing = Pt(para_data["line_spacing"])
|
|
100
|
-
|
|
101
|
-
# Apply run-level formatting
|
|
102
|
-
if not paragraph.runs:
|
|
103
|
-
run = paragraph.add_run()
|
|
104
|
-
run.text = text
|
|
105
|
-
else:
|
|
106
|
-
run = paragraph.runs[0]
|
|
107
|
-
run.text = text
|
|
108
|
-
|
|
109
|
-
# Apply font properties
|
|
110
|
-
apply_font_properties(run, para_data)
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
def apply_font_properties(run, para_data: Dict[str, Any]):
|
|
114
|
-
"""Apply font properties to a text run."""
|
|
115
|
-
if "bold" in para_data:
|
|
116
|
-
run.font.bold = para_data["bold"]
|
|
117
|
-
if "italic" in para_data:
|
|
118
|
-
run.font.italic = para_data["italic"]
|
|
119
|
-
if "underline" in para_data:
|
|
120
|
-
run.font.underline = para_data["underline"]
|
|
121
|
-
if "font_size" in para_data:
|
|
122
|
-
run.font.size = Pt(para_data["font_size"])
|
|
123
|
-
if "font_name" in para_data:
|
|
124
|
-
run.font.name = para_data["font_name"]
|
|
125
|
-
|
|
126
|
-
# Apply color - prefer RGB, fall back to theme_color
|
|
127
|
-
if "color" in para_data:
|
|
128
|
-
color_hex = para_data["color"].lstrip("#")
|
|
129
|
-
if len(color_hex) == 6:
|
|
130
|
-
r = int(color_hex[0:2], 16)
|
|
131
|
-
g = int(color_hex[2:4], 16)
|
|
132
|
-
b = int(color_hex[4:6], 16)
|
|
133
|
-
run.font.color.rgb = RGBColor(r, g, b)
|
|
134
|
-
elif "theme_color" in para_data:
|
|
135
|
-
# Get theme color by name (e.g., "DARK_1", "ACCENT_1")
|
|
136
|
-
theme_name = para_data["theme_color"]
|
|
137
|
-
try:
|
|
138
|
-
run.font.color.theme_color = getattr(MSO_THEME_COLOR, theme_name)
|
|
139
|
-
except AttributeError:
|
|
140
|
-
print(f" WARNING: Unknown theme color name '{theme_name}'")
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
def detect_frame_overflow(inventory: InventoryData) -> Dict[str, Dict[str, float]]:
|
|
144
|
-
"""Detect text overflow in shapes (text exceeding shape bounds).
|
|
145
|
-
|
|
146
|
-
Returns dict of slide_key -> shape_key -> overflow_inches.
|
|
147
|
-
Only includes shapes that have text overflow.
|
|
148
|
-
"""
|
|
149
|
-
overflow_map = {}
|
|
150
|
-
|
|
151
|
-
for slide_key, shapes_dict in inventory.items():
|
|
152
|
-
for shape_key, shape_data in shapes_dict.items():
|
|
153
|
-
# Check for frame overflow (text exceeding shape bounds)
|
|
154
|
-
if shape_data.frame_overflow_bottom is not None:
|
|
155
|
-
if slide_key not in overflow_map:
|
|
156
|
-
overflow_map[slide_key] = {}
|
|
157
|
-
overflow_map[slide_key][shape_key] = shape_data.frame_overflow_bottom
|
|
158
|
-
|
|
159
|
-
return overflow_map
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
def validate_replacements(inventory: InventoryData, replacements: Dict) -> List[str]:
|
|
163
|
-
"""Validate that all shapes in replacements exist in inventory.
|
|
164
|
-
|
|
165
|
-
Returns list of error messages.
|
|
166
|
-
"""
|
|
167
|
-
errors = []
|
|
168
|
-
|
|
169
|
-
for slide_key, shapes_data in replacements.items():
|
|
170
|
-
if not slide_key.startswith("slide-"):
|
|
171
|
-
continue
|
|
172
|
-
|
|
173
|
-
# Check if slide exists
|
|
174
|
-
if slide_key not in inventory:
|
|
175
|
-
errors.append(f"Slide '{slide_key}' not found in inventory")
|
|
176
|
-
continue
|
|
177
|
-
|
|
178
|
-
# Check each shape
|
|
179
|
-
for shape_key in shapes_data.keys():
|
|
180
|
-
if shape_key not in inventory[slide_key]:
|
|
181
|
-
# Find shapes without replacements defined and show their content
|
|
182
|
-
unused_with_content = []
|
|
183
|
-
for k in inventory[slide_key].keys():
|
|
184
|
-
if k not in shapes_data:
|
|
185
|
-
shape_data = inventory[slide_key][k]
|
|
186
|
-
# Get text from paragraphs as preview
|
|
187
|
-
paragraphs = shape_data.paragraphs
|
|
188
|
-
if paragraphs and paragraphs[0].text:
|
|
189
|
-
first_text = paragraphs[0].text[:50]
|
|
190
|
-
if len(paragraphs[0].text) > 50:
|
|
191
|
-
first_text += "..."
|
|
192
|
-
unused_with_content.append(f"{k} ('{first_text}')")
|
|
193
|
-
else:
|
|
194
|
-
unused_with_content.append(k)
|
|
195
|
-
|
|
196
|
-
errors.append(
|
|
197
|
-
f"Shape '{shape_key}' not found on '{slide_key}'. "
|
|
198
|
-
f"Shapes without replacements: {', '.join(sorted(unused_with_content)) if unused_with_content else 'none'}"
|
|
199
|
-
)
|
|
200
|
-
|
|
201
|
-
return errors
|
|
202
|
-
|
|
203
|
-
|
|
204
|
-
def check_duplicate_keys(pairs):
|
|
205
|
-
"""Check for duplicate keys when loading JSON."""
|
|
206
|
-
result = {}
|
|
207
|
-
for key, value in pairs:
|
|
208
|
-
if key in result:
|
|
209
|
-
raise ValueError(f"Duplicate key found in JSON: '{key}'")
|
|
210
|
-
result[key] = value
|
|
211
|
-
return result
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
def apply_replacements(pptx_file: str, json_file: str, output_file: str):
|
|
215
|
-
"""Apply text replacements from JSON to PowerPoint presentation."""
|
|
216
|
-
|
|
217
|
-
# Load presentation
|
|
218
|
-
prs = Presentation(pptx_file)
|
|
219
|
-
|
|
220
|
-
# Get inventory of all text shapes (returns ShapeData objects)
|
|
221
|
-
# Pass prs to use same Presentation instance
|
|
222
|
-
inventory = extract_text_inventory(Path(pptx_file), prs)
|
|
223
|
-
|
|
224
|
-
# Detect text overflow in original presentation
|
|
225
|
-
original_overflow = detect_frame_overflow(inventory)
|
|
226
|
-
|
|
227
|
-
# Load replacement data with duplicate key detection
|
|
228
|
-
with open(json_file, "r") as f:
|
|
229
|
-
replacements = json.load(f, object_pairs_hook=check_duplicate_keys)
|
|
230
|
-
|
|
231
|
-
# Validate replacements
|
|
232
|
-
errors = validate_replacements(inventory, replacements)
|
|
233
|
-
if errors:
|
|
234
|
-
print("ERROR: Invalid shapes in replacement JSON:")
|
|
235
|
-
for error in errors:
|
|
236
|
-
print(f" - {error}")
|
|
237
|
-
print("\nPlease check the inventory and update your replacement JSON.")
|
|
238
|
-
print(
|
|
239
|
-
"You can regenerate the inventory with: python inventory.py <input.pptx> <output.json>"
|
|
240
|
-
)
|
|
241
|
-
raise ValueError(f"Found {len(errors)} validation error(s)")
|
|
242
|
-
|
|
243
|
-
# Track statistics
|
|
244
|
-
shapes_processed = 0
|
|
245
|
-
shapes_cleared = 0
|
|
246
|
-
shapes_replaced = 0
|
|
247
|
-
|
|
248
|
-
# Process each slide from inventory
|
|
249
|
-
for slide_key, shapes_dict in inventory.items():
|
|
250
|
-
if not slide_key.startswith("slide-"):
|
|
251
|
-
continue
|
|
252
|
-
|
|
253
|
-
slide_index = int(slide_key.split("-")[1])
|
|
254
|
-
|
|
255
|
-
if slide_index >= len(prs.slides):
|
|
256
|
-
print(f"Warning: Slide {slide_index} not found")
|
|
257
|
-
continue
|
|
258
|
-
|
|
259
|
-
# Process each shape from inventory
|
|
260
|
-
for shape_key, shape_data in shapes_dict.items():
|
|
261
|
-
shapes_processed += 1
|
|
262
|
-
|
|
263
|
-
# Get the shape directly from ShapeData
|
|
264
|
-
shape = shape_data.shape
|
|
265
|
-
if not shape:
|
|
266
|
-
print(f"Warning: {shape_key} has no shape reference")
|
|
267
|
-
continue
|
|
268
|
-
|
|
269
|
-
# ShapeData already validates text_frame in __init__
|
|
270
|
-
text_frame = shape.text_frame # type: ignore
|
|
271
|
-
|
|
272
|
-
text_frame.clear() # type: ignore
|
|
273
|
-
shapes_cleared += 1
|
|
274
|
-
|
|
275
|
-
# Check for replacement paragraphs
|
|
276
|
-
replacement_shape_data = replacements.get(slide_key, {}).get(shape_key, {})
|
|
277
|
-
if "paragraphs" not in replacement_shape_data:
|
|
278
|
-
continue
|
|
279
|
-
|
|
280
|
-
shapes_replaced += 1
|
|
281
|
-
|
|
282
|
-
# Add replacement paragraphs
|
|
283
|
-
for i, para_data in enumerate(replacement_shape_data["paragraphs"]):
|
|
284
|
-
if i == 0:
|
|
285
|
-
p = text_frame.paragraphs[0] # type: ignore
|
|
286
|
-
else:
|
|
287
|
-
p = text_frame.add_paragraph() # type: ignore
|
|
288
|
-
|
|
289
|
-
apply_paragraph_properties(p, para_data)
|
|
290
|
-
|
|
291
|
-
# Check for issues after replacements
|
|
292
|
-
# Save to a temporary file and reload to avoid modifying the presentation during inventory
|
|
293
|
-
# (extract_text_inventory accesses font.color which adds empty <a:solidFill/> elements)
|
|
294
|
-
import tempfile
|
|
295
|
-
|
|
296
|
-
with tempfile.NamedTemporaryFile(suffix=".pptx", delete=False) as tmp:
|
|
297
|
-
tmp_path = Path(tmp.name)
|
|
298
|
-
prs.save(str(tmp_path))
|
|
299
|
-
|
|
300
|
-
try:
|
|
301
|
-
updated_inventory = extract_text_inventory(tmp_path)
|
|
302
|
-
updated_overflow = detect_frame_overflow(updated_inventory)
|
|
303
|
-
finally:
|
|
304
|
-
tmp_path.unlink() # Clean up temp file
|
|
305
|
-
|
|
306
|
-
# Check if any text overflow got worse
|
|
307
|
-
overflow_errors = []
|
|
308
|
-
for slide_key, shape_overflows in updated_overflow.items():
|
|
309
|
-
for shape_key, new_overflow in shape_overflows.items():
|
|
310
|
-
# Get original overflow (0 if there was no overflow before)
|
|
311
|
-
original = original_overflow.get(slide_key, {}).get(shape_key, 0.0)
|
|
312
|
-
|
|
313
|
-
# Error if overflow increased
|
|
314
|
-
if new_overflow > original + 0.01: # Small tolerance for rounding
|
|
315
|
-
increase = new_overflow - original
|
|
316
|
-
overflow_errors.append(
|
|
317
|
-
f'{slide_key}/{shape_key}: overflow worsened by {increase:.2f}" '
|
|
318
|
-
f'(was {original:.2f}", now {new_overflow:.2f}")'
|
|
319
|
-
)
|
|
320
|
-
|
|
321
|
-
# Collect warnings from updated shapes
|
|
322
|
-
warnings = []
|
|
323
|
-
for slide_key, shapes_dict in updated_inventory.items():
|
|
324
|
-
for shape_key, shape_data in shapes_dict.items():
|
|
325
|
-
if shape_data.warnings:
|
|
326
|
-
for warning in shape_data.warnings:
|
|
327
|
-
warnings.append(f"{slide_key}/{shape_key}: {warning}")
|
|
328
|
-
|
|
329
|
-
# Fail if there are any issues
|
|
330
|
-
if overflow_errors or warnings:
|
|
331
|
-
print("\nERROR: Issues detected in replacement output:")
|
|
332
|
-
if overflow_errors:
|
|
333
|
-
print("\nText overflow worsened:")
|
|
334
|
-
for error in overflow_errors:
|
|
335
|
-
print(f" - {error}")
|
|
336
|
-
if warnings:
|
|
337
|
-
print("\nFormatting warnings:")
|
|
338
|
-
for warning in warnings:
|
|
339
|
-
print(f" - {warning}")
|
|
340
|
-
print("\nPlease fix these issues before saving.")
|
|
341
|
-
raise ValueError(
|
|
342
|
-
f"Found {len(overflow_errors)} overflow error(s) and {len(warnings)} warning(s)"
|
|
343
|
-
)
|
|
344
|
-
|
|
345
|
-
# Save the presentation
|
|
346
|
-
prs.save(output_file)
|
|
347
|
-
|
|
348
|
-
# Report results
|
|
349
|
-
print(f"Saved updated presentation to: {output_file}")
|
|
350
|
-
print(f"Processed {len(prs.slides)} slides")
|
|
351
|
-
print(f" - Shapes processed: {shapes_processed}")
|
|
352
|
-
print(f" - Shapes cleared: {shapes_cleared}")
|
|
353
|
-
print(f" - Shapes replaced: {shapes_replaced}")
|
|
354
|
-
|
|
355
|
-
|
|
356
|
-
def main():
|
|
357
|
-
"""Main entry point for command-line usage."""
|
|
358
|
-
if len(sys.argv) != 4:
|
|
359
|
-
print(__doc__)
|
|
360
|
-
sys.exit(1)
|
|
361
|
-
|
|
362
|
-
input_pptx = Path(sys.argv[1])
|
|
363
|
-
replacements_json = Path(sys.argv[2])
|
|
364
|
-
output_pptx = Path(sys.argv[3])
|
|
365
|
-
|
|
366
|
-
if not input_pptx.exists():
|
|
367
|
-
print(f"Error: Input file '{input_pptx}' not found")
|
|
368
|
-
sys.exit(1)
|
|
369
|
-
|
|
370
|
-
if not replacements_json.exists():
|
|
371
|
-
print(f"Error: Replacements JSON file '{replacements_json}' not found")
|
|
372
|
-
sys.exit(1)
|
|
373
|
-
|
|
374
|
-
try:
|
|
375
|
-
apply_replacements(str(input_pptx), str(replacements_json), str(output_pptx))
|
|
376
|
-
except Exception as e:
|
|
377
|
-
print(f"Error applying replacements: {e}")
|
|
378
|
-
import traceback
|
|
379
|
-
|
|
380
|
-
traceback.print_exc()
|
|
381
|
-
sys.exit(1)
|
|
382
|
-
|
|
383
|
-
|
|
384
|
-
if __name__ == "__main__":
|
|
385
|
-
main()
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""Apply text replacements to PowerPoint presentation.
|
|
3
|
+
|
|
4
|
+
Usage:
|
|
5
|
+
python replace.py <input.pptx> <replacements.json> <output.pptx>
|
|
6
|
+
|
|
7
|
+
The replacements JSON should have the structure output by inventory.py.
|
|
8
|
+
ALL text shapes identified by inventory.py will have their text cleared
|
|
9
|
+
unless "paragraphs" is specified in the replacements for that shape.
|
|
10
|
+
"""
|
|
11
|
+
|
|
12
|
+
import json
|
|
13
|
+
import sys
|
|
14
|
+
from pathlib import Path
|
|
15
|
+
from typing import Any, Dict, List
|
|
16
|
+
|
|
17
|
+
from inventory import InventoryData, extract_text_inventory
|
|
18
|
+
from pptx import Presentation
|
|
19
|
+
from pptx.dml.color import RGBColor
|
|
20
|
+
from pptx.enum.dml import MSO_THEME_COLOR
|
|
21
|
+
from pptx.enum.text import PP_ALIGN
|
|
22
|
+
from pptx.oxml.xmlchemy import OxmlElement
|
|
23
|
+
from pptx.util import Pt
|
|
24
|
+
|
|
25
|
+
|
|
26
|
+
def clear_paragraph_bullets(paragraph):
|
|
27
|
+
"""Clear bullet formatting from a paragraph."""
|
|
28
|
+
pPr = paragraph._element.get_or_add_pPr()
|
|
29
|
+
|
|
30
|
+
# Remove existing bullet elements
|
|
31
|
+
for child in list(pPr):
|
|
32
|
+
if (
|
|
33
|
+
child.tag.endswith("buChar")
|
|
34
|
+
or child.tag.endswith("buNone")
|
|
35
|
+
or child.tag.endswith("buAutoNum")
|
|
36
|
+
or child.tag.endswith("buFont")
|
|
37
|
+
):
|
|
38
|
+
pPr.remove(child)
|
|
39
|
+
|
|
40
|
+
return pPr
|
|
41
|
+
|
|
42
|
+
|
|
43
|
+
def apply_paragraph_properties(paragraph, para_data: Dict[str, Any]):
|
|
44
|
+
"""Apply formatting properties to a paragraph."""
|
|
45
|
+
# Get the text but don't set it on paragraph directly yet
|
|
46
|
+
text = para_data.get("text", "")
|
|
47
|
+
|
|
48
|
+
# Get or create paragraph properties
|
|
49
|
+
pPr = clear_paragraph_bullets(paragraph)
|
|
50
|
+
|
|
51
|
+
# Handle bullet formatting
|
|
52
|
+
if para_data.get("bullet", False):
|
|
53
|
+
level = para_data.get("level", 0)
|
|
54
|
+
paragraph.level = level
|
|
55
|
+
|
|
56
|
+
# Calculate font-proportional indentation
|
|
57
|
+
font_size = para_data.get("font_size", 18.0)
|
|
58
|
+
level_indent_emu = int((font_size * (1.6 + level * 1.6)) * 12700)
|
|
59
|
+
hanging_indent_emu = int(-font_size * 0.8 * 12700)
|
|
60
|
+
|
|
61
|
+
# Set indentation
|
|
62
|
+
pPr.attrib["marL"] = str(level_indent_emu)
|
|
63
|
+
pPr.attrib["indent"] = str(hanging_indent_emu)
|
|
64
|
+
|
|
65
|
+
# Add bullet character
|
|
66
|
+
buChar = OxmlElement("a:buChar")
|
|
67
|
+
buChar.set("char", "•")
|
|
68
|
+
pPr.append(buChar)
|
|
69
|
+
|
|
70
|
+
# Default to left alignment for bullets if not specified
|
|
71
|
+
if "alignment" not in para_data:
|
|
72
|
+
paragraph.alignment = PP_ALIGN.LEFT
|
|
73
|
+
else:
|
|
74
|
+
# Remove indentation for non-bullet text
|
|
75
|
+
pPr.attrib["marL"] = "0"
|
|
76
|
+
pPr.attrib["indent"] = "0"
|
|
77
|
+
|
|
78
|
+
# Add buNone element
|
|
79
|
+
buNone = OxmlElement("a:buNone")
|
|
80
|
+
pPr.insert(0, buNone)
|
|
81
|
+
|
|
82
|
+
# Apply alignment
|
|
83
|
+
if "alignment" in para_data:
|
|
84
|
+
alignment_map = {
|
|
85
|
+
"LEFT": PP_ALIGN.LEFT,
|
|
86
|
+
"CENTER": PP_ALIGN.CENTER,
|
|
87
|
+
"RIGHT": PP_ALIGN.RIGHT,
|
|
88
|
+
"JUSTIFY": PP_ALIGN.JUSTIFY,
|
|
89
|
+
}
|
|
90
|
+
if para_data["alignment"] in alignment_map:
|
|
91
|
+
paragraph.alignment = alignment_map[para_data["alignment"]]
|
|
92
|
+
|
|
93
|
+
# Apply spacing
|
|
94
|
+
if "space_before" in para_data:
|
|
95
|
+
paragraph.space_before = Pt(para_data["space_before"])
|
|
96
|
+
if "space_after" in para_data:
|
|
97
|
+
paragraph.space_after = Pt(para_data["space_after"])
|
|
98
|
+
if "line_spacing" in para_data:
|
|
99
|
+
paragraph.line_spacing = Pt(para_data["line_spacing"])
|
|
100
|
+
|
|
101
|
+
# Apply run-level formatting
|
|
102
|
+
if not paragraph.runs:
|
|
103
|
+
run = paragraph.add_run()
|
|
104
|
+
run.text = text
|
|
105
|
+
else:
|
|
106
|
+
run = paragraph.runs[0]
|
|
107
|
+
run.text = text
|
|
108
|
+
|
|
109
|
+
# Apply font properties
|
|
110
|
+
apply_font_properties(run, para_data)
|
|
111
|
+
|
|
112
|
+
|
|
113
|
+
def apply_font_properties(run, para_data: Dict[str, Any]):
|
|
114
|
+
"""Apply font properties to a text run."""
|
|
115
|
+
if "bold" in para_data:
|
|
116
|
+
run.font.bold = para_data["bold"]
|
|
117
|
+
if "italic" in para_data:
|
|
118
|
+
run.font.italic = para_data["italic"]
|
|
119
|
+
if "underline" in para_data:
|
|
120
|
+
run.font.underline = para_data["underline"]
|
|
121
|
+
if "font_size" in para_data:
|
|
122
|
+
run.font.size = Pt(para_data["font_size"])
|
|
123
|
+
if "font_name" in para_data:
|
|
124
|
+
run.font.name = para_data["font_name"]
|
|
125
|
+
|
|
126
|
+
# Apply color - prefer RGB, fall back to theme_color
|
|
127
|
+
if "color" in para_data:
|
|
128
|
+
color_hex = para_data["color"].lstrip("#")
|
|
129
|
+
if len(color_hex) == 6:
|
|
130
|
+
r = int(color_hex[0:2], 16)
|
|
131
|
+
g = int(color_hex[2:4], 16)
|
|
132
|
+
b = int(color_hex[4:6], 16)
|
|
133
|
+
run.font.color.rgb = RGBColor(r, g, b)
|
|
134
|
+
elif "theme_color" in para_data:
|
|
135
|
+
# Get theme color by name (e.g., "DARK_1", "ACCENT_1")
|
|
136
|
+
theme_name = para_data["theme_color"]
|
|
137
|
+
try:
|
|
138
|
+
run.font.color.theme_color = getattr(MSO_THEME_COLOR, theme_name)
|
|
139
|
+
except AttributeError:
|
|
140
|
+
print(f" WARNING: Unknown theme color name '{theme_name}'")
|
|
141
|
+
|
|
142
|
+
|
|
143
|
+
def detect_frame_overflow(inventory: InventoryData) -> Dict[str, Dict[str, float]]:
|
|
144
|
+
"""Detect text overflow in shapes (text exceeding shape bounds).
|
|
145
|
+
|
|
146
|
+
Returns dict of slide_key -> shape_key -> overflow_inches.
|
|
147
|
+
Only includes shapes that have text overflow.
|
|
148
|
+
"""
|
|
149
|
+
overflow_map = {}
|
|
150
|
+
|
|
151
|
+
for slide_key, shapes_dict in inventory.items():
|
|
152
|
+
for shape_key, shape_data in shapes_dict.items():
|
|
153
|
+
# Check for frame overflow (text exceeding shape bounds)
|
|
154
|
+
if shape_data.frame_overflow_bottom is not None:
|
|
155
|
+
if slide_key not in overflow_map:
|
|
156
|
+
overflow_map[slide_key] = {}
|
|
157
|
+
overflow_map[slide_key][shape_key] = shape_data.frame_overflow_bottom
|
|
158
|
+
|
|
159
|
+
return overflow_map
|
|
160
|
+
|
|
161
|
+
|
|
162
|
+
def validate_replacements(inventory: InventoryData, replacements: Dict) -> List[str]:
|
|
163
|
+
"""Validate that all shapes in replacements exist in inventory.
|
|
164
|
+
|
|
165
|
+
Returns list of error messages.
|
|
166
|
+
"""
|
|
167
|
+
errors = []
|
|
168
|
+
|
|
169
|
+
for slide_key, shapes_data in replacements.items():
|
|
170
|
+
if not slide_key.startswith("slide-"):
|
|
171
|
+
continue
|
|
172
|
+
|
|
173
|
+
# Check if slide exists
|
|
174
|
+
if slide_key not in inventory:
|
|
175
|
+
errors.append(f"Slide '{slide_key}' not found in inventory")
|
|
176
|
+
continue
|
|
177
|
+
|
|
178
|
+
# Check each shape
|
|
179
|
+
for shape_key in shapes_data.keys():
|
|
180
|
+
if shape_key not in inventory[slide_key]:
|
|
181
|
+
# Find shapes without replacements defined and show their content
|
|
182
|
+
unused_with_content = []
|
|
183
|
+
for k in inventory[slide_key].keys():
|
|
184
|
+
if k not in shapes_data:
|
|
185
|
+
shape_data = inventory[slide_key][k]
|
|
186
|
+
# Get text from paragraphs as preview
|
|
187
|
+
paragraphs = shape_data.paragraphs
|
|
188
|
+
if paragraphs and paragraphs[0].text:
|
|
189
|
+
first_text = paragraphs[0].text[:50]
|
|
190
|
+
if len(paragraphs[0].text) > 50:
|
|
191
|
+
first_text += "..."
|
|
192
|
+
unused_with_content.append(f"{k} ('{first_text}')")
|
|
193
|
+
else:
|
|
194
|
+
unused_with_content.append(k)
|
|
195
|
+
|
|
196
|
+
errors.append(
|
|
197
|
+
f"Shape '{shape_key}' not found on '{slide_key}'. "
|
|
198
|
+
f"Shapes without replacements: {', '.join(sorted(unused_with_content)) if unused_with_content else 'none'}"
|
|
199
|
+
)
|
|
200
|
+
|
|
201
|
+
return errors
|
|
202
|
+
|
|
203
|
+
|
|
204
|
+
def check_duplicate_keys(pairs):
|
|
205
|
+
"""Check for duplicate keys when loading JSON."""
|
|
206
|
+
result = {}
|
|
207
|
+
for key, value in pairs:
|
|
208
|
+
if key in result:
|
|
209
|
+
raise ValueError(f"Duplicate key found in JSON: '{key}'")
|
|
210
|
+
result[key] = value
|
|
211
|
+
return result
|
|
212
|
+
|
|
213
|
+
|
|
214
|
+
def apply_replacements(pptx_file: str, json_file: str, output_file: str):
|
|
215
|
+
"""Apply text replacements from JSON to PowerPoint presentation."""
|
|
216
|
+
|
|
217
|
+
# Load presentation
|
|
218
|
+
prs = Presentation(pptx_file)
|
|
219
|
+
|
|
220
|
+
# Get inventory of all text shapes (returns ShapeData objects)
|
|
221
|
+
# Pass prs to use same Presentation instance
|
|
222
|
+
inventory = extract_text_inventory(Path(pptx_file), prs)
|
|
223
|
+
|
|
224
|
+
# Detect text overflow in original presentation
|
|
225
|
+
original_overflow = detect_frame_overflow(inventory)
|
|
226
|
+
|
|
227
|
+
# Load replacement data with duplicate key detection
|
|
228
|
+
with open(json_file, "r") as f:
|
|
229
|
+
replacements = json.load(f, object_pairs_hook=check_duplicate_keys)
|
|
230
|
+
|
|
231
|
+
# Validate replacements
|
|
232
|
+
errors = validate_replacements(inventory, replacements)
|
|
233
|
+
if errors:
|
|
234
|
+
print("ERROR: Invalid shapes in replacement JSON:")
|
|
235
|
+
for error in errors:
|
|
236
|
+
print(f" - {error}")
|
|
237
|
+
print("\nPlease check the inventory and update your replacement JSON.")
|
|
238
|
+
print(
|
|
239
|
+
"You can regenerate the inventory with: python inventory.py <input.pptx> <output.json>"
|
|
240
|
+
)
|
|
241
|
+
raise ValueError(f"Found {len(errors)} validation error(s)")
|
|
242
|
+
|
|
243
|
+
# Track statistics
|
|
244
|
+
shapes_processed = 0
|
|
245
|
+
shapes_cleared = 0
|
|
246
|
+
shapes_replaced = 0
|
|
247
|
+
|
|
248
|
+
# Process each slide from inventory
|
|
249
|
+
for slide_key, shapes_dict in inventory.items():
|
|
250
|
+
if not slide_key.startswith("slide-"):
|
|
251
|
+
continue
|
|
252
|
+
|
|
253
|
+
slide_index = int(slide_key.split("-")[1])
|
|
254
|
+
|
|
255
|
+
if slide_index >= len(prs.slides):
|
|
256
|
+
print(f"Warning: Slide {slide_index} not found")
|
|
257
|
+
continue
|
|
258
|
+
|
|
259
|
+
# Process each shape from inventory
|
|
260
|
+
for shape_key, shape_data in shapes_dict.items():
|
|
261
|
+
shapes_processed += 1
|
|
262
|
+
|
|
263
|
+
# Get the shape directly from ShapeData
|
|
264
|
+
shape = shape_data.shape
|
|
265
|
+
if not shape:
|
|
266
|
+
print(f"Warning: {shape_key} has no shape reference")
|
|
267
|
+
continue
|
|
268
|
+
|
|
269
|
+
# ShapeData already validates text_frame in __init__
|
|
270
|
+
text_frame = shape.text_frame # type: ignore
|
|
271
|
+
|
|
272
|
+
text_frame.clear() # type: ignore
|
|
273
|
+
shapes_cleared += 1
|
|
274
|
+
|
|
275
|
+
# Check for replacement paragraphs
|
|
276
|
+
replacement_shape_data = replacements.get(slide_key, {}).get(shape_key, {})
|
|
277
|
+
if "paragraphs" not in replacement_shape_data:
|
|
278
|
+
continue
|
|
279
|
+
|
|
280
|
+
shapes_replaced += 1
|
|
281
|
+
|
|
282
|
+
# Add replacement paragraphs
|
|
283
|
+
for i, para_data in enumerate(replacement_shape_data["paragraphs"]):
|
|
284
|
+
if i == 0:
|
|
285
|
+
p = text_frame.paragraphs[0] # type: ignore
|
|
286
|
+
else:
|
|
287
|
+
p = text_frame.add_paragraph() # type: ignore
|
|
288
|
+
|
|
289
|
+
apply_paragraph_properties(p, para_data)
|
|
290
|
+
|
|
291
|
+
# Check for issues after replacements
|
|
292
|
+
# Save to a temporary file and reload to avoid modifying the presentation during inventory
|
|
293
|
+
# (extract_text_inventory accesses font.color which adds empty <a:solidFill/> elements)
|
|
294
|
+
import tempfile
|
|
295
|
+
|
|
296
|
+
with tempfile.NamedTemporaryFile(suffix=".pptx", delete=False) as tmp:
|
|
297
|
+
tmp_path = Path(tmp.name)
|
|
298
|
+
prs.save(str(tmp_path))
|
|
299
|
+
|
|
300
|
+
try:
|
|
301
|
+
updated_inventory = extract_text_inventory(tmp_path)
|
|
302
|
+
updated_overflow = detect_frame_overflow(updated_inventory)
|
|
303
|
+
finally:
|
|
304
|
+
tmp_path.unlink() # Clean up temp file
|
|
305
|
+
|
|
306
|
+
# Check if any text overflow got worse
|
|
307
|
+
overflow_errors = []
|
|
308
|
+
for slide_key, shape_overflows in updated_overflow.items():
|
|
309
|
+
for shape_key, new_overflow in shape_overflows.items():
|
|
310
|
+
# Get original overflow (0 if there was no overflow before)
|
|
311
|
+
original = original_overflow.get(slide_key, {}).get(shape_key, 0.0)
|
|
312
|
+
|
|
313
|
+
# Error if overflow increased
|
|
314
|
+
if new_overflow > original + 0.01: # Small tolerance for rounding
|
|
315
|
+
increase = new_overflow - original
|
|
316
|
+
overflow_errors.append(
|
|
317
|
+
f'{slide_key}/{shape_key}: overflow worsened by {increase:.2f}" '
|
|
318
|
+
f'(was {original:.2f}", now {new_overflow:.2f}")'
|
|
319
|
+
)
|
|
320
|
+
|
|
321
|
+
# Collect warnings from updated shapes
|
|
322
|
+
warnings = []
|
|
323
|
+
for slide_key, shapes_dict in updated_inventory.items():
|
|
324
|
+
for shape_key, shape_data in shapes_dict.items():
|
|
325
|
+
if shape_data.warnings:
|
|
326
|
+
for warning in shape_data.warnings:
|
|
327
|
+
warnings.append(f"{slide_key}/{shape_key}: {warning}")
|
|
328
|
+
|
|
329
|
+
# Fail if there are any issues
|
|
330
|
+
if overflow_errors or warnings:
|
|
331
|
+
print("\nERROR: Issues detected in replacement output:")
|
|
332
|
+
if overflow_errors:
|
|
333
|
+
print("\nText overflow worsened:")
|
|
334
|
+
for error in overflow_errors:
|
|
335
|
+
print(f" - {error}")
|
|
336
|
+
if warnings:
|
|
337
|
+
print("\nFormatting warnings:")
|
|
338
|
+
for warning in warnings:
|
|
339
|
+
print(f" - {warning}")
|
|
340
|
+
print("\nPlease fix these issues before saving.")
|
|
341
|
+
raise ValueError(
|
|
342
|
+
f"Found {len(overflow_errors)} overflow error(s) and {len(warnings)} warning(s)"
|
|
343
|
+
)
|
|
344
|
+
|
|
345
|
+
# Save the presentation
|
|
346
|
+
prs.save(output_file)
|
|
347
|
+
|
|
348
|
+
# Report results
|
|
349
|
+
print(f"Saved updated presentation to: {output_file}")
|
|
350
|
+
print(f"Processed {len(prs.slides)} slides")
|
|
351
|
+
print(f" - Shapes processed: {shapes_processed}")
|
|
352
|
+
print(f" - Shapes cleared: {shapes_cleared}")
|
|
353
|
+
print(f" - Shapes replaced: {shapes_replaced}")
|
|
354
|
+
|
|
355
|
+
|
|
356
|
+
def main():
|
|
357
|
+
"""Main entry point for command-line usage."""
|
|
358
|
+
if len(sys.argv) != 4:
|
|
359
|
+
print(__doc__)
|
|
360
|
+
sys.exit(1)
|
|
361
|
+
|
|
362
|
+
input_pptx = Path(sys.argv[1])
|
|
363
|
+
replacements_json = Path(sys.argv[2])
|
|
364
|
+
output_pptx = Path(sys.argv[3])
|
|
365
|
+
|
|
366
|
+
if not input_pptx.exists():
|
|
367
|
+
print(f"Error: Input file '{input_pptx}' not found")
|
|
368
|
+
sys.exit(1)
|
|
369
|
+
|
|
370
|
+
if not replacements_json.exists():
|
|
371
|
+
print(f"Error: Replacements JSON file '{replacements_json}' not found")
|
|
372
|
+
sys.exit(1)
|
|
373
|
+
|
|
374
|
+
try:
|
|
375
|
+
apply_replacements(str(input_pptx), str(replacements_json), str(output_pptx))
|
|
376
|
+
except Exception as e:
|
|
377
|
+
print(f"Error applying replacements: {e}")
|
|
378
|
+
import traceback
|
|
379
|
+
|
|
380
|
+
traceback.print_exc()
|
|
381
|
+
sys.exit(1)
|
|
382
|
+
|
|
383
|
+
|
|
384
|
+
if __name__ == "__main__":
|
|
385
|
+
main()
|