@yandy0725/pi-lark 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/README.md +30 -0
- package/README.zh.md +30 -0
- package/package.json +36 -0
- package/skills/lark-approval/SKILL.md +56 -0
- package/skills/lark-approval/references/lark-approval-initiate.md +196 -0
- package/skills/lark-approval/references/lark-approval-instance-form-control-parameters.md +606 -0
- package/skills/lark-approval/references/lark-approval-instance-value-sourcing.md +108 -0
- package/skills/lark-apps/SKILL.md +78 -0
- package/skills/lark-apps/references/lark-apps-access-scope-get.md +28 -0
- package/skills/lark-apps/references/lark-apps-access-scope-set.md +40 -0
- package/skills/lark-apps/references/lark-apps-cloud-dev.md +120 -0
- package/skills/lark-apps/references/lark-apps-create.md +40 -0
- package/skills/lark-apps/references/lark-apps-db-env-create.md +31 -0
- package/skills/lark-apps/references/lark-apps-db-execute.md +40 -0
- package/skills/lark-apps/references/lark-apps-db-table-get.md +29 -0
- package/skills/lark-apps/references/lark-apps-db-table-list.md +31 -0
- package/skills/lark-apps/references/lark-apps-env-pull.md +35 -0
- package/skills/lark-apps/references/lark-apps-git-credential.md +37 -0
- package/skills/lark-apps/references/lark-apps-html-publish.md +57 -0
- package/skills/lark-apps/references/lark-apps-init.md +37 -0
- package/skills/lark-apps/references/lark-apps-list.md +37 -0
- package/skills/lark-apps/references/lark-apps-local-dev.md +76 -0
- package/skills/lark-apps/references/lark-apps-release-create.md +30 -0
- package/skills/lark-apps/references/lark-apps-release-get.md +28 -0
- package/skills/lark-apps/references/lark-apps-release-list.md +31 -0
- package/skills/lark-apps/references/lark-apps-session-messages-list.md +53 -0
- package/skills/lark-apps/references/lark-apps-update.md +30 -0
- package/skills/lark-attendance/SKILL.md +57 -0
- package/skills/lark-base/SKILL.md +157 -0
- package/skills/lark-base/references/dashboard-block-data-config.md +350 -0
- package/skills/lark-base/references/formula-field-guide.md +737 -0
- package/skills/lark-base/references/lark-base-cell-value.md +153 -0
- package/skills/lark-base/references/lark-base-dashboard-block-get-data.md +717 -0
- package/skills/lark-base/references/lark-base-dashboard.md +238 -0
- package/skills/lark-base/references/lark-base-data-analysis-sop.md +210 -0
- package/skills/lark-base/references/lark-base-data-query-guide.md +61 -0
- package/skills/lark-base/references/lark-base-data-query.md +452 -0
- package/skills/lark-base/references/lark-base-field-create.md +103 -0
- package/skills/lark-base/references/lark-base-field-json.md +489 -0
- package/skills/lark-base/references/lark-base-field-update.md +171 -0
- package/skills/lark-base/references/lark-base-form-detail.md +71 -0
- package/skills/lark-base/references/lark-base-form-questions-create.md +118 -0
- package/skills/lark-base/references/lark-base-form-questions-update.md +92 -0
- package/skills/lark-base/references/lark-base-form-submit.md +170 -0
- package/skills/lark-base/references/lark-base-record-batch-create.md +57 -0
- package/skills/lark-base/references/lark-base-record-batch-update.md +52 -0
- package/skills/lark-base/references/lark-base-record-history-list.md +43 -0
- package/skills/lark-base/references/lark-base-record-upsert.md +63 -0
- package/skills/lark-base/references/lark-base-role-guide.md +65 -0
- package/skills/lark-base/references/lark-base-view-set-filter.md +189 -0
- package/skills/lark-base/references/lark-base-workflow-guide.md +830 -0
- package/skills/lark-base/references/lark-base-workflow-schema.md +1071 -0
- package/skills/lark-base/references/lookup-field-guide.md +512 -0
- package/skills/lark-base/references/role-config.md +549 -0
- package/skills/lark-calendar/SKILL.md +137 -0
- package/skills/lark-calendar/references/lark-calendar-agenda.md +78 -0
- package/skills/lark-calendar/references/lark-calendar-create.md +106 -0
- package/skills/lark-calendar/references/lark-calendar-freebusy.md +124 -0
- package/skills/lark-calendar/references/lark-calendar-meeting.md +40 -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-search-event.md +29 -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 +59 -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 +79 -0
- package/skills/lark-doc/references/lark-doc-create.md +79 -0
- package/skills/lark-doc/references/lark-doc-fetch.md +138 -0
- package/skills/lark-doc/references/lark-doc-md.md +76 -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-resource-cover.md +70 -0
- package/skills/lark-doc/references/lark-doc-update.md +259 -0
- package/skills/lark-doc/references/lark-doc-whiteboard.md +154 -0
- package/skills/lark-doc/references/lark-doc-xml.md +181 -0
- package/skills/lark-doc/references/style/lark-doc-create-workflow.md +59 -0
- package/skills/lark-doc/references/style/lark-doc-style.md +86 -0
- package/skills/lark-doc/references/style/lark-doc-update-workflow.md +55 -0
- package/skills/lark-drive/SKILL.md +215 -0
- package/skills/lark-drive/references/lark-drive-add-comment.md +193 -0
- package/skills/lark-drive/references/lark-drive-apply-permission.md +77 -0
- package/skills/lark-drive/references/lark-drive-comment-location.md +193 -0
- package/skills/lark-drive/references/lark-drive-comments-guide.md +72 -0
- package/skills/lark-drive/references/lark-drive-cover.md +79 -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 +145 -0
- package/skills/lark-drive/references/lark-drive-files-list.md +158 -0
- package/skills/lark-drive/references/lark-drive-import.md +170 -0
- package/skills/lark-drive/references/lark-drive-inspect.md +50 -0
- package/skills/lark-drive/references/lark-drive-member-add.md +66 -0
- package/skills/lark-drive/references/lark-drive-move.md +120 -0
- package/skills/lark-drive/references/lark-drive-permission-guide.md +41 -0
- package/skills/lark-drive/references/lark-drive-preview.md +87 -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 +269 -0
- package/skills/lark-drive/references/lark-drive-secure-label.md +52 -0
- package/skills/lark-drive/references/lark-drive-status.md +137 -0
- package/skills/lark-drive/references/lark-drive-task-result.md +302 -0
- package/skills/lark-drive/references/lark-drive-upload.md +101 -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 +35 -0
- package/skills/lark-drive/references/lark-drive-workflow-knowledge-organize-analysis.md +249 -0
- package/skills/lark-drive/references/lark-drive-workflow-knowledge-organize-discovery.md +253 -0
- package/skills/lark-drive/references/lark-drive-workflow-knowledge-organize-execution.md +200 -0
- package/skills/lark-drive/references/lark-drive-workflow-knowledge-organize-planning.md +336 -0
- package/skills/lark-drive/references/lark-drive-workflow-knowledge-organize-rollback.md +308 -0
- package/skills/lark-drive/references/lark-drive-workflow-knowledge-organize.md +226 -0
- package/skills/lark-drive/references/lark-drive-workflow-permission-governance-commands.md +168 -0
- package/skills/lark-drive/references/lark-drive-workflow-permission-governance-outputs.md +424 -0
- package/skills/lark-drive/references/lark-drive-workflow-permission-governance.md +207 -0
- package/skills/lark-drive/references/lark-drive-workflow.md +130 -0
- package/skills/lark-event/SKILL.md +154 -0
- package/skills/lark-event/references/lark-event-im.md +87 -0
- package/skills/lark-event/references/lark-event-minutes.md +54 -0
- package/skills/lark-event/references/lark-event-task.md +78 -0
- package/skills/lark-event/references/lark-event-vc.md +94 -0
- package/skills/lark-event/references/lark-event-whiteboard.md +67 -0
- package/skills/lark-im/SKILL.md +247 -0
- package/skills/lark-im/references/lark-im-card-action-reply.md +175 -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 +166 -0
- package/skills/lark-im/references/lark-im-chat-messages-list.md +157 -0
- package/skills/lark-im/references/lark-im-chat-search.md +142 -0
- package/skills/lark-im/references/lark-im-chat-update.md +84 -0
- package/skills/lark-im/references/lark-im-feed-group-list-item.md +68 -0
- package/skills/lark-im/references/lark-im-feed-group-list.md +65 -0
- package/skills/lark-im/references/lark-im-feed-group-query-item.md +44 -0
- package/skills/lark-im/references/lark-im-feed-groups.md +452 -0
- package/skills/lark-im/references/lark-im-feed-shortcut-create.md +97 -0
- package/skills/lark-im/references/lark-im-feed-shortcut-list.md +103 -0
- package/skills/lark-im/references/lark-im-feed-shortcut-remove.md +48 -0
- package/skills/lark-im/references/lark-im-flag-cancel.md +67 -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-message-enrichment.md +54 -0
- package/skills/lark-im/references/lark-im-messages-mget.md +99 -0
- package/skills/lark-im/references/lark-im-messages-reply.md +267 -0
- package/skills/lark-im/references/lark-im-messages-resources-download.md +94 -0
- package/skills/lark-im/references/lark-im-messages-search.md +234 -0
- package/skills/lark-im/references/lark-im-messages-send.md +267 -0
- package/skills/lark-im/references/lark-im-reactions.md +299 -0
- package/skills/lark-im/references/lark-im-threads-messages-list.md +115 -0
- package/skills/lark-mail/SKILL.md +287 -0
- package/skills/lark-mail/assets/templates/job-application--resume.html +33 -0
- package/skills/lark-mail/assets/templates/newsletter--weekly-brief.html +50 -0
- package/skills/lark-mail/assets/templates/research--market-report.html +256 -0
- package/skills/lark-mail/assets/templates/weekly--personal-report.html +43 -0
- package/skills/lark-mail/assets/templates/weekly--team-report.html +9 -0
- package/skills/lark-mail/references/lark-mail-calendar-invite.md +36 -0
- package/skills/lark-mail/references/lark-mail-decline-receipt.md +115 -0
- package/skills/lark-mail/references/lark-mail-draft-create.md +127 -0
- package/skills/lark-mail/references/lark-mail-draft-edit.md +404 -0
- package/skills/lark-mail/references/lark-mail-forward.md +239 -0
- package/skills/lark-mail/references/lark-mail-html.md +333 -0
- package/skills/lark-mail/references/lark-mail-lint-html.md +243 -0
- package/skills/lark-mail/references/lark-mail-message.md +233 -0
- package/skills/lark-mail/references/lark-mail-messages.md +108 -0
- package/skills/lark-mail/references/lark-mail-recall.md +66 -0
- package/skills/lark-mail/references/lark-mail-recipient-search.md +59 -0
- package/skills/lark-mail/references/lark-mail-reply-all.md +213 -0
- package/skills/lark-mail/references/lark-mail-reply.md +249 -0
- package/skills/lark-mail/references/lark-mail-rules.md +31 -0
- package/skills/lark-mail/references/lark-mail-send-as.md +44 -0
- package/skills/lark-mail/references/lark-mail-send-receipt.md +120 -0
- package/skills/lark-mail/references/lark-mail-send-status.md +46 -0
- package/skills/lark-mail/references/lark-mail-send.md +222 -0
- package/skills/lark-mail/references/lark-mail-share-to-chat.md +87 -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 +150 -0
- package/skills/lark-mail/references/lark-mail-template.md +54 -0
- package/skills/lark-mail/references/lark-mail-thread.md +111 -0
- package/skills/lark-mail/references/lark-mail-triage.md +131 -0
- package/skills/lark-mail/references/lark-mail-watch.md +94 -0
- package/skills/lark-markdown/SKILL.md +69 -0
- package/skills/lark-markdown/references/lark-markdown-create.md +94 -0
- package/skills/lark-markdown/references/lark-markdown-diff.md +156 -0
- package/skills/lark-markdown/references/lark-markdown-fetch.md +79 -0
- package/skills/lark-markdown/references/lark-markdown-overwrite.md +85 -0
- package/skills/lark-markdown/references/lark-markdown-patch.md +160 -0
- package/skills/lark-minutes/SKILL.md +192 -0
- package/skills/lark-minutes/references/lark-minutes-detail.md +62 -0
- package/skills/lark-minutes/references/lark-minutes-download.md +137 -0
- package/skills/lark-minutes/references/lark-minutes-search.md +204 -0
- package/skills/lark-minutes/references/lark-minutes-speaker-replace.md +107 -0
- package/skills/lark-minutes/references/lark-minutes-summary.md +122 -0
- package/skills/lark-minutes/references/lark-minutes-todo.md +138 -0
- package/skills/lark-minutes/references/lark-minutes-update.md +41 -0
- package/skills/lark-minutes/references/lark-minutes-upload.md +104 -0
- package/skills/lark-note/SKILL.md +94 -0
- package/skills/lark-note/references/lark-note-detail.md +26 -0
- package/skills/lark-note/references/lark-note-transcript.md +23 -0
- package/skills/lark-okr/SKILL.md +115 -0
- package/skills/lark-okr/references/lark-okr-batch-create.md +106 -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-indicator-update.md +80 -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-okr/references/lark-okr-reorder.md +81 -0
- package/skills/lark-okr/references/lark-okr-weight.md +96 -0
- package/skills/lark-openapi-explorer/SKILL.md +153 -0
- package/skills/lark-shared/SKILL.md +168 -0
- package/skills/lark-shared/references/lark-wiki-token-routing.md +42 -0
- package/skills/lark-sheets/SKILL.md +165 -0
- package/skills/lark-sheets/references/lark-sheets-batch-update.md +191 -0
- package/skills/lark-sheets/references/lark-sheets-chart.md +330 -0
- package/skills/lark-sheets/references/lark-sheets-conditional-format.md +179 -0
- package/skills/lark-sheets/references/lark-sheets-core-operations.md +103 -0
- package/skills/lark-sheets/references/lark-sheets-filter-view.md +137 -0
- package/skills/lark-sheets/references/lark-sheets-filter.md +130 -0
- package/skills/lark-sheets/references/lark-sheets-float-image.md +159 -0
- package/skills/lark-sheets/references/lark-sheets-formula-translation.md +267 -0
- package/skills/lark-sheets/references/lark-sheets-pivot-table.md +166 -0
- package/skills/lark-sheets/references/lark-sheets-range-operations.md +267 -0
- package/skills/lark-sheets/references/lark-sheets-read-data.md +235 -0
- package/skills/lark-sheets/references/lark-sheets-search-replace.md +111 -0
- package/skills/lark-sheets/references/lark-sheets-sheet-structure.md +212 -0
- package/skills/lark-sheets/references/lark-sheets-sparkline.md +149 -0
- package/skills/lark-sheets/references/lark-sheets-visual-standards.md +205 -0
- package/skills/lark-sheets/references/lark-sheets-workbook.md +395 -0
- package/skills/lark-sheets/references/lark-sheets-write-cells.md +565 -0
- package/skills/lark-sheets/scripts/sheets_df.py +32 -0
- package/skills/lark-skill-maker/SKILL.md +85 -0
- package/skills/lark-slides/SKILL.md +293 -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 +1469 -0
- package/skills/lark-slides/assets/templates/marketing--marketing_strategy.xml +1484 -0
- package/skills/lark-slides/assets/templates/marketing--product_whitepaper.xml +1455 -0
- package/skills/lark-slides/assets/templates/marketing--roadshow_business_plan.xml +1506 -0
- package/skills/lark-slides/assets/templates/misc--book_sharing.xml +1338 -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 +687 -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/iconpark-index.json +41901 -0
- package/skills/lark-slides/references/iconpark.md +46 -0
- package/skills/lark-slides/references/lark-slides-create.md +137 -0
- package/skills/lark-slides/references/lark-slides-edit-workflows.md +144 -0
- package/skills/lark-slides/references/lark-slides-media-upload.md +128 -0
- package/skills/lark-slides/references/lark-slides-replace-pages.md +95 -0
- package/skills/lark-slides/references/lark-slides-replace-slide.md +240 -0
- package/skills/lark-slides/references/lark-slides-screenshot.md +94 -0
- package/skills/lark-slides/references/lark-slides-whiteboard.md +330 -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 +187 -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 +3049 -0
- package/skills/lark-slides/references/template-catalog.md +463 -0
- package/skills/lark-slides/references/template-index.json +1853 -0
- package/skills/lark-slides/references/troubleshooting.md +63 -0
- package/skills/lark-slides/references/validation-checklist.md +110 -0
- package/skills/lark-slides/references/visual-planning.md +254 -0
- package/skills/lark-slides/references/xml-format-guide.md +369 -0
- package/skills/lark-slides/references/xml-schema-quick-ref.md +245 -0
- package/skills/lark-slides/scripts/iconpark_tool.py +362 -0
- package/skills/lark-slides/scripts/iconpark_tool_test.py +177 -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 +167 -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 +61 -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-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 +202 -0
- package/skills/lark-vc/references/lark-vc-detail.md +44 -0
- package/skills/lark-vc/references/lark-vc-recording.md +154 -0
- package/skills/lark-vc/references/lark-vc-search.md +163 -0
- package/skills/lark-vc/references/vc-domain-boundaries.md +188 -0
- package/skills/lark-vc-agent/SKILL.md +169 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-events.md +287 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-join.md +141 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-leave.md +105 -0
- package/skills/lark-vc-agent/references/lark-vc-agent-meeting-list-active.md +91 -0
- package/skills/lark-whiteboard/SKILL.md +47 -0
- package/skills/lark-whiteboard/elements/connectors.md +102 -0
- package/skills/lark-whiteboard/elements/content.md +40 -0
- package/skills/lark-whiteboard/elements/image.md +80 -0
- package/skills/lark-whiteboard/elements/layout.md +374 -0
- package/skills/lark-whiteboard/elements/schema.md +357 -0
- package/skills/lark-whiteboard/elements/style.md +318 -0
- package/skills/lark-whiteboard/elements/typography.md +73 -0
- package/skills/lark-whiteboard/references/lark-whiteboard-query.md +60 -0
- package/skills/lark-whiteboard/references/lark-whiteboard-update.md +122 -0
- package/skills/lark-whiteboard/references/lark-whiteboard-workflow.md +94 -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-edit.md +85 -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 +185 -0
- package/skills/lark-whiteboard/scenes/flywheel.md +195 -0
- package/skills/lark-whiteboard/scenes/funnel.md +101 -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 +126 -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 +216 -0
- package/skills/lark-wiki/SKILL.md +110 -0
- package/skills/lark-wiki/references/lark-wiki-delete-space.md +205 -0
- package/skills/lark-wiki/references/lark-wiki-member-add.md +67 -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 +57 -0
- package/skills/lark-wiki/references/lark-wiki-node-list.md +88 -0
- package/skills/lark-wiki/references/lark-wiki-space-create.md +46 -0
- package/skills/lark-wiki/references/lark-wiki-space-list.md +68 -0
- package/skills/lark-workflow-meeting-summary/SKILL.md +122 -0
- package/skills/lark-workflow-standup-report/SKILL.md +122 -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
|