2020-09-28 17:19:34 +02:00
|
|
|
use super::selector_row::SelectorRow;
|
2020-09-28 18:12:59 +02:00
|
|
|
use super::{InstrumentSelector, PersonSelector, PartEditor, SectionEditor};
|
2020-10-09 12:22:02 +02:00
|
|
|
use crate::backend::*;
|
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()
|
|
|
|
|
}
|
2020-09-28 17:51:02 +02:00
|
|
|
|
|
|
|
|
pub fn get_title(&self) -> String {
|
|
|
|
|
if self.is_part() {
|
|
|
|
|
self.unwrap_part().title
|
|
|
|
|
} else {
|
|
|
|
|
self.unwrap_section().title
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-28 16:00:51 +02:00
|
|
|
}
|
|
|
|
|
|
2020-10-09 12:22:02 +02:00
|
|
|
pub struct WorkEditor<F>
|
|
|
|
|
where
|
|
|
|
|
F: Fn(WorkDescription) -> () + 'static, {
|
|
|
|
|
backend: Rc<Backend>,
|
2020-09-28 16:00:51 +02:00
|
|
|
window: gtk::Window,
|
2020-10-09 12:22:02 +02:00
|
|
|
callback: F,
|
2020-09-28 16:00:51 +02:00
|
|
|
save_button: gtk::Button,
|
|
|
|
|
id: i64,
|
|
|
|
|
title_entry: gtk::Entry,
|
|
|
|
|
composer: RefCell<Option<Person>>,
|
2020-09-28 16:52:59 +02:00
|
|
|
composer_label: gtk::Label,
|
2020-09-28 16:00:51 +02:00
|
|
|
instruments: RefCell<Vec<Instrument>>,
|
2020-09-28 17:19:34 +02:00
|
|
|
instrument_list: gtk::ListBox,
|
2020-09-28 16:00:51 +02:00
|
|
|
structure: RefCell<Vec<PartOrSection>>,
|
2020-09-28 17:51:02 +02:00
|
|
|
part_list: gtk::ListBox,
|
2020-09-28 16:00:51 +02:00
|
|
|
}
|
|
|
|
|
|
2020-10-09 12:22:02 +02:00
|
|
|
impl<F> WorkEditor<F>
|
|
|
|
|
where
|
|
|
|
|
F: Fn(WorkDescription) -> () + 'static, {
|
|
|
|
|
pub fn new<P: IsA<gtk::Window>>(
|
|
|
|
|
backend: Rc<Backend>,
|
2020-09-28 16:00:51 +02:00
|
|
|
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 {
|
2020-10-09 12:22:02 +02:00
|
|
|
backend: backend,
|
2020-09-28 16:00:51 +02:00
|
|
|
window: window,
|
2020-10-09 12:22:02 +02:00
|
|
|
callback: callback,
|
2020-09-28 16:00:51 +02:00
|
|
|
save_button: save_button,
|
|
|
|
|
id: id,
|
|
|
|
|
title_entry: title_entry,
|
|
|
|
|
composer: composer,
|
2020-09-28 16:52:59 +02:00
|
|
|
composer_label: composer_label,
|
2020-09-28 16:00:51 +02:00
|
|
|
instruments: instruments,
|
2020-09-28 17:19:34 +02:00
|
|
|
instrument_list: instrument_list,
|
2020-09-28 16:00:51 +02:00
|
|
|
structure: structure,
|
2020-09-28 17:51:02 +02:00
|
|
|
part_list: part_list,
|
2020-09-28 16:00:51 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
|
|
cancel_button.connect_clicked(clone!(@strong result => move |_| {
|
|
|
|
|
result.window.close();
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
result.save_button.connect_clicked(clone!(@strong result => move |_| {
|
|
|
|
|
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,
|
|
|
|
|
};
|
|
|
|
|
|
2020-10-09 12:22:02 +02:00
|
|
|
result.backend.update_work(work.clone().into(), clone!(@strong result => move |_| {
|
|
|
|
|
result.window.close();
|
|
|
|
|
(result.callback)(work.clone());
|
|
|
|
|
}));
|
2020-09-28 16:00:51 +02:00
|
|
|
}));
|
|
|
|
|
|
2020-09-28 16:52:59 +02:00
|
|
|
composer_button.connect_clicked(clone!(@strong result => move |_| {
|
2020-10-09 12:22:02 +02:00
|
|
|
PersonSelector::new(result.backend.clone(), &result.window, clone!(@strong result => move |person| {
|
2020-09-28 16:52:59 +02:00
|
|
|
result.composer.replace(Some(person.clone()));
|
|
|
|
|
result.composer_label.set_text(&person.name_fl());
|
|
|
|
|
result.save_button.set_sensitive(true);
|
|
|
|
|
})).show();
|
|
|
|
|
}));
|
|
|
|
|
|
2020-09-28 17:19:34 +02:00
|
|
|
add_instrument_button.connect_clicked(clone!(@strong result => move |_| {
|
2020-10-09 12:22:02 +02:00
|
|
|
InstrumentSelector::new(result.backend.clone(), &result.window, clone!(@strong result => move |instrument| {
|
2020-09-28 17:19:34 +02:00
|
|
|
{
|
|
|
|
|
let mut instruments = result.instruments.borrow_mut();
|
|
|
|
|
instruments.push(instrument);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result.show_instruments();
|
|
|
|
|
})).show();
|
|
|
|
|
}));
|
|
|
|
|
|
2020-09-28 17:26:23 +02:00
|
|
|
remove_instrument_button.connect_clicked(clone!(@strong result => move |_| {
|
|
|
|
|
let row = result.get_selected_instrument_row();
|
|
|
|
|
match row {
|
|
|
|
|
Some(row) => {
|
|
|
|
|
let index = row.get_index();
|
|
|
|
|
let index: usize = index.try_into().unwrap();
|
|
|
|
|
result.instruments.borrow_mut().remove(index);
|
|
|
|
|
result.show_instruments();
|
|
|
|
|
}
|
|
|
|
|
None => (),
|
|
|
|
|
}
|
|
|
|
|
}));
|
|
|
|
|
|
2020-09-28 17:51:02 +02:00
|
|
|
add_part_button.connect_clicked(clone!(@strong result => move |_| {
|
2020-10-09 12:22:02 +02:00
|
|
|
PartEditor::new(result.backend.clone(), &result.window, None, clone!(@strong result => move |part| {
|
2020-09-28 17:51:02 +02:00
|
|
|
{
|
|
|
|
|
let mut structure = result.structure.borrow_mut();
|
|
|
|
|
structure.push(PartOrSection::part(part));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result.show_parts();
|
|
|
|
|
})).show();
|
|
|
|
|
}));
|
|
|
|
|
|
2020-09-28 18:12:59 +02:00
|
|
|
add_section_button.connect_clicked(clone!(@strong result => move |_| {
|
|
|
|
|
SectionEditor::new(&result.window, None, clone!(@strong result => move |section| {
|
|
|
|
|
{
|
|
|
|
|
let mut structure = result.structure.borrow_mut();
|
|
|
|
|
structure.push(PartOrSection::section(section));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
result.show_parts();
|
|
|
|
|
})).show();
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
edit_part_button.connect_clicked(clone!(@strong result => move |_| {
|
2020-09-28 17:51:02 +02:00
|
|
|
let row = result.get_selected_part_row();
|
|
|
|
|
match row {
|
|
|
|
|
Some(row) => {
|
|
|
|
|
let index = row.get_index();
|
|
|
|
|
let index: usize = index.try_into().unwrap();
|
2020-09-28 18:12:59 +02:00
|
|
|
let pos = &result.structure.borrow()[index];
|
|
|
|
|
|
|
|
|
|
if pos.is_part() {
|
|
|
|
|
let editor =
|
2020-10-09 12:22:02 +02:00
|
|
|
PartEditor::new(result.backend.clone(), &result.window, Some(pos.unwrap_part()), clone!(@strong result => move |part| {
|
2020-09-28 18:12:59 +02:00
|
|
|
result.structure.borrow_mut()[index] = PartOrSection::part(part);
|
|
|
|
|
result.show_parts();
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
editor.show();
|
|
|
|
|
} else {
|
|
|
|
|
let editor =
|
|
|
|
|
SectionEditor::new(&result.window, Some(pos.unwrap_section()), clone!(@strong result => move |section| {
|
|
|
|
|
result.structure.borrow_mut()[index] = PartOrSection::section(section);
|
|
|
|
|
result.show_parts();
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
editor.show();
|
|
|
|
|
}
|
2020-09-28 17:51:02 +02:00
|
|
|
}
|
|
|
|
|
None => (),
|
|
|
|
|
}
|
|
|
|
|
}));
|
|
|
|
|
|
2020-09-28 18:12:59 +02:00
|
|
|
remove_part_button.connect_clicked(clone!(@strong result => move |_| {
|
2020-09-28 17:51:02 +02:00
|
|
|
let row = result.get_selected_part_row();
|
|
|
|
|
match row {
|
|
|
|
|
Some(row) => {
|
|
|
|
|
let index = row.get_index();
|
|
|
|
|
let index: usize = index.try_into().unwrap();
|
2020-09-28 18:12:59 +02:00
|
|
|
result.structure.borrow_mut().remove(index);
|
|
|
|
|
result.show_parts();
|
2020-09-28 17:51:02 +02:00
|
|
|
}
|
|
|
|
|
None => (),
|
|
|
|
|
}
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
move_part_up_button.connect_clicked(clone!(@strong result => move |_| {
|
|
|
|
|
let row = result.get_selected_part_row();
|
|
|
|
|
match row {
|
|
|
|
|
Some(row) => {
|
|
|
|
|
let index = row.get_index();
|
|
|
|
|
if index > 0 {
|
|
|
|
|
let index: usize = index.try_into().unwrap();
|
|
|
|
|
result.structure.borrow_mut().swap(index - 1, index);
|
|
|
|
|
result.show_parts();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
None => (),
|
|
|
|
|
}
|
|
|
|
|
}));
|
|
|
|
|
|
|
|
|
|
move_part_down_button.connect_clicked(clone!(@strong result => move |_| {
|
|
|
|
|
let row = result.get_selected_part_row();
|
|
|
|
|
match row {
|
|
|
|
|
Some(row) => {
|
|
|
|
|
let index = row.get_index();
|
|
|
|
|
let index: usize = index.try_into().unwrap();
|
|
|
|
|
if index < result.structure.borrow().len() - 1 {
|
|
|
|
|
result.structure.borrow_mut().swap(index, index + 1);
|
|
|
|
|
result.show_parts();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
None => (),
|
|
|
|
|
}
|
|
|
|
|
}));
|
|
|
|
|
|
2020-09-28 16:00:51 +02:00
|
|
|
result.window.set_transient_for(Some(parent));
|
|
|
|
|
|
|
|
|
|
result
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn show(&self) {
|
|
|
|
|
self.window.show();
|
|
|
|
|
}
|
2020-09-28 17:19:34 +02:00
|
|
|
|
|
|
|
|
fn show_instruments(&self) {
|
|
|
|
|
for child in self.instrument_list.get_children() {
|
|
|
|
|
self.instrument_list.remove(&child);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (index, instrument) in self.instruments.borrow().iter().enumerate() {
|
|
|
|
|
let label = gtk::Label::new(Some(&instrument.name));
|
|
|
|
|
label.set_halign(gtk::Align::Start);
|
|
|
|
|
let row = SelectorRow::new(index.try_into().unwrap(), &label);
|
|
|
|
|
row.show_all();
|
|
|
|
|
self.instrument_list.insert(&row, -1);
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-28 17:26:23 +02:00
|
|
|
|
|
|
|
|
fn get_selected_instrument_row(&self) -> Option<SelectorRow> {
|
|
|
|
|
match self.instrument_list.get_selected_rows().first() {
|
|
|
|
|
Some(row) => match row.get_child() {
|
|
|
|
|
Some(child) => Some(child.downcast().unwrap()),
|
|
|
|
|
None => None,
|
|
|
|
|
},
|
|
|
|
|
None => None,
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-28 17:51:02 +02:00
|
|
|
|
|
|
|
|
fn show_parts(&self) {
|
|
|
|
|
for child in self.part_list.get_children() {
|
|
|
|
|
self.part_list.remove(&child);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (index, part) in self.structure.borrow().iter().enumerate() {
|
|
|
|
|
let label = gtk::Label::new(Some(&part.get_title()));
|
|
|
|
|
label.set_halign(gtk::Align::Start);
|
2020-09-28 18:12:59 +02:00
|
|
|
|
|
|
|
|
if part.is_part() {
|
|
|
|
|
label.set_margin_start(6);
|
|
|
|
|
} else {
|
|
|
|
|
let attributes = pango::AttrList::new();
|
|
|
|
|
attributes.insert(pango::Attribute::new_weight(pango::Weight::Bold).unwrap());
|
|
|
|
|
label.set_attributes(Some(&attributes));
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-28 17:51:02 +02:00
|
|
|
let row = SelectorRow::new(index.try_into().unwrap(), &label);
|
|
|
|
|
row.show_all();
|
|
|
|
|
self.part_list.insert(&row, -1);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn get_selected_part_row(&self) -> Option<SelectorRow> {
|
|
|
|
|
match self.part_list.get_selected_rows().first() {
|
|
|
|
|
Some(row) => match row.get_child() {
|
|
|
|
|
Some(child) => Some(child.downcast().unwrap()),
|
|
|
|
|
None => None,
|
|
|
|
|
},
|
|
|
|
|
None => None,
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-28 16:00:51 +02:00
|
|
|
}
|