pi-feishu-cli 0.1.0

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 (407) hide show
  1. package/LICENSE +21 -0
  2. package/README.md +111 -0
  3. package/dist/src/bot.d.ts +29 -0
  4. package/dist/src/bot.js +75 -0
  5. package/dist/src/cards.d.ts +6 -0
  6. package/dist/src/cards.js +87 -0
  7. package/dist/src/config.d.ts +3 -0
  8. package/dist/src/config.js +28 -0
  9. package/dist/src/daemon.d.ts +2 -0
  10. package/dist/src/daemon.js +151 -0
  11. package/dist/src/extension.d.ts +2 -0
  12. package/dist/src/extension.js +124 -0
  13. package/dist/src/poller.d.ts +33 -0
  14. package/dist/src/poller.js +94 -0
  15. package/dist/src/renderer.d.ts +8 -0
  16. package/dist/src/renderer.js +31 -0
  17. package/dist/src/session-registry.d.ts +15 -0
  18. package/dist/src/session-registry.js +82 -0
  19. package/dist/src/types.d.ts +25 -0
  20. package/dist/src/types.js +1 -0
  21. package/dist/tests/bot.test.d.ts +1 -0
  22. package/dist/tests/bot.test.js +89 -0
  23. package/dist/tests/cards.test.d.ts +1 -0
  24. package/dist/tests/cards.test.js +39 -0
  25. package/dist/tests/config.test.d.ts +1 -0
  26. package/dist/tests/config.test.js +59 -0
  27. package/dist/tests/renderer.test.d.ts +1 -0
  28. package/dist/tests/renderer.test.js +61 -0
  29. package/dist/tests/session-registry.test.d.ts +1 -0
  30. package/dist/tests/session-registry.test.js +92 -0
  31. package/dist/tests/types.test.d.ts +1 -0
  32. package/dist/tests/types.test.js +30 -0
  33. package/package.json +35 -0
  34. package/skills/lark-approval/SKILL.md +56 -0
  35. package/skills/lark-apps/SKILL.md +92 -0
  36. package/skills/lark-apps/references/lark-apps-access-scope-get.md +104 -0
  37. package/skills/lark-apps/references/lark-apps-access-scope-set.md +126 -0
  38. package/skills/lark-apps/references/lark-apps-create.md +112 -0
  39. package/skills/lark-apps/references/lark-apps-html-publish.md +151 -0
  40. package/skills/lark-apps/references/lark-apps-list.md +95 -0
  41. package/skills/lark-apps/references/lark-apps-update.md +86 -0
  42. package/skills/lark-attendance/SKILL.md +57 -0
  43. package/skills/lark-base/SKILL.md +359 -0
  44. package/skills/lark-base/references/dashboard-block-data-config.md +350 -0
  45. package/skills/lark-base/references/examples.md +140 -0
  46. package/skills/lark-base/references/formula-field-guide.md +737 -0
  47. package/skills/lark-base/references/lark-base-advperm-disable.md +83 -0
  48. package/skills/lark-base/references/lark-base-advperm-enable.md +80 -0
  49. package/skills/lark-base/references/lark-base-base-copy.md +74 -0
  50. package/skills/lark-base/references/lark-base-base-create.md +68 -0
  51. package/skills/lark-base/references/lark-base-base-get.md +39 -0
  52. package/skills/lark-base/references/lark-base-cell-value.md +151 -0
  53. package/skills/lark-base/references/lark-base-dashboard-arrange.md +83 -0
  54. package/skills/lark-base/references/lark-base-dashboard-block-create.md +108 -0
  55. package/skills/lark-base/references/lark-base-dashboard-block-delete.md +46 -0
  56. package/skills/lark-base/references/lark-base-dashboard-block-get.md +57 -0
  57. package/skills/lark-base/references/lark-base-dashboard-block-list.md +53 -0
  58. package/skills/lark-base/references/lark-base-dashboard-block-update.md +84 -0
  59. package/skills/lark-base/references/lark-base-dashboard-create.md +73 -0
  60. package/skills/lark-base/references/lark-base-dashboard-delete.md +44 -0
  61. package/skills/lark-base/references/lark-base-dashboard-get.md +59 -0
  62. package/skills/lark-base/references/lark-base-dashboard-list.md +52 -0
  63. package/skills/lark-base/references/lark-base-dashboard-update.md +69 -0
  64. package/skills/lark-base/references/lark-base-dashboard.md +240 -0
  65. package/skills/lark-base/references/lark-base-data-analysis-sop.md +88 -0
  66. package/skills/lark-base/references/lark-base-data-query.md +375 -0
  67. package/skills/lark-base/references/lark-base-field-create.md +104 -0
  68. package/skills/lark-base/references/lark-base-field-delete.md +51 -0
  69. package/skills/lark-base/references/lark-base-field-get.md +42 -0
  70. package/skills/lark-base/references/lark-base-field-list.md +44 -0
  71. package/skills/lark-base/references/lark-base-field-search-options.md +48 -0
  72. package/skills/lark-base/references/lark-base-field-update.md +97 -0
  73. package/skills/lark-base/references/lark-base-field.md +22 -0
  74. package/skills/lark-base/references/lark-base-form-create.md +87 -0
  75. package/skills/lark-base/references/lark-base-form-delete.md +64 -0
  76. package/skills/lark-base/references/lark-base-form-detail.md +198 -0
  77. package/skills/lark-base/references/lark-base-form-get.md +68 -0
  78. package/skills/lark-base/references/lark-base-form-list.md +73 -0
  79. package/skills/lark-base/references/lark-base-form-questions-create.md +118 -0
  80. package/skills/lark-base/references/lark-base-form-questions-delete.md +68 -0
  81. package/skills/lark-base/references/lark-base-form-questions-list.md +84 -0
  82. package/skills/lark-base/references/lark-base-form-questions-update.md +92 -0
  83. package/skills/lark-base/references/lark-base-form-questions.md +23 -0
  84. package/skills/lark-base/references/lark-base-form-submit.md +171 -0
  85. package/skills/lark-base/references/lark-base-form-update.md +82 -0
  86. package/skills/lark-base/references/lark-base-form.md +25 -0
  87. package/skills/lark-base/references/lark-base-history.md +16 -0
  88. package/skills/lark-base/references/lark-base-record-batch-create.md +58 -0
  89. package/skills/lark-base/references/lark-base-record-batch-update.md +53 -0
  90. package/skills/lark-base/references/lark-base-record-delete.md +62 -0
  91. package/skills/lark-base/references/lark-base-record-history-list.md +86 -0
  92. package/skills/lark-base/references/lark-base-record-share-link-create.md +72 -0
  93. package/skills/lark-base/references/lark-base-record-upsert.md +64 -0
  94. package/skills/lark-base/references/lark-base-record.md +31 -0
  95. package/skills/lark-base/references/lark-base-role-create.md +89 -0
  96. package/skills/lark-base/references/lark-base-role-delete.md +83 -0
  97. package/skills/lark-base/references/lark-base-role-get.md +87 -0
  98. package/skills/lark-base/references/lark-base-role-list.md +81 -0
  99. package/skills/lark-base/references/lark-base-role-update.md +94 -0
  100. package/skills/lark-base/references/lark-base-shortcut-field-properties.md +481 -0
  101. package/skills/lark-base/references/lark-base-table-create.md +62 -0
  102. package/skills/lark-base/references/lark-base-table-delete.md +51 -0
  103. package/skills/lark-base/references/lark-base-table-get.md +46 -0
  104. package/skills/lark-base/references/lark-base-table-list.md +43 -0
  105. package/skills/lark-base/references/lark-base-table-update.md +49 -0
  106. package/skills/lark-base/references/lark-base-table.md +20 -0
  107. package/skills/lark-base/references/lark-base-view-create.md +50 -0
  108. package/skills/lark-base/references/lark-base-view-delete.md +48 -0
  109. package/skills/lark-base/references/lark-base-view-get-card.md +38 -0
  110. package/skills/lark-base/references/lark-base-view-get-filter.md +38 -0
  111. package/skills/lark-base/references/lark-base-view-get-group.md +38 -0
  112. package/skills/lark-base/references/lark-base-view-get-sort.md +38 -0
  113. package/skills/lark-base/references/lark-base-view-get-timebar.md +38 -0
  114. package/skills/lark-base/references/lark-base-view-get-visible-fields.md +28 -0
  115. package/skills/lark-base/references/lark-base-view-get.md +38 -0
  116. package/skills/lark-base/references/lark-base-view-list.md +44 -0
  117. package/skills/lark-base/references/lark-base-view-rename.md +44 -0
  118. package/skills/lark-base/references/lark-base-view-set-card.md +55 -0
  119. package/skills/lark-base/references/lark-base-view-set-filter.md +181 -0
  120. package/skills/lark-base/references/lark-base-view-set-group.md +65 -0
  121. package/skills/lark-base/references/lark-base-view-set-sort.md +63 -0
  122. package/skills/lark-base/references/lark-base-view-set-timebar.md +51 -0
  123. package/skills/lark-base/references/lark-base-view-set-visible-fields.md +46 -0
  124. package/skills/lark-base/references/lark-base-view.md +44 -0
  125. package/skills/lark-base/references/lark-base-workflow-create.md +180 -0
  126. package/skills/lark-base/references/lark-base-workflow-disable.md +94 -0
  127. package/skills/lark-base/references/lark-base-workflow-enable.md +94 -0
  128. package/skills/lark-base/references/lark-base-workflow-get.md +147 -0
  129. package/skills/lark-base/references/lark-base-workflow-guide.md +718 -0
  130. package/skills/lark-base/references/lark-base-workflow-list.md +124 -0
  131. package/skills/lark-base/references/lark-base-workflow-schema.md +935 -0
  132. package/skills/lark-base/references/lark-base-workflow-update.md +167 -0
  133. package/skills/lark-base/references/lark-base-workflow.md +23 -0
  134. package/skills/lark-base/references/lark-base-workspace.md +18 -0
  135. package/skills/lark-base/references/lookup-field-guide.md +512 -0
  136. package/skills/lark-base/references/role-config.md +539 -0
  137. package/skills/lark-calendar/SKILL.md +154 -0
  138. package/skills/lark-calendar/references/lark-calendar-agenda.md +78 -0
  139. package/skills/lark-calendar/references/lark-calendar-create.md +109 -0
  140. package/skills/lark-calendar/references/lark-calendar-freebusy.md +124 -0
  141. package/skills/lark-calendar/references/lark-calendar-room-find.md +113 -0
  142. package/skills/lark-calendar/references/lark-calendar-rsvp.md +42 -0
  143. package/skills/lark-calendar/references/lark-calendar-schedule-meeting.md +265 -0
  144. package/skills/lark-calendar/references/lark-calendar-suggestion.md +125 -0
  145. package/skills/lark-calendar/references/lark-calendar-update.md +105 -0
  146. package/skills/lark-contact/SKILL.md +45 -0
  147. package/skills/lark-contact/references/lark-contact-get-user.md +19 -0
  148. package/skills/lark-contact/references/lark-contact-search-user.md +124 -0
  149. package/skills/lark-doc/SKILL.md +65 -0
  150. package/skills/lark-doc/references/lark-doc-create.md +89 -0
  151. package/skills/lark-doc/references/lark-doc-fetch.md +141 -0
  152. package/skills/lark-doc/references/lark-doc-md.md +71 -0
  153. package/skills/lark-doc/references/lark-doc-media-download.md +50 -0
  154. package/skills/lark-doc/references/lark-doc-media-insert.md +114 -0
  155. package/skills/lark-doc/references/lark-doc-media-preview.md +41 -0
  156. package/skills/lark-doc/references/lark-doc-update.md +252 -0
  157. package/skills/lark-doc/references/lark-doc-whiteboard.md +100 -0
  158. package/skills/lark-doc/references/lark-doc-xml.md +169 -0
  159. package/skills/lark-doc/references/style/lark-doc-create-workflow.md +56 -0
  160. package/skills/lark-doc/references/style/lark-doc-style.md +106 -0
  161. package/skills/lark-doc/references/style/lark-doc-update-workflow.md +54 -0
  162. package/skills/lark-drive/SKILL.md +369 -0
  163. package/skills/lark-drive/references/lark-drive-add-comment.md +182 -0
  164. package/skills/lark-drive/references/lark-drive-apply-permission.md +77 -0
  165. package/skills/lark-drive/references/lark-drive-create-folder.md +73 -0
  166. package/skills/lark-drive/references/lark-drive-create-shortcut.md +103 -0
  167. package/skills/lark-drive/references/lark-drive-delete.md +79 -0
  168. package/skills/lark-drive/references/lark-drive-download.md +31 -0
  169. package/skills/lark-drive/references/lark-drive-export-download.md +50 -0
  170. package/skills/lark-drive/references/lark-drive-export.md +119 -0
  171. package/skills/lark-drive/references/lark-drive-import.md +159 -0
  172. package/skills/lark-drive/references/lark-drive-inspect.md +50 -0
  173. package/skills/lark-drive/references/lark-drive-move.md +120 -0
  174. package/skills/lark-drive/references/lark-drive-pull.md +137 -0
  175. package/skills/lark-drive/references/lark-drive-push.md +162 -0
  176. package/skills/lark-drive/references/lark-drive-reactions.md +113 -0
  177. package/skills/lark-drive/references/lark-drive-search.md +266 -0
  178. package/skills/lark-drive/references/lark-drive-status.md +198 -0
  179. package/skills/lark-drive/references/lark-drive-task-result.md +302 -0
  180. package/skills/lark-drive/references/lark-drive-upload.md +67 -0
  181. package/skills/lark-drive/references/lark-drive-version-delete.md +38 -0
  182. package/skills/lark-drive/references/lark-drive-version-get.md +71 -0
  183. package/skills/lark-drive/references/lark-drive-version-history.md +73 -0
  184. package/skills/lark-drive/references/lark-drive-version-revert.md +198 -0
  185. package/skills/lark-event/SKILL.md +145 -0
  186. package/skills/lark-event/references/lark-event-im.md +86 -0
  187. package/skills/lark-im/SKILL.md +162 -0
  188. package/skills/lark-im/references/lark-im-chat-create.md +162 -0
  189. package/skills/lark-im/references/lark-im-chat-identity.md +55 -0
  190. package/skills/lark-im/references/lark-im-chat-list.md +198 -0
  191. package/skills/lark-im/references/lark-im-chat-messages-list.md +148 -0
  192. package/skills/lark-im/references/lark-im-chat-search.md +136 -0
  193. package/skills/lark-im/references/lark-im-chat-update.md +84 -0
  194. package/skills/lark-im/references/lark-im-flag-cancel.md +198 -0
  195. package/skills/lark-im/references/lark-im-flag-create.md +67 -0
  196. package/skills/lark-im/references/lark-im-flag-list.md +100 -0
  197. package/skills/lark-im/references/lark-im-messages-mget.md +95 -0
  198. package/skills/lark-im/references/lark-im-messages-reply.md +228 -0
  199. package/skills/lark-im/references/lark-im-messages-resources-download.md +94 -0
  200. package/skills/lark-im/references/lark-im-messages-search.md +232 -0
  201. package/skills/lark-im/references/lark-im-messages-send.md +229 -0
  202. package/skills/lark-im/references/lark-im-reactions.md +297 -0
  203. package/skills/lark-im/references/lark-im-threads-messages-list.md +111 -0
  204. package/skills/lark-mail/SKILL.md +648 -0
  205. package/skills/lark-mail/references/lark-mail-decline-receipt.md +115 -0
  206. package/skills/lark-mail/references/lark-mail-draft-create.md +123 -0
  207. package/skills/lark-mail/references/lark-mail-draft-edit.md +400 -0
  208. package/skills/lark-mail/references/lark-mail-forward.md +173 -0
  209. package/skills/lark-mail/references/lark-mail-message.md +230 -0
  210. package/skills/lark-mail/references/lark-mail-messages.md +108 -0
  211. package/skills/lark-mail/references/lark-mail-reply-all.md +206 -0
  212. package/skills/lark-mail/references/lark-mail-reply.md +242 -0
  213. package/skills/lark-mail/references/lark-mail-send-receipt.md +198 -0
  214. package/skills/lark-mail/references/lark-mail-send.md +216 -0
  215. package/skills/lark-mail/references/lark-mail-share-to-chat.md +198 -0
  216. package/skills/lark-mail/references/lark-mail-signature.md +98 -0
  217. package/skills/lark-mail/references/lark-mail-template-create.md +129 -0
  218. package/skills/lark-mail/references/lark-mail-template-update.md +198 -0
  219. package/skills/lark-mail/references/lark-mail-thread.md +111 -0
  220. package/skills/lark-mail/references/lark-mail-triage.md +122 -0
  221. package/skills/lark-mail/references/lark-mail-watch.md +94 -0
  222. package/skills/lark-minutes/SKILL.md +139 -0
  223. package/skills/lark-minutes/references/lark-minutes-download.md +137 -0
  224. package/skills/lark-minutes/references/lark-minutes-search.md +206 -0
  225. package/skills/lark-minutes/references/lark-minutes-upload.md +104 -0
  226. package/skills/lark-okr/SKILL.md +133 -0
  227. package/skills/lark-okr/references/lark-okr-contentblock.md +359 -0
  228. package/skills/lark-okr/references/lark-okr-cycle-detail.md +84 -0
  229. package/skills/lark-okr/references/lark-okr-cycle-list.md +90 -0
  230. package/skills/lark-okr/references/lark-okr-entities.md +329 -0
  231. package/skills/lark-okr/references/lark-okr-image-upload.md +116 -0
  232. package/skills/lark-okr/references/lark-okr-progress-create.md +81 -0
  233. package/skills/lark-okr/references/lark-okr-progress-delete.md +47 -0
  234. package/skills/lark-okr/references/lark-okr-progress-get.md +62 -0
  235. package/skills/lark-okr/references/lark-okr-progress-list.md +80 -0
  236. package/skills/lark-okr/references/lark-okr-progress-update.md +81 -0
  237. package/skills/lark-openapi-explorer/SKILL.md +153 -0
  238. package/skills/lark-shared/SKILL.md +144 -0
  239. package/skills/lark-sheets/SKILL.md +343 -0
  240. package/skills/lark-sheets/references/lark-sheets-cell-data.md +197 -0
  241. package/skills/lark-sheets/references/lark-sheets-cell-images.md +59 -0
  242. package/skills/lark-sheets/references/lark-sheets-cell-style-and-merge.md +141 -0
  243. package/skills/lark-sheets/references/lark-sheets-dropdown.md +133 -0
  244. package/skills/lark-sheets/references/lark-sheets-filter-views.md +193 -0
  245. package/skills/lark-sheets/references/lark-sheets-float-images.md +125 -0
  246. package/skills/lark-sheets/references/lark-sheets-formula.md +88 -0
  247. package/skills/lark-sheets/references/lark-sheets-row-column-management.md +151 -0
  248. package/skills/lark-sheets/references/lark-sheets-sheet-management.md +164 -0
  249. package/skills/lark-sheets/references/lark-sheets-spreadsheet-management.md +140 -0
  250. package/skills/lark-skill-maker/SKILL.md +85 -0
  251. package/skills/lark-slides/SKILL.md +296 -0
  252. package/skills/lark-slides/assets/templates/administration--all_hands_meeting.xml +1999 -0
  253. package/skills/lark-slides/assets/templates/administration--annual_gala.xml +1160 -0
  254. package/skills/lark-slides/assets/templates/administration--company_intro.xml +1376 -0
  255. package/skills/lark-slides/assets/templates/administration--corporate_culture.xml +1765 -0
  256. package/skills/lark-slides/assets/templates/hr--employee_training.xml +912 -0
  257. package/skills/lark-slides/assets/templates/hr--employee_training_workshop.xml +1504 -0
  258. package/skills/lark-slides/assets/templates/hr--onboarding.xml +933 -0
  259. package/skills/lark-slides/assets/templates/marketing--brand_communication.xml +1367 -0
  260. package/skills/lark-slides/assets/templates/marketing--brand_logo_design.xml +1347 -0
  261. package/skills/lark-slides/assets/templates/marketing--brand_operations_plan.xml +1309 -0
  262. package/skills/lark-slides/assets/templates/marketing--business_plan.xml +1646 -0
  263. package/skills/lark-slides/assets/templates/marketing--marketing_plan.xml +198 -0
  264. package/skills/lark-slides/assets/templates/marketing--marketing_strategy.xml +1484 -0
  265. package/skills/lark-slides/assets/templates/marketing--product_whitepaper.xml +198 -0
  266. package/skills/lark-slides/assets/templates/marketing--roadshow_business_plan.xml +1506 -0
  267. package/skills/lark-slides/assets/templates/misc--book_sharing.xml +198 -0
  268. package/skills/lark-slides/assets/templates/misc--club_event_plan.xml +4885 -0
  269. package/skills/lark-slides/assets/templates/misc--student_career_plan.xml +1854 -0
  270. package/skills/lark-slides/assets/templates/office--dark_general.xml +3763 -0
  271. package/skills/lark-slides/assets/templates/office--dept_annual_report.xml +1192 -0
  272. package/skills/lark-slides/assets/templates/office--light_general.xml +3378 -0
  273. package/skills/lark-slides/assets/templates/office--project_kickoff.xml +3152 -0
  274. package/skills/lark-slides/assets/templates/office--quarterly_review.xml +1253 -0
  275. package/skills/lark-slides/assets/templates/office--work_report.xml +1099 -0
  276. package/skills/lark-slides/assets/templates/office--work_summary.xml +4420 -0
  277. package/skills/lark-slides/assets/templates/office--work_summary_report.xml +1523 -0
  278. package/skills/lark-slides/assets/templates/operations--brand_logo_design.xml +1347 -0
  279. package/skills/lark-slides/assets/templates/operations--brand_operations_plan.xml +1309 -0
  280. package/skills/lark-slides/assets/templates/operations--marketing_plan.xml +1469 -0
  281. package/skills/lark-slides/assets/templates/operations--product_promotion.xml +687 -0
  282. package/skills/lark-slides/assets/templates/personal--experience_sharing.xml +2242 -0
  283. package/skills/lark-slides/assets/templates/personal--personal_resume.xml +2047 -0
  284. package/skills/lark-slides/assets/templates/personal--promotion_defense.xml +1099 -0
  285. package/skills/lark-slides/assets/templates/personal--promotion_report.xml +1039 -0
  286. package/skills/lark-slides/assets/templates/personal--self_intro.xml +696 -0
  287. package/skills/lark-slides/assets/templates/personal--teaching_sharing.xml +3013 -0
  288. package/skills/lark-slides/assets/templates/product--business_case_analysis.xml +1341 -0
  289. package/skills/lark-slides/assets/templates/product--market_analysis.xml +898 -0
  290. package/skills/lark-slides/assets/templates/product--product_analysis.xml +1537 -0
  291. package/skills/lark-slides/assets/templates/product--product_intro.xml +2838 -0
  292. package/skills/lark-slides/assets/templates/product--product_promotion.xml +687 -0
  293. package/skills/lark-slides/assets/templates/product--product_promotion_2.xml +198 -0
  294. package/skills/lark-slides/references/asset-planning.md +124 -0
  295. package/skills/lark-slides/references/examples.md +261 -0
  296. package/skills/lark-slides/references/lark-slides-create.md +137 -0
  297. package/skills/lark-slides/references/lark-slides-edit-workflows.md +142 -0
  298. package/skills/lark-slides/references/lark-slides-media-upload.md +128 -0
  299. package/skills/lark-slides/references/lark-slides-replace-slide.md +239 -0
  300. package/skills/lark-slides/references/lark-slides-xml-presentation-slide-create.md +220 -0
  301. package/skills/lark-slides/references/lark-slides-xml-presentation-slide-delete.md +123 -0
  302. package/skills/lark-slides/references/lark-slides-xml-presentation-slide-get.md +110 -0
  303. package/skills/lark-slides/references/lark-slides-xml-presentation-slide-replace.md +186 -0
  304. package/skills/lark-slides/references/lark-slides-xml-presentations-get.md +98 -0
  305. package/skills/lark-slides/references/planning-layer.md +219 -0
  306. package/skills/lark-slides/references/slide-templates.md +201 -0
  307. package/skills/lark-slides/references/slides_demo.xml +226 -0
  308. package/skills/lark-slides/references/slides_xml_schema_definition.xml +3004 -0
  309. package/skills/lark-slides/references/template-catalog.md +463 -0
  310. package/skills/lark-slides/references/template-index.json +198 -0
  311. package/skills/lark-slides/references/troubleshooting.md +198 -0
  312. package/skills/lark-slides/references/validation-checklist.md +102 -0
  313. package/skills/lark-slides/references/visual-planning.md +250 -0
  314. package/skills/lark-slides/references/xml-format-guide.md +369 -0
  315. package/skills/lark-slides/references/xml-schema-quick-ref.md +215 -0
  316. package/skills/lark-slides/scripts/template_tool.py +970 -0
  317. package/skills/lark-slides/scripts/template_tool_test.py +177 -0
  318. package/skills/lark-slides/scripts/xml_text_overlap_lint.py +367 -0
  319. package/skills/lark-slides/scripts/xml_text_overlap_lint_test.py +263 -0
  320. package/skills/lark-task/SKILL.md +165 -0
  321. package/skills/lark-task/references/lark-task-assign.md +38 -0
  322. package/skills/lark-task/references/lark-task-comment.md +28 -0
  323. package/skills/lark-task/references/lark-task-complete.md +27 -0
  324. package/skills/lark-task/references/lark-task-create.md +57 -0
  325. package/skills/lark-task/references/lark-task-followers.md +35 -0
  326. package/skills/lark-task/references/lark-task-get-my-tasks.md +55 -0
  327. package/skills/lark-task/references/lark-task-get-related-tasks.md +53 -0
  328. package/skills/lark-task/references/lark-task-reminder.md +36 -0
  329. package/skills/lark-task/references/lark-task-reopen.md +27 -0
  330. package/skills/lark-task/references/lark-task-search.md +41 -0
  331. package/skills/lark-task/references/lark-task-set-ancestor.md +32 -0
  332. package/skills/lark-task/references/lark-task-subscribe-event.md +86 -0
  333. package/skills/lark-task/references/lark-task-tasklist-create.md +35 -0
  334. package/skills/lark-task/references/lark-task-tasklist-members.md +36 -0
  335. package/skills/lark-task/references/lark-task-tasklist-search.md +38 -0
  336. package/skills/lark-task/references/lark-task-tasklist-task-add.md +38 -0
  337. package/skills/lark-task/references/lark-task-update.md +37 -0
  338. package/skills/lark-task/references/lark-task-upload-attachment.md +59 -0
  339. package/skills/lark-vc/SKILL.md +168 -0
  340. package/skills/lark-vc/references/lark-vc-notes.md +126 -0
  341. package/skills/lark-vc/references/lark-vc-recording.md +153 -0
  342. package/skills/lark-vc/references/lark-vc-search.md +193 -0
  343. package/skills/lark-vc-agent/SKILL.md +121 -0
  344. package/skills/lark-vc-agent/references/lark-vc-agent-meeting-events.md +247 -0
  345. package/skills/lark-vc-agent/references/lark-vc-agent-meeting-join.md +133 -0
  346. package/skills/lark-vc-agent/references/lark-vc-agent-meeting-leave.md +111 -0
  347. package/skills/lark-whiteboard/SKILL.md +144 -0
  348. package/skills/lark-whiteboard/references/connectors.md +102 -0
  349. package/skills/lark-whiteboard/references/content.md +40 -0
  350. package/skills/lark-whiteboard/references/image.md +80 -0
  351. package/skills/lark-whiteboard/references/lark-whiteboard-query.md +49 -0
  352. package/skills/lark-whiteboard/references/lark-whiteboard-update.md +100 -0
  353. package/skills/lark-whiteboard/references/layout.md +374 -0
  354. package/skills/lark-whiteboard/references/schema.md +357 -0
  355. package/skills/lark-whiteboard/references/style.md +318 -0
  356. package/skills/lark-whiteboard/references/typography.md +73 -0
  357. package/skills/lark-whiteboard/routes/dsl.md +107 -0
  358. package/skills/lark-whiteboard/routes/mermaid.md +27 -0
  359. package/skills/lark-whiteboard/routes/svg.md +54 -0
  360. package/skills/lark-whiteboard/scenes/architecture.md +433 -0
  361. package/skills/lark-whiteboard/scenes/bar-chart.md +187 -0
  362. package/skills/lark-whiteboard/scenes/comparison.md +135 -0
  363. package/skills/lark-whiteboard/scenes/fishbone.md +238 -0
  364. package/skills/lark-whiteboard/scenes/flowchart.md +198 -0
  365. package/skills/lark-whiteboard/scenes/flywheel.md +195 -0
  366. package/skills/lark-whiteboard/scenes/funnel.md +198 -0
  367. package/skills/lark-whiteboard/scenes/line-chart.md +214 -0
  368. package/skills/lark-whiteboard/scenes/mermaid.md +130 -0
  369. package/skills/lark-whiteboard/scenes/milestone.md +139 -0
  370. package/skills/lark-whiteboard/scenes/organization.md +173 -0
  371. package/skills/lark-whiteboard/scenes/photo-showcase.md +198 -0
  372. package/skills/lark-whiteboard/scenes/pyramid.md +99 -0
  373. package/skills/lark-whiteboard/scenes/swimlane.md +371 -0
  374. package/skills/lark-whiteboard/scenes/treemap.md +198 -0
  375. package/skills/lark-wiki/SKILL.md +119 -0
  376. package/skills/lark-wiki/references/lark-wiki-delete-space.md +205 -0
  377. package/skills/lark-wiki/references/lark-wiki-member-add.md +66 -0
  378. package/skills/lark-wiki/references/lark-wiki-member-list.md +76 -0
  379. package/skills/lark-wiki/references/lark-wiki-member-remove.md +61 -0
  380. package/skills/lark-wiki/references/lark-wiki-move.md +183 -0
  381. package/skills/lark-wiki/references/lark-wiki-node-copy.md +72 -0
  382. package/skills/lark-wiki/references/lark-wiki-node-create.md +127 -0
  383. package/skills/lark-wiki/references/lark-wiki-node-delete.md +62 -0
  384. package/skills/lark-wiki/references/lark-wiki-node-get.md +56 -0
  385. package/skills/lark-wiki/references/lark-wiki-node-list.md +198 -0
  386. package/skills/lark-wiki/references/lark-wiki-space-create.md +46 -0
  387. package/skills/lark-wiki/references/lark-wiki-space-list.md +198 -0
  388. package/skills/lark-workflow-meeting-summary/SKILL.md +104 -0
  389. package/skills/lark-workflow-standup-report/SKILL.md +120 -0
  390. package/skills/mmx-cli/SKILL.md +440 -0
  391. package/src/bot.ts +109 -0
  392. package/src/cards.ts +105 -0
  393. package/src/config.ts +33 -0
  394. package/src/daemon.ts +217 -0
  395. package/src/extension.ts +132 -0
  396. package/src/poller.ts +135 -0
  397. package/src/renderer.ts +47 -0
  398. package/src/session-registry.ts +90 -0
  399. package/src/types.ts +29 -0
  400. package/tests/bot.test.ts +104 -0
  401. package/tests/cards.test.ts +48 -0
  402. package/tests/config.test.ts +59 -0
  403. package/tests/renderer.test.ts +74 -0
  404. package/tests/session-registry.test.ts +94 -0
  405. package/tests/types.test.ts +35 -0
  406. package/tsconfig.json +16 -0
  407. package/vitest.config.ts +7 -0
