@aslomon/effectum 0.3.3 → 0.3.4

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.
Files changed (351) hide show
  1. package/bin/install.js +27 -0
  2. package/package.json +1 -1
  3. package/system/hooks/observe.sh +164 -0
  4. package/system/hooks/quick-diff.sh +87 -0
  5. package/system/hooks/save-results.sh +56 -0
  6. package/system/hooks/scan.sh +170 -0
  7. package/system/skills/README.md +86 -0
  8. package/system/skills/SKILL.md +117 -0
  9. package/system/skills/agents/observer.md +137 -0
  10. package/system/skills/agents/start-observer.sh +143 -0
  11. package/system/skills/algorithmic-art/LICENSE.txt +202 -0
  12. package/system/skills/algorithmic-art/SKILL.md +405 -0
  13. package/system/skills/algorithmic-art/templates/generator_template.js +223 -0
  14. package/system/skills/algorithmic-art/templates/viewer.html +599 -0
  15. package/system/skills/api-endpoint/SKILL.md +106 -0
  16. package/system/skills/canvas-design/LICENSE.txt +202 -0
  17. package/system/skills/canvas-design/SKILL.md +130 -0
  18. package/system/skills/canvas-design/canvas-fonts/ArsenalSC-OFL.txt +93 -0
  19. package/system/skills/canvas-design/canvas-fonts/ArsenalSC-Regular.ttf +0 -0
  20. package/system/skills/canvas-design/canvas-fonts/BigShoulders-Bold.ttf +0 -0
  21. package/system/skills/canvas-design/canvas-fonts/BigShoulders-OFL.txt +93 -0
  22. package/system/skills/canvas-design/canvas-fonts/BigShoulders-Regular.ttf +0 -0
  23. package/system/skills/canvas-design/canvas-fonts/Boldonse-OFL.txt +93 -0
  24. package/system/skills/canvas-design/canvas-fonts/Boldonse-Regular.ttf +0 -0
  25. package/system/skills/canvas-design/canvas-fonts/BricolageGrotesque-Bold.ttf +0 -0
  26. package/system/skills/canvas-design/canvas-fonts/BricolageGrotesque-OFL.txt +93 -0
  27. package/system/skills/canvas-design/canvas-fonts/BricolageGrotesque-Regular.ttf +0 -0
  28. package/system/skills/canvas-design/canvas-fonts/CrimsonPro-Bold.ttf +0 -0
  29. package/system/skills/canvas-design/canvas-fonts/CrimsonPro-Italic.ttf +0 -0
  30. package/system/skills/canvas-design/canvas-fonts/CrimsonPro-OFL.txt +93 -0
  31. package/system/skills/canvas-design/canvas-fonts/CrimsonPro-Regular.ttf +0 -0
  32. package/system/skills/canvas-design/canvas-fonts/DMMono-OFL.txt +93 -0
  33. package/system/skills/canvas-design/canvas-fonts/DMMono-Regular.ttf +0 -0
  34. package/system/skills/canvas-design/canvas-fonts/EricaOne-OFL.txt +94 -0
  35. package/system/skills/canvas-design/canvas-fonts/EricaOne-Regular.ttf +0 -0
  36. package/system/skills/canvas-design/canvas-fonts/GeistMono-Bold.ttf +0 -0
  37. package/system/skills/canvas-design/canvas-fonts/GeistMono-OFL.txt +93 -0
  38. package/system/skills/canvas-design/canvas-fonts/GeistMono-Regular.ttf +0 -0
  39. package/system/skills/canvas-design/canvas-fonts/Gloock-OFL.txt +93 -0
  40. package/system/skills/canvas-design/canvas-fonts/Gloock-Regular.ttf +0 -0
  41. package/system/skills/canvas-design/canvas-fonts/IBMPlexMono-Bold.ttf +0 -0
  42. package/system/skills/canvas-design/canvas-fonts/IBMPlexMono-OFL.txt +93 -0
  43. package/system/skills/canvas-design/canvas-fonts/IBMPlexMono-Regular.ttf +0 -0
  44. package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-Bold.ttf +0 -0
  45. package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-BoldItalic.ttf +0 -0
  46. package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-Italic.ttf +0 -0
  47. package/system/skills/canvas-design/canvas-fonts/IBMPlexSerif-Regular.ttf +0 -0
  48. package/system/skills/canvas-design/canvas-fonts/InstrumentSans-Bold.ttf +0 -0
  49. package/system/skills/canvas-design/canvas-fonts/InstrumentSans-BoldItalic.ttf +0 -0
  50. package/system/skills/canvas-design/canvas-fonts/InstrumentSans-Italic.ttf +0 -0
  51. package/system/skills/canvas-design/canvas-fonts/InstrumentSans-OFL.txt +93 -0
  52. package/system/skills/canvas-design/canvas-fonts/InstrumentSans-Regular.ttf +0 -0
  53. package/system/skills/canvas-design/canvas-fonts/InstrumentSerif-Italic.ttf +0 -0
  54. package/system/skills/canvas-design/canvas-fonts/InstrumentSerif-Regular.ttf +0 -0
  55. package/system/skills/canvas-design/canvas-fonts/Italiana-OFL.txt +93 -0
  56. package/system/skills/canvas-design/canvas-fonts/Italiana-Regular.ttf +0 -0
  57. package/system/skills/canvas-design/canvas-fonts/JetBrainsMono-Bold.ttf +0 -0
  58. package/system/skills/canvas-design/canvas-fonts/JetBrainsMono-OFL.txt +93 -0
  59. package/system/skills/canvas-design/canvas-fonts/JetBrainsMono-Regular.ttf +0 -0
  60. package/system/skills/canvas-design/canvas-fonts/Jura-Light.ttf +0 -0
  61. package/system/skills/canvas-design/canvas-fonts/Jura-Medium.ttf +0 -0
  62. package/system/skills/canvas-design/canvas-fonts/Jura-OFL.txt +93 -0
  63. package/system/skills/canvas-design/canvas-fonts/LibreBaskerville-OFL.txt +93 -0
  64. package/system/skills/canvas-design/canvas-fonts/LibreBaskerville-Regular.ttf +0 -0
  65. package/system/skills/canvas-design/canvas-fonts/Lora-Bold.ttf +0 -0
  66. package/system/skills/canvas-design/canvas-fonts/Lora-BoldItalic.ttf +0 -0
  67. package/system/skills/canvas-design/canvas-fonts/Lora-Italic.ttf +0 -0
  68. package/system/skills/canvas-design/canvas-fonts/Lora-OFL.txt +93 -0
  69. package/system/skills/canvas-design/canvas-fonts/Lora-Regular.ttf +0 -0
  70. package/system/skills/canvas-design/canvas-fonts/NationalPark-Bold.ttf +0 -0
  71. package/system/skills/canvas-design/canvas-fonts/NationalPark-OFL.txt +93 -0
  72. package/system/skills/canvas-design/canvas-fonts/NationalPark-Regular.ttf +0 -0
  73. package/system/skills/canvas-design/canvas-fonts/NothingYouCouldDo-OFL.txt +93 -0
  74. package/system/skills/canvas-design/canvas-fonts/NothingYouCouldDo-Regular.ttf +0 -0
  75. package/system/skills/canvas-design/canvas-fonts/Outfit-Bold.ttf +0 -0
  76. package/system/skills/canvas-design/canvas-fonts/Outfit-OFL.txt +93 -0
  77. package/system/skills/canvas-design/canvas-fonts/Outfit-Regular.ttf +0 -0
  78. package/system/skills/canvas-design/canvas-fonts/PixelifySans-Medium.ttf +0 -0
  79. package/system/skills/canvas-design/canvas-fonts/PixelifySans-OFL.txt +93 -0
  80. package/system/skills/canvas-design/canvas-fonts/PoiretOne-OFL.txt +93 -0
  81. package/system/skills/canvas-design/canvas-fonts/PoiretOne-Regular.ttf +0 -0
  82. package/system/skills/canvas-design/canvas-fonts/RedHatMono-Bold.ttf +0 -0
  83. package/system/skills/canvas-design/canvas-fonts/RedHatMono-OFL.txt +93 -0
  84. package/system/skills/canvas-design/canvas-fonts/RedHatMono-Regular.ttf +0 -0
  85. package/system/skills/canvas-design/canvas-fonts/Silkscreen-OFL.txt +93 -0
  86. package/system/skills/canvas-design/canvas-fonts/Silkscreen-Regular.ttf +0 -0
  87. package/system/skills/canvas-design/canvas-fonts/SmoochSans-Medium.ttf +0 -0
  88. package/system/skills/canvas-design/canvas-fonts/SmoochSans-OFL.txt +93 -0
  89. package/system/skills/canvas-design/canvas-fonts/Tektur-Medium.ttf +0 -0
  90. package/system/skills/canvas-design/canvas-fonts/Tektur-OFL.txt +93 -0
  91. package/system/skills/canvas-design/canvas-fonts/Tektur-Regular.ttf +0 -0
  92. package/system/skills/canvas-design/canvas-fonts/WorkSans-Bold.ttf +0 -0
  93. package/system/skills/canvas-design/canvas-fonts/WorkSans-BoldItalic.ttf +0 -0
  94. package/system/skills/canvas-design/canvas-fonts/WorkSans-Italic.ttf +0 -0
  95. package/system/skills/canvas-design/canvas-fonts/WorkSans-OFL.txt +93 -0
  96. package/system/skills/canvas-design/canvas-fonts/WorkSans-Regular.ttf +0 -0
  97. package/system/skills/canvas-design/canvas-fonts/YoungSerif-OFL.txt +93 -0
  98. package/system/skills/canvas-design/canvas-fonts/YoungSerif-Regular.ttf +0 -0
  99. package/system/skills/component/SKILL.md +103 -0
  100. package/system/skills/config.json +18 -0
  101. package/system/skills/doc-coauthoring/SKILL.md +375 -0
  102. package/system/skills/docx/LICENSE.txt +30 -0
  103. package/system/skills/docx/SKILL.md +590 -0
  104. package/system/skills/docx/scripts/__init__.py +1 -0
  105. package/system/skills/docx/scripts/accept_changes.py +135 -0
  106. package/system/skills/docx/scripts/comment.py +318 -0
  107. package/system/skills/docx/scripts/office/helpers/__init__.py +0 -0
  108. package/system/skills/docx/scripts/office/helpers/merge_runs.py +199 -0
  109. package/system/skills/docx/scripts/office/helpers/simplify_redlines.py +197 -0
  110. package/system/skills/docx/scripts/office/pack.py +159 -0
  111. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  112. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  113. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  114. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  115. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  116. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  117. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  118. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  119. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  120. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  121. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  122. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  123. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  124. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  125. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  126. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  127. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  128. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  129. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  130. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  131. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  132. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  133. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  134. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  135. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  136. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  137. package/system/skills/docx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  138. package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  139. package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  140. package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  141. package/system/skills/docx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  142. package/system/skills/docx/scripts/office/schemas/mce/mc.xsd +75 -0
  143. package/system/skills/docx/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
  144. package/system/skills/docx/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
  145. package/system/skills/docx/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
  146. package/system/skills/docx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
  147. package/system/skills/docx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
  148. package/system/skills/docx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  149. package/system/skills/docx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
  150. package/system/skills/docx/scripts/office/soffice.py +183 -0
  151. package/system/skills/docx/scripts/office/unpack.py +132 -0
  152. package/system/skills/docx/scripts/office/validate.py +111 -0
  153. package/system/skills/docx/scripts/office/validators/__init__.py +15 -0
  154. package/system/skills/docx/scripts/office/validators/base.py +847 -0
  155. package/system/skills/docx/scripts/office/validators/docx.py +446 -0
  156. package/system/skills/docx/scripts/office/validators/pptx.py +275 -0
  157. package/system/skills/docx/scripts/office/validators/redlining.py +247 -0
  158. package/system/skills/docx/scripts/templates/comments.xml +3 -0
  159. package/system/skills/docx/scripts/templates/commentsExtended.xml +3 -0
  160. package/system/skills/docx/scripts/templates/commentsExtensible.xml +3 -0
  161. package/system/skills/docx/scripts/templates/commentsIds.xml +3 -0
  162. package/system/skills/docx/scripts/templates/people.xml +3 -0
  163. package/system/skills/evaluate-session.sh +69 -0
  164. package/system/skills/feature/SKILL.md +73 -0
  165. package/system/skills/frontend-design/LICENSE.txt +177 -0
  166. package/system/skills/frontend-design/SKILL.md +52 -0
  167. package/system/skills/hooks/observe.sh +164 -0
  168. package/system/skills/mcp-builder/LICENSE.txt +202 -0
  169. package/system/skills/mcp-builder/SKILL.md +236 -0
  170. package/system/skills/mcp-builder/reference/evaluation.md +602 -0
  171. package/system/skills/mcp-builder/reference/mcp_best_practices.md +249 -0
  172. package/system/skills/mcp-builder/reference/node_mcp_server.md +970 -0
  173. package/system/skills/mcp-builder/reference/python_mcp_server.md +719 -0
  174. package/system/skills/mcp-builder/scripts/connections.py +151 -0
  175. package/system/skills/mcp-builder/scripts/evaluation.py +373 -0
  176. package/system/skills/mcp-builder/scripts/example_evaluation.xml +22 -0
  177. package/system/skills/mcp-builder/scripts/requirements.txt +2 -0
  178. package/system/skills/pdf/LICENSE.txt +30 -0
  179. package/system/skills/pdf/SKILL.md +314 -0
  180. package/system/skills/pdf/forms.md +294 -0
  181. package/system/skills/pdf/reference.md +612 -0
  182. package/system/skills/pdf/scripts/check_bounding_boxes.py +65 -0
  183. package/system/skills/pdf/scripts/check_fillable_fields.py +11 -0
  184. package/system/skills/pdf/scripts/convert_pdf_to_images.py +33 -0
  185. package/system/skills/pdf/scripts/create_validation_image.py +37 -0
  186. package/system/skills/pdf/scripts/extract_form_field_info.py +122 -0
  187. package/system/skills/pdf/scripts/extract_form_structure.py +115 -0
  188. package/system/skills/pdf/scripts/fill_fillable_fields.py +98 -0
  189. package/system/skills/pdf/scripts/fill_pdf_form_with_annotations.py +107 -0
  190. package/system/skills/pptx/LICENSE.txt +30 -0
  191. package/system/skills/pptx/SKILL.md +232 -0
  192. package/system/skills/pptx/editing.md +205 -0
  193. package/system/skills/pptx/pptxgenjs.md +420 -0
  194. package/system/skills/pptx/scripts/__init__.py +0 -0
  195. package/system/skills/pptx/scripts/add_slide.py +195 -0
  196. package/system/skills/pptx/scripts/clean.py +286 -0
  197. package/system/skills/pptx/scripts/office/helpers/__init__.py +0 -0
  198. package/system/skills/pptx/scripts/office/helpers/merge_runs.py +199 -0
  199. package/system/skills/pptx/scripts/office/helpers/simplify_redlines.py +197 -0
  200. package/system/skills/pptx/scripts/office/pack.py +159 -0
  201. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  202. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  203. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  204. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  205. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  206. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  207. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  208. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  209. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  210. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  211. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  212. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  213. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  214. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  215. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  216. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  217. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  218. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  219. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  220. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  221. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  222. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  223. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  224. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  225. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  226. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  227. package/system/skills/pptx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  228. package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  229. package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  230. package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  231. package/system/skills/pptx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  232. package/system/skills/pptx/scripts/office/schemas/mce/mc.xsd +75 -0
  233. package/system/skills/pptx/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
  234. package/system/skills/pptx/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
  235. package/system/skills/pptx/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
  236. package/system/skills/pptx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
  237. package/system/skills/pptx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
  238. package/system/skills/pptx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  239. package/system/skills/pptx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
  240. package/system/skills/pptx/scripts/office/soffice.py +183 -0
  241. package/system/skills/pptx/scripts/office/unpack.py +132 -0
  242. package/system/skills/pptx/scripts/office/validate.py +111 -0
  243. package/system/skills/pptx/scripts/office/validators/__init__.py +15 -0
  244. package/system/skills/pptx/scripts/office/validators/base.py +847 -0
  245. package/system/skills/pptx/scripts/office/validators/docx.py +446 -0
  246. package/system/skills/pptx/scripts/office/validators/pptx.py +275 -0
  247. package/system/skills/pptx/scripts/office/validators/redlining.py +247 -0
  248. package/system/skills/pptx/scripts/thumbnail.py +289 -0
  249. package/system/skills/scripts/quick-diff.sh +87 -0
  250. package/system/skills/scripts/save-results.sh +56 -0
  251. package/system/skills/scripts/scan.sh +170 -0
  252. package/system/skills/security-check/SKILL.md +70 -0
  253. package/system/skills/skill-creator/LICENSE.txt +202 -0
  254. package/system/skills/skill-creator/SKILL.md +479 -0
  255. package/system/skills/skill-creator/agents/analyzer.md +274 -0
  256. package/system/skills/skill-creator/agents/comparator.md +202 -0
  257. package/system/skills/skill-creator/agents/grader.md +223 -0
  258. package/system/skills/skill-creator/assets/eval_review.html +146 -0
  259. package/system/skills/skill-creator/eval-viewer/generate_review.py +471 -0
  260. package/system/skills/skill-creator/eval-viewer/viewer.html +1325 -0
  261. package/system/skills/skill-creator/references/schemas.md +430 -0
  262. package/system/skills/skill-creator/scripts/__init__.py +0 -0
  263. package/system/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
  264. package/system/skills/skill-creator/scripts/generate_report.py +326 -0
  265. package/system/skills/skill-creator/scripts/improve_description.py +248 -0
  266. package/system/skills/skill-creator/scripts/package_skill.py +136 -0
  267. package/system/skills/skill-creator/scripts/quick_validate.py +103 -0
  268. package/system/skills/skill-creator/scripts/run_eval.py +310 -0
  269. package/system/skills/skill-creator/scripts/run_loop.py +332 -0
  270. package/system/skills/skill-creator/scripts/utils.py +47 -0
  271. package/system/skills/suggest-compact.sh +54 -0
  272. package/system/skills/supabase-migration/SKILL.md +59 -0
  273. package/system/skills/theme-factory/LICENSE.txt +202 -0
  274. package/system/skills/theme-factory/SKILL.md +59 -0
  275. package/system/skills/theme-factory/theme-showcase.pdf +0 -0
  276. package/system/skills/theme-factory/themes/arctic-frost.md +19 -0
  277. package/system/skills/theme-factory/themes/botanical-garden.md +19 -0
  278. package/system/skills/theme-factory/themes/desert-rose.md +19 -0
  279. package/system/skills/theme-factory/themes/forest-canopy.md +19 -0
  280. package/system/skills/theme-factory/themes/golden-hour.md +19 -0
  281. package/system/skills/theme-factory/themes/midnight-galaxy.md +19 -0
  282. package/system/skills/theme-factory/themes/modern-minimalist.md +19 -0
  283. package/system/skills/theme-factory/themes/ocean-depths.md +19 -0
  284. package/system/skills/theme-factory/themes/sunset-boulevard.md +19 -0
  285. package/system/skills/theme-factory/themes/tech-innovation.md +19 -0
  286. package/system/skills/web-artifacts-builder/LICENSE.txt +202 -0
  287. package/system/skills/web-artifacts-builder/SKILL.md +74 -0
  288. package/system/skills/web-artifacts-builder/scripts/bundle-artifact.sh +54 -0
  289. package/system/skills/web-artifacts-builder/scripts/init-artifact.sh +322 -0
  290. package/system/skills/web-artifacts-builder/scripts/shadcn-components.tar.gz +0 -0
  291. package/system/skills/webapp-testing/LICENSE.txt +202 -0
  292. package/system/skills/webapp-testing/SKILL.md +96 -0
  293. package/system/skills/webapp-testing/examples/console_logging.py +35 -0
  294. package/system/skills/webapp-testing/examples/element_discovery.py +40 -0
  295. package/system/skills/webapp-testing/examples/static_html_automation.py +33 -0
  296. package/system/skills/webapp-testing/scripts/with_server.py +106 -0
  297. package/system/skills/xlsx/LICENSE.txt +30 -0
  298. package/system/skills/xlsx/SKILL.md +292 -0
  299. package/system/skills/xlsx/scripts/office/helpers/__init__.py +0 -0
  300. package/system/skills/xlsx/scripts/office/helpers/merge_runs.py +199 -0
  301. package/system/skills/xlsx/scripts/office/helpers/simplify_redlines.py +197 -0
  302. package/system/skills/xlsx/scripts/office/pack.py +159 -0
  303. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chart.xsd +1499 -0
  304. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-chartDrawing.xsd +146 -0
  305. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-diagram.xsd +1085 -0
  306. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-lockedCanvas.xsd +11 -0
  307. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-main.xsd +3081 -0
  308. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-picture.xsd +23 -0
  309. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-spreadsheetDrawing.xsd +185 -0
  310. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/dml-wordprocessingDrawing.xsd +287 -0
  311. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/pml.xsd +1676 -0
  312. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-additionalCharacteristics.xsd +28 -0
  313. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-bibliography.xsd +144 -0
  314. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-commonSimpleTypes.xsd +174 -0
  315. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlDataProperties.xsd +25 -0
  316. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-customXmlSchemaProperties.xsd +18 -0
  317. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesCustom.xsd +59 -0
  318. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesExtended.xsd +56 -0
  319. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-documentPropertiesVariantTypes.xsd +195 -0
  320. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-math.xsd +582 -0
  321. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/shared-relationshipReference.xsd +25 -0
  322. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/sml.xsd +4439 -0
  323. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-main.xsd +570 -0
  324. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-officeDrawing.xsd +509 -0
  325. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-presentationDrawing.xsd +12 -0
  326. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-spreadsheetDrawing.xsd +108 -0
  327. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/vml-wordprocessingDrawing.xsd +96 -0
  328. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/wml.xsd +3646 -0
  329. package/system/skills/xlsx/scripts/office/schemas/ISO-IEC29500-4_2016/xml.xsd +116 -0
  330. package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-contentTypes.xsd +42 -0
  331. package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-coreProperties.xsd +50 -0
  332. package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-digSig.xsd +49 -0
  333. package/system/skills/xlsx/scripts/office/schemas/ecma/fouth-edition/opc-relationships.xsd +33 -0
  334. package/system/skills/xlsx/scripts/office/schemas/mce/mc.xsd +75 -0
  335. package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-2010.xsd +560 -0
  336. package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-2012.xsd +67 -0
  337. package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-2018.xsd +14 -0
  338. package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-cex-2018.xsd +20 -0
  339. package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-cid-2016.xsd +13 -0
  340. package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-sdtdatahash-2020.xsd +4 -0
  341. package/system/skills/xlsx/scripts/office/schemas/microsoft/wml-symex-2015.xsd +8 -0
  342. package/system/skills/xlsx/scripts/office/soffice.py +183 -0
  343. package/system/skills/xlsx/scripts/office/unpack.py +132 -0
  344. package/system/skills/xlsx/scripts/office/validate.py +111 -0
  345. package/system/skills/xlsx/scripts/office/validators/__init__.py +15 -0
  346. package/system/skills/xlsx/scripts/office/validators/base.py +847 -0
  347. package/system/skills/xlsx/scripts/office/validators/docx.py +446 -0
  348. package/system/skills/xlsx/scripts/office/validators/pptx.py +275 -0
  349. package/system/skills/xlsx/scripts/office/validators/redlining.py +247 -0
  350. package/system/skills/xlsx/scripts/recalc.py +184 -0
  351. package/system/teams/profiles.md +76 -0
