From eb6864bfb7d5d1ed4fcb415ae2c9d9f1c30d5b74 Mon Sep 17 00:00:00 2001 From: Rods Date: Sat, 11 Apr 2026 13:55:02 -0300 Subject: [PATCH 01/21] chore: add CLAUDE.md with project instructions for AI agents Co-Authored-By: Claude Sonnet 4.6 --- CLAUDE.md | 116 ++++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 116 insertions(+) create mode 100644 CLAUDE.md diff --git a/CLAUDE.md b/CLAUDE.md new file mode 100644 index 0000000..6ae673d --- /dev/null +++ b/CLAUDE.md @@ -0,0 +1,116 @@ +# CraftD — Claude Instructions + +## O que é o projeto + +CraftD é uma biblioteca de **Server Driven UI** multiplatforma. O servidor decide quais componentes renderizar e como; o cliente (app) apenas executa. Suporta: + +- Android Compose (`android_kmp/craftd-compose`) +- Android View System / XML (`android_kmp/craftd-xml`) +- KMP Compose Multiplatform (`android_kmp/craftd-compose` via commonMain) +- iOS SwiftUI (`ios/craftd-swiftui`) +- Flutter (`flutter/craftd_widget`) + +--- + +## Estrutura de módulos + +``` +android_kmp/ + craftd-core/ # modelos e abstrações compartilhadas (KMP) + commonMain/ # código compartilhado entre plataformas + androidMain/ # implementações Android-específicas + craftd-compose/ # implementação Compose / KMP + craftd-xml/ # implementação View System (XML) + app-sample-android/ # app de exemplo Android + app-sample-cmp/ # app de exemplo KMP Compose + build-logic/ # configuração de build compartilhada + +ios/craftd-swiftui/ # biblioteca iOS (Swift Package + CocoaPods) +flutter/craftd_widget/ # biblioteca Flutter (pub.dev) +docs/ # documentação do site (MkDocs) +``` + +--- + +## Regras arquiteturais — nunca violar + +1. **Módulos de plataforma não dependem uns dos outros.** `craftd-compose`, `craftd-xml`, `ios/` e `flutter/` dependem apenas do `craftd-core`. Jamais entre si. + +2. **Todo novo componente implementa a abstração do platform.** No Android/KMP é `CraftDBuilder`. No iOS é o equivalente Swift. No Flutter é o equivalente Dart. Nunca criar componente avulso fora dessa abstração. + +3. **`onAction` / fallback sempre coberto.** Toda implementação de componente deve tratar o callback de ação (`ActionProperties`), mesmo que seja um no-op. + +4. **`commonMain` é sagrado.** Código em `commonMain` não pode ter dependências de plataforma. Se precisar de comportamento diferente por plataforma, usa `expect/actual`. + +5. **Novos componentes seguem o padrão existente.** Consultar `CraftDButtonBuilder` e `CraftDTextBuilder` como referência de implementação antes de criar algo novo. + +6. **Nomes de classes são consistentes entre todas as plataformas.** Ex: se um componente se chama `CraftDImage` no Android, deve se chamar `CraftDImage` em iOS e Flutter também. + +7. **Todo novo componente deve ter teste unitário** (quando possível) e **documentação em `docs/` na seção da respectiva plataforma**. + +--- + +## Abstrações principais por plataforma + +As três plataformas espelham os mesmos conceitos com nomes equivalentes. + +### Android / KMP (Kotlin) + +| Classe | Papel | +|---|---| +| `CraftDBuilder` | Interface base para criar componentes | +| `CraftDBuilderManager` | Registra e resolve builders pelo `key` | +| `CraftDynamic` | Composable principal que renderiza o SDUI | +| `SimpleProperties` | Modelo base de dados (`key` + `value` JSON) | +| `ActionProperties` | Dados de ação (deeplink + analytics) | +| `CraftDComponentKey` | Enum com as chaves de componentes built-in | +| `CraftDViewListener` | Callback de ações para o consumidor | + +### iOS / SwiftUI (Swift — `ios/craftd-swiftui/`) + +| Classe | Papel | +|---|---| +| `CraftDBuilder` | Protocol base para criar componentes | +| `CraftDBuilderManager` | Registra e resolve builders pelo `key` | +| `CraftDynamic` | View principal que renderiza o SDUI | +| `SimpleProperties` | Modelo base de dados | +| `ActionProperties` | Dados de ação (deeplink + analytics) | +| `CraftDViewListener` | Callback de ações para o consumidor | + +### Flutter (Dart — `flutter/craftd_widget/`) + +| Classe | Papel | +|---|---| +| `CraftDynamic` | Widget principal que renderiza o SDUI | +| `CraftDViewListener` | Callback de ações para o consumidor | +| `SimpleProperties` | Modelo base de dados | +| `ActionProperties` | Dados de ação (deeplink + analytics) | +| `CraftDAlign` | Alinhamento de componentes | + +> Ao adicionar um novo componente, ele deve ser implementado nas três plataformas seguindo a mesma abstração de cada uma. Consultar `CraftDButton` / `CraftDButtonBuilder` como referência em todas. + +--- + +## Convenções de código + +- **Kotlin:** segue as convenções oficiais do Kotlin. Prefere `data class` para modelos. +- **Nomenclatura de componentes:** prefixo `CraftD` em tudo que é parte da lib (ex: `CraftDButton`, `CraftDButtonBuilder`). +- **Testes:** JUnit4 + MockK. Nomenclatura em backtick: `` `given X when Y then Z` ``. Path espelha o source: `src/test/java/...` +- **Commits:** mensagens em inglês, semânticas (`feat:`, `fix:`, `test:`, `chore:`, `docs:`). + +--- + +## CI / automação + +- **`pr.yml`** — build e testes, dispara em todo PR +- **`generate-tests.yml`** — gera testes unitários automaticamente via Claude API para `.kt` modificados, abre PR separado. Só roda após `pr.yml` passar. Não roda em PRs de bots. + +--- + +## O que não fazer + +- Não criar componente fora da abstração `CraftDBuilder` (ou equivalente de plataforma) +- Não adicionar dependência de plataforma em `commonMain` +- Não criar dependência entre módulos de plataforma (`craftd-compose` → `craftd-xml`, por exemplo) +- Não commitar `local.properties` nem arquivos de credenciais +- Não usar `--no-verify` para burlar hooks de CI From 1dc290760e67a8950e795c94f4d2899ceed67853 Mon Sep 17 00:00:00 2001 From: Rods Date: Sat, 11 Apr 2026 14:19:55 -0300 Subject: [PATCH 02/21] chore: setup SDD structure with OpenSpec, skills and CLAUDE.md updates - Add OpenSpec commands and skills (propose, explore, apply, archive) - Add Android/Compose skills (testing, performance, accessibility, gradle, compose-ui) - Update CLAUDE.md with folder patterns, code principles and docs rule - Add GitHub Copilot equivalents (prompts and skills) Co-Authored-By: Claude Sonnet 4.6 --- .claude/commands/opsx/apply.md | 152 +++++++++ .claude/commands/opsx/archive.md | 157 ++++++++++ .claude/commands/opsx/explore.md | 173 +++++++++++ .claude/commands/opsx/propose.md | 106 +++++++ .claude/skills/android-accessibility/SKILL.md | 34 +++ .claude/skills/android-gradle-logic/SKILL.md | 29 ++ .claude/skills/android-testing/SKILL.md | 37 +++ .../skills/compose-performance-audit/SKILL.md | 33 ++ .claude/skills/compose-ui/SKILL.md | 46 +++ .claude/skills/openspec-apply-change/SKILL.md | 156 ++++++++++ .../skills/openspec-archive-change/SKILL.md | 114 +++++++ .claude/skills/openspec-explore/SKILL.md | 288 ++++++++++++++++++ .claude/skills/openspec-propose/SKILL.md | 110 +++++++ .github/prompts/opsx-apply.prompt.md | 149 +++++++++ .github/prompts/opsx-archive.prompt.md | 154 ++++++++++ .github/prompts/opsx-explore.prompt.md | 170 +++++++++++ .github/prompts/opsx-propose.prompt.md | 103 +++++++ .github/skills/openspec-apply-change/SKILL.md | 156 ++++++++++ .../skills/openspec-archive-change/SKILL.md | 114 +++++++ .github/skills/openspec-explore/SKILL.md | 288 ++++++++++++++++++ .github/skills/openspec-propose/SKILL.md | 110 +++++++ CLAUDE.md | 55 ++++ 22 files changed, 2734 insertions(+) create mode 100644 .claude/commands/opsx/apply.md create mode 100644 .claude/commands/opsx/archive.md create mode 100644 .claude/commands/opsx/explore.md create mode 100644 .claude/commands/opsx/propose.md create mode 100644 .claude/skills/android-accessibility/SKILL.md create mode 100644 .claude/skills/android-gradle-logic/SKILL.md create mode 100644 .claude/skills/android-testing/SKILL.md create mode 100644 .claude/skills/compose-performance-audit/SKILL.md create mode 100644 .claude/skills/compose-ui/SKILL.md create mode 100644 .claude/skills/openspec-apply-change/SKILL.md create mode 100644 .claude/skills/openspec-archive-change/SKILL.md create mode 100644 .claude/skills/openspec-explore/SKILL.md create mode 100644 .claude/skills/openspec-propose/SKILL.md create mode 100644 .github/prompts/opsx-apply.prompt.md create mode 100644 .github/prompts/opsx-archive.prompt.md create mode 100644 .github/prompts/opsx-explore.prompt.md create mode 100644 .github/prompts/opsx-propose.prompt.md create mode 100644 .github/skills/openspec-apply-change/SKILL.md create mode 100644 .github/skills/openspec-archive-change/SKILL.md create mode 100644 .github/skills/openspec-explore/SKILL.md create mode 100644 .github/skills/openspec-propose/SKILL.md diff --git a/.claude/commands/opsx/apply.md b/.claude/commands/opsx/apply.md new file mode 100644 index 0000000..bf23721 --- /dev/null +++ b/.claude/commands/opsx/apply.md @@ -0,0 +1,152 @@ +--- +name: "OPSX: Apply" +description: Implement tasks from an OpenSpec change (Experimental) +category: Workflow +tags: [workflow, artifacts, experimental] +--- + +Implement tasks from an OpenSpec change. + +**Input**: Optionally specify a change name (e.g., `/opsx:apply add-auth`). If omitted, check if it can be inferred from conversation context. If vague or ambiguous you MUST prompt for available changes. + +**Steps** + +1. **Select the change** + + If a name is provided, use it. Otherwise: + - Infer from conversation context if the user mentioned a change + - Auto-select if only one active change exists + - If ambiguous, run `openspec list --json` to get available changes and use the **AskUserQuestion tool** to let the user select + + Always announce: "Using change: " and how to override (e.g., `/opsx:apply `). + +2. **Check status to understand the schema** + ```bash + openspec status --change "" --json + ``` + Parse the JSON to understand: + - `schemaName`: The workflow being used (e.g., "spec-driven") + - Which artifact contains the tasks (typically "tasks" for spec-driven, check status for others) + +3. **Get apply instructions** + + ```bash + openspec instructions apply --change "" --json + ``` + + This returns: + - Context file paths (varies by schema) + - Progress (total, complete, remaining) + - Task list with status + - Dynamic instruction based on current state + + **Handle states:** + - If `state: "blocked"` (missing artifacts): show message, suggest using `/opsx:continue` + - If `state: "all_done"`: congratulate, suggest archive + - Otherwise: proceed to implementation + +4. **Read context files** + + Read the files listed in `contextFiles` from the apply instructions output. + The files depend on the schema being used: + - **spec-driven**: proposal, specs, design, tasks + - Other schemas: follow the contextFiles from CLI output + +5. **Show current progress** + + Display: + - Schema being used + - Progress: "N/M tasks complete" + - Remaining tasks overview + - Dynamic instruction from CLI + +6. **Implement tasks (loop until done or blocked)** + + For each pending task: + - Show which task is being worked on + - Make the code changes required + - Keep changes minimal and focused + - Mark task complete in the tasks file: `- [ ]` → `- [x]` + - Continue to next task + + **Pause if:** + - Task is unclear → ask for clarification + - Implementation reveals a design issue → suggest updating artifacts + - Error or blocker encountered → report and wait for guidance + - User interrupts + +7. **On completion or pause, show status** + + Display: + - Tasks completed this session + - Overall progress: "N/M tasks complete" + - If all done: suggest archive + - If paused: explain why and wait for guidance + +**Output During Implementation** + +``` +## Implementing: (schema: ) + +Working on task 3/7: +[...implementation happening...] +✓ Task complete + +Working on task 4/7: +[...implementation happening...] +✓ Task complete +``` + +**Output On Completion** + +``` +## Implementation Complete + +**Change:** +**Schema:** +**Progress:** 7/7 tasks complete ✓ + +### Completed This Session +- [x] Task 1 +- [x] Task 2 +... + +All tasks complete! You can archive this change with `/opsx:archive`. +``` + +**Output On Pause (Issue Encountered)** + +``` +## Implementation Paused + +**Change:** +**Schema:** +**Progress:** 4/7 tasks complete + +### Issue Encountered + + +**Options:** +1.