claude-mpm 5.4.55__py3-none-any.whl → 5.4.85__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.
- claude_mpm/VERSION +1 -1
- claude_mpm/agents/CLAUDE_MPM_FOUNDERS_OUTPUT_STYLE.md +405 -0
- claude_mpm/agents/CLAUDE_MPM_OUTPUT_STYLE.md +63 -241
- claude_mpm/agents/CLAUDE_MPM_TEACHER_OUTPUT_STYLE.md +109 -1925
- claude_mpm/agents/PM_INSTRUCTIONS.md +36 -9
- claude_mpm/cli/__init__.py +5 -1
- claude_mpm/cli/commands/agents.py +2 -4
- claude_mpm/cli/commands/agents_reconcile.py +197 -0
- claude_mpm/cli/commands/configure.py +620 -21
- claude_mpm/cli/commands/skills.py +166 -14
- claude_mpm/cli/executor.py +1 -0
- claude_mpm/cli/interactive/__init__.py +10 -0
- claude_mpm/cli/interactive/agent_wizard.py +30 -50
- claude_mpm/cli/interactive/questionary_styles.py +65 -0
- claude_mpm/cli/interactive/skill_selector.py +481 -0
- claude_mpm/cli/parsers/base_parser.py +5 -0
- claude_mpm/cli/startup.py +223 -388
- claude_mpm/constants.py +1 -0
- claude_mpm/core/claude_runner.py +2 -2
- claude_mpm/core/interactive_session.py +7 -7
- claude_mpm/core/output_style_manager.py +21 -13
- claude_mpm/core/unified_config.py +50 -8
- claude_mpm/core/unified_paths.py +30 -13
- claude_mpm/scripts/start_activity_logging.py +0 -0
- claude_mpm/services/agents/deployment/agent_template_builder.py +8 -0
- claude_mpm/services/agents/deployment/deployment_reconciler.py +577 -0
- claude_mpm/services/agents/deployment/startup_reconciliation.py +138 -0
- claude_mpm/services/agents/sources/git_source_sync_service.py +7 -4
- claude_mpm/services/agents/startup_sync.py +5 -2
- claude_mpm/services/pm_skills_deployer.py +4 -0
- claude_mpm/services/skills/git_skill_source_manager.py +24 -8
- claude_mpm/services/skills/selective_skill_deployer.py +82 -83
- claude_mpm/skills/bundled/collaboration/brainstorming/SKILL.md +79 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/SKILL.md +178 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/agent-prompts.md +577 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/coordination-patterns.md +467 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/examples.md +537 -0
- claude_mpm/skills/bundled/collaboration/dispatching-parallel-agents/references/troubleshooting.md +730 -0
- claude_mpm/skills/bundled/collaboration/git-worktrees.md +317 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/SKILL.md +112 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/code-reviewer-template.md +146 -0
- claude_mpm/skills/bundled/collaboration/requesting-code-review/references/review-examples.md +412 -0
- claude_mpm/skills/bundled/collaboration/stacked-prs.md +251 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/SKILL.md +81 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/best-practices.md +362 -0
- claude_mpm/skills/bundled/collaboration/writing-plans/references/plan-structure-templates.md +312 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/SKILL.md +152 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/advanced-techniques.md +668 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/examples.md +587 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/integration.md +438 -0
- claude_mpm/skills/bundled/debugging/root-cause-tracing/references/tracing-techniques.md +391 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/CREATION-LOG.md +119 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/SKILL.md +148 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/anti-patterns.md +483 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/examples.md +452 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/troubleshooting.md +449 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/references/workflow.md +411 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-academic.md +14 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-1.md +58 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-2.md +68 -0
- claude_mpm/skills/bundled/debugging/systematic-debugging/test-pressure-3.md +69 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/SKILL.md +131 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/gate-function.md +325 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/integration-and-workflows.md +490 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/red-flags-and-failures.md +425 -0
- claude_mpm/skills/bundled/debugging/verification-before-completion/references/verification-patterns.md +499 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/INTEGRATION.md +611 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/README.md +596 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/SKILL.md +260 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/examples/nextjs-env-structure.md +315 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/frameworks.md +436 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/security.md +433 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/synchronization.md +452 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/troubleshooting.md +404 -0
- claude_mpm/skills/bundled/infrastructure/env-manager/references/validation.md +420 -0
- claude_mpm/skills/bundled/main/artifacts-builder/SKILL.md +86 -0
- claude_mpm/skills/bundled/main/internal-comms/SKILL.md +43 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/3p-updates.md +47 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/company-newsletter.md +65 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/faq-answers.md +30 -0
- claude_mpm/skills/bundled/main/internal-comms/examples/general-comms.md +16 -0
- claude_mpm/skills/bundled/main/mcp-builder/SKILL.md +160 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/design_principles.md +412 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/evaluation.md +602 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/mcp_best_practices.md +915 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/node_mcp_server.md +916 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/python_mcp_server.md +752 -0
- claude_mpm/skills/bundled/main/mcp-builder/reference/workflow.md +1237 -0
- claude_mpm/skills/bundled/main/skill-creator/SKILL.md +189 -0
- claude_mpm/skills/bundled/main/skill-creator/references/best-practices.md +500 -0
- claude_mpm/skills/bundled/main/skill-creator/references/creation-workflow.md +464 -0
- claude_mpm/skills/bundled/main/skill-creator/references/examples.md +619 -0
- claude_mpm/skills/bundled/main/skill-creator/references/progressive-disclosure.md +437 -0
- claude_mpm/skills/bundled/main/skill-creator/references/skill-structure.md +231 -0
- claude_mpm/skills/bundled/php/espocrm-development/SKILL.md +170 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/architecture.md +602 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/common-tasks.md +821 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/development-workflow.md +742 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/frontend-customization.md +726 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/hooks-and-services.md +764 -0
- claude_mpm/skills/bundled/php/espocrm-development/references/testing-debugging.md +831 -0
- claude_mpm/skills/bundled/pm/pm-bug-reporting/pm-bug-reporting.md +248 -0
- claude_mpm/skills/bundled/pm/pm-delegation-patterns/SKILL.md +167 -0
- claude_mpm/skills/bundled/pm/pm-git-file-tracking/SKILL.md +113 -0
- claude_mpm/skills/bundled/pm/pm-pr-workflow/SKILL.md +124 -0
- claude_mpm/skills/bundled/pm/pm-teaching-mode/SKILL.md +657 -0
- claude_mpm/skills/bundled/pm/pm-ticketing-integration/SKILL.md +154 -0
- claude_mpm/skills/bundled/pm/pm-verification-protocols/SKILL.md +198 -0
- claude_mpm/skills/bundled/react/flexlayout-react.md +742 -0
- claude_mpm/skills/bundled/rust/desktop-applications/SKILL.md +226 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/architecture-patterns.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/native-gui-frameworks.md +901 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/platform-integration.md +775 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/state-management.md +937 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/tauri-framework.md +770 -0
- claude_mpm/skills/bundled/rust/desktop-applications/references/testing-deployment.md +961 -0
- claude_mpm/skills/bundled/tauri/tauri-async-patterns.md +495 -0
- claude_mpm/skills/bundled/tauri/tauri-build-deploy.md +599 -0
- claude_mpm/skills/bundled/tauri/tauri-command-patterns.md +535 -0
- claude_mpm/skills/bundled/tauri/tauri-error-handling.md +613 -0
- claude_mpm/skills/bundled/tauri/tauri-event-system.md +648 -0
- claude_mpm/skills/bundled/tauri/tauri-file-system.md +673 -0
- claude_mpm/skills/bundled/tauri/tauri-frontend-integration.md +767 -0
- claude_mpm/skills/bundled/tauri/tauri-performance.md +669 -0
- claude_mpm/skills/bundled/tauri/tauri-state-management.md +573 -0
- claude_mpm/skills/bundled/tauri/tauri-testing.md +384 -0
- claude_mpm/skills/bundled/tauri/tauri-window-management.md +628 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/SKILL.md +119 -0
- claude_mpm/skills/bundled/testing/condition-based-waiting/references/patterns-and-implementation.md +253 -0
- claude_mpm/skills/bundled/testing/test-driven-development/SKILL.md +145 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/anti-patterns.md +543 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/examples.md +741 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/integration.md +470 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/philosophy.md +458 -0
- claude_mpm/skills/bundled/testing/test-driven-development/references/workflow.md +639 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/SKILL.md +458 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/examples/example-inspection-report.md +411 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/assertion-quality.md +317 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/inspection-checklist.md +270 -0
- claude_mpm/skills/bundled/testing/test-quality-inspector/references/red-flags.md +436 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/SKILL.md +140 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/completeness-anti-patterns.md +572 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/core-anti-patterns.md +411 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/detection-guide.md +569 -0
- claude_mpm/skills/bundled/testing/testing-anti-patterns/references/tdd-connection.md +695 -0
- claude_mpm/skills/bundled/testing/webapp-testing/SKILL.md +184 -0
- claude_mpm/skills/bundled/testing/webapp-testing/decision-tree.md +459 -0
- claude_mpm/skills/bundled/testing/webapp-testing/playwright-patterns.md +479 -0
- claude_mpm/skills/bundled/testing/webapp-testing/reconnaissance-pattern.md +687 -0
- claude_mpm/skills/bundled/testing/webapp-testing/server-management.md +758 -0
- claude_mpm/skills/bundled/testing/webapp-testing/troubleshooting.md +868 -0
- claude_mpm/utils/agent_dependency_loader.py +103 -4
- claude_mpm/utils/robust_installer.py +45 -24
- {claude_mpm-5.4.55.dist-info → claude_mpm-5.4.85.dist-info}/METADATA +47 -23
- {claude_mpm-5.4.55.dist-info → claude_mpm-5.4.85.dist-info}/RECORD +159 -47
- claude_mpm/hooks/claude_hooks/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/correlation_manager.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/event_handlers.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/hook_handler.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/installer.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/memory_integration.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/response_tracking.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/__pycache__/tool_analysis.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/__init__.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/connection_manager_http.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/duplicate_detector.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/state_manager.cpython-311.pyc +0 -0
- claude_mpm/hooks/claude_hooks/services/__pycache__/subagent_processor.cpython-311.pyc +0 -0
- {claude_mpm-5.4.55.dist-info → claude_mpm-5.4.85.dist-info}/WHEEL +0 -0
- {claude_mpm-5.4.55.dist-info → claude_mpm-5.4.85.dist-info}/entry_points.txt +0 -0
- {claude_mpm-5.4.55.dist-info → claude_mpm-5.4.85.dist-info}/licenses/LICENSE +0 -0
- {claude_mpm-5.4.55.dist-info → claude_mpm-5.4.85.dist-info}/licenses/LICENSE-FAQ.md +0 -0
- {claude_mpm-5.4.55.dist-info → claude_mpm-5.4.85.dist-info}/top_level.txt +0 -0
|
@@ -0,0 +1,937 @@
|
|
|
1
|
+
# State Management
|
|
2
|
+
|
|
3
|
+
Comprehensive guide to managing application state in Rust desktop applications, from simple local state to complex async operations and multi-window synchronization.
|
|
4
|
+
|
|
5
|
+
## State Management Strategies
|
|
6
|
+
|
|
7
|
+
### Local State (Single Component)
|
|
8
|
+
|
|
9
|
+
Simplest form - state lives within a single component or module.
|
|
10
|
+
|
|
11
|
+
```rust
|
|
12
|
+
// egui example
|
|
13
|
+
struct MyApp {
|
|
14
|
+
counter: i32,
|
|
15
|
+
text: String,
|
|
16
|
+
selected: Option<usize>,
|
|
17
|
+
}
|
|
18
|
+
|
|
19
|
+
impl eframe::App for MyApp {
|
|
20
|
+
fn update(&mut self, ctx: &egui::Context, _frame: &mut eframe::Frame) {
|
|
21
|
+
egui::CentralPanel::default().show(ctx, |ui| {
|
|
22
|
+
ui.label(format!("Counter: {}", self.counter));
|
|
23
|
+
|
|
24
|
+
if ui.button("Increment").clicked() {
|
|
25
|
+
self.counter += 1;
|
|
26
|
+
}
|
|
27
|
+
|
|
28
|
+
ui.text_edit_singleline(&mut self.text);
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
```
|
|
33
|
+
|
|
34
|
+
**Tauri example:**
|
|
35
|
+
```rust
|
|
36
|
+
use std::sync::Mutex;
|
|
37
|
+
|
|
38
|
+
struct AppState {
|
|
39
|
+
counter: Mutex<i32>,
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
#[tauri::command]
|
|
43
|
+
fn increment(state: tauri::State<AppState>) -> i32 {
|
|
44
|
+
let mut counter = state.counter.lock().unwrap();
|
|
45
|
+
*counter += 1;
|
|
46
|
+
*counter
|
|
47
|
+
}
|
|
48
|
+
|
|
49
|
+
#[tauri::command]
|
|
50
|
+
fn get_counter(state: tauri::State<AppState>) -> i32 {
|
|
51
|
+
*state.counter.lock().unwrap()
|
|
52
|
+
}
|
|
53
|
+
|
|
54
|
+
fn main() {
|
|
55
|
+
tauri::Builder::default()
|
|
56
|
+
.manage(AppState {
|
|
57
|
+
counter: Mutex::new(0),
|
|
58
|
+
})
|
|
59
|
+
.invoke_handler(tauri::generate_handler![increment, get_counter])
|
|
60
|
+
.run(tauri::generate_context!())
|
|
61
|
+
.expect("error while running tauri application");
|
|
62
|
+
}
|
|
63
|
+
```
|
|
64
|
+
|
|
65
|
+
### Shared State with Arc<Mutex<T>>
|
|
66
|
+
|
|
67
|
+
Thread-safe shared state for multi-threaded applications.
|
|
68
|
+
|
|
69
|
+
```rust
|
|
70
|
+
use std::sync::{Arc, Mutex};
|
|
71
|
+
|
|
72
|
+
#[derive(Clone)]
|
|
73
|
+
struct SharedState {
|
|
74
|
+
data: Arc<Mutex<AppData>>,
|
|
75
|
+
}
|
|
76
|
+
|
|
77
|
+
struct AppData {
|
|
78
|
+
users: Vec<User>,
|
|
79
|
+
settings: Settings,
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
impl SharedState {
|
|
83
|
+
fn new() -> Self {
|
|
84
|
+
Self {
|
|
85
|
+
data: Arc::new(Mutex::new(AppData {
|
|
86
|
+
users: Vec::new(),
|
|
87
|
+
settings: Settings::default(),
|
|
88
|
+
})),
|
|
89
|
+
}
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
fn add_user(&self, user: User) {
|
|
93
|
+
let mut data = self.data.lock().unwrap();
|
|
94
|
+
data.users.push(user);
|
|
95
|
+
}
|
|
96
|
+
|
|
97
|
+
fn get_users(&self) -> Vec<User> {
|
|
98
|
+
let data = self.data.lock().unwrap();
|
|
99
|
+
data.users.clone()
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
|
|
103
|
+
// Tauri commands
|
|
104
|
+
#[tauri::command]
|
|
105
|
+
fn add_user(state: tauri::State<SharedState>, name: String, email: String) {
|
|
106
|
+
let user = User {
|
|
107
|
+
id: generate_id(),
|
|
108
|
+
name,
|
|
109
|
+
email,
|
|
110
|
+
};
|
|
111
|
+
state.add_user(user);
|
|
112
|
+
}
|
|
113
|
+
|
|
114
|
+
#[tauri::command]
|
|
115
|
+
fn get_users(state: tauri::State<SharedState>) -> Vec<User> {
|
|
116
|
+
state.get_users()
|
|
117
|
+
}
|
|
118
|
+
```
|
|
119
|
+
|
|
120
|
+
### RwLock for Read-Heavy Workloads
|
|
121
|
+
|
|
122
|
+
Better performance when reads outnumber writes.
|
|
123
|
+
|
|
124
|
+
```rust
|
|
125
|
+
use std::sync::{Arc, RwLock};
|
|
126
|
+
|
|
127
|
+
struct AppState {
|
|
128
|
+
cache: Arc<RwLock<HashMap<String, String>>>,
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
impl AppState {
|
|
132
|
+
fn new() -> Self {
|
|
133
|
+
Self {
|
|
134
|
+
cache: Arc::new(RwLock::new(HashMap::new())),
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
|
|
138
|
+
// Multiple readers can access simultaneously
|
|
139
|
+
fn get(&self, key: &str) -> Option<String> {
|
|
140
|
+
let cache = self.cache.read().unwrap();
|
|
141
|
+
cache.get(key).cloned()
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
// Exclusive write access
|
|
145
|
+
fn set(&self, key: String, value: String) {
|
|
146
|
+
let mut cache = self.cache.write().unwrap();
|
|
147
|
+
cache.insert(key, value);
|
|
148
|
+
}
|
|
149
|
+
|
|
150
|
+
// Bulk read operation
|
|
151
|
+
fn get_all(&self) -> HashMap<String, String> {
|
|
152
|
+
let cache = self.cache.read().unwrap();
|
|
153
|
+
cache.clone()
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
#[tauri::command]
|
|
158
|
+
fn cache_get(state: tauri::State<AppState>, key: String) -> Option<String> {
|
|
159
|
+
state.get(&key)
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
#[tauri::command]
|
|
163
|
+
fn cache_set(state: tauri::State<AppState>, key: String, value: String) {
|
|
164
|
+
state.set(key, value);
|
|
165
|
+
}
|
|
166
|
+
```
|
|
167
|
+
|
|
168
|
+
## Async Runtime Integration
|
|
169
|
+
|
|
170
|
+
### Tokio Integration with Tauri
|
|
171
|
+
|
|
172
|
+
```rust
|
|
173
|
+
use tokio::sync::RwLock as TokioRwLock;
|
|
174
|
+
use std::sync::Arc;
|
|
175
|
+
|
|
176
|
+
struct AsyncState {
|
|
177
|
+
data: Arc<TokioRwLock<AppData>>,
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
#[derive(Clone)]
|
|
181
|
+
struct AppData {
|
|
182
|
+
items: Vec<Item>,
|
|
183
|
+
loading: bool,
|
|
184
|
+
}
|
|
185
|
+
|
|
186
|
+
impl AsyncState {
|
|
187
|
+
fn new() -> Self {
|
|
188
|
+
Self {
|
|
189
|
+
data: Arc::new(TokioRwLock::new(AppData {
|
|
190
|
+
items: Vec::new(),
|
|
191
|
+
loading: false,
|
|
192
|
+
})),
|
|
193
|
+
}
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
async fn fetch_items(&self) -> Result<Vec<Item>, String> {
|
|
197
|
+
// Set loading state
|
|
198
|
+
{
|
|
199
|
+
let mut data = self.data.write().await;
|
|
200
|
+
data.loading = true;
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
// Perform async operation
|
|
204
|
+
let items = fetch_from_api().await.map_err(|e| e.to_string())?;
|
|
205
|
+
|
|
206
|
+
// Update state
|
|
207
|
+
{
|
|
208
|
+
let mut data = self.data.write().await;
|
|
209
|
+
data.items = items.clone();
|
|
210
|
+
data.loading = false;
|
|
211
|
+
}
|
|
212
|
+
|
|
213
|
+
Ok(items)
|
|
214
|
+
}
|
|
215
|
+
|
|
216
|
+
async fn get_items(&self) -> Vec<Item> {
|
|
217
|
+
let data = self.data.read().await;
|
|
218
|
+
data.items.clone()
|
|
219
|
+
}
|
|
220
|
+
|
|
221
|
+
async fn is_loading(&self) -> bool {
|
|
222
|
+
let data = self.data.read().await;
|
|
223
|
+
data.loading
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
|
|
227
|
+
#[tauri::command]
|
|
228
|
+
async fn fetch_items(state: tauri::State<'_, AsyncState>) -> Result<Vec<Item>, String> {
|
|
229
|
+
state.fetch_items().await
|
|
230
|
+
}
|
|
231
|
+
|
|
232
|
+
#[tauri::command]
|
|
233
|
+
async fn get_items(state: tauri::State<'_, AsyncState>) -> Vec<Item> {
|
|
234
|
+
state.get_items().await
|
|
235
|
+
}
|
|
236
|
+
|
|
237
|
+
async fn fetch_from_api() -> Result<Vec<Item>, Box<dyn std::error::Error>> {
|
|
238
|
+
use reqwest;
|
|
239
|
+
|
|
240
|
+
let response = reqwest::get("https://api.example.com/items")
|
|
241
|
+
.await?
|
|
242
|
+
.json::<Vec<Item>>()
|
|
243
|
+
.await?;
|
|
244
|
+
|
|
245
|
+
Ok(response)
|
|
246
|
+
}
|
|
247
|
+
```
|
|
248
|
+
|
|
249
|
+
### Background Tasks and Channels
|
|
250
|
+
|
|
251
|
+
```rust
|
|
252
|
+
use tokio::sync::mpsc;
|
|
253
|
+
use tokio::time::{interval, Duration};
|
|
254
|
+
|
|
255
|
+
struct BackgroundWorker {
|
|
256
|
+
tx: mpsc::UnboundedSender<WorkerMessage>,
|
|
257
|
+
}
|
|
258
|
+
|
|
259
|
+
enum WorkerMessage {
|
|
260
|
+
ProcessData(String),
|
|
261
|
+
Stop,
|
|
262
|
+
}
|
|
263
|
+
|
|
264
|
+
impl BackgroundWorker {
|
|
265
|
+
fn new(app_handle: tauri::AppHandle) -> Self {
|
|
266
|
+
let (tx, mut rx) = mpsc::unbounded_channel();
|
|
267
|
+
|
|
268
|
+
tokio::spawn(async move {
|
|
269
|
+
let mut ticker = interval(Duration::from_secs(1));
|
|
270
|
+
|
|
271
|
+
loop {
|
|
272
|
+
tokio::select! {
|
|
273
|
+
_ = ticker.tick() => {
|
|
274
|
+
// Periodic task
|
|
275
|
+
let _ = app_handle.emit("tick", "Periodic update");
|
|
276
|
+
}
|
|
277
|
+
Some(msg) = rx.recv() => {
|
|
278
|
+
match msg {
|
|
279
|
+
WorkerMessage::ProcessData(data) => {
|
|
280
|
+
// Process data
|
|
281
|
+
println!("Processing: {}", data);
|
|
282
|
+
let _ = app_handle.emit("data-processed", data);
|
|
283
|
+
}
|
|
284
|
+
WorkerMessage::Stop => {
|
|
285
|
+
println!("Stopping worker");
|
|
286
|
+
break;
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
}
|
|
292
|
+
});
|
|
293
|
+
|
|
294
|
+
Self { tx }
|
|
295
|
+
}
|
|
296
|
+
|
|
297
|
+
fn send(&self, msg: WorkerMessage) {
|
|
298
|
+
let _ = self.tx.send(msg);
|
|
299
|
+
}
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
#[tauri::command]
|
|
303
|
+
fn process_data(worker: tauri::State<BackgroundWorker>, data: String) {
|
|
304
|
+
worker.send(WorkerMessage::ProcessData(data));
|
|
305
|
+
}
|
|
306
|
+
|
|
307
|
+
fn main() {
|
|
308
|
+
tauri::Builder::default()
|
|
309
|
+
.setup(|app| {
|
|
310
|
+
let worker = BackgroundWorker::new(app.handle());
|
|
311
|
+
app.manage(worker);
|
|
312
|
+
Ok(())
|
|
313
|
+
})
|
|
314
|
+
.invoke_handler(tauri::generate_handler![process_data])
|
|
315
|
+
.run(tauri::generate_context!())
|
|
316
|
+
.expect("error while running tauri application");
|
|
317
|
+
}
|
|
318
|
+
```
|
|
319
|
+
|
|
320
|
+
## Message Passing Patterns
|
|
321
|
+
|
|
322
|
+
### Command-Query Separation
|
|
323
|
+
|
|
324
|
+
```rust
|
|
325
|
+
use tokio::sync::mpsc;
|
|
326
|
+
|
|
327
|
+
// Commands (modify state)
|
|
328
|
+
enum Command {
|
|
329
|
+
AddUser { name: String, email: String },
|
|
330
|
+
RemoveUser { id: u64 },
|
|
331
|
+
UpdateSettings { key: String, value: String },
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
// Queries (read state)
|
|
335
|
+
enum Query {
|
|
336
|
+
GetUser { id: u64, response: oneshot::Sender<Option<User>> },
|
|
337
|
+
GetAllUsers { response: oneshot::Sender<Vec<User>> },
|
|
338
|
+
GetSettings { response: oneshot::Sender<Settings> },
|
|
339
|
+
}
|
|
340
|
+
|
|
341
|
+
struct StateManager {
|
|
342
|
+
command_tx: mpsc::UnboundedSender<Command>,
|
|
343
|
+
query_tx: mpsc::UnboundedSender<Query>,
|
|
344
|
+
}
|
|
345
|
+
|
|
346
|
+
impl StateManager {
|
|
347
|
+
fn new() -> Self {
|
|
348
|
+
let (command_tx, mut command_rx) = mpsc::unbounded_channel();
|
|
349
|
+
let (query_tx, mut query_rx) = mpsc::unbounded_channel();
|
|
350
|
+
|
|
351
|
+
// State lives in this task
|
|
352
|
+
tokio::spawn(async move {
|
|
353
|
+
let mut state = AppState::new();
|
|
354
|
+
|
|
355
|
+
loop {
|
|
356
|
+
tokio::select! {
|
|
357
|
+
Some(cmd) = command_rx.recv() => {
|
|
358
|
+
match cmd {
|
|
359
|
+
Command::AddUser { name, email } => {
|
|
360
|
+
state.add_user(User { id: generate_id(), name, email });
|
|
361
|
+
}
|
|
362
|
+
Command::RemoveUser { id } => {
|
|
363
|
+
state.remove_user(id);
|
|
364
|
+
}
|
|
365
|
+
Command::UpdateSettings { key, value } => {
|
|
366
|
+
state.update_setting(key, value);
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
}
|
|
370
|
+
Some(query) = query_rx.recv() => {
|
|
371
|
+
match query {
|
|
372
|
+
Query::GetUser { id, response } => {
|
|
373
|
+
let _ = response.send(state.get_user(id));
|
|
374
|
+
}
|
|
375
|
+
Query::GetAllUsers { response } => {
|
|
376
|
+
let _ = response.send(state.get_all_users());
|
|
377
|
+
}
|
|
378
|
+
Query::GetSettings { response } => {
|
|
379
|
+
let _ = response.send(state.get_settings());
|
|
380
|
+
}
|
|
381
|
+
}
|
|
382
|
+
}
|
|
383
|
+
}
|
|
384
|
+
}
|
|
385
|
+
});
|
|
386
|
+
|
|
387
|
+
Self { command_tx, query_tx }
|
|
388
|
+
}
|
|
389
|
+
|
|
390
|
+
fn send_command(&self, cmd: Command) {
|
|
391
|
+
let _ = self.command_tx.send(cmd);
|
|
392
|
+
}
|
|
393
|
+
|
|
394
|
+
async fn query_user(&self, id: u64) -> Option<User> {
|
|
395
|
+
let (tx, rx) = oneshot::channel();
|
|
396
|
+
let _ = self.query_tx.send(Query::GetUser { id, response: tx });
|
|
397
|
+
rx.await.unwrap()
|
|
398
|
+
}
|
|
399
|
+
|
|
400
|
+
async fn query_all_users(&self) -> Vec<User> {
|
|
401
|
+
let (tx, rx) = oneshot::channel();
|
|
402
|
+
let _ = self.query_tx.send(Query::GetAllUsers { response: tx });
|
|
403
|
+
rx.await.unwrap()
|
|
404
|
+
}
|
|
405
|
+
}
|
|
406
|
+
|
|
407
|
+
// Tauri commands
|
|
408
|
+
#[tauri::command]
|
|
409
|
+
fn add_user(manager: tauri::State<StateManager>, name: String, email: String) {
|
|
410
|
+
manager.send_command(Command::AddUser { name, email });
|
|
411
|
+
}
|
|
412
|
+
|
|
413
|
+
#[tauri::command]
|
|
414
|
+
async fn get_user(manager: tauri::State<'_, StateManager>, id: u64) -> Option<User> {
|
|
415
|
+
manager.query_user(id).await
|
|
416
|
+
}
|
|
417
|
+
```
|
|
418
|
+
|
|
419
|
+
### Actor Pattern
|
|
420
|
+
|
|
421
|
+
```rust
|
|
422
|
+
use tokio::sync::mpsc;
|
|
423
|
+
|
|
424
|
+
trait Actor {
|
|
425
|
+
type Message;
|
|
426
|
+
|
|
427
|
+
fn handle(&mut self, msg: Self::Message);
|
|
428
|
+
}
|
|
429
|
+
|
|
430
|
+
struct ActorHandle<M> {
|
|
431
|
+
tx: mpsc::UnboundedSender<M>,
|
|
432
|
+
}
|
|
433
|
+
|
|
434
|
+
impl<M: Send + 'static> ActorHandle<M> {
|
|
435
|
+
fn new<A>(mut actor: A) -> Self
|
|
436
|
+
where
|
|
437
|
+
A: Actor<Message = M> + Send + 'static,
|
|
438
|
+
{
|
|
439
|
+
let (tx, mut rx) = mpsc::unbounded_channel();
|
|
440
|
+
|
|
441
|
+
tokio::spawn(async move {
|
|
442
|
+
while let Some(msg) = rx.recv().await {
|
|
443
|
+
actor.handle(msg);
|
|
444
|
+
}
|
|
445
|
+
});
|
|
446
|
+
|
|
447
|
+
Self { tx }
|
|
448
|
+
}
|
|
449
|
+
|
|
450
|
+
fn send(&self, msg: M) {
|
|
451
|
+
let _ = self.tx.send(msg);
|
|
452
|
+
}
|
|
453
|
+
}
|
|
454
|
+
|
|
455
|
+
// Example actor
|
|
456
|
+
struct UserActor {
|
|
457
|
+
users: HashMap<u64, User>,
|
|
458
|
+
}
|
|
459
|
+
|
|
460
|
+
enum UserMessage {
|
|
461
|
+
Add(User),
|
|
462
|
+
Remove(u64),
|
|
463
|
+
Get { id: u64, response: oneshot::Sender<Option<User>> },
|
|
464
|
+
}
|
|
465
|
+
|
|
466
|
+
impl Actor for UserActor {
|
|
467
|
+
type Message = UserMessage;
|
|
468
|
+
|
|
469
|
+
fn handle(&mut self, msg: Self::Message) {
|
|
470
|
+
match msg {
|
|
471
|
+
UserMessage::Add(user) => {
|
|
472
|
+
self.users.insert(user.id, user);
|
|
473
|
+
}
|
|
474
|
+
UserMessage::Remove(id) => {
|
|
475
|
+
self.users.remove(&id);
|
|
476
|
+
}
|
|
477
|
+
UserMessage::Get { id, response } => {
|
|
478
|
+
let user = self.users.get(&id).cloned();
|
|
479
|
+
let _ = response.send(user);
|
|
480
|
+
}
|
|
481
|
+
}
|
|
482
|
+
}
|
|
483
|
+
}
|
|
484
|
+
|
|
485
|
+
// Usage
|
|
486
|
+
fn setup_actors() -> ActorHandle<UserMessage> {
|
|
487
|
+
let actor = UserActor {
|
|
488
|
+
users: HashMap::new(),
|
|
489
|
+
};
|
|
490
|
+
ActorHandle::new(actor)
|
|
491
|
+
}
|
|
492
|
+
```
|
|
493
|
+
|
|
494
|
+
## Reactive State Patterns
|
|
495
|
+
|
|
496
|
+
### Observable State with Signals
|
|
497
|
+
|
|
498
|
+
```rust
|
|
499
|
+
use std::sync::{Arc, Mutex};
|
|
500
|
+
use std::collections::HashMap;
|
|
501
|
+
|
|
502
|
+
type Listener<T> = Box<dyn Fn(&T) + Send + Sync>;
|
|
503
|
+
|
|
504
|
+
struct Signal<T: Clone> {
|
|
505
|
+
value: Arc<Mutex<T>>,
|
|
506
|
+
listeners: Arc<Mutex<Vec<Listener<T>>>>,
|
|
507
|
+
}
|
|
508
|
+
|
|
509
|
+
impl<T: Clone + Send + Sync + 'static> Signal<T> {
|
|
510
|
+
fn new(initial: T) -> Self {
|
|
511
|
+
Self {
|
|
512
|
+
value: Arc::new(Mutex::new(initial)),
|
|
513
|
+
listeners: Arc::new(Mutex::new(Vec::new())),
|
|
514
|
+
}
|
|
515
|
+
}
|
|
516
|
+
|
|
517
|
+
fn get(&self) -> T {
|
|
518
|
+
self.value.lock().unwrap().clone()
|
|
519
|
+
}
|
|
520
|
+
|
|
521
|
+
fn set(&self, new_value: T) {
|
|
522
|
+
{
|
|
523
|
+
let mut value = self.value.lock().unwrap();
|
|
524
|
+
*value = new_value.clone();
|
|
525
|
+
}
|
|
526
|
+
|
|
527
|
+
// Notify listeners
|
|
528
|
+
let listeners = self.listeners.lock().unwrap();
|
|
529
|
+
for listener in listeners.iter() {
|
|
530
|
+
listener(&new_value);
|
|
531
|
+
}
|
|
532
|
+
}
|
|
533
|
+
|
|
534
|
+
fn update<F>(&self, f: F)
|
|
535
|
+
where
|
|
536
|
+
F: FnOnce(&mut T),
|
|
537
|
+
{
|
|
538
|
+
let new_value = {
|
|
539
|
+
let mut value = self.value.lock().unwrap();
|
|
540
|
+
f(&mut value);
|
|
541
|
+
value.clone()
|
|
542
|
+
};
|
|
543
|
+
|
|
544
|
+
// Notify listeners
|
|
545
|
+
let listeners = self.listeners.lock().unwrap();
|
|
546
|
+
for listener in listeners.iter() {
|
|
547
|
+
listener(&new_value);
|
|
548
|
+
}
|
|
549
|
+
}
|
|
550
|
+
|
|
551
|
+
fn subscribe<F>(&self, listener: F)
|
|
552
|
+
where
|
|
553
|
+
F: Fn(&T) + Send + Sync + 'static,
|
|
554
|
+
{
|
|
555
|
+
let mut listeners = self.listeners.lock().unwrap();
|
|
556
|
+
listeners.push(Box::new(listener));
|
|
557
|
+
}
|
|
558
|
+
}
|
|
559
|
+
|
|
560
|
+
// Example usage
|
|
561
|
+
struct AppState {
|
|
562
|
+
counter: Signal<i32>,
|
|
563
|
+
username: Signal<String>,
|
|
564
|
+
}
|
|
565
|
+
|
|
566
|
+
impl AppState {
|
|
567
|
+
fn new() -> Self {
|
|
568
|
+
Self {
|
|
569
|
+
counter: Signal::new(0),
|
|
570
|
+
username: Signal::new(String::from("Guest")),
|
|
571
|
+
}
|
|
572
|
+
}
|
|
573
|
+
}
|
|
574
|
+
|
|
575
|
+
fn setup_state(app_handle: tauri::AppHandle) -> AppState {
|
|
576
|
+
let state = AppState::new();
|
|
577
|
+
|
|
578
|
+
// Subscribe to changes
|
|
579
|
+
let handle = app_handle.clone();
|
|
580
|
+
state.counter.subscribe(move |value| {
|
|
581
|
+
let _ = handle.emit("counter-changed", value);
|
|
582
|
+
});
|
|
583
|
+
|
|
584
|
+
let handle = app_handle.clone();
|
|
585
|
+
state.username.subscribe(move |value| {
|
|
586
|
+
let _ = handle.emit("username-changed", value);
|
|
587
|
+
});
|
|
588
|
+
|
|
589
|
+
state
|
|
590
|
+
}
|
|
591
|
+
|
|
592
|
+
#[tauri::command]
|
|
593
|
+
fn increment_counter(state: tauri::State<AppState>) {
|
|
594
|
+
state.counter.update(|c| *c += 1);
|
|
595
|
+
}
|
|
596
|
+
|
|
597
|
+
#[tauri::command]
|
|
598
|
+
fn set_username(state: tauri::State<AppState>, name: String) {
|
|
599
|
+
state.username.set(name);
|
|
600
|
+
}
|
|
601
|
+
|
|
602
|
+
#[tauri::command]
|
|
603
|
+
fn get_counter(state: tauri::State<AppState>) -> i32 {
|
|
604
|
+
state.counter.get()
|
|
605
|
+
}
|
|
606
|
+
```
|
|
607
|
+
|
|
608
|
+
### Computed Values
|
|
609
|
+
|
|
610
|
+
```rust
|
|
611
|
+
struct Computed<T, F>
|
|
612
|
+
where
|
|
613
|
+
T: Clone,
|
|
614
|
+
F: Fn() -> T,
|
|
615
|
+
{
|
|
616
|
+
compute: F,
|
|
617
|
+
cached: Arc<Mutex<Option<T>>>,
|
|
618
|
+
}
|
|
619
|
+
|
|
620
|
+
impl<T: Clone, F: Fn() -> T> Computed<T, F> {
|
|
621
|
+
fn new(compute: F) -> Self {
|
|
622
|
+
Self {
|
|
623
|
+
compute,
|
|
624
|
+
cached: Arc::new(Mutex::new(None)),
|
|
625
|
+
}
|
|
626
|
+
}
|
|
627
|
+
|
|
628
|
+
fn get(&self) -> T {
|
|
629
|
+
let mut cached = self.cached.lock().unwrap();
|
|
630
|
+
|
|
631
|
+
if let Some(value) = cached.as_ref() {
|
|
632
|
+
value.clone()
|
|
633
|
+
} else {
|
|
634
|
+
let value = (self.compute)();
|
|
635
|
+
*cached = Some(value.clone());
|
|
636
|
+
value
|
|
637
|
+
}
|
|
638
|
+
}
|
|
639
|
+
|
|
640
|
+
fn invalidate(&self) {
|
|
641
|
+
let mut cached = self.cached.lock().unwrap();
|
|
642
|
+
*cached = None;
|
|
643
|
+
}
|
|
644
|
+
}
|
|
645
|
+
|
|
646
|
+
// Example
|
|
647
|
+
struct TodoState {
|
|
648
|
+
todos: Signal<Vec<Todo>>,
|
|
649
|
+
completed_count: Computed<usize, Box<dyn Fn() -> usize + Send + Sync>>,
|
|
650
|
+
}
|
|
651
|
+
|
|
652
|
+
impl TodoState {
|
|
653
|
+
fn new() -> Self {
|
|
654
|
+
let todos = Signal::new(Vec::new());
|
|
655
|
+
let todos_clone = todos.clone();
|
|
656
|
+
|
|
657
|
+
let completed_count = Computed::new(Box::new(move || {
|
|
658
|
+
todos_clone
|
|
659
|
+
.get()
|
|
660
|
+
.iter()
|
|
661
|
+
.filter(|t| t.completed)
|
|
662
|
+
.count()
|
|
663
|
+
}));
|
|
664
|
+
|
|
665
|
+
Self {
|
|
666
|
+
todos,
|
|
667
|
+
completed_count,
|
|
668
|
+
}
|
|
669
|
+
}
|
|
670
|
+
|
|
671
|
+
fn add_todo(&self, todo: Todo) {
|
|
672
|
+
self.todos.update(|todos| todos.push(todo));
|
|
673
|
+
self.completed_count.invalidate();
|
|
674
|
+
}
|
|
675
|
+
|
|
676
|
+
fn toggle_todo(&self, id: u64) {
|
|
677
|
+
self.todos.update(|todos| {
|
|
678
|
+
if let Some(todo) = todos.iter_mut().find(|t| t.id == id) {
|
|
679
|
+
todo.completed = !todo.completed;
|
|
680
|
+
}
|
|
681
|
+
});
|
|
682
|
+
self.completed_count.invalidate();
|
|
683
|
+
}
|
|
684
|
+
|
|
685
|
+
fn get_completed_count(&self) -> usize {
|
|
686
|
+
self.completed_count.get()
|
|
687
|
+
}
|
|
688
|
+
}
|
|
689
|
+
```
|
|
690
|
+
|
|
691
|
+
## Persistence
|
|
692
|
+
|
|
693
|
+
### File-Based Persistence
|
|
694
|
+
|
|
695
|
+
```rust
|
|
696
|
+
use serde::{Deserialize, Serialize};
|
|
697
|
+
use std::path::PathBuf;
|
|
698
|
+
|
|
699
|
+
#[derive(Serialize, Deserialize, Clone)]
|
|
700
|
+
struct AppSettings {
|
|
701
|
+
theme: String,
|
|
702
|
+
language: String,
|
|
703
|
+
window_size: (u32, u32),
|
|
704
|
+
}
|
|
705
|
+
|
|
706
|
+
struct PersistedState {
|
|
707
|
+
settings: Signal<AppSettings>,
|
|
708
|
+
config_path: PathBuf,
|
|
709
|
+
}
|
|
710
|
+
|
|
711
|
+
impl PersistedState {
|
|
712
|
+
fn new(config_path: PathBuf) -> Self {
|
|
713
|
+
let settings = Self::load_settings(&config_path)
|
|
714
|
+
.unwrap_or_else(|_| AppSettings::default());
|
|
715
|
+
|
|
716
|
+
let state = Self {
|
|
717
|
+
settings: Signal::new(settings),
|
|
718
|
+
config_path,
|
|
719
|
+
};
|
|
720
|
+
|
|
721
|
+
// Auto-save on changes
|
|
722
|
+
let config_path = state.config_path.clone();
|
|
723
|
+
state.settings.subscribe(move |settings| {
|
|
724
|
+
let _ = Self::save_settings(&config_path, settings);
|
|
725
|
+
});
|
|
726
|
+
|
|
727
|
+
state
|
|
728
|
+
}
|
|
729
|
+
|
|
730
|
+
fn load_settings(path: &PathBuf) -> Result<AppSettings, Box<dyn std::error::Error>> {
|
|
731
|
+
let content = std::fs::read_to_string(path)?;
|
|
732
|
+
let settings = serde_json::from_str(&content)?;
|
|
733
|
+
Ok(settings)
|
|
734
|
+
}
|
|
735
|
+
|
|
736
|
+
fn save_settings(path: &PathBuf, settings: &AppSettings) -> Result<(), Box<dyn std::error::Error>> {
|
|
737
|
+
let content = serde_json::to_string_pretty(settings)?;
|
|
738
|
+
std::fs::write(path, content)?;
|
|
739
|
+
Ok(())
|
|
740
|
+
}
|
|
741
|
+
|
|
742
|
+
fn update_settings<F>(&self, f: F)
|
|
743
|
+
where
|
|
744
|
+
F: FnOnce(&mut AppSettings),
|
|
745
|
+
{
|
|
746
|
+
self.settings.update(f);
|
|
747
|
+
}
|
|
748
|
+
}
|
|
749
|
+
|
|
750
|
+
#[tauri::command]
|
|
751
|
+
fn update_theme(state: tauri::State<PersistedState>, theme: String) {
|
|
752
|
+
state.update_settings(|s| s.theme = theme);
|
|
753
|
+
}
|
|
754
|
+
|
|
755
|
+
#[tauri::command]
|
|
756
|
+
fn get_settings(state: tauri::State<PersistedState>) -> AppSettings {
|
|
757
|
+
state.settings.get()
|
|
758
|
+
}
|
|
759
|
+
```
|
|
760
|
+
|
|
761
|
+
### Database Integration with sqlx
|
|
762
|
+
|
|
763
|
+
```rust
|
|
764
|
+
use sqlx::{SqlitePool, FromRow};
|
|
765
|
+
|
|
766
|
+
#[derive(FromRow, Serialize, Clone)]
|
|
767
|
+
struct Note {
|
|
768
|
+
id: i64,
|
|
769
|
+
title: String,
|
|
770
|
+
content: String,
|
|
771
|
+
created_at: String,
|
|
772
|
+
}
|
|
773
|
+
|
|
774
|
+
struct DatabaseState {
|
|
775
|
+
pool: SqlitePool,
|
|
776
|
+
}
|
|
777
|
+
|
|
778
|
+
impl DatabaseState {
|
|
779
|
+
async fn new(database_url: &str) -> Result<Self, sqlx::Error> {
|
|
780
|
+
let pool = SqlitePool::connect(database_url).await?;
|
|
781
|
+
|
|
782
|
+
// Run migrations
|
|
783
|
+
sqlx::query(
|
|
784
|
+
r#"
|
|
785
|
+
CREATE TABLE IF NOT EXISTS notes (
|
|
786
|
+
id INTEGER PRIMARY KEY AUTOINCREMENT,
|
|
787
|
+
title TEXT NOT NULL,
|
|
788
|
+
content TEXT NOT NULL,
|
|
789
|
+
created_at DATETIME DEFAULT CURRENT_TIMESTAMP
|
|
790
|
+
)
|
|
791
|
+
"#,
|
|
792
|
+
)
|
|
793
|
+
.execute(&pool)
|
|
794
|
+
.await?;
|
|
795
|
+
|
|
796
|
+
Ok(Self { pool })
|
|
797
|
+
}
|
|
798
|
+
|
|
799
|
+
async fn create_note(&self, title: String, content: String) -> Result<Note, sqlx::Error> {
|
|
800
|
+
let note = sqlx::query_as::<_, Note>(
|
|
801
|
+
"INSERT INTO notes (title, content) VALUES (?, ?) RETURNING *",
|
|
802
|
+
)
|
|
803
|
+
.bind(title)
|
|
804
|
+
.bind(content)
|
|
805
|
+
.fetch_one(&self.pool)
|
|
806
|
+
.await?;
|
|
807
|
+
|
|
808
|
+
Ok(note)
|
|
809
|
+
}
|
|
810
|
+
|
|
811
|
+
async fn get_all_notes(&self) -> Result<Vec<Note>, sqlx::Error> {
|
|
812
|
+
sqlx::query_as::<_, Note>("SELECT * FROM notes ORDER BY created_at DESC")
|
|
813
|
+
.fetch_all(&self.pool)
|
|
814
|
+
.await
|
|
815
|
+
}
|
|
816
|
+
|
|
817
|
+
async fn update_note(&self, id: i64, title: String, content: String) -> Result<(), sqlx::Error> {
|
|
818
|
+
sqlx::query("UPDATE notes SET title = ?, content = ? WHERE id = ?")
|
|
819
|
+
.bind(title)
|
|
820
|
+
.bind(content)
|
|
821
|
+
.bind(id)
|
|
822
|
+
.execute(&self.pool)
|
|
823
|
+
.await?;
|
|
824
|
+
|
|
825
|
+
Ok(())
|
|
826
|
+
}
|
|
827
|
+
|
|
828
|
+
async fn delete_note(&self, id: i64) -> Result<(), sqlx::Error> {
|
|
829
|
+
sqlx::query("DELETE FROM notes WHERE id = ?")
|
|
830
|
+
.bind(id)
|
|
831
|
+
.execute(&self.pool)
|
|
832
|
+
.await?;
|
|
833
|
+
|
|
834
|
+
Ok(())
|
|
835
|
+
}
|
|
836
|
+
}
|
|
837
|
+
|
|
838
|
+
#[tauri::command]
|
|
839
|
+
async fn create_note(
|
|
840
|
+
state: tauri::State<'_, DatabaseState>,
|
|
841
|
+
title: String,
|
|
842
|
+
content: String,
|
|
843
|
+
) -> Result<Note, String> {
|
|
844
|
+
state
|
|
845
|
+
.create_note(title, content)
|
|
846
|
+
.await
|
|
847
|
+
.map_err(|e| e.to_string())
|
|
848
|
+
}
|
|
849
|
+
|
|
850
|
+
#[tauri::command]
|
|
851
|
+
async fn get_all_notes(state: tauri::State<'_, DatabaseState>) -> Result<Vec<Note>, String> {
|
|
852
|
+
state.get_all_notes().await.map_err(|e| e.to_string())
|
|
853
|
+
}
|
|
854
|
+
```
|
|
855
|
+
|
|
856
|
+
## Multi-Window State Sharing
|
|
857
|
+
|
|
858
|
+
```rust
|
|
859
|
+
use std::sync::Arc;
|
|
860
|
+
use tokio::sync::RwLock;
|
|
861
|
+
|
|
862
|
+
#[derive(Clone)]
|
|
863
|
+
struct SharedAppState {
|
|
864
|
+
data: Arc<RwLock<GlobalData>>,
|
|
865
|
+
}
|
|
866
|
+
|
|
867
|
+
struct GlobalData {
|
|
868
|
+
current_user: Option<User>,
|
|
869
|
+
notifications: Vec<Notification>,
|
|
870
|
+
}
|
|
871
|
+
|
|
872
|
+
impl SharedAppState {
|
|
873
|
+
fn new() -> Self {
|
|
874
|
+
Self {
|
|
875
|
+
data: Arc::new(RwLock::new(GlobalData {
|
|
876
|
+
current_user: None,
|
|
877
|
+
notifications: Vec::new(),
|
|
878
|
+
})),
|
|
879
|
+
}
|
|
880
|
+
}
|
|
881
|
+
|
|
882
|
+
async fn set_user(&self, user: User) {
|
|
883
|
+
let mut data = self.data.write().await;
|
|
884
|
+
data.current_user = Some(user);
|
|
885
|
+
}
|
|
886
|
+
|
|
887
|
+
async fn add_notification(&self, notification: Notification) {
|
|
888
|
+
let mut data = self.data.write().await;
|
|
889
|
+
data.notifications.push(notification);
|
|
890
|
+
}
|
|
891
|
+
|
|
892
|
+
async fn get_user(&self) -> Option<User> {
|
|
893
|
+
let data = self.data.read().await;
|
|
894
|
+
data.current_user.clone()
|
|
895
|
+
}
|
|
896
|
+
}
|
|
897
|
+
|
|
898
|
+
// Broadcast state changes to all windows
|
|
899
|
+
use tauri::{Emitter, Manager};
|
|
900
|
+
|
|
901
|
+
#[tauri::command]
|
|
902
|
+
async fn login_user(
|
|
903
|
+
app: tauri::AppHandle,
|
|
904
|
+
state: tauri::State<'_, SharedAppState>,
|
|
905
|
+
username: String,
|
|
906
|
+
) -> Result<(), String> {
|
|
907
|
+
let user = User {
|
|
908
|
+
id: 1,
|
|
909
|
+
name: username,
|
|
910
|
+
email: "user@example.com".to_string(),
|
|
911
|
+
};
|
|
912
|
+
|
|
913
|
+
state.set_user(user.clone()).await;
|
|
914
|
+
|
|
915
|
+
// Notify all windows
|
|
916
|
+
app.emit("user-logged-in", &user).map_err(|e| e.to_string())?;
|
|
917
|
+
|
|
918
|
+
Ok(())
|
|
919
|
+
}
|
|
920
|
+
|
|
921
|
+
// Open new window with shared state
|
|
922
|
+
#[tauri::command]
|
|
923
|
+
fn open_settings_window(app: tauri::AppHandle) -> Result<(), String> {
|
|
924
|
+
tauri::WebviewWindowBuilder::new(
|
|
925
|
+
&app,
|
|
926
|
+
"settings",
|
|
927
|
+
tauri::WebviewUrl::App("settings.html".into()),
|
|
928
|
+
)
|
|
929
|
+
.title("Settings")
|
|
930
|
+
.build()
|
|
931
|
+
.map_err(|e| e.to_string())?;
|
|
932
|
+
|
|
933
|
+
Ok(())
|
|
934
|
+
}
|
|
935
|
+
```
|
|
936
|
+
|
|
937
|
+
These state management patterns provide flexibility for applications of all sizes - from simple local state to complex distributed state with persistence and multi-window synchronization.
|