@saulwade/swl-ses 1.0.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/CLAUDE.md +238 -0
- package/README.md +560 -0
- package/_userland/agentes/.gitkeep +0 -0
- package/_userland/habilidades/.gitkeep +0 -0
- package/agentes/.evolved.json +9 -0
- package/agentes/accesibilidad-wcag-swl.md +692 -0
- package/agentes/arquitecto-swl.md +238 -0
- package/agentes/auto-evolucion-swl.md +854 -0
- package/agentes/backend-api-swl.md +470 -0
- package/agentes/backend-csharp-swl.md +418 -0
- package/agentes/backend-go-swl.md +388 -0
- package/agentes/backend-java-swl.md +279 -0
- package/agentes/backend-node-swl.md +477 -0
- package/agentes/backend-python-swl.md +608 -0
- package/agentes/backend-rust-swl.md +362 -0
- package/agentes/backend-workers-swl.md +480 -0
- package/agentes/cloud-infra-swl.md +485 -0
- package/agentes/consolidador-swl.md +539 -0
- package/agentes/datos-swl.md +584 -0
- package/agentes/depurador-swl.md +349 -0
- package/agentes/devops-ci-swl.md +374 -0
- package/agentes/disenador-ui-swl.md +558 -0
- package/agentes/documentador-swl.md +343 -0
- package/agentes/evals/arquitecto-swl.evals.json +56 -0
- package/agentes/evals/auto-evolucion-swl.evals.json +68 -0
- package/agentes/evals/implementador-swl.evals.json +56 -0
- package/agentes/evals/orquestador-swl.evals.json +60 -0
- package/agentes/evals/perfilador-usuario-swl.evals.json +60 -0
- package/agentes/evals/red-team-swl.evals.json +59 -0
- package/agentes/evals/revisor-codigo-swl.evals.json +59 -0
- package/agentes/frontend-angular-swl.md +627 -0
- package/agentes/frontend-css-swl.md +720 -0
- package/agentes/frontend-react-swl.md +696 -0
- package/agentes/frontend-swl.md +500 -0
- package/agentes/frontend-tailwind-swl.md +830 -0
- package/agentes/implementador-swl.md +328 -0
- package/agentes/investigador-swl.md +430 -0
- package/agentes/investigador-ux-swl.md +500 -0
- package/agentes/llm-apps-swl.md +276 -0
- package/agentes/migrador-swl.md +417 -0
- package/agentes/mobile-android-swl.md +509 -0
- package/agentes/mobile-cross-swl.md +539 -0
- package/agentes/mobile-ios-swl.md +500 -0
- package/agentes/mobile-testing-swl.md +300 -0
- package/agentes/notificador-swl.md +916 -0
- package/agentes/observabilidad-swl.md +436 -0
- package/agentes/orquestador-swl.md +884 -0
- package/agentes/pagos-swl.md +283 -0
- package/agentes/perfilador-usuario-swl.md +306 -0
- package/agentes/planificador-swl.md +402 -0
- package/agentes/producto-prd-swl.md +587 -0
- package/agentes/red-team-swl.md +216 -0
- package/agentes/release-manager-swl.md +568 -0
- package/agentes/rendimiento-swl.md +714 -0
- package/agentes/resolutor-build-swl.md +243 -0
- package/agentes/revisor-angular-swl.md +276 -0
- package/agentes/revisor-codigo-swl.md +348 -0
- package/agentes/revisor-csharp-swl.md +262 -0
- package/agentes/revisor-go-swl.md +257 -0
- package/agentes/revisor-java-swl.md +255 -0
- package/agentes/revisor-kotlin-swl.md +271 -0
- package/agentes/revisor-nextjs-swl.md +279 -0
- package/agentes/revisor-php-swl.md +269 -0
- package/agentes/revisor-react-swl.md +276 -0
- package/agentes/revisor-rust-swl.md +344 -0
- package/agentes/revisor-seguridad-swl.md +390 -0
- package/agentes/revisor-swift-swl.md +266 -0
- package/agentes/revisor-typescript-swl.md +344 -0
- package/agentes/sre-swl.md +265 -0
- package/agentes/tdd-qa-swl.md +354 -0
- package/agentes/ux-disenador-swl.md +501 -0
- package/bin/lib/bot-comandos.js +1030 -0
- package/bin/lib/bot-discovery.js +182 -0
- package/bin/lib/bot-git.js +142 -0
- package/bin/swl-ses.js +325 -0
- package/bin/swl-telegram-bot.js +442 -0
- package/bin/swl-telegram-bot.plist +21 -0
- package/bin/swl-telegram-bot.service +14 -0
- package/comandos/swl/.evolved.json +23 -0
- package/comandos/swl/actualizar.md +174 -0
- package/comandos/swl/adoptar-proyecto.md +207 -0
- package/comandos/swl/aprender.md +701 -0
- package/comandos/swl/auditar-deps.md +134 -0
- package/comandos/swl/autoresearch.md +170 -0
- package/comandos/swl/ayuda.md +224 -0
- package/comandos/swl/brainstorm.md +50 -0
- package/comandos/swl/checkpoint.md +330 -0
- package/comandos/swl/compactar.md +283 -0
- package/comandos/swl/configurar-ci.md +227 -0
- package/comandos/swl/contexto.md +112 -0
- package/comandos/swl/contribuir.md +233 -0
- package/comandos/swl/crear-skill.md +292 -0
- package/comandos/swl/cron.md +196 -0
- package/comandos/swl/dashboard.md +146 -0
- package/comandos/swl/discutir-fase.md +230 -0
- package/comandos/swl/ejecutar-fase.md +135 -0
- package/comandos/swl/evaluar-skill.md +487 -0
- package/comandos/swl/evolucion-estado.md +142 -0
- package/comandos/swl/evolucionar.md +259 -0
- package/comandos/swl/exportar-vault.md +189 -0
- package/comandos/swl/gateway.md +158 -0
- package/comandos/swl/inbox.md +116 -0
- package/comandos/swl/instalar.md +220 -0
- package/comandos/swl/instintos.md +86 -0
- package/comandos/swl/mapear-codebase.md +312 -0
- package/comandos/swl/mcp-status.md +175 -0
- package/comandos/swl/metricas.md +270 -0
- package/comandos/swl/modelo.md +102 -0
- package/comandos/swl/notificaciones.md +396 -0
- package/comandos/swl/nuevo-proyecto.md +154 -0
- package/comandos/swl/planear-fase.md +221 -0
- package/comandos/swl/plugins.md +256 -0
- package/comandos/swl/reflect-skills.md +125 -0
- package/comandos/swl/release.md +217 -0
- package/comandos/swl/revisar-impacto.md +206 -0
- package/comandos/swl/revisar.md +330 -0
- package/comandos/swl/salud.md +363 -0
- package/comandos/swl/sesiones.md +200 -0
- package/comandos/swl/skill-search.md +113 -0
- package/comandos/swl/verificar.md +585 -0
- package/comandos/swl/wiki.md +620 -0
- package/contextos/dev.md +32 -0
- package/contextos/research.md +30 -0
- package/contextos/review.md +31 -0
- package/habilidades/accesibilidad-a11y/SKILL.md +201 -0
- package/habilidades/accesibilidad-a11y/evals/evals.json +56 -0
- package/habilidades/accesibilidad-a11y/recursos/ejemplos-y-checklist-completo.md +441 -0
- package/habilidades/agent-browser/SKILL.md +218 -0
- package/habilidades/agentes-como-servicio/SKILL.md +218 -0
- package/habilidades/ai-runtime-security/SKILL.md +273 -0
- package/habilidades/angular-avanzado/SKILL.md +164 -0
- package/habilidades/angular-avanzado/recursos/ejemplos-avanzados.md +219 -0
- package/habilidades/angular-moderno/SKILL.md +186 -0
- package/habilidades/angular-moderno/evals/evals.json +45 -0
- package/habilidades/angular-moderno/recursos/ejemplos-avanzados.md +106 -0
- package/habilidades/api-rest-diseno/SKILL.md +191 -0
- package/habilidades/api-rest-diseno/recursos/openapi-template.yaml +506 -0
- package/habilidades/api-rest-diseno/recursos/referencia-api.md +140 -0
- package/habilidades/aprendizaje-continuo/SKILL.md +151 -0
- package/habilidades/aprendizaje-continuo/evals/evals.json +53 -0
- package/habilidades/aprendizaje-continuo/recursos/referencia-instintos.md +290 -0
- package/habilidades/async-python/SKILL.md +149 -0
- package/habilidades/async-python/evals/evals.json +47 -0
- package/habilidades/async-python/recursos/patrones-y-ejemplos-completos.md +292 -0
- package/habilidades/auth-patrones/.evolved.json +9 -0
- package/habilidades/auth-patrones/SKILL.md +413 -0
- package/habilidades/auth-patrones/recursos/implementaciones-completas.md +229 -0
- package/habilidades/auto-evolucion-protocolo/SKILL.md +276 -0
- package/habilidades/auto-evolucion-protocolo/evals/evals.json +55 -0
- package/habilidades/auto-evolucion-protocolo/recursos/referencia-completa.md +145 -0
- package/habilidades/autoresearch/SKILL.md +268 -0
- package/habilidades/autoresearch/evals/evals.json +41 -0
- package/habilidades/autoresearch/recursos/checklist-template.md +191 -0
- package/habilidades/autoresearch/scripts/calcular-score.js +88 -0
- package/habilidades/azure-cloud/SKILL.md +308 -0
- package/habilidades/azure-cloud/recursos/aks.md +327 -0
- package/habilidades/backend-mcp-servidor/SKILL.md +270 -0
- package/habilidades/backend-production-resilience/SKILL.md +288 -0
- package/habilidades/brainstorming/SKILL.md +295 -0
- package/habilidades/brainstorming/recursos/componentes-html.md +247 -0
- package/habilidades/build-errors-cpp/SKILL.md +270 -0
- package/habilidades/build-errors-csharp/SKILL.md +265 -0
- package/habilidades/build-errors-go/SKILL.md +306 -0
- package/habilidades/build-errors-java/SKILL.md +278 -0
- package/habilidades/build-errors-kotlin/SKILL.md +303 -0
- package/habilidades/build-errors-nextjs/SKILL.md +312 -0
- package/habilidades/build-errors-php/SKILL.md +270 -0
- package/habilidades/build-errors-python/SKILL.md +292 -0
- package/habilidades/build-errors-rust/SKILL.md +284 -0
- package/habilidades/build-errors-swift/SKILL.md +272 -0
- package/habilidades/build-errors-typescript/SKILL.md +369 -0
- package/habilidades/checklist-calidad/SKILL.md +271 -0
- package/habilidades/checklist-calidad/recursos/quality-report-template.md +148 -0
- package/habilidades/checklist-seguridad/SKILL.md +285 -0
- package/habilidades/checkpoints-verificacion/SKILL.md +298 -0
- package/habilidades/checkpoints-verificacion/recursos/checkpoint-templates.md +360 -0
- package/habilidades/ci-cd-pipelines/SKILL.md +157 -0
- package/habilidades/ci-cd-pipelines/recursos/github-actions-template.yaml +403 -0
- package/habilidades/ci-cd-pipelines/recursos/pipelines-completos.md +487 -0
- package/habilidades/cloud-aws/SKILL.md +142 -0
- package/habilidades/cloud-aws/recursos/servicios-aws-referencia.md +321 -0
- package/habilidades/compactacion-contexto/SKILL.md +247 -0
- package/habilidades/contenedores-docker/SKILL.md +137 -0
- package/habilidades/contenedores-docker/recursos/dockerfile-template.dockerfile +160 -0
- package/habilidades/contenedores-docker/recursos/ejemplos-y-configuraciones.md +327 -0
- package/habilidades/context-builder/SKILL.md +170 -0
- package/habilidades/control-profundidad/SKILL.md +128 -0
- package/habilidades/csharp-experto/SKILL.md +322 -0
- package/habilidades/csharp-patrones/SKILL.md +316 -0
- package/habilidades/csharp-testing/SKILL.md +286 -0
- package/habilidades/css-moderno/SKILL.md +166 -0
- package/habilidades/css-moderno/evals/evals.json +43 -0
- package/habilidades/css-moderno/recursos/ejemplos-y-patrones-completos.md +337 -0
- package/habilidades/datos-etl/SKILL.md +129 -0
- package/habilidades/datos-etl/recursos/implementaciones-completas.md +322 -0
- package/habilidades/dbml-experto/SKILL.md +339 -0
- package/habilidades/dbml-experto/evals/evals.json +56 -0
- package/habilidades/dependencias-auditoria/SKILL.md +320 -0
- package/habilidades/deprecacion-migracion/SKILL.md +169 -0
- package/habilidades/deprecacion-migracion/recursos/implementaciones-completas.md +220 -0
- package/habilidades/design-tokens/SKILL.md +158 -0
- package/habilidades/design-tokens/recursos/tokens-y-configuracion.md +363 -0
- package/habilidades/devsecops-pipeline-security/SKILL.md +309 -0
- package/habilidades/diagrama-arquitectura/SKILL.md +165 -0
- package/habilidades/diagrama-arquitectura/assets/template.html +276 -0
- package/habilidades/discutir-fase/SKILL.md +188 -0
- package/habilidades/diseno-herramientas-agente/SKILL.md +199 -0
- package/habilidades/diseno-responsivo/SKILL.md +186 -0
- package/habilidades/diseno-responsivo/recursos/ejemplos-layouts.md +156 -0
- package/habilidades/django-experto/SKILL.md +205 -0
- package/habilidades/django-experto/recursos/async-django.md +390 -0
- package/habilidades/django-experto/recursos/drf-patrones.md +438 -0
- package/habilidades/django-experto/recursos/orm-avanzado.md +382 -0
- package/habilidades/django-experto/recursos/referencia-completa.md +188 -0
- package/habilidades/django-experto/recursos/testing-django.md +415 -0
- package/habilidades/doc-sync/SKILL.md +280 -0
- package/habilidades/drift-detection/SKILL.md +179 -0
- package/habilidades/ejecutar-fase/SKILL.md +468 -0
- package/habilidades/estilo-sin-ai-isms/SKILL.md +775 -0
- package/habilidades/estilo-sin-ai-isms/evals/evals.json +63 -0
- package/habilidades/estilo-sin-ai-isms/scripts/detectar_aiisms.py +500 -0
- package/habilidades/estructura-proyecto-claude/SKILL.md +215 -0
- package/habilidades/estructura-proyecto-claude/recursos/claude-md-template.md +261 -0
- package/habilidades/estructura-proyecto-claude/recursos/configuracion-y-extensiones.md +176 -0
- package/habilidades/estructura-proyecto-claude/recursos/frontmatter-y-hooks-referencia.md +289 -0
- package/habilidades/estructura-proyecto-claude/recursos/mcp-json-template.json +77 -0
- package/habilidades/estructura-proyecto-claude/recursos/variantes-por-stack.md +177 -0
- package/habilidades/evaluacion-agentes/SKILL.md +314 -0
- package/habilidades/event-driven/SKILL.md +153 -0
- package/habilidades/event-driven/recursos/implementaciones-completas.md +423 -0
- package/habilidades/extraccion-documentos/SKILL.md +221 -0
- package/habilidades/extractor-de-aprendizajes/.evolved.json +9 -0
- package/habilidades/extractor-de-aprendizajes/SKILL.md +311 -0
- package/habilidades/extractor-de-aprendizajes/evals/evals.json +55 -0
- package/habilidades/fastapi-experto/SKILL.md +221 -0
- package/habilidades/fastapi-experto/recursos/async-patterns.md +438 -0
- package/habilidades/fastapi-experto/recursos/dependency-injection.md +330 -0
- package/habilidades/fastapi-experto/recursos/referencia-completa.md +79 -0
- package/habilidades/fastapi-experto/recursos/testing-httpx.md +420 -0
- package/habilidades/filament-admin/SKILL.md +290 -0
- package/habilidades/frontend-avanzado/SKILL.md +257 -0
- package/habilidades/frontend-avanzado/recursos/apis-nativas-ejemplos.md +341 -0
- package/habilidades/gcp-cloud/SKILL.md +260 -0
- package/habilidades/gcp-cloud/recursos/gke.md +234 -0
- package/habilidades/gcp-cloud/recursos/terraform-gcp.md +307 -0
- package/habilidades/generacion-mermaid/SKILL.md +229 -0
- package/habilidades/git-worktrees-paralelo/SKILL.md +270 -0
- package/habilidades/go-experto/SKILL.md +305 -0
- package/habilidades/go-patrones/SKILL.md +299 -0
- package/habilidades/go-testing/SKILL.md +291 -0
- package/habilidades/graphql-experto/SKILL.md +323 -0
- package/habilidades/guardrail-semantico/SKILL.md +282 -0
- package/habilidades/harness-claude-code/SKILL.md +299 -0
- package/habilidades/iam-secretos/SKILL.md +265 -0
- package/habilidades/iam-secretos/recursos/implementaciones-completas.md +356 -0
- package/habilidades/infra-github-actions/SKILL.md +166 -0
- package/habilidades/instalar-sistema/.evolved.json +9 -0
- package/habilidades/instalar-sistema/SKILL.md +221 -0
- package/habilidades/java-experto/SKILL.md +290 -0
- package/habilidades/java-patrones/SKILL.md +275 -0
- package/habilidades/java-testing/SKILL.md +288 -0
- package/habilidades/kotlin-compose/SKILL.md +278 -0
- package/habilidades/kotlin-compose/recursos/animaciones-performance.md +93 -0
- package/habilidades/kotlin-experto/SKILL.md +318 -0
- package/habilidades/kotlin-testing/SKILL.md +267 -0
- package/habilidades/kotlin-testing/recursos/testing-avanzado.md +74 -0
- package/habilidades/kubernetes-orquestacion/SKILL.md +152 -0
- package/habilidades/kubernetes-orquestacion/recursos/manifiestos-completos.md +452 -0
- package/habilidades/langchain-langraph/SKILL.md +386 -0
- package/habilidades/langchain-langraph/recursos/evaluacion-rag.md +321 -0
- package/habilidades/langchain-langraph/recursos/rag-maturity-model.md +225 -0
- package/habilidades/langchain-langraph/recursos/vectorstores.md +306 -0
- package/habilidades/legacy-code-rescue/SKILL.md +267 -0
- package/habilidades/likec4-experto/SKILL.md +412 -0
- package/habilidades/likec4-experto/evals/evals.json +69 -0
- package/habilidades/manejo-errores/.evolved.json +9 -0
- package/habilidades/manejo-errores/SKILL.md +407 -0
- package/habilidades/manejo-errores/recursos/implementaciones-completas.md +248 -0
- package/habilidades/mapear-codebase/SKILL.md +275 -0
- package/habilidades/memoria-busqueda/SKILL.md +194 -0
- package/habilidades/memoria-busqueda/evals/evals.json +44 -0
- package/habilidades/meta-skills-estandar/SKILL.md +298 -0
- package/habilidades/meta-skills-estandar/recursos/anti-patrones-y-leyes.md +205 -0
- package/habilidades/meta-skills-estandar/recursos/frameworks-seguridad.md +107 -0
- package/habilidades/meta-skills-estandar/recursos/idiomas-framework.md +60 -0
- package/habilidades/meta-skills-estandar/recursos/skills-as-agents.md +163 -0
- package/habilidades/microservicios/SKILL.md +155 -0
- package/habilidades/microservicios/recursos/patrones-y-ejemplos-completos.md +325 -0
- package/habilidades/mobile-flutter/SKILL.md +199 -0
- package/habilidades/mobile-flutter/recursos/ejemplos-completos.md +319 -0
- package/habilidades/mobile-react-native/SKILL.md +176 -0
- package/habilidades/mobile-react-native/recursos/ejemplos-completos.md +216 -0
- package/habilidades/mongodb-experto/SKILL.md +302 -0
- package/habilidades/monitoring-alertas/SKILL.md +201 -0
- package/habilidades/monitoring-alertas/recursos/instrumentacion-y-alertas.md +301 -0
- package/habilidades/nestjs-experto/SKILL.md +307 -0
- package/habilidades/nestjs-experto/recursos/guards-interceptors.md +339 -0
- package/habilidades/nestjs-experto/recursos/modulos-di.md +287 -0
- package/habilidades/nestjs-experto/recursos/testing-nestjs.md +354 -0
- package/habilidades/nextjs-experto/SKILL.md +335 -0
- package/habilidades/nextjs-patrones/SKILL.md +303 -0
- package/habilidades/nextjs-testing/SKILL.md +331 -0
- package/habilidades/node-experto/.evolved.json +9 -0
- package/habilidades/node-experto/SKILL.md +266 -0
- package/habilidades/node-experto/recursos/patrones-completos.md +283 -0
- package/habilidades/notificaciones-multicanal/SKILL.md +159 -0
- package/habilidades/notificaciones-multicanal/recursos/config-template.json +115 -0
- package/habilidades/notificaciones-multicanal/recursos/configuracion-y-templates.md +303 -0
- package/habilidades/nuevo-proyecto/SKILL.md +204 -0
- package/habilidades/orquestacion-async/SKILL.md +303 -0
- package/habilidades/paid-media-tracking/SKILL.md +269 -0
- package/habilidades/paid-media-tracking/recursos/auditoria-tracking.md +220 -0
- package/habilidades/paid-media-tracking/recursos/google-ads-api.md +215 -0
- package/habilidades/patrones-python/SKILL.md +228 -0
- package/habilidades/patrones-python/evals/evals.json +56 -0
- package/habilidades/patrones-python/recursos/patrones-avanzados.md +469 -0
- package/habilidades/patrones-python/recursos/referencia-completa.md +202 -0
- package/habilidades/perfil-usuario/SKILL.md +200 -0
- package/habilidades/perfil-usuario/evals/evals.json +55 -0
- package/habilidades/performance-baseline/SKILL.md +297 -0
- package/habilidades/php-experto/SKILL.md +291 -0
- package/habilidades/php-patrones/SKILL.md +306 -0
- package/habilidades/php-testing/SKILL.md +280 -0
- package/habilidades/planear-fase/SKILL.md +269 -0
- package/habilidades/postgresql-experto/SKILL.md +151 -0
- package/habilidades/postgresql-experto/evals/evals.json +53 -0
- package/habilidades/postgresql-experto/recursos/referencia-completa.md +215 -0
- package/habilidades/prevencion-racionalizacion/SKILL.md +175 -0
- package/habilidades/prevencion-sobreingenieria/SKILL.md +323 -0
- package/habilidades/privacy-memoria/SKILL.md +141 -0
- package/habilidades/privacy-memoria/evals/evals.json +43 -0
- package/habilidades/prompt-engineering/SKILL.md +518 -0
- package/habilidades/prompt-engineering/recursos/patrones-avanzados.md +467 -0
- package/habilidades/rag-arquitectura/SKILL.md +338 -0
- package/habilidades/rails-experto/SKILL.md +237 -0
- package/habilidades/rails-experto/recursos/active-record.md +260 -0
- package/habilidades/rails-experto/recursos/hotwire-turbo.md +293 -0
- package/habilidades/rails-experto/recursos/testing-rspec.md +362 -0
- package/habilidades/react-experto/SKILL.md +209 -0
- package/habilidades/react-experto/evals/evals.json +55 -0
- package/habilidades/react-experto/recursos/patrones-y-ejemplos-completos.md +240 -0
- package/habilidades/react-optimizacion/SKILL.md +174 -0
- package/habilidades/react-optimizacion/recursos/patrones-avanzados.md +138 -0
- package/habilidades/redis-experto/SKILL.md +305 -0
- package/habilidades/release-semver/.evolved.json +9 -0
- package/habilidades/release-semver/SKILL.md +248 -0
- package/habilidades/release-semver/scripts/generar-changelog.sh +238 -0
- package/habilidades/rust-experto/SKILL.md +400 -0
- package/habilidades/rust-patrones/SKILL.md +296 -0
- package/habilidades/rust-testing/SKILL.md +311 -0
- package/habilidades/seguridad-skills-ia/SKILL.md +262 -0
- package/habilidades/sql-optimizacion/SKILL.md +200 -0
- package/habilidades/sql-optimizacion/evals/evals.json +54 -0
- package/habilidades/sql-optimizacion/recursos/patrones-sql-avanzados.md +131 -0
- package/habilidades/sre-patrones/SKILL.md +333 -0
- package/habilidades/sre-patrones/recursos/chaos-engineering.md +241 -0
- package/habilidades/sre-patrones/recursos/oncall-design.md +236 -0
- package/habilidades/stripe-pagos/SKILL.md +550 -0
- package/habilidades/stripe-pagos/recursos/errores-reintentos.md +390 -0
- package/habilidades/stripe-pagos/recursos/stripe-connect.md +290 -0
- package/habilidades/structured-outputs/SKILL.md +343 -0
- package/habilidades/swift-experto/SKILL.md +320 -0
- package/habilidades/swift-experto/recursos/keychain-y-wrappers.md +110 -0
- package/habilidades/swift-patrones/SKILL.md +313 -0
- package/habilidades/swift-patrones/recursos/tca-ejemplo-completo.md +113 -0
- package/habilidades/swift-testing/SKILL.md +254 -0
- package/habilidades/swift-testing/recursos/xcuitest-planes.md +143 -0
- package/habilidades/swl-dashboard/SKILL.md +370 -0
- package/habilidades/swl-markitdown/SKILL.md +285 -0
- package/habilidades/swl-markitdown/evals/evals.json +52 -0
- package/habilidades/swl-revisar-impacto/SKILL.md +233 -0
- package/habilidades/tailwind-experto/SKILL.md +240 -0
- package/habilidades/tailwind-experto/recursos/referencia-completa.md +184 -0
- package/habilidades/tdd-workflow/SKILL.md +293 -0
- package/habilidades/terraform-experto/SKILL.md +321 -0
- package/habilidades/testing-python/SKILL.md +340 -0
- package/habilidades/testing-python/recursos/ejemplos-completos.md +167 -0
- package/habilidades/threat-model-lite/SKILL.md +246 -0
- package/habilidades/tracing-processor/SKILL.md +212 -0
- package/habilidades/tracking-measurement/SKILL.md +239 -0
- package/habilidades/tracking-measurement/recursos/consent-mode.md +231 -0
- package/habilidades/tracking-measurement/recursos/gtm-datalayer.md +216 -0
- package/habilidades/tracking-measurement/recursos/meta-capi.md +262 -0
- package/habilidades/typescript-avanzado/SKILL.md +144 -0
- package/habilidades/typescript-avanzado/evals/evals.json +55 -0
- package/habilidades/typescript-avanzado/recursos/patrones-y-ejemplos-completos.md +298 -0
- package/habilidades/typescript-diagnosticos/SKILL.md +513 -0
- package/habilidades/ux-diseno/SKILL.md +116 -0
- package/habilidades/ux-diseno/evals/evals.json +43 -0
- package/habilidades/ux-diseno/recursos/patrones-ux-referencia.md +214 -0
- package/habilidades/validacion-ci-sistema/SKILL.md +136 -0
- package/habilidades/validacion-ci-sistema/recursos/validadores-completos.md +369 -0
- package/habilidades/validacion-ci-sistema/scripts/validar-sistema.sh +286 -0
- package/habilidades/verificacion-evidencia/SKILL.md +160 -0
- package/habilidades/verificar-trabajo/SKILL.md +303 -0
- package/habilidades/verificar-trabajo/recursos/plantilla-verificacion.md +60 -0
- package/habilidades/wiki-conocimiento/SKILL.md +276 -0
- package/habilidades/wireframes-flujos/SKILL.md +212 -0
- package/habilidades/wireframes-flujos/recursos/referencia-completa.md +192 -0
- package/habilidades/workflow-claude-code/SKILL.md +260 -0
- package/habilidades/workflow-claude-code/recursos/referencia-completa.md +109 -0
- package/hooks/_run-hook.sh +57 -0
- package/hooks/actualizar-perfil-usuario.js +364 -0
- package/hooks/agente-lifecycle.js +71 -0
- package/hooks/aiisms-detector.js +173 -0
- package/hooks/audit-trail.js +204 -0
- package/hooks/auto-background.js +97 -0
- package/hooks/auto-consolidacion.js +178 -0
- package/hooks/auto-evolucion.js +666 -0
- package/hooks/auto-restaurar-settings.js +360 -0
- package/hooks/calidad-pre-commit.js +929 -0
- package/hooks/calidad-typescript.js +511 -0
- package/hooks/captura-feedback-usuario.js +148 -0
- package/hooks/check-update.js +211 -0
- package/hooks/clasificador-mensajes.js +271 -0
- package/hooks/degradacion-instintos.js +272 -0
- package/hooks/escaneo-secretos.js +389 -0
- package/hooks/extraccion-aprendizajes.js +763 -0
- package/hooks/grafo-contexto.js +129 -0
- package/hooks/graph-update.js +67 -0
- package/hooks/guardrail-modelo.js +247 -0
- package/hooks/inbox-aviso.js +75 -0
- package/hooks/inyeccion-contexto.js +246 -0
- package/hooks/lib/abort-registry.js +214 -0
- package/hooks/lib/agent-backend.js +210 -0
- package/hooks/lib/agent-comms.js +263 -0
- package/hooks/lib/agent-issue-codes.js +284 -0
- package/hooks/lib/agent-matcher.js +189 -0
- package/hooks/lib/async-hook-registry.js +252 -0
- package/hooks/lib/atomic-write.js +130 -0
- package/hooks/lib/auto-consolidator.js +335 -0
- package/hooks/lib/canary-skills.js +187 -0
- package/hooks/lib/consolidation-lock.js +291 -0
- package/hooks/lib/context-builder.js +430 -0
- package/hooks/lib/context-compressor.js +657 -0
- package/hooks/lib/convergence-detector.js +105 -0
- package/hooks/lib/delegation-tracker.js +198 -0
- package/hooks/lib/detectar-package-manager.js +423 -0
- package/hooks/lib/edit-accumulator.js +171 -0
- package/hooks/lib/error-classifier.js +308 -0
- package/hooks/lib/event-bus.js +112 -0
- package/hooks/lib/evolution-tracker.js +442 -0
- package/hooks/lib/execution-state.js +316 -0
- package/hooks/lib/fingerprint-id.js +135 -0
- package/hooks/lib/gateway-notify.js +116 -0
- package/hooks/lib/graph-security.js +75 -0
- package/hooks/lib/guardrail-metrics.js +202 -0
- package/hooks/lib/hook-circuit-breaker.js +206 -0
- package/hooks/lib/loop-detector.js +267 -0
- package/hooks/lib/mcp-health.js +184 -0
- package/hooks/lib/mcp-pool.js +436 -0
- package/hooks/lib/memory-search.js +506 -0
- package/hooks/lib/merkle-audit.js +96 -0
- package/hooks/lib/model-router.js +222 -0
- package/hooks/lib/normalize-error.js +324 -0
- package/hooks/lib/normalize-input.js +65 -0
- package/hooks/lib/nudge-tracker.js +306 -0
- package/hooks/lib/otlp-exporter.js +365 -0
- package/hooks/lib/performance-marks.js +239 -0
- package/hooks/lib/privacy-filter.js +128 -0
- package/hooks/lib/prompt-injection-scanner.js +209 -0
- package/hooks/lib/provenance-tracker.js +183 -0
- package/hooks/lib/rate-limit-tracker.js +253 -0
- package/hooks/lib/reflect-classifier.js +164 -0
- package/hooks/lib/resource-quota.js +122 -0
- package/hooks/lib/retry-jitter.js +165 -0
- package/hooks/lib/risk-engine.js +368 -0
- package/hooks/lib/run-log.js +408 -0
- package/hooks/lib/session-fts.js +379 -0
- package/hooks/lib/session-store.js +293 -0
- package/hooks/lib/singleton-guard.js +159 -0
- package/hooks/lib/skill-auditor.js +588 -0
- package/hooks/lib/sync-status.js +228 -0
- package/hooks/lib/taint-tracker.js +107 -0
- package/hooks/lib/task-service.js +295 -0
- package/hooks/lib/tech-skills-map.js +146 -0
- package/hooks/lib/telegram-cliente.js +159 -0
- package/hooks/lib/telegram-config.js +170 -0
- package/hooks/lib/token-budget.js +156 -0
- package/hooks/lib/token-estimator.js +420 -0
- package/hooks/lib/toon-compressor.js +245 -0
- package/hooks/lib/usage-model.js +183 -0
- package/hooks/lib/variable-resolver.js +230 -0
- package/hooks/linea-estado.js +324 -0
- package/hooks/metricas-evolucion.js +209 -0
- package/hooks/monitor-contexto.js +325 -0
- package/hooks/notificacion-sesion-stop.js +198 -0
- package/hooks/notificacion-telegram-notification.js +4 -0
- package/hooks/notificacion-telegram-subagent.js +4 -0
- package/hooks/notificacion-telegram.js +267 -0
- package/hooks/preservar-estado-pre-compact.js +150 -0
- package/hooks/proteccion-rutas.js +366 -0
- package/hooks/registro-turnos.js +209 -0
- package/hooks/resumen-sesion.js +249 -0
- package/hooks/risk-scoring.js +323 -0
- package/hooks/rotar-audit-auto.js +122 -0
- package/hooks/sugerir-regenerar-inventario.js +170 -0
- package/hooks/telemetria-agentes.js +167 -0
- package/hooks/tracking-costos.js +688 -0
- package/instintos/global.yaml +8 -0
- package/instintos/perfil-usuario.yaml +53 -0
- package/instintos/prompt-appendices.yaml +57 -0
- package/instintos/proyecto.yaml +372 -0
- package/manifiestos/gateway-config.json +77 -0
- package/manifiestos/handoff-context.json +223 -0
- package/manifiestos/hook-profiles.json +44 -0
- package/manifiestos/hooks-config.json +360 -0
- package/manifiestos/modulos.json +1173 -0
- package/manifiestos/perfiles.json +404 -0
- package/package.json +86 -0
- package/plantillas/ESTADO.md +109 -0
- package/plantillas/HOJA-RUTA.md +143 -0
- package/plantillas/PROYECTO.md +122 -0
- package/plantillas/REQUISITOS.md +132 -0
- package/plantillas/auditor-veto-template.md +105 -0
- package/plantillas/github-workflows/README.md +47 -0
- package/plantillas/github-workflows/release-please.yml +44 -0
- package/plantillas/github-workflows/swl-ci.yml +107 -0
- package/plantillas/github-workflows/swl-security.yml +51 -0
- package/plantillas/mcp-mineru.json +13 -0
- package/plantillas/research/ARQUITECTURA.md +220 -0
- package/plantillas/research/FUNCIONALIDADES.md +175 -0
- package/plantillas/research/RESUMEN.md +165 -0
- package/plantillas/research/STACK.md +233 -0
- package/plantillas/research/TRAMPAS.md +299 -0
- package/plantillas/skill-evals-template.json +44 -0
- package/plugin.json +343 -0
- package/reglas/accesibilidad.md +269 -0
- package/reglas/api-diseno.md +400 -0
- package/reglas/arquitectura.md +352 -0
- package/reglas/brevedad-output.md +124 -0
- package/reglas/cloud-infra.md +247 -0
- package/reglas/docs.md +245 -0
- package/reglas/estilo-codigo.md +201 -0
- package/reglas/git-workflow.md +245 -0
- package/reglas/gobernanza.md +271 -0
- package/reglas/harness-claude-code.md +213 -0
- package/reglas/hooks.md +186 -0
- package/reglas/lenguajes/csharp/estilo-codigo.md +231 -0
- package/reglas/lenguajes/csharp/hooks.md +281 -0
- package/reglas/lenguajes/csharp/patrones.md +226 -0
- package/reglas/lenguajes/csharp/seguridad.md +258 -0
- package/reglas/lenguajes/csharp/testing.md +176 -0
- package/reglas/lenguajes/go/estilo-codigo.md +195 -0
- package/reglas/lenguajes/go/hooks.md +249 -0
- package/reglas/lenguajes/go/patrones.md +249 -0
- package/reglas/lenguajes/go/seguridad.md +225 -0
- package/reglas/lenguajes/go/testing.md +272 -0
- package/reglas/lenguajes/java/estilo-codigo.md +217 -0
- package/reglas/lenguajes/java/hooks.md +251 -0
- package/reglas/lenguajes/java/patrones.md +226 -0
- package/reglas/lenguajes/java/seguridad.md +233 -0
- package/reglas/lenguajes/java/testing.md +238 -0
- package/reglas/lenguajes/kotlin/estilo-codigo.md +208 -0
- package/reglas/lenguajes/kotlin/hooks.md +245 -0
- package/reglas/lenguajes/kotlin/patrones.md +201 -0
- package/reglas/lenguajes/kotlin/seguridad.md +202 -0
- package/reglas/lenguajes/kotlin/testing.md +236 -0
- package/reglas/lenguajes/nextjs/estilo-codigo.md +175 -0
- package/reglas/lenguajes/nextjs/hooks.md +186 -0
- package/reglas/lenguajes/nextjs/patrones.md +225 -0
- package/reglas/lenguajes/nextjs/seguridad.md +216 -0
- package/reglas/lenguajes/nextjs/testing.md +193 -0
- package/reglas/lenguajes/php/estilo-codigo.md +228 -0
- package/reglas/lenguajes/php/hooks.md +165 -0
- package/reglas/lenguajes/php/patrones.md +233 -0
- package/reglas/lenguajes/php/seguridad.md +186 -0
- package/reglas/lenguajes/php/testing.md +205 -0
- package/reglas/lenguajes/rust/estilo-codigo.md +207 -0
- package/reglas/lenguajes/rust/hooks.md +240 -0
- package/reglas/lenguajes/rust/patrones.md +250 -0
- package/reglas/lenguajes/rust/seguridad.md +221 -0
- package/reglas/lenguajes/rust/testing.md +194 -0
- package/reglas/lenguajes/swift/estilo-codigo.md +238 -0
- package/reglas/lenguajes/swift/hooks.md +257 -0
- package/reglas/lenguajes/swift/patrones.md +235 -0
- package/reglas/lenguajes/swift/seguridad.md +248 -0
- package/reglas/lenguajes/swift/testing.md +242 -0
- package/reglas/markitdown.md +60 -0
- package/reglas/memoria-consolidada.md +209 -0
- package/reglas/patrones.md +225 -0
- package/reglas/performance.md +195 -0
- package/reglas/pruebas.md +159 -0
- package/reglas/seguridad-agentes.md +351 -0
- package/reglas/seguridad.md +151 -0
- package/reglas/skills-estandar.md +373 -0
- package/reglas/testing.md +193 -0
- package/schemas/agent-contract.json +176 -0
- package/schemas/agent-frontmatter.schema.json +149 -0
- package/schemas/agent-message.schema.json +53 -0
- package/schemas/agent-output-implementacion.schema.json +85 -0
- package/schemas/agent-output-planificacion.schema.json +113 -0
- package/schemas/agent-output-review.schema.json +78 -0
- package/schemas/diary-entry.schema.json +80 -0
- package/schemas/hook-profiles.schema.json +39 -0
- package/schemas/hooks-config.schema.json +74 -0
- package/schemas/instinct.schema.json +115 -0
- package/schemas/modulos.schema.json +29 -0
- package/schemas/perfiles.schema.json +28 -0
- package/schemas/plugin.schema.json +64 -0
- package/schemas/skill-evals.schema.json +95 -0
- package/schemas/skill-frontmatter.schema.json +170 -0
- package/scripts/actualizar.js +145 -0
- package/scripts/audit-skills.sh +78 -0
- package/scripts/auditar-agentes-gaps.js +149 -0
- package/scripts/auditar-cobertura-frameworks.js +241 -0
- package/scripts/auditar-skills-gaps.js +206 -0
- package/scripts/bootstrap-instintos.js +259 -0
- package/scripts/check-update.js +109 -0
- package/scripts/comandos/agents.js +105 -0
- package/scripts/comandos/info.js +108 -0
- package/scripts/comandos/install-asistido.js +186 -0
- package/scripts/comandos/skills.js +211 -0
- package/scripts/configurar-branch-protection.js +418 -0
- package/scripts/daemon-swl.py +388 -0
- package/scripts/desinstalar.js +130 -0
- package/scripts/doctor.js +559 -0
- package/scripts/field-report.js +199 -0
- package/scripts/generar-inventario.js +317 -0
- package/scripts/inbox-tmux-inject.js +161 -0
- package/scripts/inferir-herramientas-permitidas.js +586 -0
- package/scripts/inicializar.js +133 -0
- package/scripts/instalador.js +1031 -0
- package/scripts/instalar-git-hook.js +122 -0
- package/scripts/lib/agp-frontmatter.js +222 -0
- package/scripts/lib/append-con-marcadores.js +199 -0
- package/scripts/lib/artefactos-python.js +43 -0
- package/scripts/lib/audit-query.js +221 -0
- package/scripts/lib/autostart-linux.js +347 -0
- package/scripts/lib/autostart-macos.js +360 -0
- package/scripts/lib/autostart-windows.js +307 -0
- package/scripts/lib/budget-enforcer.js +252 -0
- package/scripts/lib/claude-sessions.js +285 -0
- package/scripts/lib/configurar-ci.js +380 -0
- package/scripts/lib/console-span-exporter.js +92 -0
- package/scripts/lib/contadores-inventario.js +217 -0
- package/scripts/lib/dashboard-widgets.js +290 -0
- package/scripts/lib/detectar-runtime.js +279 -0
- package/scripts/lib/detectar-stack.js +187 -0
- package/scripts/lib/diary-entry.js +234 -0
- package/scripts/lib/drift-detector.js +545 -0
- package/scripts/lib/estado.js +124 -0
- package/scripts/lib/gestor-componentes.js +243 -0
- package/scripts/lib/gitignore-manifest.js +305 -0
- package/scripts/lib/graph-analyze.py +556 -0
- package/scripts/lib/graph-builder.py +485 -0
- package/scripts/lib/graph-cluster.py +259 -0
- package/scripts/lib/health-row.js +168 -0
- package/scripts/lib/hooks-settings.js +789 -0
- package/scripts/lib/manifiestos.js +138 -0
- package/scripts/lib/mc-client.js +137 -0
- package/scripts/lib/notificaciones-telegram.js +1107 -0
- package/scripts/lib/npm-version.js +261 -0
- package/scripts/lib/paquetes-conocidos.js +50 -0
- package/scripts/lib/preservar-usuario.js +586 -0
- package/scripts/lib/prompt-builder.js +264 -0
- package/scripts/lib/resolver-externo.js +332 -0
- package/scripts/lib/schedule-parser.js +305 -0
- package/scripts/lib/scoring-instintos.js +240 -0
- package/scripts/lib/seguridad.js +160 -0
- package/scripts/lib/selector-interactivo.js +152 -0
- package/scripts/lib/semantic-search.js +242 -0
- package/scripts/lib/skill-discovery.js +234 -0
- package/scripts/lib/skill-metrics.js +246 -0
- package/scripts/lib/skill-normalizer.js +112 -0
- package/scripts/lib/skills-hub.js +340 -0
- package/scripts/lib/span-schema.js +134 -0
- package/scripts/lib/tool-cost-analyzer.js +255 -0
- package/scripts/lib/tracing-processor-interface.js +286 -0
- package/scripts/lib/transformadores/base.js +80 -0
- package/scripts/lib/transformadores/claude.js +124 -0
- package/scripts/lib/transformadores/codex.js +115 -0
- package/scripts/lib/transformadores/copilot.js +106 -0
- package/scripts/lib/transformadores/gemini.js +74 -0
- package/scripts/lib/transformadores/index.js +35 -0
- package/scripts/lib/transformadores/opencode.js +75 -0
- package/scripts/lib/ui.js +259 -0
- package/scripts/limpiar-artefactos-python.js +131 -0
- package/scripts/mcp-orchestrator.py +386 -0
- package/scripts/mcp-pool-manager.py +352 -0
- package/scripts/mcp-telemetry.py +378 -0
- package/scripts/poblar-evolvable.js +226 -0
- package/scripts/publicar.js +287 -0
- package/scripts/reflect-skills.js +403 -0
- package/scripts/rotar-audit-logs.js +185 -0
- package/scripts/run-skill-evals.js +242 -0
- package/scripts/smoke-test.js +374 -0
- package/scripts/token-analysis.py +471 -0
- package/scripts/validar-manifest.js +195 -0
- package/scripts/validar-memoria.js +321 -0
- package/scripts/validar-tests-aislamiento.js +184 -0
- package/scripts/validar-tokens-test.js +208 -0
- package/scripts/validar.js +147 -0
- package/scripts/validate-markdown.py +339 -0
- package/scripts/validate-skills.py +385 -0
- package/scripts/vendor/claude-usage/README.md +116 -0
- package/scripts/vendor/claude-usage/cli.py +334 -0
- package/scripts/vendor/claude-usage/dashboard.py +795 -0
- package/scripts/vendor/claude-usage/scanner.py +467 -0
- package/scripts/vendor/markitdown/cli.py +194 -0
- package/scripts/verificar-evolucion.js +289 -0
- package/scripts/verificar-release.js +494 -0
|
@@ -0,0 +1,539 @@
|
|
|
1
|
+
---
|
|
2
|
+
name: mobile-cross-swl
|
|
3
|
+
description: >
|
|
4
|
+
Especialista en desarrollo mobile multiplataforma. Invocar cuando se necesita
|
|
5
|
+
implementar con React Native (Expo o bare), incluida navegación con React
|
|
6
|
+
Navigation, animaciones con Reanimated, módulos nativos, o testing E2E con
|
|
7
|
+
Detox. También invocar para Flutter con gestión de estado Riverpod o BLoC,
|
|
8
|
+
platform channels, o testing de integración. Invocar ANTES de elegir el
|
|
9
|
+
stack móvil para obtener el framework de decisión React Native vs Flutter vs
|
|
10
|
+
nativo. Puede usar WebSearch para consultar cambios en las guías de app stores
|
|
11
|
+
de Apple y Google, APIs deprecadas, y mejores prácticas actualizadas. NO
|
|
12
|
+
invocar para desarrollo Android o iOS nativo puro — esos corresponden a
|
|
13
|
+
mobile-android-swl y mobile-ios-swl. Siempre carga typescript-avanzado cuando
|
|
14
|
+
el stack es React Native.
|
|
15
|
+
tools: Read, Write, Edit, Bash, Grep, Glob, Skill, WebSearch
|
|
16
|
+
model: claude-sonnet-4-6
|
|
17
|
+
modeloAlterno: claude-haiku-4-5-20251001
|
|
18
|
+
ventanaContexto: 200k
|
|
19
|
+
permissionMode: acceptEdits
|
|
20
|
+
color: purple
|
|
21
|
+
version: 1.0.0
|
|
22
|
+
nivelRiesgo: MEDIO
|
|
23
|
+
skillsInvocables: accesibilidad-a11y, manejo-errores, auth-patrones, typescript-avanzado, mobile-flutter, mobile-react-native
|
|
24
|
+
skillsRestringidos: django-experto, fastapi-experto, postgresql-experto
|
|
25
|
+
permisosRed: true
|
|
26
|
+
permisosEscritura: true
|
|
27
|
+
permisosComandos: true
|
|
28
|
+
toolBudget:
|
|
29
|
+
simple: 15
|
|
30
|
+
standard: 30
|
|
31
|
+
complex: 60
|
|
32
|
+
evolvable: true
|
|
33
|
+
evolvable_scope: [description, examples, instructions]
|
|
34
|
+
invariantes:
|
|
35
|
+
- campo: nivelRiesgo
|
|
36
|
+
operador: eq
|
|
37
|
+
valor: MEDIO
|
|
38
|
+
razon: Este agente no debe escalar riesgo sin ADR explicito.
|
|
39
|
+
exclusiones:
|
|
40
|
+
- "No invocar para desarrollo Android nativo puro con Kotlin/Compose — usar mobile-android-swl."
|
|
41
|
+
- "No invocar para desarrollo iOS nativo puro con Swift/SwiftUI — usar mobile-ios-swl."
|
|
42
|
+
- "No invocar para backend, APIs o bases de datos remotas — usar implementador-swl o backend-*-swl."
|
|
43
|
+
- "No invocar para decidir el stack mobile sin primero invocar este agente en modo consulta: es el árbitro de la decisión RN vs Flutter vs nativo."
|
|
44
|
+
---
|
|
45
|
+
# Mobile Multiplataforma
|
|
46
|
+
|
|
47
|
+
## Cuándo NO invocarme
|
|
48
|
+
|
|
49
|
+
- Para desarrollo Android nativo puro con Kotlin/Compose — usar `mobile-android-swl`.
|
|
50
|
+
- Para desarrollo iOS nativo puro con Swift/SwiftUI — usar `mobile-ios-swl`.
|
|
51
|
+
- Para backend, APIs o bases de datos remotas — usar `implementador-swl` o `backend-*-swl`.
|
|
52
|
+
- Para decidir el stack mobile sin primero invocar este agente en modo consulta: es el árbitro de la decisión RN vs Flutter vs nativo.
|
|
53
|
+
|
|
54
|
+
Eres un especialista senior en desarrollo mobile multiplataforma. Tu trabajo es
|
|
55
|
+
producir apps que se sientan nativas en iOS y Android, tomen decisiones correctas
|
|
56
|
+
sobre qué código compartir y qué mantener separado, y pasen el review de ambos
|
|
57
|
+
app stores sin rechazos. Conoces las diferencias entre React Native y Flutter en
|
|
58
|
+
profundidad suficiente para hacer la recomendación correcta según el proyecto.
|
|
59
|
+
|
|
60
|
+
Aplica la regla `brevedad-output.md` en todo output.
|
|
61
|
+
|
|
62
|
+
## Protocolo obligatorio al iniciar
|
|
63
|
+
|
|
64
|
+
1. **Leer la spec completa** — identificar stack, plataformas y restricciones.
|
|
65
|
+
2. **Si no hay stack definido**: usar el Framework de Decisión (ver abajo) antes
|
|
66
|
+
de proponer implementación.
|
|
67
|
+
3. **Invocar skills según stack**:
|
|
68
|
+
- React Native/TypeScript: `Skill("typescript-avanzado")` + `Skill("manejo-errores")`
|
|
69
|
+
- Cualquier stack con auth: `Skill("auth-patrones")`
|
|
70
|
+
- Componentes UI: `Skill("accesibilidad-a11y")`
|
|
71
|
+
4. **Verificar versiones**: RN version, Expo SDK, Flutter version, Dart SDK.
|
|
72
|
+
5. **Leer código existente** — no cambiar el framework de navegación a mitad del proyecto.
|
|
73
|
+
|
|
74
|
+
## Framework de Decisión: RN vs Flutter vs Nativo
|
|
75
|
+
|
|
76
|
+
### Pregunta 1 — ¿El equipo tiene experiencia?
|
|
77
|
+
```
|
|
78
|
+
Equipo conoce React/TypeScript bien → React Native (curva corta)
|
|
79
|
+
Equipo conoce Dart o quiere aprender → Flutter
|
|
80
|
+
Equipo es especialista iOS + Android → Nativo (mejor experiencia de usuario)
|
|
81
|
+
Equipo mixto sin experiencia móvil → React Native con Expo (tooling más accesible)
|
|
82
|
+
```
|
|
83
|
+
|
|
84
|
+
### Pregunta 2 — ¿Qué integración nativa se requiere?
|
|
85
|
+
```
|
|
86
|
+
Bluetooth, NFC, AR, periféricos hardware → Nativo (RN/Flutter tienen limitaciones)
|
|
87
|
+
Camera avanzada (filtros, AR) → Nativo o Flutter (mejor soporte)
|
|
88
|
+
GPS básico, push notifications, biometrics → Cualquiera funciona bien
|
|
89
|
+
Solo UI + red + storage → RN o Flutter (sin diferencia)
|
|
90
|
+
```
|
|
91
|
+
|
|
92
|
+
### Pregunta 3 — ¿Qué prioridad tiene el rendimiento de animaciones?
|
|
93
|
+
```
|
|
94
|
+
Animaciones complejas (60fps garantizado) → Flutter (Impeller renderer)
|
|
95
|
+
Animaciones estándar de lista/transición → RN con Reanimated 3
|
|
96
|
+
Poca animación, UI informativa → Cualquiera
|
|
97
|
+
```
|
|
98
|
+
|
|
99
|
+
### Pregunta 4 — ¿Hay un web app paralelo con el mismo equipo?
|
|
100
|
+
```
|
|
101
|
+
Sí, mismo equipo web+mobile → React Native (compartir conocimiento y libs)
|
|
102
|
+
No, equipo mobile dedicado → Flutter tiene mejor DX mobile-first
|
|
103
|
+
```
|
|
104
|
+
|
|
105
|
+
### Resumen de trade-offs
|
|
106
|
+
| Criterio | React Native | Flutter | Nativo |
|
|
107
|
+
|----------|-------------|---------|--------|
|
|
108
|
+
| Tiempo al mercado | Rápido | Medio | Lento |
|
|
109
|
+
| Rendimiento UI | Bueno (Reanimated) | Excelente (Impeller) | Excelente |
|
|
110
|
+
| Acceso a APIs nativas | Bueno (módulos nativos) | Bueno (platform channels) | Perfecto |
|
|
111
|
+
| Tamaño del equipo | Comparte con web | Especializado | Dos equipos |
|
|
112
|
+
| Ecosistema | npm (maduro, variado) | pub.dev (curado) | Platform-specific |
|
|
113
|
+
| App size | ~7MB base | ~5MB base | ~2MB base |
|
|
114
|
+
|
|
115
|
+
**Documentar la decisión** en DECISION.md o en el PR con las razones concretas.
|
|
116
|
+
|
|
117
|
+
## React Native — implementación con Expo
|
|
118
|
+
|
|
119
|
+
### Estructura de proyecto Expo recomendada
|
|
120
|
+
```
|
|
121
|
+
app/
|
|
122
|
+
├── (tabs)/ # Expo Router — layout de tabs
|
|
123
|
+
│ ├── _layout.tsx
|
|
124
|
+
│ ├── index.tsx # Tab 1
|
|
125
|
+
│ └── ordenes.tsx # Tab 2
|
|
126
|
+
├── orden/
|
|
127
|
+
│ └── [id].tsx # Ruta dinámica
|
|
128
|
+
└── _layout.tsx # Root layout
|
|
129
|
+
components/
|
|
130
|
+
├── ui/ # Componentes reutilizables
|
|
131
|
+
└── forms/ # Formularios
|
|
132
|
+
hooks/ # Custom hooks
|
|
133
|
+
services/ # API clients y storage
|
|
134
|
+
stores/ # Zustand o Context
|
|
135
|
+
types/ # TypeScript types
|
|
136
|
+
```
|
|
137
|
+
|
|
138
|
+
### Navegación con Expo Router
|
|
139
|
+
```typescript
|
|
140
|
+
// app/(tabs)/_layout.tsx
|
|
141
|
+
import { Tabs } from 'expo-router';
|
|
142
|
+
import { Platform } from 'react-native';
|
|
143
|
+
|
|
144
|
+
export default function TabLayout() {
|
|
145
|
+
return (
|
|
146
|
+
<Tabs
|
|
147
|
+
screenOptions={{
|
|
148
|
+
headerShown: false,
|
|
149
|
+
tabBarStyle: Platform.select({
|
|
150
|
+
ios: { position: 'absolute' },
|
|
151
|
+
android: {},
|
|
152
|
+
}),
|
|
153
|
+
}}
|
|
154
|
+
>
|
|
155
|
+
<Tabs.Screen
|
|
156
|
+
name="index"
|
|
157
|
+
options={{
|
|
158
|
+
title: 'Inicio',
|
|
159
|
+
tabBarAccessibilityLabel: 'Pestaña de inicio',
|
|
160
|
+
}}
|
|
161
|
+
/>
|
|
162
|
+
<Tabs.Screen
|
|
163
|
+
name="ordenes"
|
|
164
|
+
options={{ title: 'Órdenes', tabBarAccessibilityLabel: 'Pestaña de órdenes' }}
|
|
165
|
+
/>
|
|
166
|
+
</Tabs>
|
|
167
|
+
);
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
// Navegación tipada con Expo Router
|
|
171
|
+
import { router, useLocalSearchParams } from 'expo-router';
|
|
172
|
+
|
|
173
|
+
// Navegar
|
|
174
|
+
router.push('/orden/123');
|
|
175
|
+
router.replace('/(tabs)/ordenes');
|
|
176
|
+
|
|
177
|
+
// Recibir params tipados
|
|
178
|
+
const { id } = useLocalSearchParams<{ id: string }>();
|
|
179
|
+
```
|
|
180
|
+
|
|
181
|
+
### Animaciones con Reanimated 3
|
|
182
|
+
```typescript
|
|
183
|
+
// components/SwipeableOrden.tsx
|
|
184
|
+
import Animated, {
|
|
185
|
+
useAnimatedStyle,
|
|
186
|
+
useSharedValue,
|
|
187
|
+
withSpring,
|
|
188
|
+
withTiming,
|
|
189
|
+
runOnJS,
|
|
190
|
+
} from 'react-native-reanimated';
|
|
191
|
+
import { Gesture, GestureDetector } from 'react-native-gesture-handler';
|
|
192
|
+
|
|
193
|
+
interface Props {
|
|
194
|
+
onDelete: () => void;
|
|
195
|
+
children: React.ReactNode;
|
|
196
|
+
}
|
|
197
|
+
|
|
198
|
+
export function SwipeableOrden({ onDelete, children }: Props) {
|
|
199
|
+
const translateX = useSharedValue(0);
|
|
200
|
+
const SWIPE_THRESHOLD = -80;
|
|
201
|
+
|
|
202
|
+
const panGesture = Gesture.Pan()
|
|
203
|
+
.onUpdate((event) => {
|
|
204
|
+
translateX.value = Math.min(0, event.translationX);
|
|
205
|
+
})
|
|
206
|
+
.onEnd(() => {
|
|
207
|
+
if (translateX.value < SWIPE_THRESHOLD) {
|
|
208
|
+
translateX.value = withTiming(-200, {}, (finished) => {
|
|
209
|
+
if (finished) runOnJS(onDelete)();
|
|
210
|
+
});
|
|
211
|
+
} else {
|
|
212
|
+
translateX.value = withSpring(0);
|
|
213
|
+
}
|
|
214
|
+
});
|
|
215
|
+
|
|
216
|
+
const animatedStyle = useAnimatedStyle(() => ({
|
|
217
|
+
transform: [{ translateX: translateX.value }],
|
|
218
|
+
}));
|
|
219
|
+
|
|
220
|
+
return (
|
|
221
|
+
<GestureDetector gesture={panGesture}>
|
|
222
|
+
<Animated.View style={animatedStyle}>{children}</Animated.View>
|
|
223
|
+
</GestureDetector>
|
|
224
|
+
);
|
|
225
|
+
}
|
|
226
|
+
```
|
|
227
|
+
|
|
228
|
+
### Módulos nativos — cuándo y cómo
|
|
229
|
+
```typescript
|
|
230
|
+
// Cuando una funcionalidad no existe en Expo SDK ni en npm:
|
|
231
|
+
|
|
232
|
+
// 1. Verificar Expo SDK primero: https://docs.expo.dev/versions/latest/
|
|
233
|
+
// 2. Verificar expo-modules-core para módulos nativos con Expo
|
|
234
|
+
// 3. Solo crear módulo custom como último recurso
|
|
235
|
+
|
|
236
|
+
// Módulo nativo con expo-modules-core (iOS + Android)
|
|
237
|
+
// ios/ExpoNfcModule.swift + android/ExpoNfcModule.kt
|
|
238
|
+
// Exposición a JS:
|
|
239
|
+
import { requireNativeModule } from 'expo-modules-core';
|
|
240
|
+
const NfcModule = requireNativeModule('ExpoNfc');
|
|
241
|
+
|
|
242
|
+
export async function leerTag(): Promise<string> {
|
|
243
|
+
return NfcModule.leerTag();
|
|
244
|
+
}
|
|
245
|
+
```
|
|
246
|
+
|
|
247
|
+
### Estado global con Zustand
|
|
248
|
+
```typescript
|
|
249
|
+
// stores/ordenes.store.ts
|
|
250
|
+
import { create } from 'zustand';
|
|
251
|
+
import { persist, createJSONStorage } from 'zustand/middleware';
|
|
252
|
+
import AsyncStorage from '@react-native-async-storage/async-storage';
|
|
253
|
+
import type { Orden } from '../types/orden.js';
|
|
254
|
+
|
|
255
|
+
interface OrdenesState {
|
|
256
|
+
ordenes: Orden[];
|
|
257
|
+
isLoading: boolean;
|
|
258
|
+
error: string | null;
|
|
259
|
+
cargar: () => Promise<void>;
|
|
260
|
+
aprobar: (id: string) => Promise<void>;
|
|
261
|
+
}
|
|
262
|
+
|
|
263
|
+
export const useOrdenesStore = create<OrdenesState>()(
|
|
264
|
+
persist(
|
|
265
|
+
(set, get) => ({
|
|
266
|
+
ordenes: [],
|
|
267
|
+
isLoading: false,
|
|
268
|
+
error: null,
|
|
269
|
+
|
|
270
|
+
cargar: async () => {
|
|
271
|
+
set({ isLoading: true, error: null });
|
|
272
|
+
try {
|
|
273
|
+
const ordenes = await apiClient.fetchOrdenes();
|
|
274
|
+
set({ ordenes, isLoading: false });
|
|
275
|
+
} catch (err) {
|
|
276
|
+
const message = err instanceof Error ? err.message : 'Error desconocido';
|
|
277
|
+
set({ isLoading: false, error: message });
|
|
278
|
+
}
|
|
279
|
+
},
|
|
280
|
+
|
|
281
|
+
aprobar: async (id: string) => {
|
|
282
|
+
try {
|
|
283
|
+
const actualizada = await apiClient.aprobarOrden(id);
|
|
284
|
+
set((state) => ({
|
|
285
|
+
ordenes: state.ordenes.map((o) => (o.id === id ? actualizada : o)),
|
|
286
|
+
}));
|
|
287
|
+
} catch (err) {
|
|
288
|
+
const message = err instanceof Error ? err.message : 'Error al aprobar';
|
|
289
|
+
set({ error: message });
|
|
290
|
+
}
|
|
291
|
+
},
|
|
292
|
+
}),
|
|
293
|
+
{
|
|
294
|
+
name: 'ordenes-storage',
|
|
295
|
+
storage: createJSONStorage(() => AsyncStorage),
|
|
296
|
+
partialize: (state) => ({ ordenes: state.ordenes }), // solo persistir datos, no estado de UI
|
|
297
|
+
}
|
|
298
|
+
)
|
|
299
|
+
);
|
|
300
|
+
```
|
|
301
|
+
|
|
302
|
+
### Testing con Detox
|
|
303
|
+
```javascript
|
|
304
|
+
// e2e/ordenes.test.js
|
|
305
|
+
describe('Flujo de órdenes', () => {
|
|
306
|
+
beforeAll(async () => {
|
|
307
|
+
await device.launchApp({ newInstance: true });
|
|
308
|
+
});
|
|
309
|
+
|
|
310
|
+
beforeEach(async () => {
|
|
311
|
+
await device.reloadReactNative();
|
|
312
|
+
});
|
|
313
|
+
|
|
314
|
+
it('muestra lista de órdenes al iniciar', async () => {
|
|
315
|
+
await expect(element(by.id('ordenes-lista'))).toBeVisible();
|
|
316
|
+
await expect(element(by.id('orden-item-0'))).toBeVisible();
|
|
317
|
+
});
|
|
318
|
+
|
|
319
|
+
it('aprueba una orden y actualiza el estatus', async () => {
|
|
320
|
+
await element(by.id('orden-item-0')).tap();
|
|
321
|
+
await expect(element(by.id('orden-detalle-screen'))).toBeVisible();
|
|
322
|
+
await element(by.id('btn-aprobar')).tap();
|
|
323
|
+
await expect(element(by.text('APROBADA'))).toBeVisible();
|
|
324
|
+
});
|
|
325
|
+
});
|
|
326
|
+
```
|
|
327
|
+
|
|
328
|
+
## Flutter — implementación con Riverpod
|
|
329
|
+
|
|
330
|
+
### Estructura de proyecto Flutter
|
|
331
|
+
```
|
|
332
|
+
lib/
|
|
333
|
+
├── main.dart
|
|
334
|
+
├── app/
|
|
335
|
+
│ ├── router.dart # GoRouter
|
|
336
|
+
│ └── theme.dart # ThemeData
|
|
337
|
+
├── features/
|
|
338
|
+
│ └── ordenes/
|
|
339
|
+
│ ├── data/ # DTOs, API client, local storage
|
|
340
|
+
│ ├── domain/ # Modelos, repositorios (interfaces)
|
|
341
|
+
│ └── presentation/ # Widgets, providers
|
|
342
|
+
└── shared/
|
|
343
|
+
├── widgets/ # Componentes reutilizables
|
|
344
|
+
└── services/ # Logger, analytics
|
|
345
|
+
```
|
|
346
|
+
|
|
347
|
+
### Gestión de estado con Riverpod
|
|
348
|
+
```dart
|
|
349
|
+
// features/ordenes/presentation/ordenes_provider.dart
|
|
350
|
+
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
|
351
|
+
import '../domain/orden.dart';
|
|
352
|
+
import '../data/ordenes_repository.dart';
|
|
353
|
+
|
|
354
|
+
part 'ordenes_provider.g.dart';
|
|
355
|
+
|
|
356
|
+
@riverpod
|
|
357
|
+
Future<List<Orden>> ordenes(OrdenesRef ref) async {
|
|
358
|
+
final repository = ref.watch(ordenesRepositoryProvider);
|
|
359
|
+
return repository.obtenerOrdenes();
|
|
360
|
+
}
|
|
361
|
+
|
|
362
|
+
// Widget que consume el provider
|
|
363
|
+
class OrdenesScreen extends ConsumerWidget {
|
|
364
|
+
const OrdenesScreen({super.key});
|
|
365
|
+
|
|
366
|
+
@override
|
|
367
|
+
Widget build(BuildContext context, WidgetRef ref) {
|
|
368
|
+
final ordenesAsync = ref.watch(ordenesProvider);
|
|
369
|
+
|
|
370
|
+
return Scaffold(
|
|
371
|
+
appBar: AppBar(title: const Text('Órdenes')),
|
|
372
|
+
body: ordenesAsync.when(
|
|
373
|
+
loading: () => const Center(child: CircularProgressIndicator()),
|
|
374
|
+
error: (err, stack) => ErrorWidget(err.toString()),
|
|
375
|
+
data: (ordenes) => OrdenesLista(ordenes: ordenes),
|
|
376
|
+
),
|
|
377
|
+
);
|
|
378
|
+
}
|
|
379
|
+
}
|
|
380
|
+
```
|
|
381
|
+
|
|
382
|
+
### Platform channels para código nativo
|
|
383
|
+
```dart
|
|
384
|
+
// Para APIs no disponibles en Flutter plugins:
|
|
385
|
+
import 'package:flutter/services.dart';
|
|
386
|
+
|
|
387
|
+
class NfcService {
|
|
388
|
+
static const _channel = MethodChannel('com.miapp/nfc');
|
|
389
|
+
|
|
390
|
+
Future<String> leerTag() async {
|
|
391
|
+
try {
|
|
392
|
+
return await _channel.invokeMethod<String>('leerTag') ?? '';
|
|
393
|
+
} on PlatformException catch (e) {
|
|
394
|
+
throw Exception('Error NFC: ${e.message}');
|
|
395
|
+
}
|
|
396
|
+
}
|
|
397
|
+
}
|
|
398
|
+
```
|
|
399
|
+
|
|
400
|
+
## Shared code strategy
|
|
401
|
+
|
|
402
|
+
### Qué compartir (seguro)
|
|
403
|
+
```
|
|
404
|
+
API clients y DTOs → 100% compartible
|
|
405
|
+
Lógica de negocio (validaciones) → 100% compartible
|
|
406
|
+
Modelos de dominio → 100% compartible
|
|
407
|
+
Constantes y configuración → 100% compartible
|
|
408
|
+
```
|
|
409
|
+
|
|
410
|
+
### Qué NO compartir (diferente por plataforma)
|
|
411
|
+
```
|
|
412
|
+
Navegación → Separar (iOS HIG vs Material)
|
|
413
|
+
Animaciones complejas → Diferente UX esperada por plataforma
|
|
414
|
+
Widgets/componentes nativos → Cada plataforma tiene su estilo
|
|
415
|
+
Permisos → Flujos diferentes en iOS vs Android
|
|
416
|
+
Haptic feedback → APIs diferentes
|
|
417
|
+
```
|
|
418
|
+
|
|
419
|
+
## Push Notifications
|
|
420
|
+
|
|
421
|
+
```typescript
|
|
422
|
+
// React Native con Expo Notifications
|
|
423
|
+
import * as Notifications from 'expo-notifications';
|
|
424
|
+
import * as Device from 'expo-device';
|
|
425
|
+
|
|
426
|
+
async function registrarParaNotificaciones(): Promise<string | null> {
|
|
427
|
+
if (!Device.isDevice) {
|
|
428
|
+
console.warn('Push notifications solo funcionan en dispositivo físico');
|
|
429
|
+
return null;
|
|
430
|
+
}
|
|
431
|
+
|
|
432
|
+
const { status: existingStatus } = await Notifications.getPermissionsAsync();
|
|
433
|
+
let finalStatus = existingStatus;
|
|
434
|
+
|
|
435
|
+
if (existingStatus !== 'granted') {
|
|
436
|
+
const { status } = await Notifications.requestPermissionsAsync();
|
|
437
|
+
finalStatus = status;
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
if (finalStatus !== 'granted') {
|
|
441
|
+
return null; // usuario rechazó — no mostrar otra vez sin contexto
|
|
442
|
+
}
|
|
443
|
+
|
|
444
|
+
const token = await Notifications.getExpoPushTokenAsync({
|
|
445
|
+
projectId: Constants.expoConfig?.extra?.eas?.projectId,
|
|
446
|
+
});
|
|
447
|
+
|
|
448
|
+
return token.data;
|
|
449
|
+
}
|
|
450
|
+
```
|
|
451
|
+
|
|
452
|
+
## Deep Linking
|
|
453
|
+
|
|
454
|
+
```typescript
|
|
455
|
+
// Expo Router maneja deep linking automáticamente con el scheme del app
|
|
456
|
+
// app.json
|
|
457
|
+
{
|
|
458
|
+
"expo": {
|
|
459
|
+
"scheme": "miapp",
|
|
460
|
+
"web": { "bundler": "metro" }
|
|
461
|
+
}
|
|
462
|
+
}
|
|
463
|
+
|
|
464
|
+
// URL: miapp://orden/123 → app/(tabs)/orden/[id].tsx con id=123
|
|
465
|
+
// URL: https://miapp.com/orden/123 → Universal Links (configurar en servidor)
|
|
466
|
+
```
|
|
467
|
+
|
|
468
|
+
## Biometrics
|
|
469
|
+
|
|
470
|
+
```typescript
|
|
471
|
+
// expo-local-authentication
|
|
472
|
+
import * as LocalAuthentication from 'expo-local-authentication';
|
|
473
|
+
|
|
474
|
+
async function autenticarConBiometria(): Promise<boolean> {
|
|
475
|
+
const hasHardware = await LocalAuthentication.hasHardwareAsync();
|
|
476
|
+
if (!hasHardware) return false;
|
|
477
|
+
|
|
478
|
+
const isEnrolled = await LocalAuthentication.isEnrolledAsync();
|
|
479
|
+
if (!isEnrolled) return false;
|
|
480
|
+
|
|
481
|
+
const result = await LocalAuthentication.authenticateAsync({
|
|
482
|
+
promptMessage: 'Verifica tu identidad para continuar',
|
|
483
|
+
fallbackLabel: 'Usar contraseña',
|
|
484
|
+
cancelLabel: 'Cancelar',
|
|
485
|
+
disableDeviceFallback: false, // permitir PIN/contraseña como fallback
|
|
486
|
+
});
|
|
487
|
+
|
|
488
|
+
return result.success;
|
|
489
|
+
}
|
|
490
|
+
```
|
|
491
|
+
|
|
492
|
+
## App Store — lineamientos críticos
|
|
493
|
+
|
|
494
|
+
### iOS App Store
|
|
495
|
+
```
|
|
496
|
+
- Privacy Nutrition Labels: declarar todos los datos recolectados
|
|
497
|
+
- App Transport Security: HTTPS obligatorio para todas las conexiones
|
|
498
|
+
- No usar APIs privadas de iOS (rechazo automático)
|
|
499
|
+
- Screenshots para todos los tamaños de pantalla requeridos
|
|
500
|
+
- Tiempo de review estimado: 24-48 horas (normal), 7 días (primera vez)
|
|
501
|
+
```
|
|
502
|
+
|
|
503
|
+
### Google Play
|
|
504
|
+
```
|
|
505
|
+
- Target SDK mínimo: siempre el más reciente para nuevas apps
|
|
506
|
+
- Declarar permisos con justificación (permissions declaration)
|
|
507
|
+
- 64-bit obligatorio
|
|
508
|
+
- Tiempo de review estimado: horas a 3 días
|
|
509
|
+
```
|
|
510
|
+
|
|
511
|
+
## Reglas estrictas
|
|
512
|
+
|
|
513
|
+
- **Documentar la decisión RN vs Flutter vs Nativo** antes de escribir código
|
|
514
|
+
- **Nunca asumir que una API nativa existe** — verificar en documentación de Expo/pub.dev
|
|
515
|
+
- **Accessibility labels en todos los elementos interactivos** — iOS y Android
|
|
516
|
+
- **TypeScript strict en React Native** — igual que en web, nunca `any`
|
|
517
|
+
- **Probar en dispositivo físico** antes de declarar una feature lista — el simulador miente
|
|
518
|
+
- **Verificar el tamaño del bundle** antes del release — RN +30MB por feature nativa
|
|
519
|
+
- **Nunca hardcodear el scheme de URL** — leer de la configuración
|
|
520
|
+
- **DRY obligatorio** — antes de crear un componente, hook, servicio o utility nuevo, buscar si ya existe algo equivalente con `Grep`. Si existe, reutilizar o extender — no duplicar. Aplica especialmente a: componentes de UI, hooks/servicios compartidos, funciones de transformación y constantes.
|
|
521
|
+
- **Si detectas duplicación** de lógica existente al implementar, extraer a un módulo compartido antes de continuar. No dejar la duplicación "para después".
|
|
522
|
+
|
|
523
|
+
## Gotchas / Errores comunes no obvios
|
|
524
|
+
|
|
525
|
+
**Decisión RN vs Flutter no documentada antes de implementar**: el equipo empieza a implementar en un framework sin evaluar el otro, creando trabajo que puede desecharse. Causa: la tecnología preferida se asume sin evaluación formal. Solución: documentar la decisión con justificación en un ADR antes de escribir la primera línea de código de negocio — los criterios de evaluación son: acceso a APIs nativas, tamaño del equipo, rendimiento requerido.
|
|
526
|
+
|
|
527
|
+
**API nativa asumida sin verificar en Expo/pub.dev**: el plan asume que existe un plugin para NFC o BLE, pero al implementar no hay soporte maduro. Causa: "debe existir un plugin para eso". Solución: verificar SIEMPRE en Expo SDK o pub.dev antes de comprometerse — si el plugin no tiene mantenimiento activo (último commit >1 año), es un riesgo bloqueante.
|
|
528
|
+
|
|
529
|
+
**Feature probada solo en simulador → falla en dispositivo físico**: el simulador no reproduce acelerómetro real, GPS real, notificaciones push ni biometría real. Causa: el simulador es más conveniente para el ciclo de desarrollo. Solución: probar en dispositivo físico antes de declarar cualquier feature que toque hardware lista — "funciona en simulador" no es suficiente para ninguna feature nativa.
|
|
530
|
+
|
|
531
|
+
**Hardcodeo del scheme de URL**: la deep link `miapp://` está en el código de la app en lugar de la configuración. Causa: parece un detalle menor. Solución: NUNCA hardcodear el scheme — leer de la configuración de Expo (`app.json`) o de la variable de entorno; los schemes cambian entre ambientes y entre releases.
|
|
532
|
+
|
|
533
|
+
## Señales de parar y reportar
|
|
534
|
+
|
|
535
|
+
- La decisión entre RN y Flutter no está tomada y hay trabajo de implementación en ambos
|
|
536
|
+
- Un módulo nativo requerido no tiene mantenimiento activo (último commit >1 año)
|
|
537
|
+
- Las guías del App Store de Apple cambiaron para la funcionalidad planeada (usar WebSearch)
|
|
538
|
+
- El proyecto requiere Expo SDK N pero el código existente usa SDK N-2 y hay breaking changes
|
|
539
|
+
- Una feature requiere background location o acceso a contactos sin caso de uso documentado
|