@kolbo/kolbo-code-linux-arm64-musl 0.0.0-dev-202604161628
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/bin/kolbo +0 -0
- package/package.json +14 -0
- package/skills/brainstorming/SKILL.md +164 -0
- package/skills/brainstorming/scripts/frame-template.html +214 -0
- package/skills/brainstorming/scripts/helper.js +88 -0
- package/skills/brainstorming/scripts/server.cjs +354 -0
- package/skills/brainstorming/scripts/start-server.sh +148 -0
- package/skills/brainstorming/scripts/stop-server.sh +56 -0
- package/skills/brainstorming/spec-document-reviewer-prompt.md +49 -0
- package/skills/brainstorming/visual-companion.md +287 -0
- package/skills/color-grading/SKILL.md +152 -0
- package/skills/dispatching-parallel-agents/SKILL.md +182 -0
- package/skills/docx/.skillfish.json +10 -0
- package/skills/docx/SKILL.md +196 -0
- package/skills/docx/docx-js.md +350 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/skills/docx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/skills/docx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/skills/docx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/skills/docx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/skills/docx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/skills/docx/ooxml/schemas/mce/mc.xsd +75 -0
- package/skills/docx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
- package/skills/docx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
- package/skills/docx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
- package/skills/docx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/skills/docx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/skills/docx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/skills/docx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/skills/docx/ooxml/scripts/pack.py +159 -0
- package/skills/docx/ooxml/scripts/unpack.py +29 -0
- package/skills/docx/ooxml/scripts/validate.py +69 -0
- package/skills/docx/ooxml/scripts/validation/__init__.py +15 -0
- package/skills/docx/ooxml/scripts/validation/base.py +951 -0
- package/skills/docx/ooxml/scripts/validation/docx.py +274 -0
- package/skills/docx/ooxml/scripts/validation/pptx.py +315 -0
- package/skills/docx/ooxml/scripts/validation/redlining.py +279 -0
- package/skills/docx/ooxml.md +599 -0
- package/skills/docx/scripts/__init__.py +1 -0
- package/skills/docx/scripts/document.py +1272 -0
- package/skills/docx/scripts/templates/comments.xml +3 -0
- package/skills/docx/scripts/templates/commentsExtended.xml +3 -0
- package/skills/docx/scripts/templates/commentsExtensible.xml +3 -0
- package/skills/docx/scripts/templates/commentsIds.xml +3 -0
- package/skills/docx/scripts/templates/people.xml +3 -0
- package/skills/docx/scripts/utilities.py +374 -0
- package/skills/executing-plans/SKILL.md +70 -0
- package/skills/ffmpeg-patterns/SKILL.md +240 -0
- package/skills/finishing-a-development-branch/SKILL.md +200 -0
- package/skills/frontend-design/SKILL.md +42 -0
- package/skills/fullstack-app/SKILL.md +621 -0
- package/skills/image-prompting-guide/SKILL.md +143 -0
- package/skills/kolbo/SKILL.md +610 -0
- package/skills/music-prompting/SKILL.md +146 -0
- package/skills/pdf/.skillfish.json +10 -0
- package/skills/pdf/FORMS.md +205 -0
- package/skills/pdf/REFERENCE.md +612 -0
- package/skills/pdf/SKILL.md +293 -0
- package/skills/pdf/scripts/check_bounding_boxes.py +70 -0
- package/skills/pdf/scripts/check_bounding_boxes_test.py +226 -0
- package/skills/pdf/scripts/check_fillable_fields.py +12 -0
- package/skills/pdf/scripts/convert_pdf_to_images.py +35 -0
- package/skills/pdf/scripts/create_validation_image.py +41 -0
- package/skills/pdf/scripts/extract_form_field_info.py +152 -0
- package/skills/pdf/scripts/fill_fillable_fields.py +114 -0
- package/skills/pdf/scripts/fill_pdf_form_with_annotations.py +108 -0
- package/skills/photo-studio/SKILL.md +130 -0
- package/skills/pptx/.skillfish.json +10 -0
- package/skills/pptx/SKILL.md +483 -0
- package/skills/pptx/html2pptx.md +626 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
- package/skills/pptx/ooxml/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
- package/skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
- package/skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
- package/skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
- package/skills/pptx/ooxml/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
- package/skills/pptx/ooxml/schemas/mce/mc.xsd +75 -0
- package/skills/pptx/ooxml/schemas/microsoft/wml-2010.xsd +560 -0
- package/skills/pptx/ooxml/schemas/microsoft/wml-2012.xsd +67 -0
- package/skills/pptx/ooxml/schemas/microsoft/wml-2018.xsd +14 -0
- package/skills/pptx/ooxml/schemas/microsoft/wml-cex-2018.xsd +20 -0
- package/skills/pptx/ooxml/schemas/microsoft/wml-cid-2016.xsd +13 -0
- package/skills/pptx/ooxml/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
- package/skills/pptx/ooxml/schemas/microsoft/wml-symex-2015.xsd +8 -0
- package/skills/pptx/ooxml/scripts/pack.py +159 -0
- package/skills/pptx/ooxml/scripts/unpack.py +29 -0
- package/skills/pptx/ooxml/scripts/validate.py +69 -0
- package/skills/pptx/ooxml/scripts/validation/__init__.py +15 -0
- package/skills/pptx/ooxml/scripts/validation/base.py +951 -0
- package/skills/pptx/ooxml/scripts/validation/docx.py +274 -0
- package/skills/pptx/ooxml/scripts/validation/pptx.py +315 -0
- package/skills/pptx/ooxml/scripts/validation/redlining.py +279 -0
- package/skills/pptx/ooxml.md +427 -0
- package/skills/pptx/scripts/html2pptx.js +995 -0
- package/skills/pptx/scripts/inventory.py +1020 -0
- package/skills/pptx/scripts/rearrange.py +231 -0
- package/skills/pptx/scripts/replace.py +385 -0
- package/skills/pptx/scripts/thumbnail.py +450 -0
- package/skills/production-review/SKILL.md +152 -0
- package/skills/receiving-code-review/SKILL.md +213 -0
- package/skills/remotion-best-practices/SKILL.md +62 -0
- package/skills/remotion-best-practices/rules/3d.md +86 -0
- package/skills/remotion-best-practices/rules/animations.md +27 -0
- package/skills/remotion-best-practices/rules/assets/charts-bar-chart.tsx +173 -0
- package/skills/remotion-best-practices/rules/assets/text-animations-typewriter.tsx +100 -0
- package/skills/remotion-best-practices/rules/assets/text-animations-word-highlight.tsx +103 -0
- package/skills/remotion-best-practices/rules/assets.md +78 -0
- package/skills/remotion-best-practices/rules/audio-visualization.md +198 -0
- package/skills/remotion-best-practices/rules/audio.md +169 -0
- package/skills/remotion-best-practices/rules/calculate-metadata.md +134 -0
- package/skills/remotion-best-practices/rules/can-decode.md +81 -0
- package/skills/remotion-best-practices/rules/charts.md +120 -0
- package/skills/remotion-best-practices/rules/compositions.md +154 -0
- package/skills/remotion-best-practices/rules/display-captions.md +184 -0
- package/skills/remotion-best-practices/rules/extract-frames.md +229 -0
- package/skills/remotion-best-practices/rules/ffmpeg.md +38 -0
- package/skills/remotion-best-practices/rules/fonts.md +152 -0
- package/skills/remotion-best-practices/rules/get-audio-duration.md +58 -0
- package/skills/remotion-best-practices/rules/get-video-dimensions.md +68 -0
- package/skills/remotion-best-practices/rules/get-video-duration.md +60 -0
- package/skills/remotion-best-practices/rules/gifs.md +141 -0
- package/skills/remotion-best-practices/rules/images.md +134 -0
- package/skills/remotion-best-practices/rules/import-srt-captions.md +69 -0
- package/skills/remotion-best-practices/rules/light-leaks.md +73 -0
- package/skills/remotion-best-practices/rules/lottie.md +70 -0
- package/skills/remotion-best-practices/rules/maps.md +412 -0
- package/skills/remotion-best-practices/rules/measuring-dom-nodes.md +34 -0
- package/skills/remotion-best-practices/rules/measuring-text.md +140 -0
- package/skills/remotion-best-practices/rules/motion-design.md +215 -0
- package/skills/remotion-best-practices/rules/parameters.md +109 -0
- package/skills/remotion-best-practices/rules/sequencing.md +118 -0
- package/skills/remotion-best-practices/rules/sfx.md +30 -0
- package/skills/remotion-best-practices/rules/subtitles.md +36 -0
- package/skills/remotion-best-practices/rules/tailwind.md +11 -0
- package/skills/remotion-best-practices/rules/text-animations.md +20 -0
- package/skills/remotion-best-practices/rules/timing.md +179 -0
- package/skills/remotion-best-practices/rules/transcribe-captions.md +70 -0
- package/skills/remotion-best-practices/rules/transitions.md +197 -0
- package/skills/remotion-best-practices/rules/transparent-videos.md +106 -0
- package/skills/remotion-best-practices/rules/trimming.md +51 -0
- package/skills/remotion-best-practices/rules/videos.md +171 -0
- package/skills/remotion-best-practices/rules/voiceover.md +99 -0
- package/skills/requesting-code-review/SKILL.md +105 -0
- package/skills/requesting-code-review/code-reviewer.md +146 -0
- package/skills/short-form-video/SKILL.md +168 -0
- package/skills/sound-design/SKILL.md +154 -0
- package/skills/storytelling/SKILL.md +139 -0
- package/skills/subagent-driven-development/SKILL.md +277 -0
- package/skills/subagent-driven-development/code-quality-reviewer-prompt.md +26 -0
- package/skills/subagent-driven-development/implementer-prompt.md +113 -0
- package/skills/subagent-driven-development/spec-reviewer-prompt.md +61 -0
- package/skills/subtitle-production/SKILL.md +244 -0
- package/skills/subtitle-production/reference/burn_to_video.py +222 -0
- package/skills/subtitle-production/reference/export_srts.py +127 -0
- package/skills/subtitle-production/reference/gen_srt.py +42 -0
- package/skills/supabase/.skillfish.json +10 -0
- package/skills/supabase/SKILL.md +106 -0
- package/skills/supabase/assets/feedback-issue-template.md +17 -0
- package/skills/supabase/references/skill-feedback.md +17 -0
- package/skills/supabase-postgres-best-practices/.skillfish.json +10 -0
- package/skills/supabase-postgres-best-practices/SKILL.md +64 -0
- package/skills/supabase-postgres-best-practices/references/_contributing.md +170 -0
- package/skills/supabase-postgres-best-practices/references/_sections.md +39 -0
- package/skills/supabase-postgres-best-practices/references/_template.md +34 -0
- package/skills/supabase-postgres-best-practices/references/advanced-full-text-search.md +55 -0
- package/skills/supabase-postgres-best-practices/references/advanced-jsonb-indexing.md +49 -0
- package/skills/supabase-postgres-best-practices/references/conn-idle-timeout.md +46 -0
- package/skills/supabase-postgres-best-practices/references/conn-limits.md +44 -0
- package/skills/supabase-postgres-best-practices/references/conn-pooling.md +41 -0
- package/skills/supabase-postgres-best-practices/references/conn-prepared-statements.md +46 -0
- package/skills/supabase-postgres-best-practices/references/data-batch-inserts.md +54 -0
- package/skills/supabase-postgres-best-practices/references/data-n-plus-one.md +53 -0
- package/skills/supabase-postgres-best-practices/references/data-pagination.md +50 -0
- package/skills/supabase-postgres-best-practices/references/data-upsert.md +50 -0
- package/skills/supabase-postgres-best-practices/references/lock-advisory.md +56 -0
- package/skills/supabase-postgres-best-practices/references/lock-deadlock-prevention.md +68 -0
- package/skills/supabase-postgres-best-practices/references/lock-short-transactions.md +50 -0
- package/skills/supabase-postgres-best-practices/references/lock-skip-locked.md +54 -0
- package/skills/supabase-postgres-best-practices/references/monitor-explain-analyze.md +45 -0
- package/skills/supabase-postgres-best-practices/references/monitor-pg-stat-statements.md +55 -0
- package/skills/supabase-postgres-best-practices/references/monitor-vacuum-analyze.md +55 -0
- package/skills/supabase-postgres-best-practices/references/query-composite-indexes.md +44 -0
- package/skills/supabase-postgres-best-practices/references/query-covering-indexes.md +40 -0
- package/skills/supabase-postgres-best-practices/references/query-index-types.md +48 -0
- package/skills/supabase-postgres-best-practices/references/query-missing-indexes.md +43 -0
- package/skills/supabase-postgres-best-practices/references/query-partial-indexes.md +45 -0
- package/skills/supabase-postgres-best-practices/references/schema-constraints.md +80 -0
- package/skills/supabase-postgres-best-practices/references/schema-data-types.md +46 -0
- package/skills/supabase-postgres-best-practices/references/schema-foreign-key-indexes.md +59 -0
- package/skills/supabase-postgres-best-practices/references/schema-lowercase-identifiers.md +55 -0
- package/skills/supabase-postgres-best-practices/references/schema-partitioning.md +55 -0
- package/skills/supabase-postgres-best-practices/references/schema-primary-keys.md +61 -0
- package/skills/supabase-postgres-best-practices/references/security-privileges.md +54 -0
- package/skills/supabase-postgres-best-practices/references/security-rls-basics.md +50 -0
- package/skills/supabase-postgres-best-practices/references/security-rls-performance.md +57 -0
- package/skills/supabase-quickstart/SKILL.md +400 -0
- package/skills/systematic-debugging/CREATION-LOG.md +119 -0
- package/skills/systematic-debugging/SKILL.md +296 -0
- package/skills/systematic-debugging/condition-based-waiting-example.ts +158 -0
- package/skills/systematic-debugging/condition-based-waiting.md +115 -0
- package/skills/systematic-debugging/defense-in-depth.md +122 -0
- package/skills/systematic-debugging/find-polluter.sh +63 -0
- package/skills/systematic-debugging/root-cause-tracing.md +169 -0
- package/skills/systematic-debugging/test-academic.md +14 -0
- package/skills/systematic-debugging/test-pressure-1.md +58 -0
- package/skills/systematic-debugging/test-pressure-2.md +68 -0
- package/skills/systematic-debugging/test-pressure-3.md +69 -0
- package/skills/test-driven-development/SKILL.md +371 -0
- package/skills/test-driven-development/testing-anti-patterns.md +299 -0
- package/skills/typography-video/SKILL.md +182 -0
- package/skills/typography-video/reference/KineticTitleScene.tsx +345 -0
- package/skills/using-git-worktrees/SKILL.md +218 -0
- package/skills/using-superpowers/SKILL.md +115 -0
- package/skills/using-superpowers/references/codex-tools.md +100 -0
- package/skills/using-superpowers/references/gemini-tools.md +33 -0
- package/skills/verification-before-completion/SKILL.md +139 -0
- package/skills/video-editing/SKILL.md +128 -0
- package/skills/video-production/SKILL.md +247 -0
- package/skills/video-prompting-guide/SKILL.md +268 -0
- package/skills/writing-plans/SKILL.md +152 -0
- package/skills/writing-plans/plan-document-reviewer-prompt.md +49 -0
- package/skills/writing-skills/SKILL.md +655 -0
- package/skills/writing-skills/anthropic-best-practices.md +1150 -0
- package/skills/writing-skills/examples/CLAUDE_MD_TESTING.md +189 -0
- package/skills/writing-skills/graphviz-conventions.dot +172 -0
- package/skills/writing-skills/persuasion-principles.md +187 -0
- package/skills/writing-skills/render-graphs.js +168 -0
- package/skills/writing-skills/testing-skills-with-subagents.md +384 -0
- package/skills/xlsx/.skillfish.json +10 -0
- package/skills/xlsx/SKILL.md +288 -0
- package/skills/xlsx/recalc.py +178 -0
- package/skills/youtube-clipper/SKILL.md +187 -0
|
@@ -0,0 +1,450 @@
|
|
|
1
|
+
#!/usr/bin/env python3
|
|
2
|
+
"""
|
|
3
|
+
Create thumbnail grids from PowerPoint presentation slides.
|
|
4
|
+
|
|
5
|
+
Creates a grid layout of slide thumbnails with configurable columns (max 6).
|
|
6
|
+
Each grid contains up to cols×(cols+1) images. For presentations with more
|
|
7
|
+
slides, multiple numbered grid files are created automatically.
|
|
8
|
+
|
|
9
|
+
The program outputs the names of all files created.
|
|
10
|
+
|
|
11
|
+
Output:
|
|
12
|
+
- Single grid: {prefix}.jpg (if slides fit in one grid)
|
|
13
|
+
- Multiple grids: {prefix}-1.jpg, {prefix}-2.jpg, etc.
|
|
14
|
+
|
|
15
|
+
Grid limits by column count:
|
|
16
|
+
- 3 cols: max 12 slides per grid (3×4)
|
|
17
|
+
- 4 cols: max 20 slides per grid (4×5)
|
|
18
|
+
- 5 cols: max 30 slides per grid (5×6) [default]
|
|
19
|
+
- 6 cols: max 42 slides per grid (6×7)
|
|
20
|
+
|
|
21
|
+
Usage:
|
|
22
|
+
python thumbnail.py input.pptx [output_prefix] [--cols N] [--outline-placeholders]
|
|
23
|
+
|
|
24
|
+
Examples:
|
|
25
|
+
python thumbnail.py presentation.pptx
|
|
26
|
+
# Creates: thumbnails.jpg (using default prefix)
|
|
27
|
+
# Outputs:
|
|
28
|
+
# Created 1 grid(s):
|
|
29
|
+
# - thumbnails.jpg
|
|
30
|
+
|
|
31
|
+
python thumbnail.py large-deck.pptx grid --cols 4
|
|
32
|
+
# Creates: grid-1.jpg, grid-2.jpg, grid-3.jpg
|
|
33
|
+
# Outputs:
|
|
34
|
+
# Created 3 grid(s):
|
|
35
|
+
# - grid-1.jpg
|
|
36
|
+
# - grid-2.jpg
|
|
37
|
+
# - grid-3.jpg
|
|
38
|
+
|
|
39
|
+
python thumbnail.py template.pptx analysis --outline-placeholders
|
|
40
|
+
# Creates thumbnail grids with red outlines around text placeholders
|
|
41
|
+
"""
|
|
42
|
+
|
|
43
|
+
import argparse
|
|
44
|
+
import subprocess
|
|
45
|
+
import sys
|
|
46
|
+
import tempfile
|
|
47
|
+
from pathlib import Path
|
|
48
|
+
|
|
49
|
+
from inventory import extract_text_inventory
|
|
50
|
+
from PIL import Image, ImageDraw, ImageFont
|
|
51
|
+
from pptx import Presentation
|
|
52
|
+
|
|
53
|
+
# Constants
|
|
54
|
+
THUMBNAIL_WIDTH = 300 # Fixed thumbnail width in pixels
|
|
55
|
+
CONVERSION_DPI = 100 # DPI for PDF to image conversion
|
|
56
|
+
MAX_COLS = 6 # Maximum number of columns
|
|
57
|
+
DEFAULT_COLS = 5 # Default number of columns
|
|
58
|
+
JPEG_QUALITY = 95 # JPEG compression quality
|
|
59
|
+
|
|
60
|
+
# Grid layout constants
|
|
61
|
+
GRID_PADDING = 20 # Padding between thumbnails
|
|
62
|
+
BORDER_WIDTH = 2 # Border width around thumbnails
|
|
63
|
+
FONT_SIZE_RATIO = 0.12 # Font size as fraction of thumbnail width
|
|
64
|
+
LABEL_PADDING_RATIO = 0.4 # Label padding as fraction of font size
|
|
65
|
+
|
|
66
|
+
|
|
67
|
+
def main():
|
|
68
|
+
parser = argparse.ArgumentParser(
|
|
69
|
+
description="Create thumbnail grids from PowerPoint slides."
|
|
70
|
+
)
|
|
71
|
+
parser.add_argument("input", help="Input PowerPoint file (.pptx)")
|
|
72
|
+
parser.add_argument(
|
|
73
|
+
"output_prefix",
|
|
74
|
+
nargs="?",
|
|
75
|
+
default="thumbnails",
|
|
76
|
+
help="Output prefix for image files (default: thumbnails, will create prefix.jpg or prefix-N.jpg)",
|
|
77
|
+
)
|
|
78
|
+
parser.add_argument(
|
|
79
|
+
"--cols",
|
|
80
|
+
type=int,
|
|
81
|
+
default=DEFAULT_COLS,
|
|
82
|
+
help=f"Number of columns (default: {DEFAULT_COLS}, max: {MAX_COLS})",
|
|
83
|
+
)
|
|
84
|
+
parser.add_argument(
|
|
85
|
+
"--outline-placeholders",
|
|
86
|
+
action="store_true",
|
|
87
|
+
help="Outline text placeholders with a colored border",
|
|
88
|
+
)
|
|
89
|
+
|
|
90
|
+
args = parser.parse_args()
|
|
91
|
+
|
|
92
|
+
# Validate columns
|
|
93
|
+
cols = min(args.cols, MAX_COLS)
|
|
94
|
+
if args.cols > MAX_COLS:
|
|
95
|
+
print(f"Warning: Columns limited to {MAX_COLS} (requested {args.cols})")
|
|
96
|
+
|
|
97
|
+
# Validate input
|
|
98
|
+
input_path = Path(args.input)
|
|
99
|
+
if not input_path.exists() or input_path.suffix.lower() != ".pptx":
|
|
100
|
+
print(f"Error: Invalid PowerPoint file: {args.input}")
|
|
101
|
+
sys.exit(1)
|
|
102
|
+
|
|
103
|
+
# Construct output path (always JPG)
|
|
104
|
+
output_path = Path(f"{args.output_prefix}.jpg")
|
|
105
|
+
|
|
106
|
+
print(f"Processing: {args.input}")
|
|
107
|
+
|
|
108
|
+
try:
|
|
109
|
+
with tempfile.TemporaryDirectory() as temp_dir:
|
|
110
|
+
# Get placeholder regions if outlining is enabled
|
|
111
|
+
placeholder_regions = None
|
|
112
|
+
slide_dimensions = None
|
|
113
|
+
if args.outline_placeholders:
|
|
114
|
+
print("Extracting placeholder regions...")
|
|
115
|
+
placeholder_regions, slide_dimensions = get_placeholder_regions(
|
|
116
|
+
input_path
|
|
117
|
+
)
|
|
118
|
+
if placeholder_regions:
|
|
119
|
+
print(f"Found placeholders on {len(placeholder_regions)} slides")
|
|
120
|
+
|
|
121
|
+
# Convert slides to images
|
|
122
|
+
slide_images = convert_to_images(input_path, Path(temp_dir), CONVERSION_DPI)
|
|
123
|
+
if not slide_images:
|
|
124
|
+
print("Error: No slides found")
|
|
125
|
+
sys.exit(1)
|
|
126
|
+
|
|
127
|
+
print(f"Found {len(slide_images)} slides")
|
|
128
|
+
|
|
129
|
+
# Create grids (max cols×(cols+1) images per grid)
|
|
130
|
+
grid_files = create_grids(
|
|
131
|
+
slide_images,
|
|
132
|
+
cols,
|
|
133
|
+
THUMBNAIL_WIDTH,
|
|
134
|
+
output_path,
|
|
135
|
+
placeholder_regions,
|
|
136
|
+
slide_dimensions,
|
|
137
|
+
)
|
|
138
|
+
|
|
139
|
+
# Print saved files
|
|
140
|
+
print(f"Created {len(grid_files)} grid(s):")
|
|
141
|
+
for grid_file in grid_files:
|
|
142
|
+
print(f" - {grid_file}")
|
|
143
|
+
|
|
144
|
+
except Exception as e:
|
|
145
|
+
print(f"Error: {e}")
|
|
146
|
+
sys.exit(1)
|
|
147
|
+
|
|
148
|
+
|
|
149
|
+
def create_hidden_slide_placeholder(size):
|
|
150
|
+
"""Create placeholder image for hidden slides."""
|
|
151
|
+
img = Image.new("RGB", size, color="#F0F0F0")
|
|
152
|
+
draw = ImageDraw.Draw(img)
|
|
153
|
+
line_width = max(5, min(size) // 100)
|
|
154
|
+
draw.line([(0, 0), size], fill="#CCCCCC", width=line_width)
|
|
155
|
+
draw.line([(size[0], 0), (0, size[1])], fill="#CCCCCC", width=line_width)
|
|
156
|
+
return img
|
|
157
|
+
|
|
158
|
+
|
|
159
|
+
def get_placeholder_regions(pptx_path):
|
|
160
|
+
"""Extract ALL text regions from the presentation.
|
|
161
|
+
|
|
162
|
+
Returns a tuple of (placeholder_regions, slide_dimensions).
|
|
163
|
+
text_regions is a dict mapping slide indices to lists of text regions.
|
|
164
|
+
Each region is a dict with 'left', 'top', 'width', 'height' in inches.
|
|
165
|
+
slide_dimensions is a tuple of (width_inches, height_inches).
|
|
166
|
+
"""
|
|
167
|
+
prs = Presentation(str(pptx_path))
|
|
168
|
+
inventory = extract_text_inventory(pptx_path, prs)
|
|
169
|
+
placeholder_regions = {}
|
|
170
|
+
|
|
171
|
+
# Get actual slide dimensions in inches (EMU to inches conversion)
|
|
172
|
+
slide_width_inches = (prs.slide_width or 9144000) / 914400.0
|
|
173
|
+
slide_height_inches = (prs.slide_height or 5143500) / 914400.0
|
|
174
|
+
|
|
175
|
+
for slide_key, shapes in inventory.items():
|
|
176
|
+
# Extract slide index from "slide-N" format
|
|
177
|
+
slide_idx = int(slide_key.split("-")[1])
|
|
178
|
+
regions = []
|
|
179
|
+
|
|
180
|
+
for shape_key, shape_data in shapes.items():
|
|
181
|
+
# The inventory only contains shapes with text, so all shapes should be highlighted
|
|
182
|
+
regions.append(
|
|
183
|
+
{
|
|
184
|
+
"left": shape_data.left,
|
|
185
|
+
"top": shape_data.top,
|
|
186
|
+
"width": shape_data.width,
|
|
187
|
+
"height": shape_data.height,
|
|
188
|
+
}
|
|
189
|
+
)
|
|
190
|
+
|
|
191
|
+
if regions:
|
|
192
|
+
placeholder_regions[slide_idx] = regions
|
|
193
|
+
|
|
194
|
+
return placeholder_regions, (slide_width_inches, slide_height_inches)
|
|
195
|
+
|
|
196
|
+
|
|
197
|
+
def convert_to_images(pptx_path, temp_dir, dpi):
|
|
198
|
+
"""Convert PowerPoint to images via PDF, handling hidden slides."""
|
|
199
|
+
# Detect hidden slides
|
|
200
|
+
print("Analyzing presentation...")
|
|
201
|
+
prs = Presentation(str(pptx_path))
|
|
202
|
+
total_slides = len(prs.slides)
|
|
203
|
+
|
|
204
|
+
# Find hidden slides (1-based indexing for display)
|
|
205
|
+
hidden_slides = {
|
|
206
|
+
idx + 1
|
|
207
|
+
for idx, slide in enumerate(prs.slides)
|
|
208
|
+
if slide.element.get("show") == "0"
|
|
209
|
+
}
|
|
210
|
+
|
|
211
|
+
print(f"Total slides: {total_slides}")
|
|
212
|
+
if hidden_slides:
|
|
213
|
+
print(f"Hidden slides: {sorted(hidden_slides)}")
|
|
214
|
+
|
|
215
|
+
pdf_path = temp_dir / f"{pptx_path.stem}.pdf"
|
|
216
|
+
|
|
217
|
+
# Convert to PDF
|
|
218
|
+
print("Converting to PDF...")
|
|
219
|
+
result = subprocess.run(
|
|
220
|
+
[
|
|
221
|
+
"soffice",
|
|
222
|
+
"--headless",
|
|
223
|
+
"--convert-to",
|
|
224
|
+
"pdf",
|
|
225
|
+
"--outdir",
|
|
226
|
+
str(temp_dir),
|
|
227
|
+
str(pptx_path),
|
|
228
|
+
],
|
|
229
|
+
capture_output=True,
|
|
230
|
+
text=True,
|
|
231
|
+
)
|
|
232
|
+
if result.returncode != 0 or not pdf_path.exists():
|
|
233
|
+
raise RuntimeError("PDF conversion failed")
|
|
234
|
+
|
|
235
|
+
# Convert PDF to images
|
|
236
|
+
print(f"Converting to images at {dpi} DPI...")
|
|
237
|
+
result = subprocess.run(
|
|
238
|
+
["pdftoppm", "-jpeg", "-r", str(dpi), str(pdf_path), str(temp_dir / "slide")],
|
|
239
|
+
capture_output=True,
|
|
240
|
+
text=True,
|
|
241
|
+
)
|
|
242
|
+
if result.returncode != 0:
|
|
243
|
+
raise RuntimeError("Image conversion failed")
|
|
244
|
+
|
|
245
|
+
visible_images = sorted(temp_dir.glob("slide-*.jpg"))
|
|
246
|
+
|
|
247
|
+
# Create full list with placeholders for hidden slides
|
|
248
|
+
all_images = []
|
|
249
|
+
visible_idx = 0
|
|
250
|
+
|
|
251
|
+
# Get placeholder dimensions from first visible slide
|
|
252
|
+
if visible_images:
|
|
253
|
+
with Image.open(visible_images[0]) as img:
|
|
254
|
+
placeholder_size = img.size
|
|
255
|
+
else:
|
|
256
|
+
placeholder_size = (1920, 1080)
|
|
257
|
+
|
|
258
|
+
for slide_num in range(1, total_slides + 1):
|
|
259
|
+
if slide_num in hidden_slides:
|
|
260
|
+
# Create placeholder image for hidden slide
|
|
261
|
+
placeholder_path = temp_dir / f"hidden-{slide_num:03d}.jpg"
|
|
262
|
+
placeholder_img = create_hidden_slide_placeholder(placeholder_size)
|
|
263
|
+
placeholder_img.save(placeholder_path, "JPEG")
|
|
264
|
+
all_images.append(placeholder_path)
|
|
265
|
+
else:
|
|
266
|
+
# Use the actual visible slide image
|
|
267
|
+
if visible_idx < len(visible_images):
|
|
268
|
+
all_images.append(visible_images[visible_idx])
|
|
269
|
+
visible_idx += 1
|
|
270
|
+
|
|
271
|
+
return all_images
|
|
272
|
+
|
|
273
|
+
|
|
274
|
+
def create_grids(
|
|
275
|
+
image_paths,
|
|
276
|
+
cols,
|
|
277
|
+
width,
|
|
278
|
+
output_path,
|
|
279
|
+
placeholder_regions=None,
|
|
280
|
+
slide_dimensions=None,
|
|
281
|
+
):
|
|
282
|
+
"""Create multiple thumbnail grids from slide images, max cols×(cols+1) images per grid."""
|
|
283
|
+
# Maximum images per grid is cols × (cols + 1) for better proportions
|
|
284
|
+
max_images_per_grid = cols * (cols + 1)
|
|
285
|
+
grid_files = []
|
|
286
|
+
|
|
287
|
+
print(
|
|
288
|
+
f"Creating grids with {cols} columns (max {max_images_per_grid} images per grid)"
|
|
289
|
+
)
|
|
290
|
+
|
|
291
|
+
# Split images into chunks
|
|
292
|
+
for chunk_idx, start_idx in enumerate(
|
|
293
|
+
range(0, len(image_paths), max_images_per_grid)
|
|
294
|
+
):
|
|
295
|
+
end_idx = min(start_idx + max_images_per_grid, len(image_paths))
|
|
296
|
+
chunk_images = image_paths[start_idx:end_idx]
|
|
297
|
+
|
|
298
|
+
# Create grid for this chunk
|
|
299
|
+
grid = create_grid(
|
|
300
|
+
chunk_images, cols, width, start_idx, placeholder_regions, slide_dimensions
|
|
301
|
+
)
|
|
302
|
+
|
|
303
|
+
# Generate output filename
|
|
304
|
+
if len(image_paths) <= max_images_per_grid:
|
|
305
|
+
# Single grid - use base filename without suffix
|
|
306
|
+
grid_filename = output_path
|
|
307
|
+
else:
|
|
308
|
+
# Multiple grids - insert index before extension with dash
|
|
309
|
+
stem = output_path.stem
|
|
310
|
+
suffix = output_path.suffix
|
|
311
|
+
grid_filename = output_path.parent / f"{stem}-{chunk_idx + 1}{suffix}"
|
|
312
|
+
|
|
313
|
+
# Save grid
|
|
314
|
+
grid_filename.parent.mkdir(parents=True, exist_ok=True)
|
|
315
|
+
grid.save(str(grid_filename), quality=JPEG_QUALITY)
|
|
316
|
+
grid_files.append(str(grid_filename))
|
|
317
|
+
|
|
318
|
+
return grid_files
|
|
319
|
+
|
|
320
|
+
|
|
321
|
+
def create_grid(
|
|
322
|
+
image_paths,
|
|
323
|
+
cols,
|
|
324
|
+
width,
|
|
325
|
+
start_slide_num=0,
|
|
326
|
+
placeholder_regions=None,
|
|
327
|
+
slide_dimensions=None,
|
|
328
|
+
):
|
|
329
|
+
"""Create thumbnail grid from slide images with optional placeholder outlining."""
|
|
330
|
+
font_size = int(width * FONT_SIZE_RATIO)
|
|
331
|
+
label_padding = int(font_size * LABEL_PADDING_RATIO)
|
|
332
|
+
|
|
333
|
+
# Get dimensions
|
|
334
|
+
with Image.open(image_paths[0]) as img:
|
|
335
|
+
aspect = img.height / img.width
|
|
336
|
+
height = int(width * aspect)
|
|
337
|
+
|
|
338
|
+
# Calculate grid size
|
|
339
|
+
rows = (len(image_paths) + cols - 1) // cols
|
|
340
|
+
grid_w = cols * width + (cols + 1) * GRID_PADDING
|
|
341
|
+
grid_h = rows * (height + font_size + label_padding * 2) + (rows + 1) * GRID_PADDING
|
|
342
|
+
|
|
343
|
+
# Create grid
|
|
344
|
+
grid = Image.new("RGB", (grid_w, grid_h), "white")
|
|
345
|
+
draw = ImageDraw.Draw(grid)
|
|
346
|
+
|
|
347
|
+
# Load font with size based on thumbnail width
|
|
348
|
+
try:
|
|
349
|
+
# Use Pillow's default font with size
|
|
350
|
+
font = ImageFont.load_default(size=font_size)
|
|
351
|
+
except Exception:
|
|
352
|
+
# Fall back to basic default font if size parameter not supported
|
|
353
|
+
font = ImageFont.load_default()
|
|
354
|
+
|
|
355
|
+
# Place thumbnails
|
|
356
|
+
for i, img_path in enumerate(image_paths):
|
|
357
|
+
row, col = i // cols, i % cols
|
|
358
|
+
x = col * width + (col + 1) * GRID_PADDING
|
|
359
|
+
y_base = (
|
|
360
|
+
row * (height + font_size + label_padding * 2) + (row + 1) * GRID_PADDING
|
|
361
|
+
)
|
|
362
|
+
|
|
363
|
+
# Add label with actual slide number
|
|
364
|
+
label = f"{start_slide_num + i}"
|
|
365
|
+
bbox = draw.textbbox((0, 0), label, font=font)
|
|
366
|
+
text_w = bbox[2] - bbox[0]
|
|
367
|
+
draw.text(
|
|
368
|
+
(x + (width - text_w) // 2, y_base + label_padding),
|
|
369
|
+
label,
|
|
370
|
+
fill="black",
|
|
371
|
+
font=font,
|
|
372
|
+
)
|
|
373
|
+
|
|
374
|
+
# Add thumbnail below label with proportional spacing
|
|
375
|
+
y_thumbnail = y_base + label_padding + font_size + label_padding
|
|
376
|
+
|
|
377
|
+
with Image.open(img_path) as img:
|
|
378
|
+
# Get original dimensions before thumbnail
|
|
379
|
+
orig_w, orig_h = img.size
|
|
380
|
+
|
|
381
|
+
# Apply placeholder outlines if enabled
|
|
382
|
+
if placeholder_regions and (start_slide_num + i) in placeholder_regions:
|
|
383
|
+
# Convert to RGBA for transparency support
|
|
384
|
+
if img.mode != "RGBA":
|
|
385
|
+
img = img.convert("RGBA")
|
|
386
|
+
|
|
387
|
+
# Get the regions for this slide
|
|
388
|
+
regions = placeholder_regions[start_slide_num + i]
|
|
389
|
+
|
|
390
|
+
# Calculate scale factors using actual slide dimensions
|
|
391
|
+
if slide_dimensions:
|
|
392
|
+
slide_width_inches, slide_height_inches = slide_dimensions
|
|
393
|
+
else:
|
|
394
|
+
# Fallback: estimate from image size at CONVERSION_DPI
|
|
395
|
+
slide_width_inches = orig_w / CONVERSION_DPI
|
|
396
|
+
slide_height_inches = orig_h / CONVERSION_DPI
|
|
397
|
+
|
|
398
|
+
x_scale = orig_w / slide_width_inches
|
|
399
|
+
y_scale = orig_h / slide_height_inches
|
|
400
|
+
|
|
401
|
+
# Create a highlight overlay
|
|
402
|
+
overlay = Image.new("RGBA", img.size, (255, 255, 255, 0))
|
|
403
|
+
overlay_draw = ImageDraw.Draw(overlay)
|
|
404
|
+
|
|
405
|
+
# Highlight each placeholder region
|
|
406
|
+
for region in regions:
|
|
407
|
+
# Convert from inches to pixels in the original image
|
|
408
|
+
px_left = int(region["left"] * x_scale)
|
|
409
|
+
px_top = int(region["top"] * y_scale)
|
|
410
|
+
px_width = int(region["width"] * x_scale)
|
|
411
|
+
px_height = int(region["height"] * y_scale)
|
|
412
|
+
|
|
413
|
+
# Draw highlight outline with red color and thick stroke
|
|
414
|
+
# Using a bright red outline instead of fill
|
|
415
|
+
stroke_width = max(
|
|
416
|
+
5, min(orig_w, orig_h) // 150
|
|
417
|
+
) # Thicker proportional stroke width
|
|
418
|
+
overlay_draw.rectangle(
|
|
419
|
+
[(px_left, px_top), (px_left + px_width, px_top + px_height)],
|
|
420
|
+
outline=(255, 0, 0, 255), # Bright red, fully opaque
|
|
421
|
+
width=stroke_width,
|
|
422
|
+
)
|
|
423
|
+
|
|
424
|
+
# Composite the overlay onto the image using alpha blending
|
|
425
|
+
img = Image.alpha_composite(img, overlay)
|
|
426
|
+
# Convert back to RGB for JPEG saving
|
|
427
|
+
img = img.convert("RGB")
|
|
428
|
+
|
|
429
|
+
img.thumbnail((width, height), Image.Resampling.LANCZOS)
|
|
430
|
+
w, h = img.size
|
|
431
|
+
tx = x + (width - w) // 2
|
|
432
|
+
ty = y_thumbnail + (height - h) // 2
|
|
433
|
+
grid.paste(img, (tx, ty))
|
|
434
|
+
|
|
435
|
+
# Add border
|
|
436
|
+
if BORDER_WIDTH > 0:
|
|
437
|
+
draw.rectangle(
|
|
438
|
+
[
|
|
439
|
+
(tx - BORDER_WIDTH, ty - BORDER_WIDTH),
|
|
440
|
+
(tx + w + BORDER_WIDTH - 1, ty + h + BORDER_WIDTH - 1),
|
|
441
|
+
],
|
|
442
|
+
outline="gray",
|
|
443
|
+
width=BORDER_WIDTH,
|
|
444
|
+
)
|
|
445
|
+
|
|
446
|
+
return grid
|
|
447
|
+
|
|
448
|
+
|
|
449
|
+
if __name__ == "__main__":
|
|
450
|
+
main()
|
|
@@ -0,0 +1,152 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: production-review
|
|
3
|
+
description: >
|
|
4
|
+
Self-review quality gates for video production: post-render verification protocol, pre-delivery
|
|
5
|
+
checklist, audio verification, visual inspection, severity classification (critical/suggestion/nitpick),
|
|
6
|
+
review workflow. Use after completing any production stage to verify quality before delivery.
|
|
7
|
+
Keywords: review, quality, verification, checklist, render, audio check, video check, delivery,
|
|
8
|
+
QA, quality gate, self-review, post-render
|
|
9
|
+
---
|
|
10
|
+
|
|
11
|
+
# Production Review — Quality Gates
|
|
12
|
+
|
|
13
|
+
## When to Use
|
|
14
|
+
|
|
15
|
+
After completing any major production stage — especially after rendering, before delivering to the user. Read this skill and run through the relevant checklist.
|
|
16
|
+
|
|
17
|
+
## Severity Levels
|
|
18
|
+
|
|
19
|
+
| Severity | Definition | Action |
|
|
20
|
+
|----------|-----------|--------|
|
|
21
|
+
| **CRITICAL** | Breaks the output, incomplete, or dangerously wrong | Must fix. Blocks delivery. |
|
|
22
|
+
| **SUGGESTION** | Improves quality significantly but doesn't block | Note it, fix if time allows |
|
|
23
|
+
| **NITPICK** | Nice-to-have polish | Log it, move on |
|
|
24
|
+
|
|
25
|
+
## Decision Flow
|
|
26
|
+
|
|
27
|
+
1. Run the relevant checklist below
|
|
28
|
+
2. Count critical findings
|
|
29
|
+
3. **0 critical** → PASS (note suggestions)
|
|
30
|
+
4. **1+ critical** → REVISE (max 2 revision rounds)
|
|
31
|
+
5. After 2 rounds, still critical → PASS_WITH_WARNINGS (inform user of known issues)
|
|
32
|
+
|
|
33
|
+
---
|
|
34
|
+
|
|
35
|
+
## Post-Render Verification (Video)
|
|
36
|
+
|
|
37
|
+
### Step 1: Probe the Output (GATE — blocks all other steps)
|
|
38
|
+
```bash
|
|
39
|
+
ffprobe -v quiet -print_format json -show_format -show_streams rendered_video.mp4
|
|
40
|
+
```
|
|
41
|
+
|
|
42
|
+
Verify ALL of:
|
|
43
|
+
- [ ] Video stream exists with correct resolution and FPS
|
|
44
|
+
- [ ] **Audio stream exists** — if missing, STOP. Fix audio config, re-render
|
|
45
|
+
- [ ] Duration within +/-5% of target
|
|
46
|
+
- [ ] File size is reasonable (not 0 bytes, not suspiciously small)
|
|
47
|
+
|
|
48
|
+
**If audio stream is missing, do NOT proceed.** Most common cause: audio sources mixed externally but never embedded in the composition.
|
|
49
|
+
|
|
50
|
+
### Step 2: Extract Review Frames
|
|
51
|
+
Sample frames at scene midpoints and visually inspect:
|
|
52
|
+
```bash
|
|
53
|
+
ffmpeg -i rendered_video.mp4 -vf "fps=1/5" frame_%04d.png
|
|
54
|
+
```
|
|
55
|
+
- [ ] No visual artifacts or glitches
|
|
56
|
+
- [ ] Text overlays readable and within safe zones
|
|
57
|
+
- [ ] Color grade consistent across scenes
|
|
58
|
+
- [ ] No black frames or flash frames at cuts
|
|
59
|
+
|
|
60
|
+
### Step 3: Audio Verification
|
|
61
|
+
- [ ] Play back and confirm narration is audible over music
|
|
62
|
+
- [ ] No audio pops or clicks at cut points
|
|
63
|
+
- [ ] Music volume appropriate (18-20 dB below dialogue)
|
|
64
|
+
- [ ] Audio loudness within platform target (-14 LUFS for social)
|
|
65
|
+
|
|
66
|
+
### Step 4: Present Review to User
|
|
67
|
+
Structured summary with: file stats, audio verification, visual findings, caption status.
|
|
68
|
+
|
|
69
|
+
---
|
|
70
|
+
|
|
71
|
+
## Pre-Delivery Checklist by Content Type
|
|
72
|
+
|
|
73
|
+
### Explainer Video
|
|
74
|
+
- [ ] Hook lands in first 3 seconds
|
|
75
|
+
- [ ] Core concept clearly explained (the "aha" moment)
|
|
76
|
+
- [ ] Captions present and synced
|
|
77
|
+
- [ ] Background music doesn't overpower narration
|
|
78
|
+
- [ ] Duration matches target (+/-10%)
|
|
79
|
+
- [ ] Output plays correctly on target platform
|
|
80
|
+
|
|
81
|
+
### Short-Form (TikTok/Reels/Shorts)
|
|
82
|
+
- [ ] 9:16 aspect ratio, 1080x1920
|
|
83
|
+
- [ ] Important content within safe zones (900x1400)
|
|
84
|
+
- [ ] Hook in first 1-2 seconds
|
|
85
|
+
- [ ] Captions mandatory (85% watch muted)
|
|
86
|
+
- [ ] File size under platform limit
|
|
87
|
+
- [ ] H.264 High Profile, 8+ Mbps
|
|
88
|
+
|
|
89
|
+
### Talking Head
|
|
90
|
+
- [ ] Filler words removed
|
|
91
|
+
- [ ] No awkward jump cuts (covered by B-roll or transition)
|
|
92
|
+
- [ ] Speaker's face never covered by overlays
|
|
93
|
+
- [ ] Audio clean — no background noise
|
|
94
|
+
- [ ] Eye-level framing maintained
|
|
95
|
+
|
|
96
|
+
### Music/Audio
|
|
97
|
+
- [ ] Correct duration
|
|
98
|
+
- [ ] Instrumental if for background use
|
|
99
|
+
- [ ] BPM matches content energy
|
|
100
|
+
- [ ] No clipping or distortion
|
|
101
|
+
- [ ] Loudness normalized to target
|
|
102
|
+
|
|
103
|
+
---
|
|
104
|
+
|
|
105
|
+
## Remotion-Specific Verification
|
|
106
|
+
|
|
107
|
+
Before declaring a Remotion render complete:
|
|
108
|
+
|
|
109
|
+
- [ ] Run `composition_validator` before rendering
|
|
110
|
+
- [ ] All `staticFile()` references resolve to existing assets
|
|
111
|
+
- [ ] Composition duration matches sum of scene durations minus transition overlaps
|
|
112
|
+
- [ ] No CSS animations used (must use `useCurrentFrame()` + `interpolate()`)
|
|
113
|
+
- [ ] No Tailwind `animate-*` classes (break frame-based rendering)
|
|
114
|
+
- [ ] `interpolate()` calls use `extrapolateLeft: 'clamp', extrapolateRight: 'clamp'`
|
|
115
|
+
- [ ] Audio layers in sync with visual scenes
|
|
116
|
+
- [ ] Theme colors match the active style
|
|
117
|
+
- [ ] Text scenes use Remotion components, NOT AI-generated images with text
|
|
118
|
+
|
|
119
|
+
## Review Log Format
|
|
120
|
+
|
|
121
|
+
When logging a review finding:
|
|
122
|
+
```
|
|
123
|
+
[SEVERITY] Finding description
|
|
124
|
+
- What: specific issue observed
|
|
125
|
+
- Where: timestamp or scene reference
|
|
126
|
+
- Fix: recommended action
|
|
127
|
+
```
|
|
128
|
+
|
|
129
|
+
---
|
|
130
|
+
|
|
131
|
+
## Kolbo MCP Integration
|
|
132
|
+
|
|
133
|
+
Use these tools during review:
|
|
134
|
+
|
|
135
|
+
| Review Step | Kolbo MCP Tool | What to Check |
|
|
136
|
+
|-------------|---------------|---------------|
|
|
137
|
+
| Audio verification | `transcribe_audio` | Transcribe the rendered video — if 0 words, audio is silent |
|
|
138
|
+
| Visual analysis | `chat_send_message` + Gemini | "Review this video for quality issues" |
|
|
139
|
+
| Credit check | `check_credits` | Verify budget before re-renders |
|
|
140
|
+
|
|
141
|
+
**Post-render verification with Kolbo:**
|
|
142
|
+
1. `ffprobe` the output (always first — check streams exist)
|
|
143
|
+
2. `transcribe_audio` the rendered video → compare word count to script
|
|
144
|
+
3. If word count < 80% of script → audio is cut off → investigate
|
|
145
|
+
4. `chat_send_message` with Gemini + video URL → visual quality review
|
|
146
|
+
5. Present structured findings to user
|
|
147
|
+
|
|
148
|
+
**Re-generation workflow (if review finds critical issues):**
|
|
149
|
+
1. Identify the failed asset (video clip, audio, image)
|
|
150
|
+
2. Re-generate with adjusted prompt via the appropriate Kolbo MCP tool
|
|
151
|
+
3. Re-compose with FFmpeg or Remotion
|
|
152
|
+
4. Run review again (max 2 revision rounds)
|