This commit is contained in:
Julian Ospald 2017-07-01 01:56:32 +02:00
parent 92514a2fd0
commit c1df692fa8
No known key found for this signature in database
GPG Key ID: 511B62C09D50CD28
3 changed files with 106 additions and 15 deletions

View File

@ -15,6 +15,7 @@ use std::mem;
use std::ptr; use std::ptr;
use std::rc::Rc; use std::rc::Rc;
use std::u8; use std::u8;
use std::f64;
@ -46,6 +47,7 @@ pub enum AlsaEvent {
} }
// TODO: Audio struct? Separate more cleanly
// TODO: implement free/destructor // TODO: implement free/destructor
pub struct AlsaCard { pub struct AlsaCard {
_cannot_construct: (), _cannot_construct: (),
@ -54,7 +56,8 @@ pub struct AlsaCard {
pub selem_id: SelemId, pub selem_id: SelemId,
pub watch_ids: Vec<u32>, pub watch_ids: Vec<u32>,
pub last_action_timestamp: RefCell<i64>, pub last_action_timestamp: RefCell<i64>,
pub handlers: RefCell<Vec<Box<Fn(&AlsaCard, AudioSignal, AudioUser)>>>, pub handlers: RefCell<Vec<Box<Fn(AudioSignal, AudioUser)>>>,
pub scroll_step: RefCell<u32>,
} }
@ -77,6 +80,7 @@ impl AlsaCard {
.get_id(); .get_id();
let vec_pollfd = PollDescriptors::get(&mixer)?; let vec_pollfd = PollDescriptors::get(&mixer)?;
// TODO: rework, should probably be a Box?
let acard = Rc::new(RefCell::new(AlsaCard { let acard = Rc::new(RefCell::new(AlsaCard {
_cannot_construct: (), _cannot_construct: (),
card: card, card: card,
@ -86,19 +90,27 @@ impl AlsaCard {
last_action_timestamp: last_action_timestamp:
RefCell::new(0), RefCell::new(0),
handlers: RefCell::new(vec![]), handlers: RefCell::new(vec![]),
scroll_step: RefCell::new(3),
})); }));
/* TODO: callback is registered here, which must be unregistered /* TODO: callback is registered here, which must be unregistered
* when the mixer is destroyed!! * when the mixer is destroyed!!
* poll descriptors must be unwatched too */ * poll descriptors must be unwatched too */
let watch_ids = watch_poll_descriptors(vec_pollfd, let watch_ids = watch_poll_descriptors(vec_pollfd, acard.as_ptr());
acard.clone().as_ptr());
acard.borrow_mut().watch_ids = watch_ids; acard.borrow_mut().watch_ids = watch_ids;
return Ok(acard.clone()); return Ok(acard.clone());
} }
pub fn switch_card(old_card: Rc<RefCell<AlsaCard>>,
card_name: Option<String>,
elem_name: Option<String>)
-> Result<Rc<RefCell<AlsaCard>>> {
}
pub fn selem(&self) -> Selem { pub fn selem(&self) -> Selem {
return get_selems(&self.mixer) return get_selems(&self.mixer)
.nth(self.selem_id.get_index() as usize) .nth(self.selem_id.get_index() as usize)
@ -123,6 +135,34 @@ impl AlsaCard {
} }
pub fn increase_vol(&self, user: AudioUser) -> Result<()> {
{
let mut rc = self.last_action_timestamp.borrow_mut();
*rc = glib::get_monotonic_time();
}
let old_vol = self.vol()?;
let new_vol = f64::ceil(old_vol + (*self.scroll_step.borrow() as f64));
debug!("Increase vol by {:?} to {:?}", (new_vol - old_vol), new_vol);
return self.set_vol(new_vol, user);
}
pub fn decrease_vol(&self, user: AudioUser) -> Result<()> {
{
let mut rc = self.last_action_timestamp.borrow_mut();
*rc = glib::get_monotonic_time();
}
let old_vol = self.vol()?;
let new_vol = old_vol - (*self.scroll_step.borrow() as f64);
debug!("Decrease vol by {:?} to {:?}", (new_vol - old_vol), new_vol);
return self.set_vol(new_vol, user);
}
pub fn has_mute(&self) -> bool { pub fn has_mute(&self) -> bool {
return has_mute(&self.selem()); return has_mute(&self.selem());
} }
@ -176,22 +216,35 @@ impl AlsaCard {
signal, signal,
user); user);
let handlers = self.handlers.borrow(); let handlers = self.handlers.borrow();
let x: &Vec<Box<Fn(&AlsaCard, AudioSignal, AudioUser)>> = &*handlers; let handlers_ref: &Vec<Box<Fn(AudioSignal, AudioUser)>> =
for handler in x { handlers.as_ref();
for handler in handlers_ref {
let unboxed = handler.as_ref(); let unboxed = handler.as_ref();
unboxed(&self, signal, user); unboxed(signal, user);
} }
} }
pub fn connect_handler(&self, pub fn connect_handler(&self, cb: Box<Fn(AudioSignal, AudioUser)>) {
cb: Box<Fn(&AlsaCard, AudioSignal, AudioUser)>) {
self.handlers.borrow_mut().push(cb); self.handlers.borrow_mut().push(cb);
} }
} }
impl Drop for AlsaCard { impl Drop for AlsaCard {
// call Box::new(x), transmute the Box into a raw pointer, and then
// std::mem::forget
//
// if you unregister the callback, you should keep a raw pointer to the
// box
//
// For instance, `register` could return a raw pointer to the
// Box + a std::marker::PhantomData with the appropriate
// lifetime (if applicable)
//
// The struct could implement Drop, which unregisters the
// callback and frees the Box, by simply transmuting the
// raw pointer to a Box<T>
fn drop(&mut self) { fn drop(&mut self) {
debug!("Destructing watch_ids: {:?}", self.watch_ids); debug!("Destructing watch_ids: {:?}", self.watch_ids);
unwatch_poll_descriptors(&self.watch_ids); unwatch_poll_descriptors(&self.watch_ids);

View File

@ -10,7 +10,7 @@ use ui_tray_icon::*;
pub fn init(appstate: Rc<AppS>) { pub fn init(appstate: Rc<AppS>) {
{ {
let apps = appstate.clone(); let apps = appstate.clone();
appstate.acard.borrow().connect_handler(Box::new(move |a, s, u| { appstate.acard.borrow().connect_handler(Box::new(move |s, u| {
match (s, u) { match (s, u) {
(AudioSignal::AudioValuesChanged, (AudioSignal::AudioValuesChanged,
AudioUser::AudioUserUnknown) => { AudioUser::AudioUserUnknown) => {

View File

@ -1,18 +1,35 @@
use app_state::*; use app_state::*;
use gdk;
use gdk_sys;
use glib;
use glib_sys;
use std::mem;
use gtk::prelude::*; use gtk::prelude::*;
use std::rc::Rc; use std::rc::Rc;
use std::cell::RefCell;
use libc;
use audio::AudioUser::*;
use errors::*;
pub fn init_tray_icon(appstate: Rc<AppS>) { pub fn init_tray_icon(appstate: Rc<AppS>) {
{
let apps = appstate.clone();
let tray_icon = &appstate.gui.status_icon;
tray_icon.connect_activate(move |_| on_tray_icon_activate(&apps));
tray_icon.set_visible(true);
}
{
let tray_icon = &appstate.clone().gui.status_icon; let tray_icon = &appstate.clone().gui.status_icon;
tray_icon.connect_activate(move |_| { tray_icon.connect_scroll_event(move |_, e| {
on_tray_icon_activate(&appstate.clone()) on_tray_icon_scroll_event(&appstate.clone(), &e)
}); });
tray_icon.set_visible(true); tray_icon.set_visible(true);
} }
}
fn on_tray_icon_activate(appstate: &AppS) { fn on_tray_icon_activate(appstate: &AppS) {
let popup_window = &appstate.gui.popup_window.window; let popup_window = &appstate.gui.popup_window.window;
@ -23,3 +40,24 @@ fn on_tray_icon_activate(appstate: &AppS) {
popup_window.show_now(); popup_window.show_now();
} }
} }
fn on_tray_icon_scroll_event(appstate: &AppS,
event: &gdk::EventScroll)
-> bool {
let scroll_dir = event.as_ref().direction;
match scroll_dir {
gdk_sys::GdkScrollDirection::Up => {
try_wr!(appstate.acard.borrow().increase_vol(AudioUserTrayIcon),
false);
}
gdk_sys::GdkScrollDirection::Down => {
try_wr!(appstate.acard.borrow().decrease_vol(AudioUserTrayIcon),
false);
}
_ => (),
}
return false;
}