musicus/src/import/track_set_editor.rs

222 lines
7.3 KiB
Rust
Raw Normal View History

use super::source::Source;
2021-01-13 19:27:22 +01:00
use super::track_editor::TrackEditor;
use super::track_selector::TrackSelector;
use crate::backend::Backend;
2021-01-25 14:00:57 +01:00
use crate::database::Recording;
2021-02-03 14:09:17 +01:00
use crate::navigator::{NavigationHandle, Screen};
2021-02-03 15:46:48 +01:00
use crate::selectors::{PersonSelector, RecordingSelector};
2021-02-03 14:09:17 +01:00
use crate::widgets::{List, Widget};
2021-01-13 19:27:22 +01:00
use gettextrs::gettext;
use glib::clone;
use gtk::prelude::*;
use gtk_macros::get_widget;
2021-01-29 15:17:27 +01:00
use libadwaita::prelude::*;
2021-01-25 14:00:57 +01:00
use std::cell::RefCell;
2021-01-13 19:27:22 +01:00
use std::rc::Rc;
/// A track set before being imported.
#[derive(Clone, Debug)]
pub struct TrackSetData {
pub recording: Recording,
pub tracks: Vec<TrackData>,
}
/// A track before being imported.
#[derive(Clone, Debug)]
pub struct TrackData {
/// Index of the track source within the medium source's tracks.
pub track_source: usize,
/// Actual track data.
pub work_parts: Vec<usize>,
}
/// A screen for editing a set of tracks for one recording.
pub struct TrackSetEditor {
2021-02-03 14:09:17 +01:00
handle: NavigationHandle<TrackSetData>,
source: Rc<Box<dyn Source>>,
2021-01-13 19:27:22 +01:00
widget: gtk::Box,
save_button: gtk::Button,
2021-01-29 15:17:27 +01:00
recording_row: libadwaita::ActionRow,
2021-01-25 14:00:57 +01:00
track_list: Rc<List>,
2021-01-13 19:27:22 +01:00
recording: RefCell<Option<Recording>>,
tracks: RefCell<Vec<TrackData>>,
}
2021-02-03 14:09:17 +01:00
impl Screen<Rc<Box<dyn Source>>, TrackSetData> for TrackSetEditor {
2021-01-13 19:27:22 +01:00
/// Create a new track set editor.
2021-02-03 14:09:17 +01:00
fn new(source: Rc<Box<dyn Source>>, handle: NavigationHandle<TrackSetData>) -> Rc<Self> {
2021-01-13 19:27:22 +01:00
// Create UI
let builder = gtk::Builder::from_resource("/de/johrpan/musicus/ui/track_set_editor.ui");
get_widget!(builder, gtk::Box, widget);
get_widget!(builder, gtk::Button, back_button);
get_widget!(builder, gtk::Button, save_button);
2021-01-29 15:17:27 +01:00
get_widget!(builder, libadwaita::ActionRow, recording_row);
2021-01-13 19:27:22 +01:00
get_widget!(builder, gtk::Button, select_recording_button);
get_widget!(builder, gtk::Button, edit_tracks_button);
get_widget!(builder, gtk::Frame, tracks_frame);
2021-01-25 14:00:57 +01:00
let track_list = List::new();
tracks_frame.set_child(Some(&track_list.widget));
2021-01-13 19:27:22 +01:00
let this = Rc::new(Self {
2021-02-03 14:09:17 +01:00
handle,
2021-01-13 19:27:22 +01:00
source,
widget,
save_button,
recording_row,
track_list,
recording: RefCell::new(None),
tracks: RefCell::new(Vec::new()),
});
// Connect signals and callbacks
2021-02-03 14:09:17 +01:00
back_button.connect_clicked(clone!(@weak this => move |_| {
this.handle.pop(None);
2021-01-13 19:27:22 +01:00
}));
2021-02-03 14:09:17 +01:00
this.save_button.connect_clicked(clone!(@weak this => move |_| {
let data = TrackSetData {
recording: this.recording.borrow().clone().unwrap(),
tracks: this.tracks.borrow().clone(),
};
2021-01-13 19:56:27 +01:00
2021-02-03 14:09:17 +01:00
this.handle.pop(Some(data));
2021-01-13 19:27:22 +01:00
}));
2021-02-03 14:09:17 +01:00
select_recording_button.connect_clicked(clone!(@weak this => move |_| {
2021-02-03 15:46:48 +01:00
spawn!(@clone this, async move {
if let Some(recording) = push!(this.handle, RecordingSelector).await {
this.recording.replace(Some(recording));
this.recording_selected();
}
});
2021-01-13 19:27:22 +01:00
}));
2021-02-03 14:09:17 +01:00
edit_tracks_button.connect_clicked(clone!(@weak this => move |_| {
spawn!(@clone this, async move {
if let Some(selection) = push!(this.handle, TrackSelector, Rc::clone(&this.source)).await {
2021-01-13 19:27:22 +01:00
let mut tracks = Vec::new();
for index in selection {
let data = TrackData {
track_source: index,
work_parts: Vec::new(),
};
tracks.push(data);
}
let length = tracks.len();
this.tracks.replace(tracks);
this.track_list.update(length);
this.autofill_parts();
2021-02-03 14:09:17 +01:00
}
});
2021-01-13 19:27:22 +01:00
}));
2021-02-03 14:09:17 +01:00
this.track_list.set_make_widget_cb(clone!(@weak this => move |index| {
2021-01-13 19:27:22 +01:00
let track = &this.tracks.borrow()[index];
let mut title_parts = Vec::<String>::new();
if let Some(recording) = &*this.recording.borrow() {
for part in &track.work_parts {
title_parts.push(recording.work.parts[*part].title.clone());
}
}
let title = if title_parts.is_empty() {
gettext("Unknown")
} else {
title_parts.join(", ")
};
let tracks = this.source.tracks().unwrap();
2021-01-29 17:51:44 +01:00
let track_name = &tracks[track.track_source].name;
2021-01-13 19:27:22 +01:00
2021-01-25 14:00:57 +01:00
let edit_image = gtk::Image::from_icon_name(Some("document-edit-symbolic"));
2021-01-13 19:27:22 +01:00
let edit_button = gtk::Button::new();
2021-01-25 14:00:57 +01:00
edit_button.set_has_frame(false);
2021-01-13 19:27:22 +01:00
edit_button.set_valign(gtk::Align::Center);
2021-01-25 14:00:57 +01:00
edit_button.set_child(Some(&edit_image));
2021-01-13 19:27:22 +01:00
2021-01-29 15:17:27 +01:00
let row = libadwaita::ActionRow::new();
2021-01-13 19:27:22 +01:00
row.set_activatable(true);
row.set_title(Some(&title));
2021-01-29 17:51:44 +01:00
row.set_subtitle(Some(track_name));
2021-01-25 14:00:57 +01:00
row.add_suffix(&edit_button);
2021-01-13 19:27:22 +01:00
row.set_activatable_widget(Some(&edit_button));
2021-02-03 14:09:17 +01:00
edit_button.connect_clicked(clone!(@weak this => move |_| {
2021-01-13 19:27:22 +01:00
let recording = this.recording.borrow().clone();
2021-02-03 14:09:17 +01:00
if let Some(recording) = recording {
spawn!(@clone this, async move {
let track = &this.tracks.borrow()[index];
if let Some(selection) = push!(this.handle, TrackEditor, (recording, track.work_parts.clone())).await {
{
let mut tracks = this.tracks.borrow_mut();
let mut track = &mut tracks[index];
track.work_parts = selection;
};
this.update_tracks();
}
});
2021-01-13 19:27:22 +01:00
}
}));
row.upcast()
}));
this
}
2021-02-03 14:09:17 +01:00
}
2021-01-13 19:27:22 +01:00
2021-02-03 14:09:17 +01:00
impl TrackSetEditor {
2021-01-13 19:27:22 +01:00
/// Set everything up after selecting a recording.
fn recording_selected(&self) {
if let Some(recording) = &*self.recording.borrow() {
self.recording_row.set_title(Some(&recording.work.get_title()));
self.recording_row.set_subtitle(Some(&recording.get_performers()));
self.save_button.set_sensitive(true);
}
self.autofill_parts();
}
/// Automatically try to put work part information from the selected recording into the
/// selected tracks.
fn autofill_parts(&self) {
if let Some(recording) = &*self.recording.borrow() {
let mut tracks = self.tracks.borrow_mut();
for (index, _) in recording.work.parts.iter().enumerate() {
if let Some(mut track) = tracks.get_mut(index) {
track.work_parts = vec![index];
} else {
break;
}
}
}
self.update_tracks();
}
/// Update the track list.
fn update_tracks(&self) {
let length = self.tracks.borrow().len();
self.track_list.update(length);
}
}
2021-02-03 14:09:17 +01:00
impl Widget for TrackSetEditor {
2021-01-13 19:27:22 +01:00
fn get_widget(&self) -> gtk::Widget {
self.widget.clone().upcast()
}
}