@@ -0,0 +1,177 @@
1
+ # Copyright (c) 2026 Lark Technologies Pte. Ltd.
2
+ # SPDX-License-Identifier: MIT
3
+ from __future__ import annotations
4
+
5
+ import tempfile
6
+ import unittest
7
+ from pathlib import Path
8
+
9
+ import template_tool
10
+
11
+
12
+ class TemplateToolTest(unittest.TestCase):
13
+ @classmethod
14
+ def setUpClass(cls) -> None:
15
+ cls.index_data = template_tool.build_index_data()
16
+
17
+ def test_build_index_data_exposes_light_general_metadata(self) -> None:
18
+ template = next(
19
+ entry for entry in self.index_data["templates"] if entry["template_id"] == "office--light_general"
20
+ )
21
+ expected_keys = {
22
+ "template_id",
23
+ "category",
24
+ "category_label",
25
+ "scene",
26
+ "tone",
27
+ "formality",
28
+ "is_general_template",
29
+ "slide_count",
30
+ "presentation_title",
31
+ "palette",
32
+ "structure",
33
+ "page_types",
34
+ "layout_tags",
35
+ "use_cases",
36
+ "ranges",
37
+ }
38
+ self.assertEqual(set(template.keys()), expected_keys)
39
+ self.assertEqual(template["tone"], "light")
40
+ self.assertEqual(template["formality"], "formal")
41
+ self.assertEqual(template["slide_count"], 54)
42
+ self.assertEqual(template["presentation_title"], "白底通用模板")
43
+ self.assertIsInstance(template["layout_tags"], list)
44
+ self.assertNotIn("theme_summary", template)
45
+ self.assertNotIn("editable_regions", template)
46
+ self.assertNotIn("bbox_summary", template)
47
+
48
+ def test_search_templates_keeps_work_report_templates_in_top_results(self) -> None:
49
+ results = template_tool.search_templates(self.index_data, {"query": "工作汇报", "limit": 3})
50
+ self.assertTrue(results)
51
+ self.assertTrue(any(entry["template_id"] == "office--work_report" for entry in results))
52
+
53
+ def test_search_templates_extracts_scene_from_long_chinese_prompt(self) -> None:
54
+ results = template_tool.search_templates(
55
+ self.index_data,
56
+ {"query": "帮我做一个季度工作汇报PPT,偏正式", "limit": 3},
57
+ )
58
+ self.assertTrue(results)
59
+ self.assertTrue(any(entry["template_id"] == "office--work_report" for entry in results))
60
+
61
+ def test_search_templates_maps_chinese_tone_words(self) -> None:
62
+ results = template_tool.search_templates(
63
+ self.index_data,
64
+ {"query": "深色科技感产品发布", "limit": 5},
65
+ )
66
+ self.assertTrue(results)
67
+ self.assertTrue(any(entry["tone"] == "dark" for entry in results))
68
+
69
+ def test_search_templates_finds_product_launch_and_promotion_defense(self) -> None:
70
+ product_results = template_tool.search_templates(
71
+ self.index_data,
72
+ {"query": "产品发布会新品介绍", "limit": 5},
73
+ )
74
+ self.assertTrue(product_results)
75
+ self.assertTrue(
76
+ any(
77
+ entry["template_id"]
78
+ in {"office--project_kickoff", "product--product_intro", "product--product_promotion"}
79
+ for entry in product_results
80
+ )
81
+ )
82
+
83
+ defense_results = template_tool.search_templates(
84
+ self.index_data,
85
+ {"query": "晋升答辩 个人述职", "limit": 5},
86
+ )
87
+ self.assertTrue(defense_results)
88
+ self.assertTrue(any(entry["template_id"] == "personal--promotion_defense" for entry in defense_results))
89
+
90
+ def test_extract_selection_xml_keeps_only_requested_slides_and_theme(self) -> None:
91
+ xml = template_tool.extract_selection_xml(self.index_data, "office--light_general", {"label": "封面"})
92
+ self.assertEqual(len(template_tool.re.findall(r"<slide\b", xml)), 2)
93
+ self.assertIn("<theme>", xml)
94
+ self.assertIn("<title>白底通用模板</title>", xml)
95
+
96
+ def test_summarize_selection_aggregates_slide_titles_and_counts(self) -> None:
97
+ summary = template_tool.summarize_selection(self.index_data, "office--light_general", {"label": "封面"})
98
+ self.assertEqual(summary["selection"]["range"], "1-2")
99
+ self.assertEqual(summary["summary"]["slide_count"], 2)
100
+ self.assertTrue(summary["theme_summary"]["has_theme_node"])
101
+ self.assertIn("通用模板", summary["summary"]["title_hints"])
102
+ self.assertGreater(summary["summary"]["element_totals"]["shape"], 0)
103
+ self.assertIsInstance(summary["slides"][0]["layout_tags"], list)
104
+ self.assertIn("bbox_summary", summary["slides"][0])
105
+ self.assertIn("editable_regions", summary["slides"][0])
106
+
107
+ def test_template_selector_accepts_catalog_visible_filename(self) -> None:
108
+ entry = template_tool.resolve_template_entry(self.index_data, "work_report.xml")
109
+ self.assertEqual(entry["template_id"], "office--work_report")
110
+
111
+ def test_template_path_uses_user_supplied_file(self) -> None:
112
+ source_path = template_tool.TEMPLATES_DIR / "office--work_report.xml"
113
+ with tempfile.TemporaryDirectory() as temp_dir:
114
+ copied_path = Path(temp_dir) / "work_report.xml"
115
+ copied_path.write_text(
116
+ source_path.read_text(encoding="utf-8").replace(
117
+ "<title>工作汇报</title>",
118
+ "<title>Copied Path Template</title>",
119
+ 1,
120
+ ),
121
+ encoding="utf-8",
122
+ )
123
+
124
+ xml = template_tool.extract_selection_xml(
125
+ self.index_data,
126
+ str(copied_path),
127
+ {"range": "1"},
128
+ )
129
+
130
+ self.assertIn("<title>Copied Path Template</title>", xml)
131
+
132
+ def test_template_path_accepts_unindexed_xml_with_range(self) -> None:
133
+ xml = (
134
+ '<presentation xmlns="http://www.larkoffice.com/sml/2.0">'
135
+ "<title>Generated Template</title>"
136
+ "<slide><data></data></slide>"
137
+ "</presentation>"
138
+ )
139
+ with tempfile.TemporaryDirectory() as temp_dir:
140
+ template_path = Path(temp_dir) / "generated.xml"
141
+ template_path.write_text(xml, encoding="utf-8")
142
+
143
+ extracted = template_tool.extract_selection_xml(
144
+ self.index_data,
145
+ str(template_path),
146
+ {"range": "1"},
147
+ )
148
+
149
+ self.assertIn("<title>Generated Template</title>", extracted)
150
+
151
+ def test_search_templates_supports_layout_tag_filtering(self) -> None:
152
+ results = template_tool.search_templates(
153
+ self.index_data,
154
+ {"query": "", "layout-tag": "full-bleed-image-caption", "limit": 10},
155
+ )
156
+ self.assertTrue(results)
157
+ self.assertTrue(
158
+ any("full-bleed-image-caption" in entry["layout_tags"] for entry in results)
159
+ )
160
+
161
+ def test_all_template_files_are_cataloged_and_indexed(self) -> None:
162
+ template_files = sorted(path.stem for path in template_tool.TEMPLATES_DIR.glob("*.xml"))
163
+ indexed_templates = sorted(entry["template_id"] for entry in self.index_data["templates"])
164
+ self.assertEqual(indexed_templates, template_files)
165
+ self.assertEqual(self.index_data["template_count"], len(template_files))
166
+ self.assertTrue(template_files)
167
+
168
+ def test_catalog_range_parser_keeps_comma_separated_ranges(self) -> None:
169
+ template = next(
170
+ entry for entry in self.index_data["templates"] if entry["template_id"] == "operations--product_promotion"
171
+ )
172
+ content_range = next(item for item in template["ranges"] if item["label"] == "内容")
173
+ self.assertEqual(content_range["range"], "3-8, 10-12")
174
+
175
+
176
+ if __name__ == "__main__":
177
+ unittest.main()
@@ -0,0 +1,367 @@
1
+ #!/usr/bin/env python3
2
+ # Copyright (c) 2026 Lark Technologies Pte. Ltd.
3
+ # SPDX-License-Identifier: MIT
4
+
5
+ from __future__ import annotations
6
+
7
+ import json
8
+ import re
9
+ import sys
10
+ import xml.etree.ElementTree as ET
11
+ from difflib import SequenceMatcher
12
+ from pathlib import Path
13
+ from typing import Any
14
+
15
+
16
+ class XmlTextOverlapLintError(Exception):
17
+ pass
18
+
19
+
20
+ def fail(message: str) -> None:
21
+ raise XmlTextOverlapLintError(message)
22
+
23
+
24
+ def read_file(file_path: str | Path) -> str:
25
+ return Path(file_path).read_text(encoding="utf-8")
26
+
27
+
28
+ def parse_args(argv: list[str]) -> dict[str, Any]:
29
+ options: dict[str, Any] = {}
30
+ index = 0
31
+ while index < len(argv):
32
+ token = argv[index]
33
+ if not token.startswith("--"):
34
+ fail(f"unexpected argument: {token}")
35
+ key = token[2:]
36
+ next_token = argv[index + 1] if index + 1 < len(argv) else None
37
+ if next_token is None or next_token.startswith("--"):
38
+ options[key] = True
39
+ index += 1
40
+ continue
41
+ options[key] = next_token
42
+ index += 2
43
+ return options
44
+
45
+
46
+ def extract_attribute(tag_source: str, name: str) -> str | None:
47
+ match = re.search(fr'{re.escape(name)}="([^"]+)"', tag_source)
48
+ return match.group(1) if match else None
49
+
50
+
51
+ def extract_numeric_attribute(tag_source: str, name: str) -> int | float | None:
52
+ raw = extract_attribute(tag_source, name)
53
+ if raw is None:
54
+ return None
55
+ try:
56
+ value = float(raw)
57
+ except ValueError:
58
+ return None
59
+ return int(value) if value.is_integer() else value
60
+
61
+
62
+ def strip_xml(value: str) -> str:
63
+ stripped = re.sub(r"<!\[CDATA\[([\s\S]*?)\]\]>", r"\1", value)
64
+ stripped = re.sub(r"<[^>]+>", " ", stripped)
65
+ stripped = stripped.replace("&nbsp;", " ")
66
+ stripped = stripped.replace("&amp;", "&")
67
+ stripped = stripped.replace("&lt;", "<")
68
+ stripped = stripped.replace("&gt;", ">")
69
+ stripped = stripped.replace("&quot;", '"')
70
+ stripped = stripped.replace("&#39;", "'")
71
+ return re.sub(r"\s+", " ", stripped).strip()
72
+
73
+
74
+ def xml_local_name(tag: str) -> str:
75
+ return tag.rsplit("}", 1)[-1] if tag.startswith("{") else tag
76
+
77
+
78
+ def extract_error_context(xml: str, line: int | None, column: int | None, radius: int = 40) -> str | None:
79
+ if line is None or column is None:
80
+ return None
81
+ lines = xml.splitlines()
82
+ if line < 1 or line > len(lines):
83
+ return None
84
+ source_line = lines[line - 1]
85
+ start = max(column - radius, 0)
86
+ end = min(column + radius, len(source_line))
87
+ return source_line[start:end].strip()
88
+
89
+
90
+ def build_xml_error_issue(error: ET.ParseError, xml: str) -> dict[str, Any]:
91
+ line, column = getattr(error, "position", (None, None))
92
+ return {
93
+ "level": "error",
94
+ "code": "xml_not_well_formed",
95
+ "message": f"XML is not well-formed: {error}",
96
+ "line": line,
97
+ "column": column,
98
+ "context": extract_error_context(xml, line, column),
99
+ "hint": (
100
+ "Escape raw user text before placing it in XML. In text nodes and attribute values, bare & must be "
101
+ "written as &amp;. In text nodes, write < as &lt; and > as &gt;. For attribute URLs, use a=1&amp;b=2."
102
+ ),
103
+ }
104
+
105
+
106
+ def validate_xml_well_formed(xml: str) -> dict[str, Any] | None:
107
+ try:
108
+ root = ET.fromstring(xml)
109
+ except ET.ParseError as error:
110
+ return build_xml_error_issue(error, xml)
111
+
112
+ root_name = xml_local_name(root.tag)
113
+ if root_name not in {"presentation", "slide"}:
114
+ fail("input must contain a <presentation> or <slide> root")
115
+ return None
116
+
117
+
118
+ def parse_presentation(xml: str) -> dict[str, Any]:
119
+ presentation_match = re.search(r"<presentation\b([^>]*)>", xml)
120
+ if presentation_match:
121
+ return {
122
+ "width": int(float(extract_attribute(presentation_match.group(1), "width") or 960)),
123
+ "height": int(float(extract_attribute(presentation_match.group(1), "height") or 540)),
124
+ "slides": re.findall(r"<slide\b[\s\S]*?</slide>", xml),
125
+ }
126
+ slide_match = re.findall(r"<slide\b[\s\S]*?</slide>", xml)
127
+ if slide_match:
128
+ return {"width": 960, "height": 540, "slides": slide_match}
129
+ fail("input must contain a <presentation> or <slide> root")
130
+
131
+
132
+ def extract_elements(slide_xml: str) -> list[dict[str, Any]]:
133
+ elements: list[dict[str, Any]] = []
134
+ for match in re.finditer(r"<shape\b([^>]*)>([\s\S]*?)</shape>", slide_xml):
135
+ attrs, content = match.group(1), match.group(2)
136
+ x = extract_numeric_attribute(attrs, "topLeftX")
137
+ y = extract_numeric_attribute(attrs, "topLeftY")
138
+ width = extract_numeric_attribute(attrs, "width")
139
+ height = extract_numeric_attribute(attrs, "height")
140
+ if all(value is not None for value in [x, y, width, height]):
141
+ font_size = float(extract_attribute(content, "fontSize") or extract_attribute(attrs, "fontSize") or 16)
142
+ elements.append(
143
+ {
144
+ "id": f"shape-{len(elements) + 1}",
145
+ "kind": "shape",
146
+ "type": extract_attribute(attrs, "type") or "shape",
147
+ "textType": extract_attribute(content, "textType"),
148
+ "x": x,
149
+ "y": y,
150
+ "width": width,
151
+ "height": height,
152
+ "fontSize": font_size,
153
+ "text": strip_xml(content),
154
+ }
155
+ )
156
+
157
+ for match in re.finditer(r"<(img|table|chart)\b([^>]*)/?>", slide_xml):
158
+ attrs = match.group(2)
159
+ x = extract_numeric_attribute(attrs, "topLeftX")
160
+ y = extract_numeric_attribute(attrs, "topLeftY")
161
+ width = extract_numeric_attribute(attrs, "width")
162
+ height = extract_numeric_attribute(attrs, "height")
163
+ if all(value is not None for value in [x, y, width, height]):
164
+ elements.append(
165
+ {
166
+ "id": f"{match.group(1)}-{len(elements) + 1}",
167
+ "kind": match.group(1),
168
+ "type": match.group(1),
169
+ "x": x,
170
+ "y": y,
171
+ "width": width,
172
+ "height": height,
173
+ }
174
+ )
175
+ return elements
176
+
177
+
178
+ def intersects(left: dict[str, Any], right: dict[str, Any]) -> bool:
179
+ return (
180
+ left["x"] < right["x"] + right["width"]
181
+ and left["x"] + left["width"] > right["x"]
182
+ and left["y"] < right["y"] + right["height"]
183
+ and left["y"] + left["height"] > right["y"]
184
+ )
185
+
186
+
187
+ def is_text_element(element: dict[str, Any]) -> bool:
188
+ return element["kind"] == "shape" and element["type"] == "text"
189
+
190
+
191
+ def has_text_content(element: dict[str, Any]) -> bool:
192
+ return bool(element.get("text"))
193
+
194
+
195
+ def is_decorative_text(element: dict[str, Any]) -> bool:
196
+ text = element.get("text") or ""
197
+ return bool(text) and re.search(r"[A-Za-z0-9\u4e00-\u9fff]", text) is None
198
+
199
+
200
+ def normalize_text_for_overlap(text: str) -> str:
201
+ return re.sub(r"\s+", "", text)
202
+
203
+
204
+ def is_similar_text_overlay(left: dict[str, Any], right: dict[str, Any]) -> bool:
205
+ left_text = normalize_text_for_overlap(left.get("text") or "")
206
+ right_text = normalize_text_for_overlap(right.get("text") or "")
207
+ if not left_text or not right_text:
208
+ return False
209
+ if left_text == right_text or left_text in right_text or right_text in left_text:
210
+ return True
211
+ return SequenceMatcher(None, left_text, right_text).ratio() >= 0.75
212
+
213
+
214
+ def estimate_text_line_count(element: dict[str, Any]) -> int:
215
+ font_size = element["fontSize"] if isinstance(element["fontSize"], (int, float)) else 16
216
+ chars_per_line = max(1, int(element["width"] // max(font_size * 0.55, 1)))
217
+ paragraphs = [paragraph for paragraph in re.split(r"\n+", element["text"]) if paragraph]
218
+ line_count = 0
219
+ for paragraph in paragraphs:
220
+ logical_length = max(len(paragraph), 1)
221
+ line_count += max(1, -(-logical_length // chars_per_line))
222
+ return max(line_count, 1)
223
+
224
+
225
+ def estimate_text_visual_bbox(element: dict[str, Any]) -> dict[str, int | float] | None:
226
+ if not is_text_element(element) or not has_text_content(element) or is_decorative_text(element):
227
+ return None
228
+
229
+ font_size = element["fontSize"] if isinstance(element["fontSize"], (int, float)) else 16
230
+ char_width = max(font_size * 0.55, 1)
231
+ line_count = estimate_text_line_count(element)
232
+ visual_width = min(element["width"], max(1, len(element["text"]) * char_width))
233
+ visual_height = min(element["height"], max(1, line_count * font_size * 1.2))
234
+ return {
235
+ "x": element["x"],
236
+ "y": element["y"],
237
+ "width": visual_width,
238
+ "height": visual_height,
239
+ }
240
+
241
+
242
+ def intersection_area(left: dict[str, Any], right: dict[str, Any]) -> int | float:
243
+ width = min(left["x"] + left["width"], right["x"] + right["width"]) - max(left["x"], right["x"])
244
+ height = min(left["y"] + left["height"], right["y"] + right["height"]) - max(left["y"], right["y"])
245
+ if width <= 0 or height <= 0:
246
+ return 0
247
+ return width * height
248
+
249
+
250
+ def is_template_text_stack(left: dict[str, Any], right: dict[str, Any]) -> bool:
251
+ if not (is_text_element(left) and is_text_element(right)):
252
+ return False
253
+ if not (has_text_content(left) and has_text_content(right)):
254
+ return True
255
+ top, bottom = sorted([left, right], key=lambda element: element["y"])
256
+ top_type = top.get("textType")
257
+ bottom_type = bottom.get("textType")
258
+ allowed_pairs = {
259
+ ("title", "sub-headline"),
260
+ ("title", None),
261
+ ("headline", "headline"),
262
+ ("headline", None),
263
+ }
264
+ if (top_type, bottom_type) not in allowed_pairs:
265
+ return False
266
+ same_column = abs(top["x"] - bottom["x"]) <= 4
267
+ vertical_offset = bottom["y"] - top["y"]
268
+ top_font_size = float(top.get("fontSize", 16))
269
+ return same_column and vertical_offset >= top_font_size * 0.75
270
+
271
+
272
+ def should_flag_overlap(left: dict[str, Any], right: dict[str, Any]) -> bool:
273
+ if is_text_element(left) and not has_text_content(left):
274
+ return False
275
+ if is_text_element(right) and not has_text_content(right):
276
+ return False
277
+ if is_template_text_stack(left, right):
278
+ return False
279
+ if is_text_element(left) and is_text_element(right):
280
+ if is_similar_text_overlay(left, right):
281
+ return False
282
+ left_visual = estimate_text_visual_bbox(left)
283
+ right_visual = estimate_text_visual_bbox(right)
284
+ if left_visual is None or right_visual is None:
285
+ return False
286
+ overlap_area = intersection_area(left_visual, right_visual)
287
+ if overlap_area <= 0:
288
+ return False
289
+ smaller_area = min(
290
+ left_visual["width"] * left_visual["height"],
291
+ right_visual["width"] * right_visual["height"],
292
+ )
293
+ return smaller_area > 0 and overlap_area / smaller_area >= 0.30
294
+ return False
295
+
296
+
297
+ def lint_slide(slide_xml: str, slide_number: int) -> dict[str, Any]:
298
+ elements = extract_elements(slide_xml)
299
+ issues: list[dict[str, Any]] = []
300
+
301
+ for index, left in enumerate(elements):
302
+ for right in elements[index + 1 :]:
303
+ if not intersects(left, right) or not should_flag_overlap(left, right):
304
+ continue
305
+ issues.append(
306
+ {
307
+ "level": "error",
308
+ "code": "bbox_overlap",
309
+ "elements": [left["id"], right["id"]],
310
+ "message": f'{left["id"]} overlaps {right["id"]}',
311
+ }
312
+ )
313
+
314
+ return {"slide_number": slide_number, "element_count": len(elements), "issues": issues}
315
+
316
+
317
+ def lint_xml(xml: str, source_path: str | None = None) -> dict[str, Any]:
318
+ xml_error = validate_xml_well_formed(xml)
319
+ if xml_error:
320
+ return {
321
+ "file": source_path,
322
+ "slide_size": {"width": 960, "height": 540},
323
+ "summary": {"slide_count": 0, "error_count": 1, "warning_count": 0},
324
+ "issues": [xml_error],
325
+ "slides": [],
326
+ }
327
+
328
+ presentation = parse_presentation(xml)
329
+ slides = [
330
+ lint_slide(slide_xml, index + 1)
331
+ for index, slide_xml in enumerate(presentation["slides"])
332
+ ]
333
+ error_count = sum(1 for slide in slides for issue in slide["issues"] if issue["level"] == "error")
334
+ warning_count = sum(1 for slide in slides for issue in slide["issues"] if issue["level"] == "warning")
335
+ return {
336
+ "file": source_path,
337
+ "slide_size": {"width": presentation["width"], "height": presentation["height"]},
338
+ "summary": {"slide_count": len(slides), "error_count": error_count, "warning_count": warning_count},
339
+ "slides": slides,
340
+ }
341
+
342
+
343
+ def print_usage() -> None:
344
+ print("Usage:\n python3 xml_text_overlap_lint.py --input <presentation.xml>", file=sys.stderr)
345
+
346
+
347
+ def run_cli(argv: list[str] | None = None) -> None:
348
+ options = parse_args(argv or sys.argv[1:])
349
+ if options.get("help") or options.get("--help"):
350
+ print_usage()
351
+ raise SystemExit(0)
352
+ if not options.get("input"):
353
+ print_usage()
354
+ fail("--input is required")
355
+ input_path = Path(options["input"]).resolve()
356
+ result = lint_xml(read_file(input_path), str(input_path))
357
+ print(json.dumps(result, ensure_ascii=False, indent=2))
358
+ if result["summary"]["error_count"] > 0:
359
+ raise SystemExit(1)
360
+
361
+
362
+ if __name__ == "__main__":
363
+ try:
364
+ run_cli()
365
+ except XmlTextOverlapLintError as error:
366
+ print(f"xml-text-overlap-lint error: {error}", file=sys.stderr)
367
+ raise SystemExit(1) from error