mirror of
https://github.com/johrpan/musicus.git
synced 2025-10-26 11:47:25 +01:00
252 lines
7.5 KiB
Rust
252 lines
7.5 KiB
Rust
use std::cell::{Cell, OnceCell};
|
|
|
|
use gtk::{
|
|
gdk,
|
|
glib::{self, clone, subclass::Signal, Properties},
|
|
prelude::*,
|
|
subclass::prelude::*,
|
|
};
|
|
use once_cell::sync::Lazy;
|
|
|
|
use crate::player::Player;
|
|
|
|
mod imp {
|
|
use super::*;
|
|
|
|
#[derive(Properties, Debug, Default, gtk::CompositeTemplate)]
|
|
#[properties(wrapper_type = super::PlayerBar)]
|
|
#[template(file = "data/ui/player_bar.blp")]
|
|
pub struct PlayerBar {
|
|
#[property(get, construct_only)]
|
|
pub player: OnceCell<Player>,
|
|
|
|
pub seeking: Cell<bool>,
|
|
|
|
#[template_child]
|
|
pub title_label: TemplateChild<gtk::Label>,
|
|
#[template_child]
|
|
pub subtitle_label: TemplateChild<gtk::Label>,
|
|
#[template_child]
|
|
pub back_button: TemplateChild<gtk::Button>,
|
|
#[template_child]
|
|
pub playlist_button: TemplateChild<gtk::ToggleButton>,
|
|
#[template_child]
|
|
pub forward_button: TemplateChild<gtk::Button>,
|
|
#[template_child]
|
|
pub play_button: TemplateChild<gtk::Button>,
|
|
#[template_child]
|
|
pub current_time_label: TemplateChild<gtk::Label>,
|
|
#[template_child]
|
|
pub slider: TemplateChild<gtk::Scale>,
|
|
#[template_child]
|
|
pub remaining_time_label: TemplateChild<gtk::Label>,
|
|
}
|
|
|
|
impl PlayerBar {
|
|
fn update_item(&self) {
|
|
if let Some(item) = self.player.get().unwrap().current_item() {
|
|
self.title_label.set_label(&item.make_title());
|
|
|
|
if let Some(subtitle) = item.make_subtitle() {
|
|
self.subtitle_label.set_label(&subtitle);
|
|
self.subtitle_label.set_visible(true);
|
|
} else {
|
|
self.subtitle_label.set_visible(false);
|
|
}
|
|
}
|
|
}
|
|
|
|
fn update_time(&self) {
|
|
let player = self.player.get().unwrap();
|
|
|
|
let current_time_ms = if self.seeking.get() {
|
|
(self.slider.value() * player.duration_ms() as f64) as u64
|
|
} else {
|
|
let current_time_ms = player.position_ms();
|
|
self.slider
|
|
.set_value(current_time_ms as f64 / player.duration_ms() as f64);
|
|
current_time_ms
|
|
};
|
|
|
|
self.current_time_label
|
|
.set_label(&format_time(current_time_ms));
|
|
|
|
self.remaining_time_label.set_label(&format_time(
|
|
player.duration_ms().saturating_sub(current_time_ms),
|
|
));
|
|
}
|
|
}
|
|
|
|
#[glib::object_subclass]
|
|
impl ObjectSubclass for PlayerBar {
|
|
const NAME: &'static str = "MusicusPlayerBar";
|
|
type Type = super::PlayerBar;
|
|
type ParentType = gtk::Box;
|
|
|
|
fn class_init(klass: &mut Self::Class) {
|
|
klass.bind_template();
|
|
klass.bind_template_instance_callbacks();
|
|
}
|
|
|
|
fn instance_init(obj: &glib::subclass::InitializingObject<Self>) {
|
|
obj.init_template();
|
|
}
|
|
}
|
|
|
|
#[glib::derived_properties]
|
|
impl ObjectImpl for PlayerBar {
|
|
fn signals() -> &'static [Signal] {
|
|
static SIGNALS: Lazy<Vec<Signal>> = Lazy::new(|| {
|
|
vec![Signal::builder("show-playlist")
|
|
.param_types([glib::Type::BOOL])
|
|
.build()]
|
|
});
|
|
|
|
SIGNALS.as_ref()
|
|
}
|
|
|
|
fn constructed(&self) {
|
|
self.parent_constructed();
|
|
|
|
let player = self.player.get().unwrap();
|
|
|
|
player
|
|
.bind_property("playing", &self.play_button.get(), "icon-name")
|
|
.transform_to(|_, playing| {
|
|
Some(if playing {
|
|
"media-playback-pause-symbolic"
|
|
} else {
|
|
"media-playback-start-symbolic"
|
|
})
|
|
})
|
|
.sync_create()
|
|
.build();
|
|
|
|
let obj = self.obj();
|
|
|
|
player.connect_current_index_notify(clone!(
|
|
#[weak]
|
|
obj,
|
|
move |_| obj.imp().update_item()
|
|
));
|
|
player.playlist().connect_items_changed(clone!(
|
|
#[weak]
|
|
obj,
|
|
move |_, _, _, _| obj.imp().update_item()
|
|
));
|
|
|
|
player.connect_position_ms_notify(clone!(
|
|
#[weak]
|
|
obj,
|
|
move |_| obj.imp().update_time()
|
|
));
|
|
player.connect_duration_ms_notify(clone!(
|
|
#[weak]
|
|
obj,
|
|
move |_| obj.imp().update_time()
|
|
));
|
|
|
|
let seeking_controller = gtk::EventControllerLegacy::new();
|
|
|
|
seeking_controller.connect_event(clone!(
|
|
#[weak]
|
|
obj,
|
|
#[upgrade_or]
|
|
glib::Propagation::Proceed,
|
|
move |_, event| {
|
|
if let Some(event) = event.downcast_ref::<gdk::ButtonEvent>() {
|
|
let imp = obj.imp();
|
|
if event.button() == gdk::BUTTON_PRIMARY {
|
|
match event.event_type() {
|
|
gdk::EventType::ButtonPress => {
|
|
imp.seeking.set(true);
|
|
}
|
|
gdk::EventType::ButtonRelease => {
|
|
let player = obj.player();
|
|
player.seek_to(
|
|
(imp.slider.value() * player.duration_ms() as f64) as u64,
|
|
);
|
|
imp.seeking.set(false);
|
|
}
|
|
_ => (),
|
|
}
|
|
}
|
|
}
|
|
|
|
glib::Propagation::Proceed
|
|
}
|
|
));
|
|
|
|
self.slider.add_controller(seeking_controller);
|
|
|
|
self.slider.connect_value_changed(clone!(
|
|
#[weak]
|
|
obj,
|
|
move |_| obj.imp().update_time()
|
|
));
|
|
}
|
|
}
|
|
|
|
impl WidgetImpl for PlayerBar {}
|
|
impl BoxImpl for PlayerBar {}
|
|
}
|
|
|
|
glib::wrapper! {
|
|
pub struct PlayerBar(ObjectSubclass<imp::PlayerBar>)
|
|
@extends gtk::Widget, adw::Bin;
|
|
}
|
|
|
|
#[gtk::template_callbacks]
|
|
impl PlayerBar {
|
|
pub fn new(player: &Player) -> Self {
|
|
glib::Object::builder().property("player", player).build()
|
|
}
|
|
|
|
pub fn connect_show_playlist<F: Fn(&Self, bool) + 'static>(
|
|
&self,
|
|
f: F,
|
|
) -> glib::SignalHandlerId {
|
|
self.connect_local("show-playlist", true, move |values| {
|
|
let obj = values[0].get::<Self>().unwrap();
|
|
let show = values[1].get::<bool>().unwrap();
|
|
f(&obj, show);
|
|
None
|
|
})
|
|
}
|
|
|
|
pub fn playlist_hidden(&self) {
|
|
self.imp().playlist_button.set_active(false);
|
|
}
|
|
|
|
#[template_callback]
|
|
fn previous(&self) {
|
|
self.player().previous();
|
|
}
|
|
|
|
#[template_callback]
|
|
fn show_playlist(&self, button: >k::ToggleButton) {
|
|
self.emit_by_name::<()>("show-playlist", &[&button.is_active()]);
|
|
}
|
|
|
|
#[template_callback]
|
|
fn next(&self) {
|
|
self.player().next();
|
|
}
|
|
|
|
#[template_callback]
|
|
fn play_pause(&self) {
|
|
self.player().play_pause();
|
|
}
|
|
}
|
|
|
|
fn format_time(time_ms: u64) -> String {
|
|
let s = time_ms / 1000;
|
|
let (m, s) = (s / 60, s % 60);
|
|
let (h, m) = (m / 60, m % 60);
|
|
|
|
if h > 0 {
|
|
format!("{h:0>2}:{m:0>2}:{s:0>2}")
|
|
} else {
|
|
format!("{m:0>2}:{s:0>2}")
|
|
}
|
|
}
|