@@ -0,0 +1,135 @@
1
+ """Accept all tracked changes in a DOCX file using LibreOffice.
2
+
3
+ Requires LibreOffice (soffice) to be installed.
4
+ """
5
+
6
+ import argparse
7
+ import logging
8
+ import shutil
9
+ import subprocess
10
+ from pathlib import Path
11
+
12
+ from office.soffice import get_soffice_env
13
+
14
+ logger = logging.getLogger(__name__)
15
+
16
+ LIBREOFFICE_PROFILE = "/tmp/libreoffice_docx_profile"
17
+ MACRO_DIR = f"{LIBREOFFICE_PROFILE}/user/basic/Standard"
18
+
19
+ ACCEPT_CHANGES_MACRO = """<?xml version="1.0" encoding="UTF-8"?>
20
+ <!DOCTYPE script:module PUBLIC "-//OpenOffice.org//DTD OfficeDocument 1.0//EN" "module.dtd">
21
+ <script:module xmlns:script="http://openoffice.org/2000/script" script:name="Module1" script:language="StarBasic">
22
+ Sub AcceptAllTrackedChanges()
23
+ Dim document As Object
24
+ Dim dispatcher As Object
25
+
26
+ document = ThisComponent.CurrentController.Frame
27
+ dispatcher = createUnoService("com.sun.star.frame.DispatchHelper")
28
+
29
+ dispatcher.executeDispatch(document, ".uno:AcceptAllTrackedChanges", "", 0, Array())
30
+ ThisComponent.store()
31
+ ThisComponent.close(True)
32
+ End Sub
33
+ </script:module>"""
34
+
35
+
36
+ def accept_changes(
37
+ input_file: str,
38
+ output_file: str,
39
+ ) -> tuple[None, str]:
40
+ input_path = Path(input_file)
41
+ output_path = Path(output_file)
42
+
43
+ if not input_path.exists():
44
+ return None, f"Error: Input file not found: {input_file}"
45
+
46
+ if not input_path.suffix.lower() == ".docx":
47
+ return None, f"Error: Input file is not a DOCX file: {input_file}"
48
+
49
+ try:
50
+ output_path.parent.mkdir(parents=True, exist_ok=True)
51
+ shutil.copy2(input_path, output_path)
52
+ except Exception as e:
53
+ return None, f"Error: Failed to copy input file to output location: {e}"
54
+
55
+ if not _setup_libreoffice_macro():
56
+ return None, "Error: Failed to setup LibreOffice macro"
57
+
58
+ cmd = [
59
+ "soffice",
60
+ "--headless",
61
+ f"-env:UserInstallation=file://{LIBREOFFICE_PROFILE}",
62
+ "--norestore",
63
+ "vnd.sun.star.script:Standard.Module1.AcceptAllTrackedChanges?language=Basic&location=application",
64
+ str(output_path.absolute()),
65
+ ]
66
+
67
+ try:
68
+ result = subprocess.run(
69
+ cmd,
70
+ capture_output=True,
71
+ text=True,
72
+ timeout=30,
73
+ check=False,
74
+ env=get_soffice_env(),
75
+ )
76
+ except subprocess.TimeoutExpired:
77
+ return (
78
+ None,
79
+ f"Successfully accepted all tracked changes: {input_file} -> {output_file}",
80
+ )
81
+
82
+ if result.returncode != 0:
83
+ return None, f"Error: LibreOffice failed: {result.stderr}"
84
+
85
+ return (
86
+ None,
87
+ f"Successfully accepted all tracked changes: {input_file} -> {output_file}",
88
+ )
89
+
90
+
91
+ def _setup_libreoffice_macro() -> bool:
92
+ macro_dir = Path(MACRO_DIR)
93
+ macro_file = macro_dir / "Module1.xba"
94
+
95
+ if macro_file.exists() and "AcceptAllTrackedChanges" in macro_file.read_text():
96
+ return True
97
+
98
+ if not macro_dir.exists():
99
+ subprocess.run(
100
+ [
101
+ "soffice",
102
+ "--headless",
103
+ f"-env:UserInstallation=file://{LIBREOFFICE_PROFILE}",
104
+ "--terminate_after_init",
105
+ ],
106
+ capture_output=True,
107
+ timeout=10,
108
+ check=False,
109
+ env=get_soffice_env(),
110
+ )
111
+ macro_dir.mkdir(parents=True, exist_ok=True)
112
+
113
+ try:
114
+ macro_file.write_text(ACCEPT_CHANGES_MACRO)
115
+ return True
116
+ except Exception as e:
117
+ logger.warning(f"Failed to setup LibreOffice macro: {e}")
118
+ return False
119
+
120
+
121
+ if __name__ == "__main__":
122
+ parser = argparse.ArgumentParser(
123
+ description="Accept all tracked changes in a DOCX file"
124
+ )
125
+ parser.add_argument("input_file", help="Input DOCX file with tracked changes")
126
+ parser.add_argument(
127
+ "output_file", help="Output DOCX file (clean, no tracked changes)"
128
+ )
129
+ args = parser.parse_args()
130
+
131
+ _, message = accept_changes(args.input_file, args.output_file)
132
+ print(message)
133
+
134
+ if "Error" in message:
135
+ raise SystemExit(1)
@@ -0,0 +1,318 @@
1
+ """Add comments to DOCX documents.
2
+
3
+ Usage:
4
+ python comment.py unpacked/ 0 "Comment text"
5
+ python comment.py unpacked/ 1 "Reply text" --parent 0
6
+
7
+ Text should be pre-escaped XML (e.g., &amp; for &, &#x2019; for smart quotes).
8
+
9
+ After running, add markers to document.xml:
10
+ <w:commentRangeStart w:id="0"/>
11
+ ... commented content ...
12
+ <w:commentRangeEnd w:id="0"/>
13
+ <w:r><w:rPr><w:rStyle w:val="CommentReference"/></w:rPr><w:commentReference w:id="0"/></w:r>
14
+ """
15
+
16
+ import argparse
17
+ import random
18
+ import shutil
19
+ import sys
20
+ from datetime import datetime, timezone
21
+ from pathlib import Path
22
+
23
+ import defusedxml.minidom
24
+
25
+ TEMPLATE_DIR = Path(__file__).parent / "templates"
26
+ NS = {
27
+ "w": "http://schemas.openxmlformats.org/wordprocessingml/2006/main",
28
+ "w14": "http://schemas.microsoft.com/office/word/2010/wordml",
29
+ "w15": "http://schemas.microsoft.com/office/word/2012/wordml",
30
+ "w16cid": "http://schemas.microsoft.com/office/word/2016/wordml/cid",
31
+ "w16cex": "http://schemas.microsoft.com/office/word/2018/wordml/cex",
32
+ }
33
+
34
+ COMMENT_XML = """\
35
+ <w:comment w:id="{id}" w:author="{author}" w:date="{date}" w:initials="{initials}">
36
+ <w:p w14:paraId="{para_id}" w14:textId="77777777">
37
+ <w:r>
38
+ <w:rPr><w:rStyle w:val="CommentReference"/></w:rPr>
39
+ <w:annotationRef/>
40
+ </w:r>
41
+ <w:r>
42
+ <w:rPr>
43
+ <w:color w:val="000000"/>
44
+ <w:sz w:val="20"/>
45
+ <w:szCs w:val="20"/>
46
+ </w:rPr>
47
+ <w:t>{text}</w:t>
48
+ </w:r>
49
+ </w:p>
50
+ </w:comment>"""
51
+
52
+ COMMENT_MARKER_TEMPLATE = """
53
+ Add to document.xml (markers must be direct children of w:p, never inside w:r):
54
+ <w:commentRangeStart w:id="{cid}"/>
55
+ <w:r>...</w:r>
56
+ <w:commentRangeEnd w:id="{cid}"/>
57
+ <w:r><w:rPr><w:rStyle w:val="CommentReference"/></w:rPr><w:commentReference w:id="{cid}"/></w:r>"""
58
+
59
+ REPLY_MARKER_TEMPLATE = """
60
+ Nest markers inside parent {pid}'s markers (markers must be direct children of w:p, never inside w:r):
61
+ <w:commentRangeStart w:id="{pid}"/><w:commentRangeStart w:id="{cid}"/>
62
+ <w:r>...</w:r>
63
+ <w:commentRangeEnd w:id="{cid}"/><w:commentRangeEnd w:id="{pid}"/>
64
+ <w:r><w:rPr><w:rStyle w:val="CommentReference"/></w:rPr><w:commentReference w:id="{pid}"/></w:r>
65
+ <w:r><w:rPr><w:rStyle w:val="CommentReference"/></w:rPr><w:commentReference w:id="{cid}"/></w:r>"""
66
+
67
+
68
+ def _generate_hex_id() -> str:
69
+ return f"{random.randint(0, 0x7FFFFFFE):08X}"
70
+
71
+
72
+ SMART_QUOTE_ENTITIES = {
73
+ "\u201c": "&#x201C;",
74
+ "\u201d": "&#x201D;",
75
+ "\u2018": "&#x2018;",
76
+ "\u2019": "&#x2019;",
77
+ }
78
+
79
+
80
+ def _encode_smart_quotes(text: str) -> str:
81
+ for char, entity in SMART_QUOTE_ENTITIES.items():
82
+ text = text.replace(char, entity)
83
+ return text
84
+
85
+
86
+ def _append_xml(xml_path: Path, root_tag: str, content: str) -> None:
87
+ dom = defusedxml.minidom.parseString(xml_path.read_text(encoding="utf-8"))
88
+ root = dom.getElementsByTagName(root_tag)[0]
89
+ ns_attrs = " ".join(f'xmlns:{k}="{v}"' for k, v in NS.items())
90
+ wrapper_dom = defusedxml.minidom.parseString(f"<root {ns_attrs}>{content}</root>")
91
+ for child in wrapper_dom.documentElement.childNodes:
92
+ if child.nodeType == child.ELEMENT_NODE:
93
+ root.appendChild(dom.importNode(child, True))
94
+ output = _encode_smart_quotes(dom.toxml(encoding="UTF-8").decode("utf-8"))
95
+ xml_path.write_text(output, encoding="utf-8")
96
+
97
+
98
+ def _find_para_id(comments_path: Path, comment_id: int) -> str | None:
99
+ dom = defusedxml.minidom.parseString(comments_path.read_text(encoding="utf-8"))
100
+ for c in dom.getElementsByTagName("w:comment"):
101
+ if c.getAttribute("w:id") == str(comment_id):
102
+ for p in c.getElementsByTagName("w:p"):
103
+ if pid := p.getAttribute("w14:paraId"):
104
+ return pid
105
+ return None
106
+
107
+
108
+ def _get_next_rid(rels_path: Path) -> int:
109
+ dom = defusedxml.minidom.parseString(rels_path.read_text(encoding="utf-8"))
110
+ max_rid = 0
111
+ for rel in dom.getElementsByTagName("Relationship"):
112
+ rid = rel.getAttribute("Id")
113
+ if rid and rid.startswith("rId"):
114
+ try:
115
+ max_rid = max(max_rid, int(rid[3:]))
116
+ except ValueError:
117
+ pass
118
+ return max_rid + 1
119
+
120
+
121
+ def _has_relationship(rels_path: Path, target: str) -> bool:
122
+ dom = defusedxml.minidom.parseString(rels_path.read_text(encoding="utf-8"))
123
+ for rel in dom.getElementsByTagName("Relationship"):
124
+ if rel.getAttribute("Target") == target:
125
+ return True
126
+ return False
127
+
128
+
129
+ def _has_content_type(ct_path: Path, part_name: str) -> bool:
130
+ dom = defusedxml.minidom.parseString(ct_path.read_text(encoding="utf-8"))
131
+ for override in dom.getElementsByTagName("Override"):
132
+ if override.getAttribute("PartName") == part_name:
133
+ return True
134
+ return False
135
+
136
+
137
+ def _ensure_comment_relationships(unpacked_dir: Path) -> None:
138
+ rels_path = unpacked_dir / "word" / "_rels" / "document.xml.rels"
139
+ if not rels_path.exists():
140
+ return
141
+
142
+ if _has_relationship(rels_path, "comments.xml"):
143
+ return
144
+
145
+ dom = defusedxml.minidom.parseString(rels_path.read_text(encoding="utf-8"))
146
+ root = dom.documentElement
147
+ next_rid = _get_next_rid(rels_path)
148
+
149
+ rels = [
150
+ (
151
+ "http://schemas.openxmlformats.org/officeDocument/2006/relationships/comments",
152
+ "comments.xml",
153
+ ),
154
+ (
155
+ "http://schemas.microsoft.com/office/2011/relationships/commentsExtended",
156
+ "commentsExtended.xml",
157
+ ),
158
+ (
159
+ "http://schemas.microsoft.com/office/2016/09/relationships/commentsIds",
160
+ "commentsIds.xml",
161
+ ),
162
+ (
163
+ "http://schemas.microsoft.com/office/2018/08/relationships/commentsExtensible",
164
+ "commentsExtensible.xml",
165
+ ),
166
+ ]
167
+
168
+ for rel_type, target in rels:
169
+ rel = dom.createElement("Relationship")
170
+ rel.setAttribute("Id", f"rId{next_rid}")
171
+ rel.setAttribute("Type", rel_type)
172
+ rel.setAttribute("Target", target)
173
+ root.appendChild(rel)
174
+ next_rid += 1
175
+
176
+ rels_path.write_bytes(dom.toxml(encoding="UTF-8"))
177
+
178
+
179
+ def _ensure_comment_content_types(unpacked_dir: Path) -> None:
180
+ ct_path = unpacked_dir / "[Content_Types].xml"
181
+ if not ct_path.exists():
182
+ return
183
+
184
+ if _has_content_type(ct_path, "/word/comments.xml"):
185
+ return
186
+
187
+ dom = defusedxml.minidom.parseString(ct_path.read_text(encoding="utf-8"))
188
+ root = dom.documentElement
189
+
190
+ overrides = [
191
+ (
192
+ "/word/comments.xml",
193
+ "application/vnd.openxmlformats-officedocument.wordprocessingml.comments+xml",
194
+ ),
195
+ (
196
+ "/word/commentsExtended.xml",
197
+ "application/vnd.openxmlformats-officedocument.wordprocessingml.commentsExtended+xml",
198
+ ),
199
+ (
200
+ "/word/commentsIds.xml",
201
+ "application/vnd.openxmlformats-officedocument.wordprocessingml.commentsIds+xml",
202
+ ),
203
+ (
204
+ "/word/commentsExtensible.xml",
205
+ "application/vnd.openxmlformats-officedocument.wordprocessingml.commentsExtensible+xml",
206
+ ),
207
+ ]
208
+
209
+ for part_name, content_type in overrides:
210
+ override = dom.createElement("Override")
211
+ override.setAttribute("PartName", part_name)
212
+ override.setAttribute("ContentType", content_type)
213
+ root.appendChild(override)
214
+
215
+ ct_path.write_bytes(dom.toxml(encoding="UTF-8"))
216
+
217
+
218
+ def add_comment(
219
+ unpacked_dir: str,
220
+ comment_id: int,
221
+ text: str,
222
+ author: str = "Claude",
223
+ initials: str = "C",
224
+ parent_id: int | None = None,
225
+ ) -> tuple[str, str]:
226
+ word = Path(unpacked_dir) / "word"
227
+ if not word.exists():
228
+ return "", f"Error: {word} not found"
229
+
230
+ para_id, durable_id = _generate_hex_id(), _generate_hex_id()
231
+ ts = datetime.now(timezone.utc).strftime("%Y-%m-%dT%H:%M:%SZ")
232
+
233
+ comments = word / "comments.xml"
234
+ first_comment = not comments.exists()
235
+ if first_comment:
236
+ shutil.copy(TEMPLATE_DIR / "comments.xml", comments)
237
+ _ensure_comment_relationships(Path(unpacked_dir))
238
+ _ensure_comment_content_types(Path(unpacked_dir))
239
+ _append_xml(
240
+ comments,
241
+ "w:comments",
242
+ COMMENT_XML.format(
243
+ id=comment_id,
244
+ author=author,
245
+ date=ts,
246
+ initials=initials,
247
+ para_id=para_id,
248
+ text=text,
249
+ ),
250
+ )
251
+
252
+ ext = word / "commentsExtended.xml"
253
+ if not ext.exists():
254
+ shutil.copy(TEMPLATE_DIR / "commentsExtended.xml", ext)
255
+ if parent_id is not None:
256
+ parent_para = _find_para_id(comments, parent_id)
257
+ if not parent_para:
258
+ return "", f"Error: Parent comment {parent_id} not found"
259
+ _append_xml(
260
+ ext,
261
+ "w15:commentsEx",
262
+ f'<w15:commentEx w15:paraId="{para_id}" w15:paraIdParent="{parent_para}" w15:done="0"/>',
263
+ )
264
+ else:
265
+ _append_xml(
266
+ ext,
267
+ "w15:commentsEx",
268
+ f'<w15:commentEx w15:paraId="{para_id}" w15:done="0"/>',
269
+ )
270
+
271
+ ids = word / "commentsIds.xml"
272
+ if not ids.exists():
273
+ shutil.copy(TEMPLATE_DIR / "commentsIds.xml", ids)
274
+ _append_xml(
275
+ ids,
276
+ "w16cid:commentsIds",
277
+ f'<w16cid:commentId w16cid:paraId="{para_id}" w16cid:durableId="{durable_id}"/>',
278
+ )
279
+
280
+ extensible = word / "commentsExtensible.xml"
281
+ if not extensible.exists():
282
+ shutil.copy(TEMPLATE_DIR / "commentsExtensible.xml", extensible)
283
+ _append_xml(
284
+ extensible,
285
+ "w16cex:commentsExtensible",
286
+ f'<w16cex:commentExtensible w16cex:durableId="{durable_id}" w16cex:dateUtc="{ts}"/>',
287
+ )
288
+
289
+ action = "reply" if parent_id is not None else "comment"
290
+ return para_id, f"Added {action} {comment_id} (para_id={para_id})"
291
+
292
+
293
+ if __name__ == "__main__":
294
+ p = argparse.ArgumentParser(description="Add comments to DOCX documents")
295
+ p.add_argument("unpacked_dir", help="Unpacked DOCX directory")
296
+ p.add_argument("comment_id", type=int, help="Comment ID (must be unique)")
297
+ p.add_argument("text", help="Comment text")
298
+ p.add_argument("--author", default="Claude", help="Author name")
299
+ p.add_argument("--initials", default="C", help="Author initials")
300
+ p.add_argument("--parent", type=int, help="Parent comment ID (for replies)")
301
+ args = p.parse_args()
302
+
303
+ para_id, msg = add_comment(
304
+ args.unpacked_dir,
305
+ args.comment_id,
306
+ args.text,
307
+ args.author,
308
+ args.initials,
309
+ args.parent,
310
+ )
311
+ print(msg)
312
+ if "Error" in msg:
313
+ sys.exit(1)
314
+ cid = args.comment_id
315
+ if args.parent is not None:
316
+ print(REPLY_MARKER_TEMPLATE.format(pid=args.parent, cid=cid))
317
+ else:
318
+ print(COMMENT_MARKER_TEMPLATE.format(cid=cid))
@@ -0,0 +1,199 @@
1
+ """Merge adjacent runs with identical formatting in DOCX.
2
+
3
+ Merges adjacent <w:r> elements that have identical <w:rPr> properties.
4
+ Works on runs in paragraphs and inside tracked changes (<w:ins>, <w:del>).
5
+
6
+ Also:
7
+ - Removes rsid attributes from runs (revision metadata that doesn't affect rendering)
8
+ - Removes proofErr elements (spell/grammar markers that block merging)
9
+ """
10
+
11
+ from pathlib import Path
12
+
13
+ import defusedxml.minidom
14
+
15
+
16
+ def merge_runs(input_dir: str) -> tuple[int, str]:
17
+ doc_xml = Path(input_dir) / "word" / "document.xml"
18
+
19
+ if not doc_xml.exists():
20
+ return 0, f"Error: {doc_xml} not found"
21
+
22
+ try:
23
+ dom = defusedxml.minidom.parseString(doc_xml.read_text(encoding="utf-8"))
24
+ root = dom.documentElement
25
+
26
+ _remove_elements(root, "proofErr")
27
+ _strip_run_rsid_attrs(root)
28
+
29
+ containers = {run.parentNode for run in _find_elements(root, "r")}
30
+
31
+ merge_count = 0
32
+ for container in containers:
33
+ merge_count += _merge_runs_in(container)
34
+
35
+ doc_xml.write_bytes(dom.toxml(encoding="UTF-8"))
36
+ return merge_count, f"Merged {merge_count} runs"
37
+
38
+ except Exception as e:
39
+ return 0, f"Error: {e}"
40
+
41
+
42
+
43
+
44
+ def _find_elements(root, tag: str) -> list:
45
+ results = []
46
+
47
+ def traverse(node):
48
+ if node.nodeType == node.ELEMENT_NODE:
49
+ name = node.localName or node.tagName
50
+ if name == tag or name.endswith(f":{tag}"):
51
+ results.append(node)
52
+ for child in node.childNodes:
53
+ traverse(child)
54
+
55
+ traverse(root)
56
+ return results
57
+
58
+
59
+ def _get_child(parent, tag: str):
60
+ for child in parent.childNodes:
61
+ if child.nodeType == child.ELEMENT_NODE:
62
+ name = child.localName or child.tagName
63
+ if name == tag or name.endswith(f":{tag}"):
64
+ return child
65
+ return None
66
+
67
+
68
+ def _get_children(parent, tag: str) -> list:
69
+ results = []
70
+ for child in parent.childNodes:
71
+ if child.nodeType == child.ELEMENT_NODE:
72
+ name = child.localName or child.tagName
73
+ if name == tag or name.endswith(f":{tag}"):
74
+ results.append(child)
75
+ return results
76
+
77
+
78
+ def _is_adjacent(elem1, elem2) -> bool:
79
+ node = elem1.nextSibling
80
+ while node:
81
+ if node == elem2:
82
+ return True
83
+ if node.nodeType == node.ELEMENT_NODE:
84
+ return False
85
+ if node.nodeType == node.TEXT_NODE and node.data.strip():
86
+ return False
87
+ node = node.nextSibling
88
+ return False
89
+
90
+
91
+
92
+
93
+ def _remove_elements(root, tag: str):
94
+ for elem in _find_elements(root, tag):
95
+ if elem.parentNode:
96
+ elem.parentNode.removeChild(elem)
97
+
98
+
99
+ def _strip_run_rsid_attrs(root):
100
+ for run in _find_elements(root, "r"):
101
+ for attr in list(run.attributes.values()):
102
+ if "rsid" in attr.name.lower():
103
+ run.removeAttribute(attr.name)
104
+
105
+
106
+
107
+
108
+ def _merge_runs_in(container) -> int:
109
+ merge_count = 0
110
+ run = _first_child_run(container)
111
+
112
+ while run:
113
+ while True:
114
+ next_elem = _next_element_sibling(run)
115
+ if next_elem and _is_run(next_elem) and _can_merge(run, next_elem):
116
+ _merge_run_content(run, next_elem)
117
+ container.removeChild(next_elem)
118
+ merge_count += 1
119
+ else:
120
+ break
121
+
122
+ _consolidate_text(run)
123
+ run = _next_sibling_run(run)
124
+
125
+ return merge_count
126
+
127
+
128
+ def _first_child_run(container):
129
+ for child in container.childNodes:
130
+ if child.nodeType == child.ELEMENT_NODE and _is_run(child):
131
+ return child
132
+ return None
133
+
134
+
135
+ def _next_element_sibling(node):
136
+ sibling = node.nextSibling
137
+ while sibling:
138
+ if sibling.nodeType == sibling.ELEMENT_NODE:
139
+ return sibling
140
+ sibling = sibling.nextSibling
141
+ return None
142
+
143
+
144
+ def _next_sibling_run(node):
145
+ sibling = node.nextSibling
146
+ while sibling:
147
+ if sibling.nodeType == sibling.ELEMENT_NODE:
148
+ if _is_run(sibling):
149
+ return sibling
150
+ sibling = sibling.nextSibling
151
+ return None
152
+
153
+
154
+ def _is_run(node) -> bool:
155
+ name = node.localName or node.tagName
156
+ return name == "r" or name.endswith(":r")
157
+
158
+
159
+ def _can_merge(run1, run2) -> bool:
160
+ rpr1 = _get_child(run1, "rPr")
161
+ rpr2 = _get_child(run2, "rPr")
162
+
163
+ if (rpr1 is None) != (rpr2 is None):
164
+ return False
165
+ if rpr1 is None:
166
+ return True
167
+ return rpr1.toxml() == rpr2.toxml()
168
+
169
+
170
+ def _merge_run_content(target, source):
171
+ for child in list(source.childNodes):
172
+ if child.nodeType == child.ELEMENT_NODE:
173
+ name = child.localName or child.tagName
174
+ if name != "rPr" and not name.endswith(":rPr"):
175
+ target.appendChild(child)
176
+
177
+
178
+ def _consolidate_text(run):
179
+ t_elements = _get_children(run, "t")
180
+
181
+ for i in range(len(t_elements) - 1, 0, -1):
182
+ curr, prev = t_elements[i], t_elements[i - 1]
183
+
184
+ if _is_adjacent(prev, curr):
185
+ prev_text = prev.firstChild.data if prev.firstChild else ""
186
+ curr_text = curr.firstChild.data if curr.firstChild else ""
187
+ merged = prev_text + curr_text
188
+
189
+ if prev.firstChild:
190
+ prev.firstChild.data = merged
191
+ else:
192
+ prev.appendChild(run.ownerDocument.createTextNode(merged))
193
+
194
+ if merged.startswith(" ") or merged.endswith(" "):
195
+ prev.setAttribute("xml:space", "preserve")
196
+ elif prev.hasAttribute("xml:space"):
197
+ prev.removeAttribute("xml:space")
198
+
199
+ run.removeChild(curr)