attune-ai 2.0.0__py3-none-any.whl
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.
- attune/__init__.py +358 -0
- attune/adaptive/__init__.py +13 -0
- attune/adaptive/task_complexity.py +127 -0
- attune/agent_monitoring.py +414 -0
- attune/cache/__init__.py +117 -0
- attune/cache/base.py +166 -0
- attune/cache/dependency_manager.py +256 -0
- attune/cache/hash_only.py +251 -0
- attune/cache/hybrid.py +457 -0
- attune/cache/storage.py +285 -0
- attune/cache_monitor.py +356 -0
- attune/cache_stats.py +298 -0
- attune/cli/__init__.py +152 -0
- attune/cli/__main__.py +12 -0
- attune/cli/commands/__init__.py +1 -0
- attune/cli/commands/batch.py +264 -0
- attune/cli/commands/cache.py +248 -0
- attune/cli/commands/help.py +331 -0
- attune/cli/commands/info.py +140 -0
- attune/cli/commands/inspect.py +436 -0
- attune/cli/commands/inspection.py +57 -0
- attune/cli/commands/memory.py +48 -0
- attune/cli/commands/metrics.py +92 -0
- attune/cli/commands/orchestrate.py +184 -0
- attune/cli/commands/patterns.py +207 -0
- attune/cli/commands/profiling.py +202 -0
- attune/cli/commands/provider.py +98 -0
- attune/cli/commands/routing.py +285 -0
- attune/cli/commands/setup.py +96 -0
- attune/cli/commands/status.py +235 -0
- attune/cli/commands/sync.py +166 -0
- attune/cli/commands/tier.py +121 -0
- attune/cli/commands/utilities.py +114 -0
- attune/cli/commands/workflow.py +579 -0
- attune/cli/core.py +32 -0
- attune/cli/parsers/__init__.py +68 -0
- attune/cli/parsers/batch.py +118 -0
- attune/cli/parsers/cache.py +65 -0
- attune/cli/parsers/help.py +41 -0
- attune/cli/parsers/info.py +26 -0
- attune/cli/parsers/inspect.py +66 -0
- attune/cli/parsers/metrics.py +42 -0
- attune/cli/parsers/orchestrate.py +61 -0
- attune/cli/parsers/patterns.py +54 -0
- attune/cli/parsers/provider.py +40 -0
- attune/cli/parsers/routing.py +110 -0
- attune/cli/parsers/setup.py +42 -0
- attune/cli/parsers/status.py +47 -0
- attune/cli/parsers/sync.py +31 -0
- attune/cli/parsers/tier.py +33 -0
- attune/cli/parsers/workflow.py +77 -0
- attune/cli/utils/__init__.py +1 -0
- attune/cli/utils/data.py +242 -0
- attune/cli/utils/helpers.py +68 -0
- attune/cli_legacy.py +3957 -0
- attune/cli_minimal.py +1159 -0
- attune/cli_router.py +437 -0
- attune/cli_unified.py +814 -0
- attune/config/__init__.py +66 -0
- attune/config/xml_config.py +286 -0
- attune/config.py +545 -0
- attune/coordination.py +870 -0
- attune/core.py +1511 -0
- attune/core_modules/__init__.py +15 -0
- attune/cost_tracker.py +626 -0
- attune/dashboard/__init__.py +41 -0
- attune/dashboard/app.py +512 -0
- attune/dashboard/simple_server.py +435 -0
- attune/dashboard/standalone_server.py +547 -0
- attune/discovery.py +306 -0
- attune/emergence.py +306 -0
- attune/exceptions.py +123 -0
- attune/feedback_loops.py +373 -0
- attune/hot_reload/README.md +473 -0
- attune/hot_reload/__init__.py +62 -0
- attune/hot_reload/config.py +83 -0
- attune/hot_reload/integration.py +229 -0
- attune/hot_reload/reloader.py +298 -0
- attune/hot_reload/watcher.py +183 -0
- attune/hot_reload/websocket.py +177 -0
- attune/levels.py +577 -0
- attune/leverage_points.py +441 -0
- attune/logging_config.py +261 -0
- attune/mcp/__init__.py +10 -0
- attune/mcp/server.py +506 -0
- attune/memory/__init__.py +237 -0
- attune/memory/claude_memory.py +469 -0
- attune/memory/config.py +224 -0
- attune/memory/control_panel.py +1290 -0
- attune/memory/control_panel_support.py +145 -0
- attune/memory/cross_session.py +845 -0
- attune/memory/edges.py +179 -0
- attune/memory/encryption.py +159 -0
- attune/memory/file_session.py +770 -0
- attune/memory/graph.py +570 -0
- attune/memory/long_term.py +913 -0
- attune/memory/long_term_types.py +99 -0
- attune/memory/mixins/__init__.py +25 -0
- attune/memory/mixins/backend_init_mixin.py +249 -0
- attune/memory/mixins/capabilities_mixin.py +208 -0
- attune/memory/mixins/handoff_mixin.py +208 -0
- attune/memory/mixins/lifecycle_mixin.py +49 -0
- attune/memory/mixins/long_term_mixin.py +352 -0
- attune/memory/mixins/promotion_mixin.py +109 -0
- attune/memory/mixins/short_term_mixin.py +182 -0
- attune/memory/nodes.py +179 -0
- attune/memory/redis_bootstrap.py +540 -0
- attune/memory/security/__init__.py +31 -0
- attune/memory/security/audit_logger.py +932 -0
- attune/memory/security/pii_scrubber.py +640 -0
- attune/memory/security/secrets_detector.py +678 -0
- attune/memory/short_term.py +2192 -0
- attune/memory/simple_storage.py +302 -0
- attune/memory/storage/__init__.py +15 -0
- attune/memory/storage_backend.py +167 -0
- attune/memory/summary_index.py +583 -0
- attune/memory/types.py +446 -0
- attune/memory/unified.py +182 -0
- attune/meta_workflows/__init__.py +74 -0
- attune/meta_workflows/agent_creator.py +248 -0
- attune/meta_workflows/builtin_templates.py +567 -0
- attune/meta_workflows/cli_commands/__init__.py +56 -0
- attune/meta_workflows/cli_commands/agent_commands.py +321 -0
- attune/meta_workflows/cli_commands/analytics_commands.py +442 -0
- attune/meta_workflows/cli_commands/config_commands.py +232 -0
- attune/meta_workflows/cli_commands/memory_commands.py +182 -0
- attune/meta_workflows/cli_commands/template_commands.py +354 -0
- attune/meta_workflows/cli_commands/workflow_commands.py +382 -0
- attune/meta_workflows/cli_meta_workflows.py +59 -0
- attune/meta_workflows/form_engine.py +292 -0
- attune/meta_workflows/intent_detector.py +409 -0
- attune/meta_workflows/models.py +569 -0
- attune/meta_workflows/pattern_learner.py +738 -0
- attune/meta_workflows/plan_generator.py +384 -0
- attune/meta_workflows/session_context.py +397 -0
- attune/meta_workflows/template_registry.py +229 -0
- attune/meta_workflows/workflow.py +984 -0
- attune/metrics/__init__.py +12 -0
- attune/metrics/collector.py +31 -0
- attune/metrics/prompt_metrics.py +194 -0
- attune/models/__init__.py +172 -0
- attune/models/__main__.py +13 -0
- attune/models/adaptive_routing.py +437 -0
- attune/models/auth_cli.py +444 -0
- attune/models/auth_strategy.py +450 -0
- attune/models/cli.py +655 -0
- attune/models/empathy_executor.py +354 -0
- attune/models/executor.py +257 -0
- attune/models/fallback.py +762 -0
- attune/models/provider_config.py +282 -0
- attune/models/registry.py +472 -0
- attune/models/tasks.py +359 -0
- attune/models/telemetry/__init__.py +71 -0
- attune/models/telemetry/analytics.py +594 -0
- attune/models/telemetry/backend.py +196 -0
- attune/models/telemetry/data_models.py +431 -0
- attune/models/telemetry/storage.py +489 -0
- attune/models/token_estimator.py +420 -0
- attune/models/validation.py +280 -0
- attune/monitoring/__init__.py +52 -0
- attune/monitoring/alerts.py +946 -0
- attune/monitoring/alerts_cli.py +448 -0
- attune/monitoring/multi_backend.py +271 -0
- attune/monitoring/otel_backend.py +362 -0
- attune/optimization/__init__.py +19 -0
- attune/optimization/context_optimizer.py +272 -0
- attune/orchestration/__init__.py +67 -0
- attune/orchestration/agent_templates.py +707 -0
- attune/orchestration/config_store.py +499 -0
- attune/orchestration/execution_strategies.py +2111 -0
- attune/orchestration/meta_orchestrator.py +1168 -0
- attune/orchestration/pattern_learner.py +696 -0
- attune/orchestration/real_tools.py +931 -0
- attune/pattern_cache.py +187 -0
- attune/pattern_library.py +542 -0
- attune/patterns/debugging/all_patterns.json +81 -0
- attune/patterns/debugging/workflow_20260107_1770825e.json +77 -0
- attune/patterns/refactoring_memory.json +89 -0
- attune/persistence.py +564 -0
- attune/platform_utils.py +265 -0
- attune/plugins/__init__.py +28 -0
- attune/plugins/base.py +361 -0
- attune/plugins/registry.py +268 -0
- attune/project_index/__init__.py +32 -0
- attune/project_index/cli.py +335 -0
- attune/project_index/index.py +667 -0
- attune/project_index/models.py +504 -0
- attune/project_index/reports.py +474 -0
- attune/project_index/scanner.py +777 -0
- attune/project_index/scanner_parallel.py +291 -0
- attune/prompts/__init__.py +61 -0
- attune/prompts/config.py +77 -0
- attune/prompts/context.py +177 -0
- attune/prompts/parser.py +285 -0
- attune/prompts/registry.py +313 -0
- attune/prompts/templates.py +208 -0
- attune/redis_config.py +302 -0
- attune/redis_memory.py +799 -0
- attune/resilience/__init__.py +56 -0
- attune/resilience/circuit_breaker.py +256 -0
- attune/resilience/fallback.py +179 -0
- attune/resilience/health.py +300 -0
- attune/resilience/retry.py +209 -0
- attune/resilience/timeout.py +135 -0
- attune/routing/__init__.py +43 -0
- attune/routing/chain_executor.py +433 -0
- attune/routing/classifier.py +217 -0
- attune/routing/smart_router.py +234 -0
- attune/routing/workflow_registry.py +343 -0
- attune/scaffolding/README.md +589 -0
- attune/scaffolding/__init__.py +35 -0
- attune/scaffolding/__main__.py +14 -0
- attune/scaffolding/cli.py +240 -0
- attune/scaffolding/templates/base_wizard.py.jinja2 +121 -0
- attune/scaffolding/templates/coach_wizard.py.jinja2 +321 -0
- attune/scaffolding/templates/domain_wizard.py.jinja2 +408 -0
- attune/scaffolding/templates/linear_flow_wizard.py.jinja2 +203 -0
- attune/socratic/__init__.py +256 -0
- attune/socratic/ab_testing.py +958 -0
- attune/socratic/blueprint.py +533 -0
- attune/socratic/cli.py +703 -0
- attune/socratic/collaboration.py +1114 -0
- attune/socratic/domain_templates.py +924 -0
- attune/socratic/embeddings.py +738 -0
- attune/socratic/engine.py +794 -0
- attune/socratic/explainer.py +682 -0
- attune/socratic/feedback.py +772 -0
- attune/socratic/forms.py +629 -0
- attune/socratic/generator.py +732 -0
- attune/socratic/llm_analyzer.py +637 -0
- attune/socratic/mcp_server.py +702 -0
- attune/socratic/session.py +312 -0
- attune/socratic/storage.py +667 -0
- attune/socratic/success.py +730 -0
- attune/socratic/visual_editor.py +860 -0
- attune/socratic/web_ui.py +958 -0
- attune/telemetry/__init__.py +39 -0
- attune/telemetry/agent_coordination.py +475 -0
- attune/telemetry/agent_tracking.py +367 -0
- attune/telemetry/approval_gates.py +545 -0
- attune/telemetry/cli.py +1231 -0
- attune/telemetry/commands/__init__.py +14 -0
- attune/telemetry/commands/dashboard_commands.py +696 -0
- attune/telemetry/event_streaming.py +409 -0
- attune/telemetry/feedback_loop.py +567 -0
- attune/telemetry/usage_tracker.py +591 -0
- attune/templates.py +754 -0
- attune/test_generator/__init__.py +38 -0
- attune/test_generator/__main__.py +14 -0
- attune/test_generator/cli.py +234 -0
- attune/test_generator/generator.py +355 -0
- attune/test_generator/risk_analyzer.py +216 -0
- attune/test_generator/templates/unit_test.py.jinja2 +272 -0
- attune/tier_recommender.py +384 -0
- attune/tools.py +183 -0
- attune/trust/__init__.py +28 -0
- attune/trust/circuit_breaker.py +579 -0
- attune/trust_building.py +527 -0
- attune/validation/__init__.py +19 -0
- attune/validation/xml_validator.py +281 -0
- attune/vscode_bridge.py +173 -0
- attune/workflow_commands.py +780 -0
- attune/workflow_patterns/__init__.py +33 -0
- attune/workflow_patterns/behavior.py +249 -0
- attune/workflow_patterns/core.py +76 -0
- attune/workflow_patterns/output.py +99 -0
- attune/workflow_patterns/registry.py +255 -0
- attune/workflow_patterns/structural.py +288 -0
- attune/workflows/__init__.py +539 -0
- attune/workflows/autonomous_test_gen.py +1268 -0
- attune/workflows/base.py +2667 -0
- attune/workflows/batch_processing.py +342 -0
- attune/workflows/bug_predict.py +1084 -0
- attune/workflows/builder.py +273 -0
- attune/workflows/caching.py +253 -0
- attune/workflows/code_review.py +1048 -0
- attune/workflows/code_review_adapters.py +312 -0
- attune/workflows/code_review_pipeline.py +722 -0
- attune/workflows/config.py +645 -0
- attune/workflows/dependency_check.py +644 -0
- attune/workflows/document_gen/__init__.py +25 -0
- attune/workflows/document_gen/config.py +30 -0
- attune/workflows/document_gen/report_formatter.py +162 -0
- attune/workflows/document_gen/workflow.py +1426 -0
- attune/workflows/document_manager.py +216 -0
- attune/workflows/document_manager_README.md +134 -0
- attune/workflows/documentation_orchestrator.py +1205 -0
- attune/workflows/history.py +510 -0
- attune/workflows/keyboard_shortcuts/__init__.py +39 -0
- attune/workflows/keyboard_shortcuts/generators.py +391 -0
- attune/workflows/keyboard_shortcuts/parsers.py +416 -0
- attune/workflows/keyboard_shortcuts/prompts.py +295 -0
- attune/workflows/keyboard_shortcuts/schema.py +193 -0
- attune/workflows/keyboard_shortcuts/workflow.py +509 -0
- attune/workflows/llm_base.py +363 -0
- attune/workflows/manage_docs.py +87 -0
- attune/workflows/manage_docs_README.md +134 -0
- attune/workflows/manage_documentation.py +821 -0
- attune/workflows/new_sample_workflow1.py +149 -0
- attune/workflows/new_sample_workflow1_README.md +150 -0
- attune/workflows/orchestrated_health_check.py +849 -0
- attune/workflows/orchestrated_release_prep.py +600 -0
- attune/workflows/output.py +413 -0
- attune/workflows/perf_audit.py +863 -0
- attune/workflows/pr_review.py +762 -0
- attune/workflows/progress.py +785 -0
- attune/workflows/progress_server.py +322 -0
- attune/workflows/progressive/README 2.md +454 -0
- attune/workflows/progressive/README.md +454 -0
- attune/workflows/progressive/__init__.py +82 -0
- attune/workflows/progressive/cli.py +219 -0
- attune/workflows/progressive/core.py +488 -0
- attune/workflows/progressive/orchestrator.py +723 -0
- attune/workflows/progressive/reports.py +520 -0
- attune/workflows/progressive/telemetry.py +274 -0
- attune/workflows/progressive/test_gen.py +495 -0
- attune/workflows/progressive/workflow.py +589 -0
- attune/workflows/refactor_plan.py +694 -0
- attune/workflows/release_prep.py +895 -0
- attune/workflows/release_prep_crew.py +969 -0
- attune/workflows/research_synthesis.py +404 -0
- attune/workflows/routing.py +168 -0
- attune/workflows/secure_release.py +593 -0
- attune/workflows/security_adapters.py +297 -0
- attune/workflows/security_audit.py +1329 -0
- attune/workflows/security_audit_phase3.py +355 -0
- attune/workflows/seo_optimization.py +633 -0
- attune/workflows/step_config.py +234 -0
- attune/workflows/telemetry_mixin.py +269 -0
- attune/workflows/test5.py +125 -0
- attune/workflows/test5_README.md +158 -0
- attune/workflows/test_coverage_boost_crew.py +849 -0
- attune/workflows/test_gen/__init__.py +52 -0
- attune/workflows/test_gen/ast_analyzer.py +249 -0
- attune/workflows/test_gen/config.py +88 -0
- attune/workflows/test_gen/data_models.py +38 -0
- attune/workflows/test_gen/report_formatter.py +289 -0
- attune/workflows/test_gen/test_templates.py +381 -0
- attune/workflows/test_gen/workflow.py +655 -0
- attune/workflows/test_gen.py +54 -0
- attune/workflows/test_gen_behavioral.py +477 -0
- attune/workflows/test_gen_parallel.py +341 -0
- attune/workflows/test_lifecycle.py +526 -0
- attune/workflows/test_maintenance.py +627 -0
- attune/workflows/test_maintenance_cli.py +590 -0
- attune/workflows/test_maintenance_crew.py +840 -0
- attune/workflows/test_runner.py +622 -0
- attune/workflows/tier_tracking.py +531 -0
- attune/workflows/xml_enhanced_crew.py +285 -0
- attune_ai-2.0.0.dist-info/METADATA +1026 -0
- attune_ai-2.0.0.dist-info/RECORD +457 -0
- attune_ai-2.0.0.dist-info/WHEEL +5 -0
- attune_ai-2.0.0.dist-info/entry_points.txt +26 -0
- attune_ai-2.0.0.dist-info/licenses/LICENSE +201 -0
- attune_ai-2.0.0.dist-info/licenses/LICENSE_CHANGE_ANNOUNCEMENT.md +101 -0
- attune_ai-2.0.0.dist-info/top_level.txt +5 -0
- attune_healthcare/__init__.py +13 -0
- attune_healthcare/monitors/__init__.py +9 -0
- attune_healthcare/monitors/clinical_protocol_monitor.py +315 -0
- attune_healthcare/monitors/monitoring/__init__.py +44 -0
- attune_healthcare/monitors/monitoring/protocol_checker.py +300 -0
- attune_healthcare/monitors/monitoring/protocol_loader.py +214 -0
- attune_healthcare/monitors/monitoring/sensor_parsers.py +306 -0
- attune_healthcare/monitors/monitoring/trajectory_analyzer.py +389 -0
- attune_llm/README.md +553 -0
- attune_llm/__init__.py +28 -0
- attune_llm/agent_factory/__init__.py +53 -0
- attune_llm/agent_factory/adapters/__init__.py +85 -0
- attune_llm/agent_factory/adapters/autogen_adapter.py +312 -0
- attune_llm/agent_factory/adapters/crewai_adapter.py +483 -0
- attune_llm/agent_factory/adapters/haystack_adapter.py +298 -0
- attune_llm/agent_factory/adapters/langchain_adapter.py +362 -0
- attune_llm/agent_factory/adapters/langgraph_adapter.py +333 -0
- attune_llm/agent_factory/adapters/native.py +228 -0
- attune_llm/agent_factory/adapters/wizard_adapter.py +423 -0
- attune_llm/agent_factory/base.py +305 -0
- attune_llm/agent_factory/crews/__init__.py +67 -0
- attune_llm/agent_factory/crews/code_review.py +1113 -0
- attune_llm/agent_factory/crews/health_check.py +1262 -0
- attune_llm/agent_factory/crews/refactoring.py +1128 -0
- attune_llm/agent_factory/crews/security_audit.py +1018 -0
- attune_llm/agent_factory/decorators.py +287 -0
- attune_llm/agent_factory/factory.py +558 -0
- attune_llm/agent_factory/framework.py +193 -0
- attune_llm/agent_factory/memory_integration.py +328 -0
- attune_llm/agent_factory/resilient.py +320 -0
- attune_llm/agents_md/__init__.py +22 -0
- attune_llm/agents_md/loader.py +218 -0
- attune_llm/agents_md/parser.py +271 -0
- attune_llm/agents_md/registry.py +307 -0
- attune_llm/claude_memory.py +466 -0
- attune_llm/cli/__init__.py +8 -0
- attune_llm/cli/sync_claude.py +487 -0
- attune_llm/code_health.py +1313 -0
- attune_llm/commands/__init__.py +51 -0
- attune_llm/commands/context.py +375 -0
- attune_llm/commands/loader.py +301 -0
- attune_llm/commands/models.py +231 -0
- attune_llm/commands/parser.py +371 -0
- attune_llm/commands/registry.py +429 -0
- attune_llm/config/__init__.py +29 -0
- attune_llm/config/unified.py +291 -0
- attune_llm/context/__init__.py +22 -0
- attune_llm/context/compaction.py +455 -0
- attune_llm/context/manager.py +434 -0
- attune_llm/contextual_patterns.py +361 -0
- attune_llm/core.py +907 -0
- attune_llm/git_pattern_extractor.py +435 -0
- attune_llm/hooks/__init__.py +24 -0
- attune_llm/hooks/config.py +306 -0
- attune_llm/hooks/executor.py +289 -0
- attune_llm/hooks/registry.py +302 -0
- attune_llm/hooks/scripts/__init__.py +39 -0
- attune_llm/hooks/scripts/evaluate_session.py +201 -0
- attune_llm/hooks/scripts/first_time_init.py +285 -0
- attune_llm/hooks/scripts/pre_compact.py +207 -0
- attune_llm/hooks/scripts/session_end.py +183 -0
- attune_llm/hooks/scripts/session_start.py +163 -0
- attune_llm/hooks/scripts/suggest_compact.py +225 -0
- attune_llm/learning/__init__.py +30 -0
- attune_llm/learning/evaluator.py +438 -0
- attune_llm/learning/extractor.py +514 -0
- attune_llm/learning/storage.py +560 -0
- attune_llm/levels.py +227 -0
- attune_llm/pattern_confidence.py +414 -0
- attune_llm/pattern_resolver.py +272 -0
- attune_llm/pattern_summary.py +350 -0
- attune_llm/providers.py +967 -0
- attune_llm/routing/__init__.py +32 -0
- attune_llm/routing/model_router.py +362 -0
- attune_llm/security/IMPLEMENTATION_SUMMARY.md +413 -0
- attune_llm/security/PHASE2_COMPLETE.md +384 -0
- attune_llm/security/PHASE2_SECRETS_DETECTOR_COMPLETE.md +271 -0
- attune_llm/security/QUICK_REFERENCE.md +316 -0
- attune_llm/security/README.md +262 -0
- attune_llm/security/__init__.py +62 -0
- attune_llm/security/audit_logger.py +929 -0
- attune_llm/security/audit_logger_example.py +152 -0
- attune_llm/security/pii_scrubber.py +640 -0
- attune_llm/security/secrets_detector.py +678 -0
- attune_llm/security/secrets_detector_example.py +304 -0
- attune_llm/security/secure_memdocs.py +1192 -0
- attune_llm/security/secure_memdocs_example.py +278 -0
- attune_llm/session_status.py +745 -0
- attune_llm/state.py +246 -0
- attune_llm/utils/__init__.py +5 -0
- attune_llm/utils/tokens.py +349 -0
- attune_software/SOFTWARE_PLUGIN_README.md +57 -0
- attune_software/__init__.py +13 -0
- attune_software/cli/__init__.py +120 -0
- attune_software/cli/inspect.py +362 -0
- attune_software/cli.py +574 -0
- attune_software/plugin.py +188 -0
- workflow_scaffolding/__init__.py +11 -0
- workflow_scaffolding/__main__.py +12 -0
- workflow_scaffolding/cli.py +206 -0
- workflow_scaffolding/generator.py +265 -0
|
@@ -0,0 +1,295 @@
|
|
|
1
|
+
"""XML-Enhanced Prompts for Keyboard Shortcut Generation.
|
|
2
|
+
|
|
3
|
+
These prompts follow the Keyboard Conductor design principles:
|
|
4
|
+
- Musical scale pattern (Scale 1: Daily, Scale 2: Frequent, Scale 3: Advanced)
|
|
5
|
+
- Ergonomic optimization (left-hand home row preference)
|
|
6
|
+
- Mnemonic quality (letter matches, memorable phrases)
|
|
7
|
+
"""
|
|
8
|
+
|
|
9
|
+
# Stage 1: Feature Analysis Prompt
|
|
10
|
+
ANALYZE_FEATURES_PROMPT = """
|
|
11
|
+
<prompt name="analyze_features" tier="capable">
|
|
12
|
+
<system>
|
|
13
|
+
You are a UX expert specializing in keyboard ergonomics and mnemonic design.
|
|
14
|
+
Your goal is to analyze project features and recommend optimal keyboard shortcuts
|
|
15
|
+
following the "Keyboard Conductor" musical scale pattern.
|
|
16
|
+
|
|
17
|
+
Design Principles:
|
|
18
|
+
- Scale 1 (Daily): 4 most-used features on home row
|
|
19
|
+
- Scale 2 (Frequent): Next 4 features on adjacent keys
|
|
20
|
+
- Scale 3 (Advanced): Remaining features logically placed
|
|
21
|
+
- Left-hand priority for primary actions
|
|
22
|
+
- Memorable mnemonics (letter matches like M=Morning, S=Ship)
|
|
23
|
+
</system>
|
|
24
|
+
|
|
25
|
+
<context>
|
|
26
|
+
<project_name>{project_name}</project_name>
|
|
27
|
+
<project_type>{project_type}</project_type>
|
|
28
|
+
<total_features>{feature_count}</total_features>
|
|
29
|
+
<target_layouts>qwerty, dvorak, colemak</target_layouts>
|
|
30
|
+
</context>
|
|
31
|
+
|
|
32
|
+
<input>
|
|
33
|
+
<features format="yaml">
|
|
34
|
+
{features_yaml}
|
|
35
|
+
</features>
|
|
36
|
+
</input>
|
|
37
|
+
|
|
38
|
+
<task>
|
|
39
|
+
Analyze these features and provide:
|
|
40
|
+
|
|
41
|
+
1. **Frequency Classification**: Categorize each feature as:
|
|
42
|
+
- daily: Used multiple times per day (4 features max)
|
|
43
|
+
- frequent: Used regularly but not constantly (4 features)
|
|
44
|
+
- advanced: Used occasionally or for power users (remaining)
|
|
45
|
+
|
|
46
|
+
2. **Groupings**: Identify natural feature groups (4-6 max):
|
|
47
|
+
- Quick Actions (morning routines, ship checks)
|
|
48
|
+
- Workflows (code review, testing)
|
|
49
|
+
- Views (dashboard, costs)
|
|
50
|
+
- Analysis (security, bugs, health)
|
|
51
|
+
|
|
52
|
+
3. **Mnemonic Suggestions**: For each feature, suggest:
|
|
53
|
+
- Primary letter (first letter match if available)
|
|
54
|
+
- Alternative letter (if conflict)
|
|
55
|
+
- Mnemonic phrase (e.g., "M = Morning briefing")
|
|
56
|
+
|
|
57
|
+
4. **Conflict Detection**: Flag features that:
|
|
58
|
+
- Have same first letter
|
|
59
|
+
- Should share modifier patterns
|
|
60
|
+
- Might conflict with OS/IDE shortcuts
|
|
61
|
+
</task>
|
|
62
|
+
|
|
63
|
+
<output format="yaml">
|
|
64
|
+
Return a YAML document with this structure:
|
|
65
|
+
|
|
66
|
+
analyzed_features:
|
|
67
|
+
- id: "morning"
|
|
68
|
+
frequency: "daily"
|
|
69
|
+
primary_letter: "m"
|
|
70
|
+
alt_letter: null
|
|
71
|
+
mnemonic: "M = Morning briefing"
|
|
72
|
+
group: "Quick Actions"
|
|
73
|
+
|
|
74
|
+
suggested_groups:
|
|
75
|
+
- name: "Quick Actions"
|
|
76
|
+
icon: "$(zap)"
|
|
77
|
+
features: ["morning", "ship", "fix", "dashboard"]
|
|
78
|
+
|
|
79
|
+
conflict_warnings:
|
|
80
|
+
- type: "letter_conflict"
|
|
81
|
+
features: ["review", "refactor"]
|
|
82
|
+
message: "Both start with 'r' - suggest using 'r' for review, 'f' for refactor"
|
|
83
|
+
|
|
84
|
+
phrase_mnemonic: "My Ship Floats Daily"
|
|
85
|
+
</output>
|
|
86
|
+
</prompt>
|
|
87
|
+
"""
|
|
88
|
+
|
|
89
|
+
# Stage 2: Shortcut Generation Prompt
|
|
90
|
+
GENERATE_SHORTCUTS_PROMPT = """
|
|
91
|
+
<prompt name="generate_shortcuts" tier="capable">
|
|
92
|
+
<system>
|
|
93
|
+
You are a keyboard layout specialist. Generate ergonomic keyboard shortcuts
|
|
94
|
+
following the "Keyboard Conductor" musical scale pattern.
|
|
95
|
+
|
|
96
|
+
The goal is to create shortcuts that:
|
|
97
|
+
1. Are physically comfortable (minimize finger travel)
|
|
98
|
+
2. Are memorable (use meaningful letter associations)
|
|
99
|
+
3. Progress naturally (learn 4 keys, then 4 more, then rest)
|
|
100
|
+
4. Work across keyboard layouts (QWERTY, Dvorak, Colemak)
|
|
101
|
+
</system>
|
|
102
|
+
|
|
103
|
+
<constraints>
|
|
104
|
+
<physical_layout>
|
|
105
|
+
- Prioritize left-hand home row for Scale 1 (daily) actions
|
|
106
|
+
- Use adjacent keys for related actions
|
|
107
|
+
- Avoid awkward stretches or pinky overuse
|
|
108
|
+
- Reserve right hand for less frequent actions
|
|
109
|
+
</physical_layout>
|
|
110
|
+
|
|
111
|
+
<mnemonic_quality>
|
|
112
|
+
- Prefer direct letter matches (M=Morning, S=Ship, F=Fix)
|
|
113
|
+
- Use sound-alikes when letters conflict (B=Bugs, not P=Predict)
|
|
114
|
+
- Create memorable phrases for groups ("My Ship Floats Daily")
|
|
115
|
+
- Consider keyboard position for alternatives (adjacent keys)
|
|
116
|
+
</mnemonic_quality>
|
|
117
|
+
|
|
118
|
+
<layout_adaptation>
|
|
119
|
+
QWERTY home row: A-S-D-F (left), J-K-L-; (right)
|
|
120
|
+
Dvorak home row: A-O-E-U (left), H-T-N-S (right)
|
|
121
|
+
Colemak home row: A-R-S-T (left), N-E-I-O (right)
|
|
122
|
+
|
|
123
|
+
For Dvorak/Colemak, adapt mnemonics:
|
|
124
|
+
- Dvorak: "AOEU - vowels for daily actions"
|
|
125
|
+
- Colemak: "ARST - your daily rhythm"
|
|
126
|
+
</layout_adaptation>
|
|
127
|
+
</constraints>
|
|
128
|
+
|
|
129
|
+
<input>
|
|
130
|
+
<analyzed_features>{analyzed_yaml}</analyzed_features>
|
|
131
|
+
<existing_shortcuts>{existing_shortcuts}</existing_shortcuts>
|
|
132
|
+
<reserved_keys>{reserved_keys}</reserved_keys>
|
|
133
|
+
</input>
|
|
134
|
+
|
|
135
|
+
<output format="json">
|
|
136
|
+
Generate shortcuts for each layout. Return JSON with this structure:
|
|
137
|
+
|
|
138
|
+
{{
|
|
139
|
+
"qwerty": {{
|
|
140
|
+
"shortcuts": [
|
|
141
|
+
{{
|
|
142
|
+
"feature_id": "morning",
|
|
143
|
+
"key": "m",
|
|
144
|
+
"mnemonic": "M = Morning",
|
|
145
|
+
"hand": "left",
|
|
146
|
+
"finger": "index",
|
|
147
|
+
"row": "home"
|
|
148
|
+
}}
|
|
149
|
+
],
|
|
150
|
+
"scale_assignments": {{
|
|
151
|
+
"daily": ["m", "s", "f", "d"],
|
|
152
|
+
"frequent": ["w", "r", "t", "c"],
|
|
153
|
+
"advanced": ["h", "b", "a", "g", "l", "v", "p", "z"]
|
|
154
|
+
}},
|
|
155
|
+
"phrase_mnemonic": "My Ship Floats Daily, With Really Tested Code"
|
|
156
|
+
}},
|
|
157
|
+
"dvorak": {{
|
|
158
|
+
"shortcuts": [...],
|
|
159
|
+
"scale_assignments": {{
|
|
160
|
+
"daily": ["a", "o", "e", "u"],
|
|
161
|
+
"frequent": ["i", "h", "t", "n"],
|
|
162
|
+
"advanced": [...]
|
|
163
|
+
}},
|
|
164
|
+
"phrase_mnemonic": "AOEU - vowels for daily actions"
|
|
165
|
+
}},
|
|
166
|
+
"colemak": {{
|
|
167
|
+
"shortcuts": [...],
|
|
168
|
+
"scale_assignments": {{
|
|
169
|
+
"daily": ["a", "r", "s", "t"],
|
|
170
|
+
"frequent": ["g", "n", "e", "i"],
|
|
171
|
+
"advanced": [...]
|
|
172
|
+
}},
|
|
173
|
+
"phrase_mnemonic": "ARST - your daily rhythm"
|
|
174
|
+
}}
|
|
175
|
+
}}
|
|
176
|
+
</output>
|
|
177
|
+
</prompt>
|
|
178
|
+
"""
|
|
179
|
+
|
|
180
|
+
# Stage 3: Conflict Validation Prompt
|
|
181
|
+
VALIDATE_SHORTCUTS_PROMPT = """
|
|
182
|
+
<prompt name="validate_shortcuts" tier="cheap">
|
|
183
|
+
<system>
|
|
184
|
+
Validate generated keyboard shortcuts for conflicts and ergonomic issues.
|
|
185
|
+
Your role is to ensure the shortcuts are safe, comfortable, and memorable.
|
|
186
|
+
</system>
|
|
187
|
+
|
|
188
|
+
<input>
|
|
189
|
+
<generated_shortcuts>{shortcuts_json}</generated_shortcuts>
|
|
190
|
+
|
|
191
|
+
<platform_reserved>
|
|
192
|
+
macOS: Cmd+Q (quit), Cmd+W (close), Cmd+H (hide), Cmd+M (minimize)
|
|
193
|
+
Windows: Ctrl+W (close), Ctrl+Q (quit in some apps)
|
|
194
|
+
Linux: Similar to Windows
|
|
195
|
+
|
|
196
|
+
IDE (VSCode): Ctrl+S (save), Ctrl+Z (undo), Ctrl+N (new), Ctrl+O (open)
|
|
197
|
+
</platform_reserved>
|
|
198
|
+
</input>
|
|
199
|
+
|
|
200
|
+
<checks>
|
|
201
|
+
Verify each of these conditions:
|
|
202
|
+
|
|
203
|
+
1. **No Duplicates**: Each key is used only once per layout
|
|
204
|
+
|
|
205
|
+
2. **No OS Conflicts**: Shortcuts don't override critical OS shortcuts
|
|
206
|
+
(after the Ctrl+Shift+E prefix, check the final key)
|
|
207
|
+
|
|
208
|
+
3. **No IDE Conflicts**: Don't conflict with common VSCode shortcuts
|
|
209
|
+
|
|
210
|
+
4. **Balanced Fingers**: No single finger used more than 4 times
|
|
211
|
+
(distribute across fingers evenly)
|
|
212
|
+
|
|
213
|
+
5. **No Awkward Combos**: Avoid:
|
|
214
|
+
- Keys requiring hand crossing
|
|
215
|
+
- Pinky + ring finger stretches
|
|
216
|
+
- Consecutive same-finger keys
|
|
217
|
+
|
|
218
|
+
6. **Mnemonic Quality**: Each shortcut has a clear, memorable association
|
|
219
|
+
|
|
220
|
+
7. **Progressive Learning**: Scale 1 keys are easier to reach than Scale 3
|
|
221
|
+
</checks>
|
|
222
|
+
|
|
223
|
+
<output format="json">
|
|
224
|
+
Return validation results:
|
|
225
|
+
|
|
226
|
+
{{
|
|
227
|
+
"valid": true,
|
|
228
|
+
"conflicts": [
|
|
229
|
+
{{
|
|
230
|
+
"type": "duplicate",
|
|
231
|
+
"key": "r",
|
|
232
|
+
"features": ["review", "refactor"],
|
|
233
|
+
"severity": "error"
|
|
234
|
+
}}
|
|
235
|
+
],
|
|
236
|
+
"warnings": [
|
|
237
|
+
{{
|
|
238
|
+
"type": "ergonomic",
|
|
239
|
+
"key": "q",
|
|
240
|
+
"message": "Pinky stretch - consider moving to Scale 3",
|
|
241
|
+
"severity": "warning"
|
|
242
|
+
}}
|
|
243
|
+
],
|
|
244
|
+
"suggestions": [
|
|
245
|
+
{{
|
|
246
|
+
"feature": "refactor",
|
|
247
|
+
"current_key": "r",
|
|
248
|
+
"suggested_key": "x",
|
|
249
|
+
"reason": "Conflicts with 'review', 'x' is available and adjacent"
|
|
250
|
+
}}
|
|
251
|
+
],
|
|
252
|
+
"finger_distribution": {{
|
|
253
|
+
"pinky": 2,
|
|
254
|
+
"ring": 3,
|
|
255
|
+
"middle": 4,
|
|
256
|
+
"index": 5
|
|
257
|
+
}},
|
|
258
|
+
"mnemonic_quality_score": 0.85
|
|
259
|
+
}}
|
|
260
|
+
</output>
|
|
261
|
+
</prompt>
|
|
262
|
+
"""
|
|
263
|
+
|
|
264
|
+
|
|
265
|
+
def format_analyze_prompt(
|
|
266
|
+
project_name: str,
|
|
267
|
+
project_type: str,
|
|
268
|
+
feature_count: int,
|
|
269
|
+
features_yaml: str,
|
|
270
|
+
) -> str:
|
|
271
|
+
"""Format the feature analysis prompt with provided data."""
|
|
272
|
+
return ANALYZE_FEATURES_PROMPT.format(
|
|
273
|
+
project_name=project_name,
|
|
274
|
+
project_type=project_type,
|
|
275
|
+
feature_count=feature_count,
|
|
276
|
+
features_yaml=features_yaml,
|
|
277
|
+
)
|
|
278
|
+
|
|
279
|
+
|
|
280
|
+
def format_generate_prompt(
|
|
281
|
+
analyzed_yaml: str,
|
|
282
|
+
existing_shortcuts: str = "[]",
|
|
283
|
+
reserved_keys: str = "[]",
|
|
284
|
+
) -> str:
|
|
285
|
+
"""Format the shortcut generation prompt with provided data."""
|
|
286
|
+
return GENERATE_SHORTCUTS_PROMPT.format(
|
|
287
|
+
analyzed_yaml=analyzed_yaml,
|
|
288
|
+
existing_shortcuts=existing_shortcuts,
|
|
289
|
+
reserved_keys=reserved_keys,
|
|
290
|
+
)
|
|
291
|
+
|
|
292
|
+
|
|
293
|
+
def format_validate_prompt(shortcuts_json: str) -> str:
|
|
294
|
+
"""Format the validation prompt with provided data."""
|
|
295
|
+
return VALIDATE_SHORTCUTS_PROMPT.format(shortcuts_json=shortcuts_json)
|
|
@@ -0,0 +1,193 @@
|
|
|
1
|
+
"""Schema definitions for keyboard shortcut generation.
|
|
2
|
+
|
|
3
|
+
Uses Pydantic models for validation and serialization.
|
|
4
|
+
"""
|
|
5
|
+
|
|
6
|
+
from dataclasses import dataclass, field
|
|
7
|
+
from enum import Enum
|
|
8
|
+
from typing import Literal
|
|
9
|
+
|
|
10
|
+
|
|
11
|
+
class FrequencyTier(str, Enum):
|
|
12
|
+
"""Usage frequency tiers for feature prioritization."""
|
|
13
|
+
|
|
14
|
+
DAILY = "daily" # Scale 1: 4 most-used features
|
|
15
|
+
FREQUENT = "frequent" # Scale 2: Next 4 features
|
|
16
|
+
ADVANCED = "advanced" # Scale 3: Remaining features
|
|
17
|
+
|
|
18
|
+
|
|
19
|
+
class KeyboardLayout(str, Enum):
|
|
20
|
+
"""Supported keyboard layouts."""
|
|
21
|
+
|
|
22
|
+
QWERTY = "qwerty"
|
|
23
|
+
DVORAK = "dvorak"
|
|
24
|
+
COLEMAK = "colemak"
|
|
25
|
+
|
|
26
|
+
|
|
27
|
+
class HandPosition(str, Enum):
|
|
28
|
+
"""Which hand operates the key."""
|
|
29
|
+
|
|
30
|
+
LEFT = "left"
|
|
31
|
+
RIGHT = "right"
|
|
32
|
+
|
|
33
|
+
|
|
34
|
+
class FingerPosition(str, Enum):
|
|
35
|
+
"""Which finger operates the key."""
|
|
36
|
+
|
|
37
|
+
PINKY = "pinky"
|
|
38
|
+
RING = "ring"
|
|
39
|
+
MIDDLE = "middle"
|
|
40
|
+
INDEX = "index"
|
|
41
|
+
THUMB = "thumb"
|
|
42
|
+
|
|
43
|
+
|
|
44
|
+
class RowPosition(str, Enum):
|
|
45
|
+
"""Keyboard row position."""
|
|
46
|
+
|
|
47
|
+
TOP = "top" # QWERTY row
|
|
48
|
+
HOME = "home" # ASDF row
|
|
49
|
+
BOTTOM = "bottom" # ZXCV row
|
|
50
|
+
|
|
51
|
+
|
|
52
|
+
@dataclass
|
|
53
|
+
class Feature:
|
|
54
|
+
"""A single feature that needs a keyboard shortcut."""
|
|
55
|
+
|
|
56
|
+
id: str # Unique identifier (e.g., "morning", "ship")
|
|
57
|
+
name: str # Human-readable name (e.g., "Morning Briefing")
|
|
58
|
+
description: str = ""
|
|
59
|
+
command: str = "" # VSCode command or CLI command
|
|
60
|
+
cli_alias: str = "" # CLI equivalent (e.g., "empathy morning")
|
|
61
|
+
frequency: FrequencyTier = FrequencyTier.FREQUENT
|
|
62
|
+
context: Literal["global", "editor", "explorer"] = "global"
|
|
63
|
+
icon: str = "$(symbol-misc)"
|
|
64
|
+
|
|
65
|
+
|
|
66
|
+
@dataclass
|
|
67
|
+
class Category:
|
|
68
|
+
"""A group of related features."""
|
|
69
|
+
|
|
70
|
+
name: str
|
|
71
|
+
icon: str = "$(folder)"
|
|
72
|
+
tier: FrequencyTier = FrequencyTier.FREQUENT
|
|
73
|
+
features: list[Feature] = field(default_factory=list)
|
|
74
|
+
|
|
75
|
+
|
|
76
|
+
@dataclass
|
|
77
|
+
class LayoutConfig:
|
|
78
|
+
"""Configuration for a specific keyboard layout."""
|
|
79
|
+
|
|
80
|
+
layout: KeyboardLayout
|
|
81
|
+
home_row: list[str] = field(default_factory=list)
|
|
82
|
+
mnemonic_base: str = ""
|
|
83
|
+
|
|
84
|
+
def __post_init__(self):
|
|
85
|
+
if not self.home_row:
|
|
86
|
+
if self.layout == KeyboardLayout.QWERTY:
|
|
87
|
+
self.home_row = ["a", "s", "d", "f"]
|
|
88
|
+
self.mnemonic_base = "natural English letters"
|
|
89
|
+
elif self.layout == KeyboardLayout.DVORAK:
|
|
90
|
+
self.home_row = ["a", "o", "e", "u"]
|
|
91
|
+
self.mnemonic_base = "vowel-centric patterns"
|
|
92
|
+
elif self.layout == KeyboardLayout.COLEMAK:
|
|
93
|
+
self.home_row = ["a", "r", "s", "t"]
|
|
94
|
+
self.mnemonic_base = "ARST patterns"
|
|
95
|
+
|
|
96
|
+
|
|
97
|
+
@dataclass
|
|
98
|
+
class ShortcutAssignment:
|
|
99
|
+
"""A keyboard shortcut assignment for a feature."""
|
|
100
|
+
|
|
101
|
+
feature_id: str
|
|
102
|
+
key: str # Single letter key (e.g., "m", "s")
|
|
103
|
+
mnemonic: str # Memory aid (e.g., "M = Morning")
|
|
104
|
+
hand: HandPosition = HandPosition.LEFT
|
|
105
|
+
finger: FingerPosition = FingerPosition.INDEX
|
|
106
|
+
row: RowPosition = RowPosition.HOME
|
|
107
|
+
layout: KeyboardLayout = KeyboardLayout.QWERTY
|
|
108
|
+
|
|
109
|
+
|
|
110
|
+
@dataclass
|
|
111
|
+
class ScaleAssignments:
|
|
112
|
+
"""Shortcut assignments organized by scale (learning progression)."""
|
|
113
|
+
|
|
114
|
+
daily: list[str] = field(default_factory=list) # Scale 1: 4 keys
|
|
115
|
+
frequent: list[str] = field(default_factory=list) # Scale 2: 4 keys
|
|
116
|
+
advanced: list[str] = field(default_factory=list) # Scale 3: remaining
|
|
117
|
+
|
|
118
|
+
|
|
119
|
+
@dataclass
|
|
120
|
+
class LayoutShortcuts:
|
|
121
|
+
"""Complete shortcut set for a specific layout."""
|
|
122
|
+
|
|
123
|
+
layout: KeyboardLayout
|
|
124
|
+
shortcuts: list[ShortcutAssignment] = field(default_factory=list)
|
|
125
|
+
scale_assignments: ScaleAssignments = field(default_factory=ScaleAssignments)
|
|
126
|
+
phrase_mnemonic: str = "" # e.g., "My Ship Floats Daily"
|
|
127
|
+
|
|
128
|
+
|
|
129
|
+
@dataclass
|
|
130
|
+
class FeatureManifest:
|
|
131
|
+
"""Complete manifest for keyboard shortcut generation."""
|
|
132
|
+
|
|
133
|
+
project_name: str
|
|
134
|
+
project_type: Literal["vscode-extension", "python-cli", "custom"] = "custom"
|
|
135
|
+
prefix: str = "ctrl+shift+e" # Base chord
|
|
136
|
+
categories: list[Category] = field(default_factory=list)
|
|
137
|
+
layouts: list[LayoutConfig] = field(default_factory=list)
|
|
138
|
+
|
|
139
|
+
def __post_init__(self):
|
|
140
|
+
if not self.layouts:
|
|
141
|
+
self.layouts = [
|
|
142
|
+
LayoutConfig(layout=KeyboardLayout.QWERTY),
|
|
143
|
+
LayoutConfig(layout=KeyboardLayout.DVORAK),
|
|
144
|
+
LayoutConfig(layout=KeyboardLayout.COLEMAK),
|
|
145
|
+
]
|
|
146
|
+
|
|
147
|
+
def all_features(self) -> list[Feature]:
|
|
148
|
+
"""Get all features across all categories."""
|
|
149
|
+
features = []
|
|
150
|
+
for category in self.categories:
|
|
151
|
+
features.extend(category.features)
|
|
152
|
+
return features
|
|
153
|
+
|
|
154
|
+
def features_by_tier(self, tier: FrequencyTier) -> list[Feature]:
|
|
155
|
+
"""Get features filtered by frequency tier."""
|
|
156
|
+
return [f for f in self.all_features() if f.frequency == tier]
|
|
157
|
+
|
|
158
|
+
|
|
159
|
+
@dataclass
|
|
160
|
+
class GeneratedShortcuts:
|
|
161
|
+
"""Complete output from keyboard shortcut generation."""
|
|
162
|
+
|
|
163
|
+
manifest: FeatureManifest
|
|
164
|
+
layouts: dict[KeyboardLayout, LayoutShortcuts] = field(default_factory=dict)
|
|
165
|
+
validation_passed: bool = True
|
|
166
|
+
conflicts: list[str] = field(default_factory=list)
|
|
167
|
+
warnings: list[str] = field(default_factory=list)
|
|
168
|
+
|
|
169
|
+
|
|
170
|
+
# Keyboard layout reference data
|
|
171
|
+
KEYBOARD_LAYOUTS = {
|
|
172
|
+
KeyboardLayout.QWERTY: {
|
|
173
|
+
"top_row": list("qwertyuiop"),
|
|
174
|
+
"home_row": list("asdfghjkl"),
|
|
175
|
+
"bottom_row": list("zxcvbnm"),
|
|
176
|
+
},
|
|
177
|
+
KeyboardLayout.DVORAK: {
|
|
178
|
+
"top_row": list("',.pyfgcrl"),
|
|
179
|
+
"home_row": list("aoeuidhtns"),
|
|
180
|
+
"bottom_row": list(";qjkxbmwvz"),
|
|
181
|
+
},
|
|
182
|
+
KeyboardLayout.COLEMAK: {
|
|
183
|
+
"top_row": list("qwfpgjluy;"),
|
|
184
|
+
"home_row": list("arstdhneio"),
|
|
185
|
+
"bottom_row": list("zxcvbkm,."),
|
|
186
|
+
},
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
# Reserved keys that should not be used (conflicts with OS/IDE)
|
|
190
|
+
RESERVED_KEYS = {
|
|
191
|
+
"global": ["q", "w", "e", "x"], # Commonly used by OS
|
|
192
|
+
"ide": ["n", "o", "s", "z"], # Commonly used by IDE (new, open, save, undo)
|
|
193
|
+
}
|