175 lines
7.1 KiB
Rust
175 lines
7.1 KiB
Rust
// SPDX-License-Identifier: GPL-3.0-or-later
|
|
|
|
#[macro_use]
|
|
extern crate relm4_macros;
|
|
|
|
mod icons;
|
|
mod input;
|
|
mod now_playing;
|
|
mod output;
|
|
mod pa;
|
|
use pa::PA;
|
|
mod task;
|
|
mod volume;
|
|
|
|
use futures::{channel::mpsc, stream::StreamExt};
|
|
use gtk4::{
|
|
gio::ApplicationFlags,
|
|
glib::{self, clone, MainContext, PRIORITY_DEFAULT},
|
|
prelude::*,
|
|
Align, Application, ApplicationWindow, Box as GtkBox, Button, Image, Label, ListBox,
|
|
Orientation, PositionType, Revealer, RevealerTransitionType, Scale, SelectionMode, Separator,
|
|
};
|
|
use libpulse_binding::{
|
|
context::subscribe::{Facility, InterestMaskSet, Operation},
|
|
volume::Volume,
|
|
};
|
|
use mpris2_zbus::metadata::Metadata;
|
|
use once_cell::sync::Lazy;
|
|
use std::rc::Rc;
|
|
use tokio::runtime::Runtime;
|
|
|
|
static RT: Lazy<Runtime> = Lazy::new(|| Runtime::new().expect("failed to build tokio runtime"));
|
|
|
|
fn main() {
|
|
let application = Application::new(
|
|
Some("com.system76.cosmic.applets.audio"),
|
|
ApplicationFlags::default(),
|
|
);
|
|
application.connect_activate(app);
|
|
application.run();
|
|
}
|
|
|
|
fn app(application: &Application) {
|
|
// XXX handle no pulseaudio daemon?
|
|
let mut pa = PA::new().unwrap();
|
|
let (refresh_output_tx, mut refresh_output_rx) = mpsc::unbounded();
|
|
let (refresh_input_tx, mut refresh_input_rx) = mpsc::unbounded();
|
|
let (now_playing_tx, mut now_playing_rx) = mpsc::unbounded::<Vec<Metadata>>();
|
|
pa.context
|
|
.set_subscribe_callback(Some(Box::new(clone!(@strong refresh_output_tx, @strong refresh_input_tx => move |facility, operation, _idx| {
|
|
if !matches!(operation, Some(Operation::Changed)) {
|
|
return;
|
|
}
|
|
match facility {
|
|
Some(Facility::Sink) => {
|
|
refresh_output_tx.unbounded_send(()).expect("failed to send output refresh message");
|
|
}
|
|
Some(Facility::Source) => {
|
|
refresh_input_tx.unbounded_send(()).expect("failed to send output refresh message");
|
|
}
|
|
_ => {}
|
|
}
|
|
}))));
|
|
pa.context
|
|
.subscribe(InterestMaskSet::SINK | InterestMaskSet::SOURCE, |_| {});
|
|
let pa = Rc::new(pa);
|
|
view! {
|
|
window = ApplicationWindow {
|
|
set_application: Some(application),
|
|
set_title: Some("COSMIC Network Applet"),
|
|
set_default_width: 400,
|
|
set_default_height: 300,
|
|
|
|
set_child: window_box = Some(&GtkBox) {
|
|
set_orientation: Orientation::Vertical,
|
|
set_spacing: 24,
|
|
append: output_box = &GtkBox {
|
|
set_orientation: Orientation::Horizontal,
|
|
set_spacing: 16,
|
|
append: output_icon = &Image {
|
|
set_icon_name: Some("audio-speakers-symbolic"),
|
|
},
|
|
append: output_volume = &Scale::with_range(Orientation::Horizontal, 0., 100., 1.) {
|
|
set_format_value_func: |_, value| {
|
|
format!("{:.0}%", value)
|
|
},
|
|
set_value_pos: PositionType::Right,
|
|
set_hexpand: true
|
|
}
|
|
},
|
|
append: input_box = &GtkBox {
|
|
set_orientation: Orientation::Horizontal,
|
|
set_spacing: 16,
|
|
append: input_icon = &Image {
|
|
set_icon_name: Some("audio-input-microphone-symbolic"),
|
|
},
|
|
append: input_volume = &Scale::with_range(Orientation::Horizontal, 0., 100., 1.) {
|
|
set_format_value_func: |_, value| {
|
|
format!("{:.0}%", value)
|
|
},
|
|
set_value_pos: PositionType::Right,
|
|
set_hexpand: true
|
|
}
|
|
},
|
|
append: _sep = &Separator {
|
|
set_orientation: Orientation::Horizontal,
|
|
},
|
|
append: output_list_box = &GtkBox {
|
|
set_orientation: Orientation::Vertical,
|
|
append: current_output_button = &Button {
|
|
set_child: current_output = Some(&Label) {},
|
|
connect_clicked(outputs_revealer) => move |_| {
|
|
outputs_revealer.set_reveal_child(!outputs_revealer.reveals_child());
|
|
}
|
|
},
|
|
append: outputs_revealer = &Revealer {
|
|
set_transition_type: RevealerTransitionType::SlideDown,
|
|
set_child: outputs = Some(&ListBox) {
|
|
set_selection_mode: SelectionMode::None,
|
|
set_activate_on_single_click: true
|
|
}
|
|
}
|
|
},
|
|
append: _sep = &Separator {
|
|
set_orientation: Orientation::Horizontal,
|
|
},
|
|
append: input_list_box = &GtkBox {
|
|
set_orientation: Orientation::Vertical,
|
|
append: current_input_button = &Button {
|
|
set_child: current_input = Some(&Label) {},
|
|
connect_clicked(inputs_revealer) => move |_| {
|
|
inputs_revealer.set_reveal_child(!inputs_revealer.reveals_child());
|
|
}
|
|
},
|
|
append: inputs_revealer = &Revealer {
|
|
set_transition_type: RevealerTransitionType::SlideDown,
|
|
set_child: inputs = Some(&ListBox) {
|
|
set_selection_mode: SelectionMode::None,
|
|
set_activate_on_single_click: true
|
|
}
|
|
}
|
|
},
|
|
append: _sep = &Separator {
|
|
set_orientation: Orientation::Horizontal,
|
|
},
|
|
append: playing_apps = &ListBox {
|
|
set_selection_mode: SelectionMode::None,
|
|
}
|
|
}
|
|
}
|
|
}
|
|
glib::MainContext::default().spawn_local(
|
|
clone!(@weak inputs, @weak current_input, @weak input_volume, @strong pa => async move {
|
|
while let Some(()) = refresh_input_rx.next().await {
|
|
input::refresh_input_widgets(&pa, &inputs);
|
|
let default_input = input::refresh_default_input(&pa, ¤t_input);
|
|
// XXX volume::update_volume(&default_input, &input_volume);
|
|
}
|
|
}),
|
|
);
|
|
glib::MainContext::default().spawn_local(
|
|
clone!(@weak outputs, @weak current_output, @weak output_volume, @strong pa => async move {
|
|
while let Some(()) = refresh_output_rx.next().await {
|
|
output::refresh_output_widgets(&pa, &outputs);
|
|
let default_output = output::refresh_default_output(&pa, ¤t_output);
|
|
// XXX volume::update_volume(&default_output, &output_volume);
|
|
}
|
|
}),
|
|
);
|
|
glib::MainContext::default().spawn_local(clone!(@weak playing_apps => async move {
|
|
while let Some(all_metadata) = now_playing_rx.next().await {
|
|
}
|
|
}));
|
|
window.show();
|
|
}
|