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.
- package/LICENSE +21 -0
- package/README.md +111 -0
- package/dist/src/bot.d.ts +29 -0
- package/dist/src/bot.js +75 -0
- package/dist/src/cards.d.ts +6 -0
- package/dist/src/cards.js +87 -0
- package/dist/src/config.d.ts +3 -0
- package/dist/src/config.js +28 -0
- package/dist/src/daemon.d.ts +2 -0
- package/dist/src/daemon.js +151 -0
- package/dist/src/extension.d.ts +2 -0
- package/dist/src/extension.js +124 -0
- package/dist/src/poller.d.ts +33 -0
- package/dist/src/poller.js +94 -0
- package/dist/src/renderer.d.ts +8 -0
- package/dist/src/renderer.js +31 -0
- package/dist/src/session-registry.d.ts +15 -0
- package/dist/src/session-registry.js +82 -0
- package/dist/src/types.d.ts +25 -0
- package/dist/src/types.js +1 -0
- package/dist/tests/bot.test.d.ts +1 -0
- package/dist/tests/bot.test.js +89 -0
- package/dist/tests/cards.test.d.ts +1 -0
- package/dist/tests/cards.test.js +39 -0
- package/dist/tests/config.test.d.ts +1 -0
- package/dist/tests/config.test.js +59 -0
- package/dist/tests/renderer.test.d.ts +1 -0
- package/dist/tests/renderer.test.js +61 -0
- package/dist/tests/session-registry.test.d.ts +1 -0
- package/dist/tests/session-registry.test.js +92 -0
- package/dist/tests/types.test.d.ts +1 -0
- package/dist/tests/types.test.js +30 -0
- package/package.json +35 -0
- package/skills/lark-approval/SKILL.md +56 -0
- package/skills/lark-apps/SKILL.md +92 -0
- package/skills/lark-apps/references/lark-apps-access-scope-get.md +104 -0
- package/skills/lark-apps/references/lark-apps-access-scope-set.md +126 -0
- package/skills/lark-apps/references/lark-apps-create.md +112 -0
- package/skills/lark-apps/references/lark-apps-html-publish.md +151 -0
- package/skills/lark-apps/references/lark-apps-list.md +95 -0
- package/skills/lark-apps/references/lark-apps-update.md +86 -0
- package/skills/lark-attendance/SKILL.md +57 -0
- package/skills/lark-base/SKILL.md +359 -0
- package/skills/lark-base/references/dashboard-block-data-config.md +350 -0
- package/skills/lark-base/references/examples.md +140 -0
- package/skills/lark-base/references/formula-field-guide.md +737 -0
- package/skills/lark-base/references/lark-base-advperm-disable.md +83 -0
- package/skills/lark-base/references/lark-base-advperm-enable.md +80 -0
- package/skills/lark-base/references/lark-base-base-copy.md +74 -0
- package/skills/lark-base/references/lark-base-base-create.md +68 -0
- package/skills/lark-base/references/lark-base-base-get.md +39 -0
- package/skills/lark-base/references/lark-base-cell-value.md +151 -0
- package/skills/lark-base/references/lark-base-dashboard-arrange.md +83 -0
- package/skills/lark-base/references/lark-base-dashboard-block-create.md +108 -0
- package/skills/lark-base/references/lark-base-dashboard-block-delete.md +46 -0
- package/skills/lark-base/references/lark-base-dashboard-block-get.md +57 -0
- package/skills/lark-base/references/lark-base-dashboard-block-list.md +53 -0
- package/skills/lark-base/references/lark-base-dashboard-block-update.md +84 -0
- package/skills/lark-base/references/lark-base-dashboard-create.md +73 -0
- package/skills/lark-base/references/lark-base-dashboard-delete.md +44 -0
- package/skills/lark-base/references/lark-base-dashboard-get.md +59 -0
- package/skills/lark-base/references/lark-base-dashboard-list.md +52 -0
- package/skills/lark-base/references/lark-base-dashboard-update.md +69 -0
- package/skills/lark-base/references/lark-base-dashboard.md +240 -0
- package/skills/lark-base/references/lark-base-data-analysis-sop.md +88 -0
- package/skills/lark-base/references/lark-base-data-query.md +375 -0
- package/skills/lark-base/references/lark-base-field-create.md +104 -0
- package/skills/lark-base/references/lark-base-field-delete.md +51 -0
- package/skills/lark-base/references/lark-base-field-get.md +42 -0
- package/skills/lark-base/references/lark-base-field-list.md +44 -0
- package/skills/lark-base/references/lark-base-field-search-options.md +48 -0
- package/skills/lark-base/references/lark-base-field-update.md +97 -0
- package/skills/lark-base/references/lark-base-field.md +22 -0
- package/skills/lark-base/references/lark-base-form-create.md +87 -0
- package/skills/lark-base/references/lark-base-form-delete.md +64 -0
- package/skills/lark-base/references/lark-base-form-detail.md +198 -0
- package/skills/lark-base/references/lark-base-form-get.md +68 -0
- package/skills/lark-base/references/lark-base-form-list.md +73 -0
- package/skills/lark-base/references/lark-base-form-questions-create.md +118 -0
- package/skills/lark-base/references/lark-base-form-questions-delete.md +68 -0
- package/skills/lark-base/references/lark-base-form-questions-list.md +84 -0
- package/skills/lark-base/references/lark-base-form-questions-update.md +92 -0
- package/skills/lark-base/references/lark-base-form-questions.md +23 -0
- package/skills/lark-base/references/lark-base-form-submit.md +171 -0
- package/skills/lark-base/references/lark-base-form-update.md +82 -0
- package/skills/lark-base/references/lark-base-form.md +25 -0
- package/skills/lark-base/references/lark-base-history.md +16 -0
- package/skills/lark-base/references/lark-base-record-batch-create.md +58 -0
- package/skills/lark-base/references/lark-base-record-batch-update.md +53 -0
- package/skills/lark-base/references/lark-base-record-delete.md +62 -0
- package/skills/lark-base/references/lark-base-record-history-list.md +86 -0
- package/skills/lark-base/references/lark-base-record-share-link-create.md +72 -0
- package/skills/lark-base/references/lark-base-record-upsert.md +64 -0
- package/skills/lark-base/references/lark-base-record.md +31 -0
- package/skills/lark-base/references/lark-base-role-create.md +89 -0
- package/skills/lark-base/references/lark-base-role-delete.md +83 -0
- package/skills/lark-base/references/lark-base-role-get.md +87 -0
- package/skills/lark-base/references/lark-base-role-list.md +81 -0
- package/skills/lark-base/references/lark-base-role-update.md +94 -0
- package/skills/lark-base/references/lark-base-shortcut-field-properties.md +481 -0
- package/skills/lark-base/references/lark-base-table-create.md +62 -0
- package/skills/lark-base/references/lark-base-table-delete.md +51 -0
- package/skills/lark-base/references/lark-base-table-get.md +46 -0
- package/skills/lark-base/references/lark-base-table-list.md +43 -0
- package/skills/lark-base/references/lark-base-table-update.md +49 -0
- package/skills/lark-base/references/lark-base-table.md +20 -0
- package/skills/lark-base/references/lark-base-view-create.md +50 -0
- package/skills/lark-base/references/lark-base-view-delete.md +48 -0
- package/skills/lark-base/references/lark-base-view-get-card.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-filter.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-group.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-sort.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-timebar.md +38 -0
- package/skills/lark-base/references/lark-base-view-get-visible-fields.md +28 -0
- package/skills/lark-base/references/lark-base-view-get.md +38 -0
- package/skills/lark-base/references/lark-base-view-list.md +44 -0
- package/skills/lark-base/references/lark-base-view-rename.md +44 -0
- package/skills/lark-base/references/lark-base-view-set-card.md +55 -0
- package/skills/lark-base/references/lark-base-view-set-filter.md +181 -0
- package/skills/lark-base/references/lark-base-view-set-group.md +65 -0
- package/skills/lark-base/references/lark-base-view-set-sort.md +63 -0
- package/skills/lark-base/references/lark-base-view-set-timebar.md +51 -0
- package/skills/lark-base/references/lark-base-view-set-visible-fields.md +46 -0
- package/skills/lark-base/references/lark-base-view.md +44 -0
- package/skills/lark-base/references/lark-base-workflow-create.md +180 -0
- package/skills/lark-base/references/lark-base-workflow-disable.md +94 -0
- package/skills/lark-base/references/lark-base-workflow-enable.md +94 -0
- package/skills/lark-base/references/lark-base-workflow-get.md +147 -0
- package/skills/lark-base/references/lark-base-workflow-guide.md +718 -0
- package/skills/lark-base/references/lark-base-workflow-list.md +124 -0
- package/skills/lark-base/references/lark-base-workflow-schema.md +935 -0
- package/skills/lark-base/references/lark-base-workflow-update.md +167 -0
- package/skills/lark-base/references/lark-base-workflow.md +23 -0
- package/skills/lark-base/references/lark-base-workspace.md +18 -0
- package/skills/lark-base/references/lookup-field-guide.md +512 -0
- package/skills/lark-base/references/role-config.md +539 -0
- package/skills/lark-calendar/SKILL.md +154 -0
- package/skills/lark-calendar/references/lark-calendar-agenda.md +78 -0
- package/skills/lark-calendar/references/lark-calendar-create.md +109 -0
- package/skills/lark-calendar/references/lark-calendar-freebusy.md +124 -0
- package/skills/lark-calendar/references/lark-calendar-room-find.md +113 -0
- package/skills/lark-calendar/references/lark-calendar-rsvp.md +42 -0
- package/skills/lark-calendar/references/lark-calendar-schedule-meeting.md +265 -0
- package/skills/lark-calendar/references/lark-calendar-suggestion.md +125 -0
- package/skills/lark-calendar/references/lark-calendar-update.md +105 -0
- package/skills/lark-contact/SKILL.md +45 -0
- package/skills/lark-contact/references/lark-contact-get-user.md +19 -0
- package/skills/lark-contact/references/lark-contact-search-user.md +124 -0
- package/skills/lark-doc/SKILL.md +65 -0
- package/skills/lark-doc/references/lark-doc-create.md +89 -0
- package/skills/lark-doc/references/lark-doc-fetch.md +141 -0
- package/skills/lark-doc/references/lark-doc-md.md +71 -0
- package/skills/lark-doc/references/lark-doc-media-download.md +50 -0
- package/skills/lark-doc/references/lark-doc-media-insert.md +114 -0
- package/skills/lark-doc/references/lark-doc-media-preview.md +41 -0
- package/skills/lark-doc/references/lark-doc-update.md +252 -0
- package/skills/lark-doc/references/lark-doc-whiteboard.md +100 -0
- package/skills/lark-doc/references/lark-doc-xml.md +169 -0
- package/skills/lark-doc/references/style/lark-doc-create-workflow.md +56 -0
- package/skills/lark-doc/references/style/lark-doc-style.md +106 -0
- package/skills/lark-doc/references/style/lark-doc-update-workflow.md +54 -0
- package/skills/lark-drive/SKILL.md +369 -0
- package/skills/lark-drive/references/lark-drive-add-comment.md +182 -0
- package/skills/lark-drive/references/lark-drive-apply-permission.md +77 -0
- package/skills/lark-drive/references/lark-drive-create-folder.md +73 -0
- package/skills/lark-drive/references/lark-drive-create-shortcut.md +103 -0
- package/skills/lark-drive/references/lark-drive-delete.md +79 -0
- package/skills/lark-drive/references/lark-drive-download.md +31 -0
- package/skills/lark-drive/references/lark-drive-export-download.md +50 -0
- package/skills/lark-drive/references/lark-drive-export.md +119 -0
- package/skills/lark-drive/references/lark-drive-import.md +159 -0
- package/skills/lark-drive/references/lark-drive-inspect.md +50 -0
- package/skills/lark-drive/references/lark-drive-move.md +120 -0
- package/skills/lark-drive/references/lark-drive-pull.md +137 -0
- package/skills/lark-drive/references/lark-drive-push.md +162 -0
- package/skills/lark-drive/references/lark-drive-reactions.md +113 -0
- package/skills/lark-drive/references/lark-drive-search.md +266 -0
- package/skills/lark-drive/references/lark-drive-status.md +198 -0
- package/skills/lark-drive/references/lark-drive-task-result.md +302 -0
- package/skills/lark-drive/references/lark-drive-upload.md +67 -0
- package/skills/lark-drive/references/lark-drive-version-delete.md +38 -0
- package/skills/lark-drive/references/lark-drive-version-get.md +71 -0
- package/skills/lark-drive/references/lark-drive-version-history.md +73 -0
- package/skills/lark-drive/references/lark-drive-version-revert.md +198 -0
- package/skills/lark-event/SKILL.md +145 -0
- package/skills/lark-event/references/lark-event-im.md +86 -0
- package/skills/lark-im/SKILL.md +162 -0
- package/skills/lark-im/references/lark-im-chat-create.md +162 -0
- package/skills/lark-im/references/lark-im-chat-identity.md +55 -0
- package/skills/lark-im/references/lark-im-chat-list.md +198 -0
- package/skills/lark-im/references/lark-im-chat-messages-list.md +148 -0
- package/skills/lark-im/references/lark-im-chat-search.md +136 -0
- package/skills/lark-im/references/lark-im-chat-update.md +84 -0
- package/skills/lark-im/references/lark-im-flag-cancel.md +198 -0
- package/skills/lark-im/references/lark-im-flag-create.md +67 -0
- package/skills/lark-im/references/lark-im-flag-list.md +100 -0
- package/skills/lark-im/references/lark-im-messages-mget.md +95 -0
- package/skills/lark-im/references/lark-im-messages-reply.md +228 -0
- package/skills/lark-im/references/lark-im-messages-resources-download.md +94 -0
- package/skills/lark-im/references/lark-im-messages-search.md +232 -0
- package/skills/lark-im/references/lark-im-messages-send.md +229 -0
- package/skills/lark-im/references/lark-im-reactions.md +297 -0
- package/skills/lark-im/references/lark-im-threads-messages-list.md +111 -0
- package/skills/lark-mail/SKILL.md +648 -0
- package/skills/lark-mail/references/lark-mail-decline-receipt.md +115 -0
- package/skills/lark-mail/references/lark-mail-draft-create.md +123 -0
- package/skills/lark-mail/references/lark-mail-draft-edit.md +400 -0
- package/skills/lark-mail/references/lark-mail-forward.md +173 -0
- package/skills/lark-mail/references/lark-mail-message.md +230 -0
- package/skills/lark-mail/references/lark-mail-messages.md +108 -0
- package/skills/lark-mail/references/lark-mail-reply-all.md +206 -0
- package/skills/lark-mail/references/lark-mail-reply.md +242 -0
- package/skills/lark-mail/references/lark-mail-send-receipt.md +198 -0
- package/skills/lark-mail/references/lark-mail-send.md +216 -0
- package/skills/lark-mail/references/lark-mail-share-to-chat.md +198 -0
- package/skills/lark-mail/references/lark-mail-signature.md +98 -0
- package/skills/lark-mail/references/lark-mail-template-create.md +129 -0
- package/skills/lark-mail/references/lark-mail-template-update.md +198 -0
- package/skills/lark-mail/references/lark-mail-thread.md +111 -0
- package/skills/lark-mail/references/lark-mail-triage.md +122 -0
- package/skills/lark-mail/references/lark-mail-watch.md +94 -0
- package/skills/lark-minutes/SKILL.md +139 -0
- package/skills/lark-minutes/references/lark-minutes-download.md +137 -0
- package/skills/lark-minutes/references/lark-minutes-search.md +206 -0
- package/skills/lark-minutes/references/lark-minutes-upload.md +104 -0
- package/skills/lark-okr/SKILL.md +133 -0
- package/skills/lark-okr/references/lark-okr-contentblock.md +359 -0
- package/skills/lark-okr/references/lark-okr-cycle-detail.md +84 -0
- package/skills/lark-okr/references/lark-okr-cycle-list.md +90 -0
- package/skills/lark-okr/references/lark-okr-entities.md +329 -0
- package/skills/lark-okr/references/lark-okr-image-upload.md +116 -0
- package/skills/lark-okr/references/lark-okr-progress-create.md +81 -0
- package/skills/lark-okr/references/lark-okr-progress-delete.md +47 -0
- package/skills/lark-okr/references/lark-okr-progress-get.md +62 -0
- package/skills/lark-okr/references/lark-okr-progress-list.md +80 -0
- package/skills/lark-okr/references/lark-okr-progress-update.md +81 -0
- package/skills/lark-openapi-explorer/SKILL.md +153 -0
- package/skills/lark-shared/SKILL.md +144 -0
- package/skills/lark-sheets/SKILL.md +343 -0
- package/skills/lark-sheets/references/lark-sheets-cell-data.md +197 -0
- package/skills/lark-sheets/references/lark-sheets-cell-images.md +59 -0
- package/skills/lark-sheets/references/lark-sheets-cell-style-and-merge.md +141 -0
- package/skills/lark-sheets/references/lark-sheets-dropdown.md +133 -0
- package/skills/lark-sheets/references/lark-sheets-filter-views.md +193 -0
- package/skills/lark-sheets/references/lark-sheets-float-images.md +125 -0
- package/skills/lark-sheets/references/lark-sheets-formula.md +88 -0
- package/skills/lark-sheets/references/lark-sheets-row-column-management.md +151 -0
- package/skills/lark-sheets/references/lark-sheets-sheet-management.md +164 -0
- package/skills/lark-sheets/references/lark-sheets-spreadsheet-management.md +140 -0
- package/skills/lark-skill-maker/SKILL.md +85 -0
- package/skills/lark-slides/SKILL.md +296 -0
- package/skills/lark-slides/assets/templates/administration--all_hands_meeting.xml +1999 -0
- package/skills/lark-slides/assets/templates/administration--annual_gala.xml +1160 -0
- package/skills/lark-slides/assets/templates/administration--company_intro.xml +1376 -0
- package/skills/lark-slides/assets/templates/administration--corporate_culture.xml +1765 -0
- package/skills/lark-slides/assets/templates/hr--employee_training.xml +912 -0
- package/skills/lark-slides/assets/templates/hr--employee_training_workshop.xml +1504 -0
- package/skills/lark-slides/assets/templates/hr--onboarding.xml +933 -0
- package/skills/lark-slides/assets/templates/marketing--brand_communication.xml +1367 -0
- package/skills/lark-slides/assets/templates/marketing--brand_logo_design.xml +1347 -0
- package/skills/lark-slides/assets/templates/marketing--brand_operations_plan.xml +1309 -0
- package/skills/lark-slides/assets/templates/marketing--business_plan.xml +1646 -0
- package/skills/lark-slides/assets/templates/marketing--marketing_plan.xml +198 -0
- package/skills/lark-slides/assets/templates/marketing--marketing_strategy.xml +1484 -0
- package/skills/lark-slides/assets/templates/marketing--product_whitepaper.xml +198 -0
- package/skills/lark-slides/assets/templates/marketing--roadshow_business_plan.xml +1506 -0
- package/skills/lark-slides/assets/templates/misc--book_sharing.xml +198 -0
- package/skills/lark-slides/assets/templates/misc--club_event_plan.xml +4885 -0
- package/skills/lark-slides/assets/templates/misc--student_career_plan.xml +1854 -0
- package/skills/lark-slides/assets/templates/office--dark_general.xml +3763 -0
- package/skills/lark-slides/assets/templates/office--dept_annual_report.xml +1192 -0
- package/skills/lark-slides/assets/templates/office--light_general.xml +3378 -0
- package/skills/lark-slides/assets/templates/office--project_kickoff.xml +3152 -0
- package/skills/lark-slides/assets/templates/office--quarterly_review.xml +1253 -0
- package/skills/lark-slides/assets/templates/office--work_report.xml +1099 -0
- package/skills/lark-slides/assets/templates/office--work_summary.xml +4420 -0
- package/skills/lark-slides/assets/templates/office--work_summary_report.xml +1523 -0
- package/skills/lark-slides/assets/templates/operations--brand_logo_design.xml +1347 -0
- package/skills/lark-slides/assets/templates/operations--brand_operations_plan.xml +1309 -0
- package/skills/lark-slides/assets/templates/operations--marketing_plan.xml +1469 -0
- package/skills/lark-slides/assets/templates/operations--product_promotion.xml +687 -0
- package/skills/lark-slides/assets/templates/personal--experience_sharing.xml +2242 -0
- package/skills/lark-slides/assets/templates/personal--personal_resume.xml +2047 -0
- package/skills/lark-slides/assets/templates/personal--promotion_defense.xml +1099 -0
- package/skills/lark-slides/assets/templates/personal--promotion_report.xml +1039 -0
- package/skills/lark-slides/assets/templates/personal--self_intro.xml +696 -0
- package/skills/lark-slides/assets/templates/personal--teaching_sharing.xml +3013 -0
- package/skills/lark-slides/assets/templates/product--business_case_analysis.xml +1341 -0
- package/skills/lark-slides/assets/templates/product--market_analysis.xml +898 -0
- package/skills/lark-slides/assets/templates/product--product_analysis.xml +1537 -0
- package/skills/lark-slides/assets/templates/product--product_intro.xml +2838 -0
- package/skills/lark-slides/assets/templates/product--product_promotion.xml +687 -0
- package/skills/lark-slides/assets/templates/product--product_promotion_2.xml +198 -0
- package/skills/lark-slides/references/asset-planning.md +124 -0
- package/skills/lark-slides/references/examples.md +261 -0
- package/skills/lark-slides/references/lark-slides-create.md +137 -0
- package/skills/lark-slides/references/lark-slides-edit-workflows.md +142 -0
- package/skills/lark-slides/references/lark-slides-media-upload.md +128 -0
- package/skills/lark-slides/references/lark-slides-replace-slide.md +239 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-create.md +220 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-delete.md +123 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-get.md +110 -0
- package/skills/lark-slides/references/lark-slides-xml-presentation-slide-replace.md +186 -0
- package/skills/lark-slides/references/lark-slides-xml-presentations-get.md +98 -0
- package/skills/lark-slides/references/planning-layer.md +219 -0
- package/skills/lark-slides/references/slide-templates.md +201 -0
- package/skills/lark-slides/references/slides_demo.xml +226 -0
- package/skills/lark-slides/references/slides_xml_schema_definition.xml +3004 -0
- package/skills/lark-slides/references/template-catalog.md +463 -0
- package/skills/lark-slides/references/template-index.json +198 -0
- package/skills/lark-slides/references/troubleshooting.md +198 -0
- package/skills/lark-slides/references/validation-checklist.md +102 -0
- package/skills/lark-slides/references/visual-planning.md +250 -0
- package/skills/lark-slides/references/xml-format-guide.md +369 -0
- package/skills/lark-slides/references/xml-schema-quick-ref.md +215 -0
- package/skills/lark-slides/scripts/template_tool.py +970 -0
- package/skills/lark-slides/scripts/template_tool_test.py +177 -0
- package/skills/lark-slides/scripts/xml_text_overlap_lint.py +367 -0
- package/skills/lark-slides/scripts/xml_text_overlap_lint_test.py +263 -0
- package/skills/lark-task/SKILL.md +165 -0
- package/skills/lark-task/references/lark-task-assign.md +38 -0
- package/skills/lark-task/references/lark-task-comment.md +28 -0
- package/skills/lark-task/references/lark-task-complete.md +27 -0
- package/skills/lark-task/references/lark-task-create.md +57 -0
- package/skills/lark-task/references/lark-task-followers.md +35 -0
- package/skills/lark-task/references/lark-task-get-my-tasks.md +55 -0
- package/skills/lark-task/references/lark-task-get-related-tasks.md +53 -0
- package/skills/lark-task/references/lark-task-reminder.md +36 -0
- package/skills/lark-task/references/lark-task-reopen.md +27 -0
- package/skills/lark-task/references/lark-task-search.md +41 -0
- package/skills/lark-task/references/lark-task-set-ancestor.md +32 -0
- package/skills/lark-task/references/lark-task-subscribe-event.md +86 -0
- package/skills/lark-task/references/lark-task-tasklist-create.md +35 -0
- package/skills/lark-task/references/lark-task-tasklist-members.md +36 -0
- package/skills/lark-task/references/lark-task-tasklist-search.md +38 -0
- package/skills/lark-task/references/lark-task-tasklist-task-add.md +38 -0
- package/skills/lark-task/references/lark-task-update.md +37 -0
- package/skills/lark-task/references/lark-task-upload-attachment.md +59 -0
- package/skills/lark-vc/SKILL.md +168 -0
- package/skills/lark-vc/references/lark-vc-notes.md +126 -0
- package/skills/lark-vc/references/lark-vc-recording.md +153 -0
- package/skills/lark-vc/references/lark-vc-search.md +193 -0
- package/skills/lark-vc-agent/SKILL.md +121 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-events.md +247 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-join.md +133 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-leave.md +111 -0
- package/skills/lark-whiteboard/SKILL.md +144 -0
- package/skills/lark-whiteboard/references/connectors.md +102 -0
- package/skills/lark-whiteboard/references/content.md +40 -0
- package/skills/lark-whiteboard/references/image.md +80 -0
- package/skills/lark-whiteboard/references/lark-whiteboard-query.md +49 -0
- package/skills/lark-whiteboard/references/lark-whiteboard-update.md +100 -0
- package/skills/lark-whiteboard/references/layout.md +374 -0
- package/skills/lark-whiteboard/references/schema.md +357 -0
- package/skills/lark-whiteboard/references/style.md +318 -0
- package/skills/lark-whiteboard/references/typography.md +73 -0
- package/skills/lark-whiteboard/routes/dsl.md +107 -0
- package/skills/lark-whiteboard/routes/mermaid.md +27 -0
- package/skills/lark-whiteboard/routes/svg.md +54 -0
- package/skills/lark-whiteboard/scenes/architecture.md +433 -0
- package/skills/lark-whiteboard/scenes/bar-chart.md +187 -0
- package/skills/lark-whiteboard/scenes/comparison.md +135 -0
- package/skills/lark-whiteboard/scenes/fishbone.md +238 -0
- package/skills/lark-whiteboard/scenes/flowchart.md +198 -0
- package/skills/lark-whiteboard/scenes/flywheel.md +195 -0
- package/skills/lark-whiteboard/scenes/funnel.md +198 -0
- package/skills/lark-whiteboard/scenes/line-chart.md +214 -0
- package/skills/lark-whiteboard/scenes/mermaid.md +130 -0
- package/skills/lark-whiteboard/scenes/milestone.md +139 -0
- package/skills/lark-whiteboard/scenes/organization.md +173 -0
- package/skills/lark-whiteboard/scenes/photo-showcase.md +198 -0
- package/skills/lark-whiteboard/scenes/pyramid.md +99 -0
- package/skills/lark-whiteboard/scenes/swimlane.md +371 -0
- package/skills/lark-whiteboard/scenes/treemap.md +198 -0
- package/skills/lark-wiki/SKILL.md +119 -0
- package/skills/lark-wiki/references/lark-wiki-delete-space.md +205 -0
- package/skills/lark-wiki/references/lark-wiki-member-add.md +66 -0
- package/skills/lark-wiki/references/lark-wiki-member-list.md +76 -0
- package/skills/lark-wiki/references/lark-wiki-member-remove.md +61 -0
- package/skills/lark-wiki/references/lark-wiki-move.md +183 -0
- package/skills/lark-wiki/references/lark-wiki-node-copy.md +72 -0
- package/skills/lark-wiki/references/lark-wiki-node-create.md +127 -0
- package/skills/lark-wiki/references/lark-wiki-node-delete.md +62 -0
- package/skills/lark-wiki/references/lark-wiki-node-get.md +56 -0
- package/skills/lark-wiki/references/lark-wiki-node-list.md +198 -0
- package/skills/lark-wiki/references/lark-wiki-space-create.md +46 -0
- package/skills/lark-wiki/references/lark-wiki-space-list.md +198 -0
- package/skills/lark-workflow-meeting-summary/SKILL.md +104 -0
- package/skills/lark-workflow-standup-report/SKILL.md +120 -0
- package/skills/mmx-cli/SKILL.md +440 -0
- package/src/bot.ts +109 -0
- package/src/cards.ts +105 -0
- package/src/config.ts +33 -0
- package/src/daemon.ts +217 -0
- package/src/extension.ts +132 -0
- package/src/poller.ts +135 -0
- package/src/renderer.ts +47 -0
- package/src/session-registry.ts +90 -0
- package/src/types.ts +29 -0
- package/tests/bot.test.ts +104 -0
- package/tests/cards.test.ts +48 -0
- package/tests/config.test.ts +59 -0
- package/tests/renderer.test.ts +74 -0
- package/tests/session-registry.test.ts +94 -0
- package/tests/types.test.ts +35 -0
- package/tsconfig.json +16 -0
- 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(" ", " ")
|
|
66
|
+
stripped = stripped.replace("&", "&")
|
|
67
|
+
stripped = stripped.replace("<", "<")
|
|
68
|
+
stripped = stripped.replace(">", ">")
|
|
69
|
+
stripped = stripped.replace(""", '"')
|
|
70
|
+
stripped = stripped.replace("'", "'")
|
|
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 &. In text nodes, write < as < and > as >. For attribute URLs, use a=1&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
|