musicus/src/dialogs/work_editor.rs

177 lines
5.4 KiB
Rust
Raw Normal View History

2020-09-28 16:00:51 +02:00
use crate::database::*;
use glib::clone;
use gtk::prelude::*;
use gtk_macros::get_widget;
use std::cell::RefCell;
use std::convert::TryInto;
use std::rc::Rc;
struct PartOrSection {
part: Option<WorkPartDescription>,
section: Option<WorkSectionDescription>,
}
impl PartOrSection {
pub fn part(part: WorkPartDescription) -> Self {
PartOrSection {
part: Some(part),
section: None,
}
}
pub fn section(section: WorkSectionDescription) -> Self {
PartOrSection {
part: None,
section: Some(section),
}
}
pub fn is_part(&self) -> bool {
self.part.is_some()
}
pub fn unwrap_part(&self) -> WorkPartDescription {
self.part.as_ref().unwrap().clone()
}
pub fn unwrap_section(&self) -> WorkSectionDescription {
self.section.as_ref().unwrap().clone()
}
}
pub struct WorkEditor {
window: gtk::Window,
save_button: gtk::Button,
id: i64,
title_entry: gtk::Entry,
composer: RefCell<Option<Person>>,
instruments: RefCell<Vec<Instrument>>,
structure: RefCell<Vec<PartOrSection>>,
}
impl WorkEditor {
pub fn new<F: Fn(WorkDescription) -> () + 'static, P: IsA<gtk::Window>>(
db: Rc<Database>,
parent: &P,
work: Option<WorkDescription>,
callback: F,
) -> Rc<Self> {
let builder = gtk::Builder::from_resource("/de/johrpan/musicus_editor/ui/work_editor.ui");
get_widget!(builder, gtk::Window, window);
get_widget!(builder, gtk::Button, cancel_button);
get_widget!(builder, gtk::Button, save_button);
get_widget!(builder, gtk::Entry, title_entry);
get_widget!(builder, gtk::Button, composer_button);
get_widget!(builder, gtk::Label, composer_label);
get_widget!(builder, gtk::ListBox, instrument_list);
get_widget!(builder, gtk::Button, add_instrument_button);
get_widget!(builder, gtk::Button, remove_instrument_button);
get_widget!(builder, gtk::ListBox, part_list);
get_widget!(builder, gtk::Button, add_part_button);
get_widget!(builder, gtk::Button, remove_part_button);
get_widget!(builder, gtk::Button, add_section_button);
get_widget!(builder, gtk::Button, edit_part_button);
get_widget!(builder, gtk::Button, move_part_up_button);
get_widget!(builder, gtk::Button, move_part_down_button);
let id = match work.clone() {
Some(work) => {
title_entry.set_text(&work.title);
work.id
}
None => rand::random::<u32>().into(),
};
let composer = RefCell::new(match work.clone() {
Some(work) => {
save_button.set_sensitive(true);
Some(work.composer)
}
None => None,
});
let instruments = RefCell::new(match work.clone() {
Some(work) => work.instruments,
None => Vec::new(),
});
let structure = RefCell::new(match work.clone() {
Some(work) => {
let mut result = Vec::new();
for part in work.parts {
result.push(PartOrSection::part(part));
}
for section in work.sections {
result.insert(
section
.before_index
.try_into()
.expect("Section with unrealistic before_index!"),
PartOrSection::section(section),
);
}
result
}
None => Vec::new(),
});
let result = Rc::new(WorkEditor {
window: window,
save_button: save_button,
id: id,
title_entry: title_entry,
composer: composer,
instruments: instruments,
structure: structure,
});
cancel_button.connect_clicked(clone!(@strong result => move |_| {
result.window.close();
}));
result.save_button.connect_clicked(clone!(@strong result => move |_| {
result.window.close();
let mut section_count: i64 = 0;
let mut parts: Vec<WorkPartDescription> = Vec::new();
let mut sections: Vec<WorkSectionDescription> = Vec::new();
for (index, pos) in result.structure.borrow().iter().enumerate() {
if pos.is_part() {
parts.push(pos.unwrap_part());
} else {
let mut section = pos.unwrap_section();
let index: i64 = index.try_into().unwrap();
section.before_index = index - section_count;
sections.push(section);
section_count += 1;
}
}
let work = WorkDescription {
id: result.id,
title: result.title_entry.get_text().to_string(),
composer: result.composer.borrow().clone().expect("Tried to create work without composer!"),
instruments: result.instruments.borrow().to_vec(),
parts: parts,
sections: sections,
};
db.update_work(work.clone().into());
callback(work);
}));
result.window.set_transient_for(Some(parent));
result
}
pub fn show(&self) {
self.window.show();
}
}