feat: update pulse connection every time the audio applet popup is opened
This commit is contained in:
parent
62ec66ab4e
commit
a76353981f
5 changed files with 262 additions and 159 deletions
2
Cargo.lock
generated
2
Cargo.lock
generated
|
|
@ -524,6 +524,8 @@ dependencies = [
|
||||||
"libcosmic",
|
"libcosmic",
|
||||||
"libpulse-binding",
|
"libpulse-binding",
|
||||||
"libpulse-glib-binding",
|
"libpulse-glib-binding",
|
||||||
|
"log",
|
||||||
|
"pretty_env_logger",
|
||||||
"smithay-client-toolkit",
|
"smithay-client-toolkit",
|
||||||
"tokio",
|
"tokio",
|
||||||
]
|
]
|
||||||
|
|
|
||||||
|
|
@ -12,3 +12,5 @@ libpulse-glib-binding = "2.25.0"
|
||||||
tokio = { version = "1.20.1", features=["full"] }
|
tokio = { version = "1.20.1", features=["full"] }
|
||||||
libcosmic = { git = "https://github.com/pop-os/libcosmic/", branch = "master", default-features = false, features = ["tokio", "wayland", "applet"] }
|
libcosmic = { git = "https://github.com/pop-os/libcosmic/", branch = "master", default-features = false, features = ["tokio", "wayland", "applet"] }
|
||||||
sctk = { package = "smithay-client-toolkit", git = "https://github.com/Smithay/client-toolkit", rev = "3776d4a" }
|
sctk = { package = "smithay-client-toolkit", git = "https://github.com/Smithay/client-toolkit", rev = "3776d4a" }
|
||||||
|
log = "0.4.14"
|
||||||
|
pretty_env_logger = "0.4.0"
|
||||||
|
|
@ -25,6 +25,8 @@ use crate::pulse::DeviceInfo;
|
||||||
use libpulse_binding::volume::VolumeLinear;
|
use libpulse_binding::volume::VolumeLinear;
|
||||||
|
|
||||||
pub fn main() -> cosmic::iced::Result {
|
pub fn main() -> cosmic::iced::Result {
|
||||||
|
pretty_env_logger::init();
|
||||||
|
|
||||||
let helper = CosmicAppletHelper::default();
|
let helper = CosmicAppletHelper::default();
|
||||||
Audio::run(helper.window_settings())
|
Audio::run(helper.window_settings())
|
||||||
}
|
}
|
||||||
|
|
@ -80,7 +82,6 @@ impl Application for Audio {
|
||||||
current_input: None,
|
current_input: None,
|
||||||
outputs: vec![],
|
outputs: vec![],
|
||||||
inputs: vec![],
|
inputs: vec![],
|
||||||
pulse_state: PulseState::Disconnected,
|
|
||||||
icon_name: "audio-volume-high-symbolic".to_string(),
|
icon_name: "audio-volume-high-symbolic".to_string(),
|
||||||
..Default::default()
|
..Default::default()
|
||||||
},
|
},
|
||||||
|
|
@ -113,6 +114,9 @@ impl Application for Audio {
|
||||||
if let Some(p) = self.popup.take() {
|
if let Some(p) = self.popup.take() {
|
||||||
return destroy_popup(p);
|
return destroy_popup(p);
|
||||||
} else {
|
} else {
|
||||||
|
if let Some(conn) = self.pulse_state.connection() {
|
||||||
|
conn.send(pulse::Message::UpdateConnection);
|
||||||
|
}
|
||||||
self.id_ctr += 1;
|
self.id_ctr += 1;
|
||||||
let new_id = window::Id::new(self.id_ctr);
|
let new_id = window::Id::new(self.id_ctr);
|
||||||
self.popup.replace(new_id);
|
self.popup.replace(new_id);
|
||||||
|
|
@ -156,7 +160,7 @@ impl Application for Audio {
|
||||||
if let PulseState::Connected(connection) = &mut self.pulse_state {
|
if let PulseState::Connected(connection) = &mut self.pulse_state {
|
||||||
if let Some(device) = &self.current_input {
|
if let Some(device) = &self.current_input {
|
||||||
if let Some(name) = &device.name {
|
if let Some(name) = &device.name {
|
||||||
println!("increasing volume of {}", name);
|
log::info!("increasing volume of {}", name);
|
||||||
connection.send(pulse::Message::SetSourceVolumeByName(
|
connection.send(pulse::Message::SetSourceVolumeByName(
|
||||||
name.clone(),
|
name.clone(),
|
||||||
device.volume,
|
device.volume,
|
||||||
|
|
@ -165,8 +169,8 @@ impl Application for Audio {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Message::OutputChanged(val) => println!("changed output {}", val),
|
Message::OutputChanged(val) => log::info!("changed output {}", val),
|
||||||
Message::InputChanged(val) => println!("changed input {}", val),
|
Message::InputChanged(val) => log::info!("changed input {}", val),
|
||||||
Message::OutputToggle => {
|
Message::OutputToggle => {
|
||||||
self.is_open = if self.is_open == IsOpen::Output {
|
self.is_open = if self.is_open == IsOpen::Output {
|
||||||
IsOpen::None
|
IsOpen::None
|
||||||
|
|
@ -182,12 +186,16 @@ impl Application for Audio {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Message::Pulse(event) => match event {
|
Message::Pulse(event) => match event {
|
||||||
pulse::Event::Connected(mut connection) => {
|
pulse::Event::Init(conn) => self.pulse_state = PulseState::Disconnected(conn),
|
||||||
connection.send(pulse::Message::GetSinks);
|
pulse::Event::Connected => {
|
||||||
connection.send(pulse::Message::GetSources);
|
self.pulse_state.connected();
|
||||||
connection.send(pulse::Message::GetDefaultSink);
|
|
||||||
connection.send(pulse::Message::GetDefaultSource);
|
if let Some(conn) = self.pulse_state.connection() {
|
||||||
self.pulse_state = PulseState::Connected(connection);
|
conn.send(pulse::Message::GetSinks);
|
||||||
|
conn.send(pulse::Message::GetSources);
|
||||||
|
conn.send(pulse::Message::GetDefaultSink);
|
||||||
|
conn.send(pulse::Message::GetDefaultSource);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
pulse::Event::MessageReceived(msg) => {
|
pulse::Event::MessageReceived(msg) => {
|
||||||
match msg {
|
match msg {
|
||||||
|
|
@ -215,15 +223,11 @@ impl Application for Audio {
|
||||||
panic!("Subscriton error handling is bad. This should never happen.")
|
panic!("Subscriton error handling is bad. This should never happen.")
|
||||||
}
|
}
|
||||||
_ => {
|
_ => {
|
||||||
println!("Received misc message")
|
log::trace!("Received misc message")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// TODO: view() should gray out buttons/slider when state is disconnected
|
pulse::Event::Disconnected => self.pulse_state.disconnected(),
|
||||||
pulse::Event::Disconnected => {
|
|
||||||
println!("setting state to disconnected");
|
|
||||||
self.pulse_state = PulseState::Disconnected
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
Message::Ignore => {}
|
Message::Ignore => {}
|
||||||
Message::ToggleMediaControlsInTopPanel(enabled) => {
|
Message::ToggleMediaControlsInTopPanel(enabled) => {
|
||||||
|
|
@ -247,6 +251,7 @@ impl Application for Audio {
|
||||||
.on_press(Message::TogglePopup)
|
.on_press(Message::TogglePopup)
|
||||||
.into(),
|
.into(),
|
||||||
SurfaceIdWrapper::Popup(_) => {
|
SurfaceIdWrapper::Popup(_) => {
|
||||||
|
let audio_disabled = matches!(self.pulse_state, PulseState::Disconnected(_));
|
||||||
let out_f64 = VolumeLinear::from(
|
let out_f64 = VolumeLinear::from(
|
||||||
self.current_output
|
self.current_output
|
||||||
.as_ref()
|
.as_ref()
|
||||||
|
|
@ -262,68 +267,79 @@ impl Application for Audio {
|
||||||
)
|
)
|
||||||
.0 * 100.0;
|
.0 * 100.0;
|
||||||
|
|
||||||
|
let audio_content = if audio_disabled {
|
||||||
|
column![text("PulseAudio Disconnected")
|
||||||
|
.width(Length::Fill)
|
||||||
|
.horizontal_alignment(Horizontal::Center)
|
||||||
|
.size(24),]
|
||||||
|
} else {
|
||||||
|
column![
|
||||||
|
row![
|
||||||
|
icon("audio-volume-high-symbolic", 32)
|
||||||
|
.width(Length::Units(24))
|
||||||
|
.height(Length::Units(24))
|
||||||
|
.style(Svg::Symbolic),
|
||||||
|
slider(0.0..=100.0, out_f64, Message::SetOutputVolume)
|
||||||
|
.width(Length::FillPortion(5)),
|
||||||
|
text(format!("{}%", out_f64.round()))
|
||||||
|
.width(Length::FillPortion(1))
|
||||||
|
.horizontal_alignment(Horizontal::Right)
|
||||||
|
]
|
||||||
|
.spacing(12)
|
||||||
|
.align_items(Alignment::Center)
|
||||||
|
.padding([8, 24]),
|
||||||
|
row![
|
||||||
|
icon("audio-input-microphone-symbolic", 32)
|
||||||
|
.width(Length::Units(24))
|
||||||
|
.height(Length::Units(24))
|
||||||
|
.style(Svg::Symbolic),
|
||||||
|
slider(0.0..=100.0, in_f64, Message::SetInputVolume)
|
||||||
|
.width(Length::FillPortion(5)),
|
||||||
|
text(format!("{}%", in_f64.round()))
|
||||||
|
.width(Length::FillPortion(1))
|
||||||
|
.horizontal_alignment(Horizontal::Right)
|
||||||
|
]
|
||||||
|
.spacing(12)
|
||||||
|
.align_items(Alignment::Center)
|
||||||
|
.padding([8, 24]),
|
||||||
|
container(horizontal_rule(1))
|
||||||
|
.padding([12, 24])
|
||||||
|
.width(Length::Fill),
|
||||||
|
revealer(
|
||||||
|
self.is_open == IsOpen::Output,
|
||||||
|
"Output",
|
||||||
|
match &self.current_output {
|
||||||
|
Some(output) => pretty_name(output.description.clone()),
|
||||||
|
None => String::from("No device selected"),
|
||||||
|
},
|
||||||
|
self.outputs
|
||||||
|
.clone()
|
||||||
|
.into_iter()
|
||||||
|
.map(|output| pretty_name(output.description))
|
||||||
|
.collect(),
|
||||||
|
Message::OutputToggle,
|
||||||
|
Message::OutputChanged(String::from("test")),
|
||||||
|
),
|
||||||
|
revealer(
|
||||||
|
self.is_open == IsOpen::Input,
|
||||||
|
"Input",
|
||||||
|
match &self.current_input {
|
||||||
|
Some(input) => pretty_name(input.description.clone()),
|
||||||
|
None => String::from("No device selected"),
|
||||||
|
},
|
||||||
|
self.inputs
|
||||||
|
.clone()
|
||||||
|
.into_iter()
|
||||||
|
.map(|input| pretty_name(input.description))
|
||||||
|
.collect(),
|
||||||
|
Message::InputToggle,
|
||||||
|
Message::InputChanged(String::from("test")),
|
||||||
|
)
|
||||||
|
]
|
||||||
|
.align_items(Alignment::Start)
|
||||||
|
};
|
||||||
let content = column![
|
let content = column![
|
||||||
row![
|
audio_content,
|
||||||
icon("audio-volume-high-symbolic", 32)
|
|
||||||
.width(Length::Units(24))
|
|
||||||
.height(Length::Units(24))
|
|
||||||
.style(Svg::Symbolic),
|
|
||||||
slider(0.0..=100.0, out_f64, Message::SetOutputVolume)
|
|
||||||
.width(Length::FillPortion(5)),
|
|
||||||
text(format!("{}%", out_f64.round()))
|
|
||||||
.width(Length::FillPortion(1))
|
|
||||||
.horizontal_alignment(Horizontal::Right)
|
|
||||||
]
|
|
||||||
.spacing(12)
|
|
||||||
.align_items(Alignment::Center)
|
|
||||||
.padding([8, 24]),
|
|
||||||
row![
|
|
||||||
icon("audio-input-microphone-symbolic", 32)
|
|
||||||
.width(Length::Units(24))
|
|
||||||
.height(Length::Units(24))
|
|
||||||
.style(Svg::Symbolic),
|
|
||||||
slider(0.0..=100.0, in_f64, Message::SetInputVolume)
|
|
||||||
.width(Length::FillPortion(5)),
|
|
||||||
text(format!("{}%", in_f64.round()))
|
|
||||||
.width(Length::FillPortion(1))
|
|
||||||
.horizontal_alignment(Horizontal::Right)
|
|
||||||
]
|
|
||||||
.spacing(12)
|
|
||||||
.align_items(Alignment::Center)
|
|
||||||
.padding([8, 24]),
|
|
||||||
container(horizontal_rule(1))
|
|
||||||
.padding([12, 24])
|
|
||||||
.width(Length::Fill),
|
|
||||||
revealer(
|
|
||||||
self.is_open == IsOpen::Output,
|
|
||||||
"Output",
|
|
||||||
match &self.current_output {
|
|
||||||
Some(output) => pretty_name(output.description.clone()),
|
|
||||||
None => String::from("No device selected"),
|
|
||||||
},
|
|
||||||
self.outputs
|
|
||||||
.clone()
|
|
||||||
.into_iter()
|
|
||||||
.map(|output| pretty_name(output.description))
|
|
||||||
.collect(),
|
|
||||||
Message::OutputToggle,
|
|
||||||
Message::OutputChanged(String::from("test")),
|
|
||||||
),
|
|
||||||
revealer(
|
|
||||||
self.is_open == IsOpen::Input,
|
|
||||||
"Input",
|
|
||||||
match &self.current_input {
|
|
||||||
Some(input) => pretty_name(input.description.clone()),
|
|
||||||
None => String::from("No device selected"),
|
|
||||||
},
|
|
||||||
self.inputs
|
|
||||||
.clone()
|
|
||||||
.into_iter()
|
|
||||||
.map(|input| pretty_name(input.description))
|
|
||||||
.collect(),
|
|
||||||
Message::InputToggle,
|
|
||||||
Message::InputChanged(String::from("test")),
|
|
||||||
),
|
|
||||||
container(horizontal_rule(1))
|
container(horizontal_rule(1))
|
||||||
.padding([12, 24])
|
.padding([12, 24])
|
||||||
.width(Length::Fill),
|
.width(Length::Fill),
|
||||||
|
|
@ -393,14 +409,33 @@ fn pretty_name(name: Option<String>) -> String {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
enum PulseState {
|
enum PulseState {
|
||||||
Disconnected,
|
#[default]
|
||||||
|
Init,
|
||||||
|
Disconnected(pulse::Connection),
|
||||||
Connected(pulse::Connection),
|
Connected(pulse::Connection),
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Default for PulseState {
|
impl PulseState {
|
||||||
fn default() -> Self {
|
fn connection(&mut self) -> Option<&mut pulse::Connection> {
|
||||||
Self::Disconnected
|
match self {
|
||||||
|
PulseState::Disconnected(c) => Some(c),
|
||||||
|
PulseState::Connected(c) => Some(c),
|
||||||
|
PulseState::Init => None,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn connected(&mut self) {
|
||||||
|
if let PulseState::Disconnected(c) = self {
|
||||||
|
*self = PulseState::Connected(c.clone());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn disconnected(&mut self) {
|
||||||
|
if let PulseState::Connected(c) = self {
|
||||||
|
*self = PulseState::Disconnected(c.clone());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -20,40 +20,61 @@ pub fn connect() -> Subscription<Event> {
|
||||||
|
|
||||||
subscription::unfold(
|
subscription::unfold(
|
||||||
std::any::TypeId::of::<Connect>(),
|
std::any::TypeId::of::<Connect>(),
|
||||||
State::Disconnected,
|
State::Init,
|
||||||
|state| async move {
|
|state| async move {
|
||||||
match state {
|
match state {
|
||||||
// if app just started, or we are re-trying match here. Returns coenncting
|
State::Init => {
|
||||||
// message. We should store this in our app's state, but it isn't safe to
|
let PulseHandle {
|
||||||
// send messages until we get a conencted message. Which will be received
|
to_pulse,
|
||||||
// by the `State::Connecting` message below
|
from_pulse,
|
||||||
State::Disconnected => match PulseHandle::create() {
|
} = PulseHandle::new();
|
||||||
Ok(pulse_handle) => (None, State::Connecting(pulse_handle)),
|
(
|
||||||
Err(_) => (Some(Event::Disconnected), State::Disconnected),
|
Some(Event::Init(Connection(to_pulse))),
|
||||||
},
|
State::Connecting(from_pulse),
|
||||||
// Just a buffer to make sure the GUI doesn't send messages until pulse is ready
|
)
|
||||||
|
}
|
||||||
|
// Waiting for Connection to succeed
|
||||||
// The GUI doesn't have to monitor this state, as it is never sent to the GUI
|
// The GUI doesn't have to monitor this state, as it is never sent to the GUI
|
||||||
State::Connecting(mut pulse_handle) => {
|
State::Connecting(mut from_pulse) => match from_pulse.recv().await {
|
||||||
match pulse_handle.from_pulse.recv().await {
|
Some(Message::Connected) => {
|
||||||
Some(Message::Connected) => {(
|
(Some(Event::Connected), State::Connected(from_pulse))
|
||||||
Some(Event::Connected(Connection(pulse_handle.to_pulse))),
|
}
|
||||||
State::Connected(pulse_handle.from_pulse),
|
Some(Message::Disconnected) => {
|
||||||
)}
|
(Some(Event::Disconnected), State::Connecting(from_pulse))
|
||||||
Some(Message::Disconnected) => (Some(Event::Disconnected), State::Disconnected),
|
}
|
||||||
_ => panic!("Pulse subscription logic is faulty as the PulseServer shouldn't send unique messages until connection is successful")
|
Some(m) => {
|
||||||
}
|
log::error!("Unexpected message: {:?}", m);
|
||||||
|
(None, State::Connecting(from_pulse))
|
||||||
|
}
|
||||||
|
None => {
|
||||||
|
panic!("Pulse Sender dropped, something has gone wrong!");
|
||||||
|
}
|
||||||
},
|
},
|
||||||
State::Connected(mut from_pulse) => {
|
State::Connected(mut from_pulse) => {
|
||||||
// This is where we match messages from the pulse server to pass to the gui
|
// This is where we match messages from the pulse server to pass to the gui
|
||||||
match from_pulse.recv().await {
|
match from_pulse.recv().await {
|
||||||
Some(Message::SetSinks(sinks)) => (Some(Event::MessageReceived(Message::SetSinks(sinks))), State::Connected(from_pulse)),
|
Some(Message::SetSinks(sinks)) => (
|
||||||
Some(Message::SetSources(sources)) => (Some(Event::MessageReceived(Message::SetSources(sources))), State::Connected(from_pulse)),
|
Some(Event::MessageReceived(Message::SetSinks(sinks))),
|
||||||
Some(Message::SetDefaultSink(sink)) => (Some(Event::MessageReceived(Message::SetDefaultSink(sink))), State::Connected(from_pulse)),
|
State::Connected(from_pulse),
|
||||||
Some(Message::SetDefaultSource(source)) => (Some(Event::MessageReceived(Message::SetDefaultSource(source))), State::Connected(from_pulse)),
|
),
|
||||||
Some(Message::Disconnected) => (Some(Event::Disconnected), State::Disconnected),
|
Some(Message::SetSources(sources)) => (
|
||||||
None => (Some(Event::Disconnected), State::Disconnected),
|
Some(Event::MessageReceived(Message::SetSources(sources))),
|
||||||
_ => (None, State::Connected(from_pulse)),
|
State::Connected(from_pulse),
|
||||||
|
),
|
||||||
|
Some(Message::SetDefaultSink(sink)) => (
|
||||||
|
Some(Event::MessageReceived(Message::SetDefaultSink(sink))),
|
||||||
|
State::Connected(from_pulse),
|
||||||
|
),
|
||||||
|
Some(Message::SetDefaultSource(source)) => (
|
||||||
|
Some(Event::MessageReceived(Message::SetDefaultSource(source))),
|
||||||
|
State::Connected(from_pulse),
|
||||||
|
),
|
||||||
|
Some(Message::Disconnected) => {
|
||||||
|
(Some(Event::Disconnected), State::Connecting(from_pulse))
|
||||||
}
|
}
|
||||||
|
None => (Some(Event::Disconnected), State::Connecting(from_pulse)),
|
||||||
|
_ => (None, State::Connected(from_pulse)),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
@ -62,14 +83,15 @@ pub fn connect() -> Subscription<Event> {
|
||||||
|
|
||||||
// #[derive(Debug)]
|
// #[derive(Debug)]
|
||||||
enum State {
|
enum State {
|
||||||
Disconnected,
|
Init,
|
||||||
Connecting(PulseHandle),
|
Connecting(tokio::sync::mpsc::Receiver<Message>),
|
||||||
Connected(tokio::sync::mpsc::Receiver<Message>),
|
Connected(tokio::sync::mpsc::Receiver<Message>),
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub enum Event {
|
pub enum Event {
|
||||||
Connected(Connection),
|
Init(Connection),
|
||||||
|
Connected,
|
||||||
Disconnected,
|
Disconnected,
|
||||||
MessageReceived(Message),
|
MessageReceived(Message),
|
||||||
}
|
}
|
||||||
|
|
@ -91,6 +113,7 @@ pub enum Message {
|
||||||
Disconnected,
|
Disconnected,
|
||||||
GetSinks,
|
GetSinks,
|
||||||
GetSources,
|
GetSources,
|
||||||
|
UpdateConnection,
|
||||||
SetSinks(Vec<DeviceInfo>),
|
SetSinks(Vec<DeviceInfo>),
|
||||||
SetSources(Vec<DeviceInfo>),
|
SetSources(Vec<DeviceInfo>),
|
||||||
GetDefaultSink,
|
GetDefaultSink,
|
||||||
|
|
@ -108,34 +131,40 @@ struct PulseHandle {
|
||||||
|
|
||||||
impl PulseHandle {
|
impl PulseHandle {
|
||||||
// Create pulse server thread, and bidirectional comms
|
// Create pulse server thread, and bidirectional comms
|
||||||
pub fn create() -> Result<PulseHandle, PAErr> {
|
pub fn new() -> PulseHandle {
|
||||||
let (to_pulse, mut to_pulse_recv) = tokio::sync::mpsc::channel(10);
|
let (to_pulse, mut to_pulse_recv) = tokio::sync::mpsc::channel(10);
|
||||||
let (mut from_pulse_send, from_pulse) = tokio::sync::mpsc::channel(10);
|
let (mut from_pulse_send, from_pulse) = tokio::sync::mpsc::channel(10);
|
||||||
//let from_pulse = Arc::new(Mutex::new(vec![]));
|
// get initial connection status
|
||||||
//let mut from_pulse2 = from_pulse.clone();
|
to_pulse
|
||||||
|
.try_send(Message::UpdateConnection)
|
||||||
|
.expect("Failed to send initial connection update message");
|
||||||
// this thread should complete by pushing a completed message,
|
// this thread should complete by pushing a completed message,
|
||||||
// or fail message. This should never complete/fail without pushing
|
// or fail message. This should never complete/fail without pushing
|
||||||
// a message. This lets the iced subscription go to sleep while init
|
// a message. This lets the iced subscription go to sleep while init
|
||||||
// finishes. TLDR: be very careful with error handling
|
// finishes. TLDR: be very careful with error handling
|
||||||
thread::spawn(move || {
|
thread::spawn(move || {
|
||||||
if let Ok(mut server) = PulseServer::connect().and_then(|server| server.init()) {
|
let rt = tokio::runtime::Builder::new_current_thread()
|
||||||
PulseHandle::blocking_send_connected(&mut from_pulse_send);
|
.enable_all()
|
||||||
|
.build()
|
||||||
|
.unwrap();
|
||||||
|
|
||||||
// take `PulseServer` and handle reciver into async context
|
// take `PulseServer` and handle reciver into async context
|
||||||
// to listen for messages that need to be passed to the pulseserver
|
// to listen for messages that need to be passed to the pulseserver
|
||||||
// this lets us put the thread to sleep, but keep hold a single
|
// this lets us put the thread to sleep, but keep hold a single
|
||||||
// thread, because pulse audio's API is not multithreaded... at all
|
// thread, because pulse audio's API is not multithreaded... at all
|
||||||
let rt = tokio::runtime::Builder::new_current_thread()
|
rt.block_on(async {
|
||||||
.enable_all()
|
let mut server: Option<PulseServer> = None;
|
||||||
.build()
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
rt.block_on(async {
|
loop {
|
||||||
loop {
|
// This is where the we match messages from the GUI to pass to the pulse server
|
||||||
// This is where the we match messages from the GUI to pass to the pulse server
|
if let Some(msg) = to_pulse_recv.recv().await {
|
||||||
if let Some(msg) = to_pulse_recv.recv().await {
|
match msg {
|
||||||
match msg {
|
Message::GetDefaultSink => {
|
||||||
Message::GetDefaultSink => match server.get_default_sink() {
|
let server = match server.as_mut() {
|
||||||
|
Some(s) => s,
|
||||||
|
None => continue,
|
||||||
|
};
|
||||||
|
match server.get_default_sink() {
|
||||||
Ok(sink) => from_pulse_send
|
Ok(sink) => from_pulse_send
|
||||||
.send(Message::SetDefaultSink(sink))
|
.send(Message::SetDefaultSink(sink))
|
||||||
.await
|
.await
|
||||||
|
|
@ -143,18 +172,30 @@ impl PulseHandle {
|
||||||
Err(_) => {
|
Err(_) => {
|
||||||
PulseHandle::send_disconnected(&mut from_pulse_send).await
|
PulseHandle::send_disconnected(&mut from_pulse_send).await
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
Message::GetDefaultSource => match server.get_default_source() {
|
}
|
||||||
|
Message::GetDefaultSource => {
|
||||||
|
let server = match server.as_mut() {
|
||||||
|
Some(s) => s,
|
||||||
|
None => continue,
|
||||||
|
};
|
||||||
|
match server.get_default_source() {
|
||||||
Ok(source) => from_pulse_send
|
Ok(source) => from_pulse_send
|
||||||
.send(Message::SetDefaultSource(source))
|
.send(Message::SetDefaultSource(source))
|
||||||
.await
|
.await
|
||||||
.unwrap(),
|
.unwrap(),
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
println!("ERROR! {:?}", e);
|
log::error!("ERROR! {:?}", e);
|
||||||
PulseHandle::send_disconnected(&mut from_pulse_send).await;
|
PulseHandle::send_disconnected(&mut from_pulse_send).await;
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
Message::GetSinks => match server.get_sinks() {
|
}
|
||||||
|
Message::GetSinks => {
|
||||||
|
let server = match server.as_mut() {
|
||||||
|
Some(s) => s,
|
||||||
|
None => continue,
|
||||||
|
};
|
||||||
|
match server.get_sinks() {
|
||||||
Ok(sinks) => from_pulse_send
|
Ok(sinks) => from_pulse_send
|
||||||
.send(Message::SetSinks(sinks))
|
.send(Message::SetSinks(sinks))
|
||||||
.await
|
.await
|
||||||
|
|
@ -162,8 +203,14 @@ impl PulseHandle {
|
||||||
Err(_) => {
|
Err(_) => {
|
||||||
PulseHandle::send_disconnected(&mut from_pulse_send).await
|
PulseHandle::send_disconnected(&mut from_pulse_send).await
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
Message::GetSources => match server.get_sources() {
|
}
|
||||||
|
Message::GetSources => {
|
||||||
|
let server = match server.as_mut() {
|
||||||
|
Some(s) => s,
|
||||||
|
None => continue,
|
||||||
|
};
|
||||||
|
match server.get_sources() {
|
||||||
Ok(sinks) => from_pulse_send
|
Ok(sinks) => from_pulse_send
|
||||||
.send(Message::SetSources(sinks))
|
.send(Message::SetSources(sinks))
|
||||||
.await
|
.await
|
||||||
|
|
@ -171,36 +218,51 @@ impl PulseHandle {
|
||||||
Err(_) => {
|
Err(_) => {
|
||||||
PulseHandle::send_disconnected(&mut from_pulse_send).await
|
PulseHandle::send_disconnected(&mut from_pulse_send).await
|
||||||
}
|
}
|
||||||
},
|
|
||||||
Message::SetSinkVolumeByName(name, channel_volumes) => {
|
|
||||||
server.set_sink_volume_by_name(&name, &channel_volumes)
|
|
||||||
}
|
}
|
||||||
Message::SetSourceVolumeByName(name, channel_volumes) => {
|
}
|
||||||
server.set_source_volume_by_name(&name, &channel_volumes)
|
Message::SetSinkVolumeByName(name, channel_volumes) => {
|
||||||
}
|
let server = match server.as_mut() {
|
||||||
_ => {
|
Some(s) => s,
|
||||||
println!("message doesn't match")
|
None => continue,
|
||||||
|
};
|
||||||
|
server.set_sink_volume_by_name(&name, &channel_volumes)
|
||||||
|
}
|
||||||
|
Message::SetSourceVolumeByName(name, channel_volumes) => {
|
||||||
|
let server = match server.as_mut() {
|
||||||
|
Some(s) => s,
|
||||||
|
None => continue,
|
||||||
|
};
|
||||||
|
server.set_source_volume_by_name(&name, &channel_volumes)
|
||||||
|
}
|
||||||
|
Message::UpdateConnection => {
|
||||||
|
log::trace!("Updating Connection {:?}", server.is_some());
|
||||||
|
if let Some(mut cur_server) = server.take() {
|
||||||
|
log::trace!("getting server info...");
|
||||||
|
if let Err(_) = cur_server.get_server_info() {
|
||||||
|
PulseHandle::send_disconnected(&mut from_pulse_send).await;
|
||||||
|
} else {
|
||||||
|
server = Some(cur_server);
|
||||||
|
}
|
||||||
|
} else if let Ok(new_server) =
|
||||||
|
PulseServer::connect().and_then(|server| server.init())
|
||||||
|
{
|
||||||
|
log::trace!("got new server...");
|
||||||
|
PulseHandle::send_connected(&mut from_pulse_send).await;
|
||||||
|
server = Some(new_server);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
_ => {
|
||||||
|
log::warn!("message doesn't match")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
}
|
});
|
||||||
// Always report that server is disconnected
|
|
||||||
PulseHandle::blocking_send_disconnected(&mut from_pulse_send);
|
|
||||||
});
|
});
|
||||||
Ok(PulseHandle {
|
PulseHandle {
|
||||||
to_pulse,
|
to_pulse,
|
||||||
from_pulse,
|
from_pulse,
|
||||||
})
|
}
|
||||||
}
|
|
||||||
|
|
||||||
fn blocking_send_disconnected(sender: &mut tokio::sync::mpsc::Sender<Message>) {
|
|
||||||
sender.blocking_send(Message::Disconnected).unwrap()
|
|
||||||
}
|
|
||||||
|
|
||||||
fn blocking_send_connected(sender: &mut tokio::sync::mpsc::Sender<Message>) {
|
|
||||||
sender.blocking_send(Message::Connected).unwrap()
|
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn send_disconnected(sender: &mut tokio::sync::mpsc::Sender<Message>) {
|
async fn send_disconnected(sender: &mut tokio::sync::mpsc::Sender<Message>) {
|
||||||
|
|
|
||||||
2
debian/control
vendored
2
debian/control
vendored
|
|
@ -21,4 +21,6 @@ Architecture: amd64 arm64
|
||||||
Depends:
|
Depends:
|
||||||
${misc:Depends},
|
${misc:Depends},
|
||||||
${shlibs:Depends}
|
${shlibs:Depends}
|
||||||
|
Recommends:
|
||||||
|
pipewire-pulse
|
||||||
Description: Cosmic Applets
|
Description: Cosmic Applets
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue