Big refactor, file change notifications added
This commit is contained in:
@@ -20,5 +20,7 @@ features = [
|
|||||||
"Win32_UI_WindowsAndMessaging",
|
"Win32_UI_WindowsAndMessaging",
|
||||||
"Win32_System_Com",
|
"Win32_System_Com",
|
||||||
"Win32_System_Com_StructuredStorage",
|
"Win32_System_Com_StructuredStorage",
|
||||||
"Win32_Devices_FunctionDiscovery"
|
"Win32_Devices_FunctionDiscovery",
|
||||||
|
"Win32_Storage_FileSystem",
|
||||||
|
"Win32_System_WindowsProgramming",
|
||||||
]
|
]
|
||||||
1
mute.txt
1
mute.txt
@@ -14,4 +14,3 @@ sm64.us.f3dex2e.exe
|
|||||||
teardown.exe
|
teardown.exe
|
||||||
underrail.exe
|
underrail.exe
|
||||||
valheim.exe
|
valheim.exe
|
||||||
Spotify.exe
|
|
||||||
@@ -6,9 +6,13 @@ use windows::{
|
|||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
pub trait DeviceNotificationObserver {
|
||||||
|
fn add_device(&self, device_id : &windows::core::PCWSTR);
|
||||||
|
}
|
||||||
|
|
||||||
#[windows::core::implement(IMMNotificationClient)]
|
#[windows::core::implement(IMMNotificationClient)]
|
||||||
pub(crate) struct DeviceNotificationClient {
|
pub(crate) struct DeviceNotificationClient {
|
||||||
pub(crate) callback: Box<dyn Fn(&windows::core::PCWSTR)>
|
pub observer: Box<dyn DeviceNotificationObserver>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl IMMNotificationClient_Impl for DeviceNotificationClient {
|
impl IMMNotificationClient_Impl for DeviceNotificationClient {
|
||||||
@@ -21,7 +25,7 @@ impl IMMNotificationClient_Impl for DeviceNotificationClient {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn OnDeviceAdded(&self, pwstrdeviceid: &windows::core::PCWSTR) -> windows::core::Result<()> {
|
fn OnDeviceAdded(&self, pwstrdeviceid: &windows::core::PCWSTR) -> windows::core::Result<()> {
|
||||||
(self.callback)(pwstrdeviceid);
|
self.observer.add_device(pwstrdeviceid);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
115
src/main.rs
115
src/main.rs
@@ -12,57 +12,74 @@ use std::{
|
|||||||
io::{BufRead, BufReader},
|
io::{BufRead, BufReader},
|
||||||
path::Path,
|
path::Path,
|
||||||
ptr::null_mut,
|
ptr::null_mut,
|
||||||
sync::{Arc, Mutex},
|
sync::mpsc::{self, Receiver, Sender},
|
||||||
|
thread::{self, JoinHandle},
|
||||||
};
|
};
|
||||||
|
|
||||||
use sm_session_notifier::SMSessionNotifier;
|
use sm_session_notifier::SMSessionNotifierThread;
|
||||||
use window_change::WindowChangeMonitor;
|
use window_change::WindowChangeMonitor;
|
||||||
use windows::{
|
use windows::{
|
||||||
core::Interface,
|
core::{Interface},
|
||||||
Win32::{
|
Win32::{
|
||||||
Media::Audio::{IAudioSessionControl2, ISimpleAudioVolume},
|
Media::Audio::{IAudioSessionControl2, ISimpleAudioVolume},
|
||||||
System::Com::{CoInitializeEx, COINIT_MULTITHREADED},
|
System::{Com::{CoInitializeEx, COINIT_MULTITHREADED}, Threading::WaitForSingleObject, WindowsProgramming::INFINITE}, Storage::FileSystem::{FindFirstChangeNotificationW, FILE_NOTIFY_CHANGE_LAST_WRITE, FindCloseChangeNotification}, Foundation::HANDLE,
|
||||||
},
|
}, w,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::pid_to_exe::pid_to_exe_path;
|
use crate::pid_to_exe::pid_to_exe_path;
|
||||||
|
|
||||||
|
enum MuterMessage {
|
||||||
|
WindowChange(String),
|
||||||
|
AddSession(IAudioSessionControl2),
|
||||||
|
Exit(),
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe impl Send for MuterMessage {}
|
||||||
|
|
||||||
struct SessionMuter {
|
struct SessionMuter {
|
||||||
sessions: Vec<IAudioSessionControl2>,
|
sessions: Vec<IAudioSessionControl2>,
|
||||||
mute_executables: HashSet<String>,
|
mute_executables: HashSet<String>,
|
||||||
mute_flag: bool,
|
mute_flag: bool,
|
||||||
win: Option<WindowChangeMonitor>
|
_sn: SMSessionNotifierThread,
|
||||||
|
_wn: WindowChangeMonitor,
|
||||||
|
rx: Receiver<MuterMessage>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SessionMuter {
|
impl SessionMuter {
|
||||||
fn start(mute_executables: HashSet<String>) -> Arc<Mutex<SessionMuter>> {
|
fn new(
|
||||||
let muter = Arc::new(Mutex::new(SessionMuter {
|
mute_executables: HashSet<String>,
|
||||||
|
rx: Receiver<MuterMessage>,
|
||||||
|
tx: Sender<MuterMessage>,
|
||||||
|
) -> SessionMuter {
|
||||||
|
SessionMuter {
|
||||||
sessions: Vec::new(),
|
sessions: Vec::new(),
|
||||||
mute_executables,
|
mute_executables,
|
||||||
mute_flag: true,
|
mute_flag: true,
|
||||||
win: None
|
_sn: {
|
||||||
}));
|
let tx = tx.clone();
|
||||||
{
|
SMSessionNotifierThread::new(Box::new(move |session| {
|
||||||
let muter = Arc::downgrade(&muter);
|
tx.send(MuterMessage::AddSession(session)).unwrap();
|
||||||
let muter_internal = muter.clone();
|
}))
|
||||||
muter.upgrade().unwrap().lock().as_mut().unwrap().win = Some(WindowChangeMonitor::start(Box::new(move |s| {
|
},
|
||||||
muter_internal.upgrade().map(|f| f.lock().unwrap().notify_window_changed(&s));
|
_wn: {
|
||||||
})));
|
let tx = tx.clone();
|
||||||
|
WindowChangeMonitor::start(Box::new(move |s| {
|
||||||
|
tx.send(MuterMessage::WindowChange(s.to_owned())).unwrap();
|
||||||
|
}))
|
||||||
|
},
|
||||||
|
rx,
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
{
|
fn run(&mut self) {
|
||||||
let muter = Arc::downgrade(&muter);
|
loop {
|
||||||
let sn = SMSessionNotifier::new(Box::new(move |session| {
|
let msg = self.rx.recv().unwrap();
|
||||||
muter.upgrade().unwrap().lock().unwrap().add_session(session).unwrap()
|
match msg {
|
||||||
}));
|
MuterMessage::WindowChange(win) => self.notify_window_changed(&win),
|
||||||
sn.lock()
|
MuterMessage::AddSession(session) => self.add_session(session).unwrap(),
|
||||||
.unwrap()
|
MuterMessage::Exit() => break,
|
||||||
.as_mut()
|
}
|
||||||
.unwrap()
|
|
||||||
.boot_devices()
|
|
||||||
.expect("failed to get initial audio devices and sessions");
|
|
||||||
}
|
}
|
||||||
return muter;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn add_session(
|
fn add_session(
|
||||||
@@ -132,15 +149,45 @@ impl SessionMuter {
|
|||||||
|
|
||||||
unsafe impl Send for SessionMuter {}
|
unsafe impl Send for SessionMuter {}
|
||||||
|
|
||||||
|
pub struct MuterThread {
|
||||||
|
handle: Option<JoinHandle<()>>,
|
||||||
|
sender: Sender<MuterMessage>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl MuterThread {
|
||||||
|
pub fn new(s: HashSet<String>) -> MuterThread {
|
||||||
|
let (sender, receiver) = mpsc::channel::<MuterMessage>();
|
||||||
|
MuterThread {
|
||||||
|
sender: sender.clone(),
|
||||||
|
handle: Some(thread::spawn(move || {
|
||||||
|
let mut muter = SessionMuter::new(s, receiver, sender);
|
||||||
|
muter.run();
|
||||||
|
})),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for MuterThread {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
if let Some(handle) = self.handle.take() {
|
||||||
|
self.sender.send(MuterMessage::Exit()).unwrap();
|
||||||
|
handle.join().unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn load_mute_txt() -> HashSet<String> {
|
fn load_mute_txt() -> HashSet<String> {
|
||||||
let file = File::open("mute.txt").unwrap();
|
let file = File::open("mute.txt").unwrap();
|
||||||
return HashSet::from_iter(BufReader::new(file).lines().map(|line| line.unwrap()));
|
return HashSet::from_iter(BufReader::new(file).lines().map(|line| line.unwrap()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn await_file_change() -> Result<(), Box<dyn Error>> {
|
||||||
fn test_muter() {
|
unsafe {
|
||||||
SessionMuter::start(load_mute_txt());
|
let handle = FindFirstChangeNotificationW(w!("."), false, FILE_NOTIFY_CHANGE_LAST_WRITE)?;
|
||||||
|
WaitForSingleObject(HANDLE{0: handle.0}, INFINITE).ok()?;
|
||||||
|
FindCloseChangeNotification(handle).ok()?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
@@ -148,7 +195,9 @@ fn main() {
|
|||||||
CoInitializeEx(None, COINIT_MULTITHREADED).unwrap();
|
CoInitializeEx(None, COINIT_MULTITHREADED).unwrap();
|
||||||
}
|
}
|
||||||
loop {
|
loop {
|
||||||
test_muter();
|
let mut _mt = MuterThread::new(load_mute_txt());
|
||||||
println!("Test_muter done");
|
await_file_change().unwrap();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -10,21 +10,25 @@ use windows::{
|
|||||||
|
|
||||||
#[windows::core::implement(IAudioSessionNotification)]
|
#[windows::core::implement(IAudioSessionNotification)]
|
||||||
pub(crate) struct SessionNotification {
|
pub(crate) struct SessionNotification {
|
||||||
pub(crate) callback: Box<dyn Fn(IAudioSessionControl2)>
|
pub(crate) observer: Box<dyn SessionObserver>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl IAudioSessionNotification_Impl for SessionNotification {
|
pub trait SessionObserver {
|
||||||
|
fn add_session(&self, session: IAudioSessionControl2);
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> IAudioSessionNotification_Impl for SessionNotification {
|
||||||
fn OnSessionCreated(
|
fn OnSessionCreated(
|
||||||
self: &SessionNotification,
|
self: &SessionNotification,
|
||||||
newsession: &core::option::Option<IAudioSessionControl>,
|
newsession: &core::option::Option<IAudioSessionControl>,
|
||||||
) -> windows::core::Result<()> {
|
) -> windows::core::Result<()> {
|
||||||
let ses: IAudioSessionControl2 = newsession.as_ref().unwrap().cast().unwrap();
|
let ses: IAudioSessionControl2 = newsession.as_ref().unwrap().cast().unwrap();
|
||||||
(self.callback)(ses);
|
self.observer.add_session(ses);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Drop for SessionNotification {
|
impl<'a> Drop for SessionNotification {
|
||||||
fn drop(&mut self) {
|
fn drop(&mut self) {
|
||||||
println!("SN drop");
|
println!("SN drop");
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,10 +1,12 @@
|
|||||||
use std::{
|
use std::{
|
||||||
error::Error,
|
error::Error,
|
||||||
sync::{Arc, Mutex},
|
sync::mpsc::{self, Receiver, Sender},
|
||||||
|
thread::{self, JoinHandle},
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
device_notification_client::DeviceNotificationClient, session_notification::SessionNotification,
|
device_notification_client::{DeviceNotificationClient, DeviceNotificationObserver},
|
||||||
|
session_notification::{SessionNotification, SessionObserver},
|
||||||
};
|
};
|
||||||
use windows::{
|
use windows::{
|
||||||
core::{Interface, PCWSTR},
|
core::{Interface, PCWSTR},
|
||||||
@@ -17,46 +19,72 @@ use windows::{
|
|||||||
System::Com::{CoCreateInstance, CLSCTX_ALL, STGM_READ},
|
System::Com::{CoCreateInstance, CLSCTX_ALL, STGM_READ},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
pub enum SMMessage {
|
||||||
|
Session(IAudioSessionControl2),
|
||||||
|
Device(PCWSTR),
|
||||||
|
Exit(),
|
||||||
|
}
|
||||||
|
|
||||||
|
unsafe impl Send for SMMessage {}
|
||||||
|
|
||||||
|
struct SessionToMessage {
|
||||||
|
sender: Sender<SMMessage>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SessionObserver for SessionToMessage {
|
||||||
|
fn add_session(&self, session: IAudioSessionControl2) {
|
||||||
|
self.sender
|
||||||
|
.send(SMMessage::Session(session))
|
||||||
|
.unwrap_or_else(|_| println!("Failed to add new session"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
struct DeviceToMessage {
|
||||||
|
sender: Sender<SMMessage>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DeviceNotificationObserver for DeviceToMessage {
|
||||||
|
fn add_device(&self, device_id: &windows::core::PCWSTR) {
|
||||||
|
self.sender
|
||||||
|
.send(SMMessage::Device(*device_id))
|
||||||
|
.unwrap_or_else(|_| println!("Failed to add new device"));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub(crate) struct SMSessionNotifier {
|
pub(crate) struct SMSessionNotifier {
|
||||||
device_enumerator: IMMDeviceEnumerator,
|
device_enumerator: IMMDeviceEnumerator,
|
||||||
device_notification_client: IMMNotificationClient,
|
device_notification_client: IMMNotificationClient,
|
||||||
session_notification: IAudioSessionNotification,
|
session_notification: IAudioSessionNotification,
|
||||||
session_managers: Vec<IAudioSessionManager2>,
|
session_managers: Vec<IAudioSessionManager2>,
|
||||||
notification_function: Box<dyn Fn(IAudioSessionControl2)>,
|
notification_function: Box<dyn Fn(IAudioSessionControl2)>,
|
||||||
|
receiver: Receiver<SMMessage>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SMSessionNotifier {
|
impl SMSessionNotifier {
|
||||||
pub(crate) fn new(
|
pub(crate) fn new(
|
||||||
callback: Box<dyn Fn(IAudioSessionControl2)>,
|
callback: Box<dyn Fn(IAudioSessionControl2)>,
|
||||||
) -> Arc<Mutex<Option<SMSessionNotifier>>> {
|
sender: mpsc::Sender<SMMessage>,
|
||||||
let session_notifier_arc: Arc<Mutex<Option<SMSessionNotifier>>> =
|
receiver: mpsc::Receiver<SMMessage>,
|
||||||
Arc::new(Mutex::new(None));
|
) -> SMSessionNotifier {
|
||||||
let sn2 = Arc::downgrade(&session_notifier_arc);
|
SMSessionNotifier {
|
||||||
let sn3 = sn2.clone();
|
|
||||||
let notifier = SMSessionNotifier {
|
|
||||||
session_managers: Vec::new(),
|
session_managers: Vec::new(),
|
||||||
device_enumerator: unsafe {
|
device_enumerator: unsafe {
|
||||||
CoCreateInstance(&MMDeviceEnumerator, None, CLSCTX_ALL).unwrap()
|
CoCreateInstance(&MMDeviceEnumerator, None, CLSCTX_ALL).unwrap()
|
||||||
},
|
},
|
||||||
device_notification_client: IMMNotificationClient::from(DeviceNotificationClient {
|
device_notification_client: IMMNotificationClient::from(DeviceNotificationClient {
|
||||||
callback: Box::new(move |id| {
|
observer: Box::new(DeviceToMessage {
|
||||||
match sn2.upgrade().unwrap().lock().unwrap().as_mut().unwrap().add_device_by_id(id) {
|
sender: sender.clone(),
|
||||||
Ok(_) => {},
|
|
||||||
Err(e) => {
|
|
||||||
println!("Failed to add new device: {:?}", e);
|
|
||||||
},
|
|
||||||
}
|
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
session_notification: IAudioSessionNotification::from(SessionNotification {
|
session_notification: IAudioSessionNotification::from(SessionNotification {
|
||||||
callback: Box::new(move |session| {
|
observer: Box::new(SessionToMessage {
|
||||||
(sn3.upgrade().unwrap().lock().unwrap().as_mut().unwrap().notification_function)(session)
|
sender: sender.clone(),
|
||||||
}),
|
}),
|
||||||
}),
|
}),
|
||||||
notification_function: callback,
|
notification_function: callback,
|
||||||
};
|
receiver,
|
||||||
*session_notifier_arc.lock().unwrap() = Some(notifier);
|
}
|
||||||
return session_notifier_arc;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub(crate) fn boot_devices(self: &mut SMSessionNotifier) -> Result<(), Box<dyn Error>> {
|
pub(crate) fn boot_devices(self: &mut SMSessionNotifier) -> Result<(), Box<dyn Error>> {
|
||||||
@@ -105,8 +133,57 @@ impl SMSessionNotifier {
|
|||||||
}
|
}
|
||||||
let prop_store = device.OpenPropertyStore(STGM_READ)?;
|
let prop_store = device.OpenPropertyStore(STGM_READ)?;
|
||||||
let prop_var = prop_store.GetValue(&PKEY_Device_FriendlyName)?;
|
let prop_var = prop_store.GetValue(&PKEY_Device_FriendlyName)?;
|
||||||
println!("Device Added: {} Existing Sessions: {}", prop_var.Anonymous.Anonymous.Anonymous.pwszVal.to_string()?, session_count);
|
println!(
|
||||||
|
"Device Added: {} Existing Sessions: {}",
|
||||||
|
prop_var.Anonymous.Anonymous.Anonymous.pwszVal.to_string()?,
|
||||||
|
session_count
|
||||||
|
);
|
||||||
self.session_managers.push(sm);
|
self.session_managers.push(sm);
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn run(&mut self) -> Result<(), Box<dyn Error>> {
|
||||||
|
self.boot_devices()?;
|
||||||
|
|
||||||
|
loop {
|
||||||
|
let msg = self.receiver.recv()?;
|
||||||
|
match msg {
|
||||||
|
SMMessage::Session(session) => {
|
||||||
|
(self.notification_function)(session);
|
||||||
|
}
|
||||||
|
SMMessage::Device(id) => {
|
||||||
|
self.add_device_by_id(&id)?;
|
||||||
|
}
|
||||||
|
SMMessage::Exit() => break,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct SMSessionNotifierThread {
|
||||||
|
handle: Option<JoinHandle<()>>,
|
||||||
|
sender: Sender<SMMessage>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SMSessionNotifierThread {
|
||||||
|
pub fn new(s: Box<dyn Fn(IAudioSessionControl2) + Send>) -> SMSessionNotifierThread {
|
||||||
|
let (sender, receiver) = mpsc::channel::<SMMessage>();
|
||||||
|
SMSessionNotifierThread {
|
||||||
|
sender: sender.clone(),
|
||||||
|
handle: Some(thread::spawn(move || {
|
||||||
|
let mut session_notifier = SMSessionNotifier::new(s, sender, receiver);
|
||||||
|
session_notifier.run().unwrap();
|
||||||
|
})),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Drop for SMSessionNotifierThread {
|
||||||
|
fn drop(&mut self) {
|
||||||
|
if let Some(handle) = self.handle.take() {
|
||||||
|
self.sender.send(SMMessage::Exit()).unwrap();
|
||||||
|
handle.join().unwrap()
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user