fix: 停止跟随屏幕时,程序崩溃的问题。

This commit is contained in:
Ivan Li 2023-01-15 19:35:14 +08:00
parent 5384a30872
commit a12c503e81
2 changed files with 75 additions and 86 deletions

View File

@ -1,14 +1,12 @@
use futures::future::join_all; use futures::future::join_all;
use once_cell::sync::OnceCell; use once_cell::sync::OnceCell;
use paris::info; use paris::{error, info};
use serde::{Deserialize, Serialize}; use serde::{Deserialize, Serialize};
use std::{collections::HashMap, sync::Arc, time::Duration}; use std::{collections::HashMap, sync::Arc, time::Duration};
use tauri::async_runtime::RwLock; use tauri::async_runtime::RwLock;
use tokio::{ use tokio::{
join,
sync::mpsc, sync::mpsc,
task,
time::{sleep, Instant}, time::{sleep, Instant},
}; };
use tracing::warn; use tracing::warn;
@ -21,7 +19,7 @@ use crate::{
rpc, rpc,
}; };
#[derive(Debug, Serialize, Deserialize)] #[derive(Debug, Serialize, Deserialize, Clone, Copy)]
pub enum AmbientLightMode { pub enum AmbientLightMode {
None, None,
Follow, Follow,
@ -40,25 +38,25 @@ impl CoreManager {
ambient_light_mode: Arc::new(RwLock::new(AmbientLightMode::None)), ambient_light_mode: Arc::new(RwLock::new(AmbientLightMode::None)),
}); });
tokio::spawn(async {
loop {
core.play_flowing_light().await;
match core.play_follow().await {
Ok(_) => {}
Err(error) => {
warn!("Can not following displays. {}", error);
sleep(Duration::from_millis(1000)).await;
}
};
sleep(Duration::from_millis(10)).await;
}
});
core core
} }
pub async fn set_ambient_light(&self, target_mode: AmbientLightMode) { pub async fn set_ambient_light(&self, target_mode: AmbientLightMode) {
let mut mode = self.ambient_light_mode.write().await; let mut mode = self.ambient_light_mode.write().await;
*mode = target_mode; *mode = target_mode;
drop(mode);
match target_mode {
AmbientLightMode::Flowing => self.play_flowing_light().await,
AmbientLightMode::None => {}
AmbientLightMode::Follow => match self.play_follow().await {
Ok(_) => {}
Err(error) => {
warn!("Can not following displays. {}", error);
}
},
};
} }
pub async fn play_flowing_light(&self) { pub async fn play_flowing_light(&self) {
@ -91,76 +89,67 @@ impl CoreManager {
} }
pub async fn play_follow(&self) -> anyhow::Result<()> { pub async fn play_follow(&self) -> anyhow::Result<()> {
let lock = self.ambient_light_mode.read().await;
let mut futs = vec![]; let mut futs = vec![];
if let AmbientLightMode::Follow = *lock { let configs = Picker::global().await.display_configs.lock().await;
drop(lock); info!("piker display configs: {:?}", configs);
let configs = Picker::global().await.display_configs.lock().await;
let (tx, mut rx) = mpsc::channel(10); let (tx, mut rx) = mpsc::channel(10);
for config in configs.to_owned() { for config in configs.to_owned() {
let tx = tx.clone(); let tx = tx.clone();
let fut = tokio::spawn(async move { let fut = tokio::spawn(async move {
match Self::follow_display_by_config(config, tx).await { match Self::follow_display_by_config(config, tx).await {
Ok(_) => {} Ok(_) => {}
Err(error) => { Err(error) => {
warn!("following failed. {}", error); warn!("following failed. {}", error);
}
}
});
futs.push(fut);
}
let configs = configs.clone();
tokio::spawn(async move {
let mut global_colors = HashMap::new();
while let Some(screenshot) = rx.recv().await {
let start_at = Instant::now();
let colors = screenshot.get_top_colors();
let start = screenshot
.get_top_of_led_start_at()
.min(screenshot.get_top_of_led_end_at());
let colors_len = colors.len();
for (index, color) in colors.into_iter().enumerate() {
global_colors.insert(index + start, color);
}
info!(
"led count: {}, spend: {:?}",
global_colors.len(),
start_at.elapsed()
);
if global_colors.len() == 60 {
let mut colors = vec![];
for index in 0..global_colors.len() {
colors.push(*global_colors.get(&index).unwrap());
}
global_colors = HashMap::new();
match rpc::manager::Manager::global()
.publish_led_colors(&colors)
.await
{
Ok(_) => {
info!("publish successful",);
}
Err(error) => {
warn!("publish led colors failed. {}", error);
}
}
} }
} }
}); });
futs.push(fut);
join_all(futs).await;
} else {
drop(lock);
return Ok(());
} }
tokio::spawn(async move {
let mut global_colors = HashMap::new();
while let Some(screenshot) = rx.recv().await {
let start_at = Instant::now();
let colors = screenshot.get_top_colors();
let start = screenshot
.get_top_of_led_start_at()
.min(screenshot.get_top_of_led_end_at());
for (index, color) in colors.into_iter().enumerate() {
global_colors.insert(index + start, color);
}
info!(
"led count: {}, spend: {:?}",
global_colors.len(),
start_at.elapsed()
);
if global_colors.len() == 60 {
let mut colors = vec![];
for index in 0..global_colors.len() {
colors.push(*global_colors.get(&index).unwrap());
}
global_colors = HashMap::new();
match rpc::manager::Manager::global()
.publish_led_colors(&colors)
.await
{
Ok(_) => {
info!("publish successful",);
}
Err(error) => {
warn!("publish led colors failed. {}", error);
}
}
}
}
});
join_all(futs).await;
Ok(()) Ok(())
} }
@ -179,17 +168,17 @@ impl CoreManager {
drop(lock); drop(lock);
let screenshot = picker.take_screenshot()?; let screenshot = picker.take_screenshot()?;
info!("Take Screenshot Spend: {:?}", start.elapsed()); info!("Take Screenshot Spend: {:?}", start.elapsed());
tx.send(screenshot).await; match tx.send(screenshot).await {
Ok(_) => {}
Err(err) => {
error!("send screenshot to main thread was failed. {:?}", err);
}
};
} else { } else {
break; break;
} }
tokio::time::sleep_until(next_tick).await; tokio::time::sleep_until(next_tick).await;
} }
// // Picker::global().take_screenshots_for_all().await?;
// // let colors = Picker::global().get_led_strip_colors().await?;
// // let colors = colors.into_iter().rev().collect();
Ok(()) Ok(())
} }
} }

View File

@ -78,7 +78,7 @@ async fn write_picker_config(config: picker::config::Configuration) -> Result<()
async fn play_mode(target_mode: AmbientLightMode) { async fn play_mode(target_mode: AmbientLightMode) {
info!("target mode: {:?}", target_mode); info!("target mode: {:?}", target_mode);
CoreManager::global().set_ambient_light(target_mode).await; tokio::spawn(async move { CoreManager::global().set_ambient_light(target_mode).await });
} }
#[tokio::main] #[tokio::main]