meli/meli/src/contacts/editor.rs

302 lines
10 KiB
Rust

/*
* meli - contacts module
*
* Copyright 2019 Manos Pitsidianakis
*
* This file is part of meli.
*
* meli is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* meli is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with meli. If not, see <http://www.gnu.org/licenses/>.
*/
use std::collections::HashMap;
use melib::Card;
use crate::{
terminal::*, Action::*, CellBuffer, Component, ComponentId, Context, Field, FormWidget, Key,
StatusEvent, TabAction, ThemeAttribute, UIDialog, UIEvent,
};
#[derive(Debug)]
enum ViewMode {
ReadOnly,
Discard(Box<UIDialog<char>>),
Edit,
//New,
}
#[derive(Debug)]
pub struct ContactManager {
id: ComponentId,
parent_id: Option<ComponentId>,
pub card: Card,
mode: ViewMode,
form: FormWidget<bool>,
pub account_pos: usize,
content: CellBuffer,
theme_default: ThemeAttribute,
dirty: bool,
has_changes: bool,
initialized: bool,
}
impl std::fmt::Display for ContactManager {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
write!(f, "{}", self.card)
}
}
impl ContactManager {
pub fn new(context: &Context) -> Self {
let theme_default: ThemeAttribute = crate::conf::value(context, "theme_default");
ContactManager {
id: ComponentId::default(),
parent_id: None,
card: Card::new(),
mode: ViewMode::Edit,
form: FormWidget::default(),
account_pos: 0,
content: CellBuffer::new_with_context(100, 1, None, context),
theme_default,
dirty: true,
has_changes: false,
initialized: false,
}
}
fn initialize(&mut self, context: &Context) {
let (width, _) = self.content.size();
let (x, _) = self.content.write_string(
"Last edited: ",
self.theme_default.fg,
self.theme_default.bg,
self.theme_default.attrs,
((0, 0), (width - 1, 0)),
None,
);
let (x, y) = self.content.write_string(
&self.card.last_edited(),
self.theme_default.fg,
self.theme_default.bg,
self.theme_default.attrs,
((x, 0), (width - 1, 0)),
None,
);
if self.card.external_resource() {
self.mode = ViewMode::ReadOnly;
_ = self.content.resize(self.content.size().0, 2, None);
self.content.write_string(
"This contact's origin is external and cannot be edited within meli.",
self.theme_default.fg,
self.theme_default.bg,
self.theme_default.attrs,
((x, y), (width - 1, y)),
None,
);
}
self.form = FormWidget::new(
("Save".into(), true),
/* cursor_up_shortcut */ context.settings.shortcuts.general.scroll_up.clone(),
/* cursor_down_shortcut */
context.settings.shortcuts.general.scroll_down.clone(),
);
self.form.add_button(("Cancel(Esc)".into(), false));
self.form
.push(("NAME".into(), self.card.name().to_string()));
self.form.push((
"ADDITIONAL NAME".into(),
self.card.additionalname().to_string(),
));
self.form
.push(("NAME PREFIX".into(), self.card.name_prefix().to_string()));
self.form
.push(("NAME SUFFIX".into(), self.card.name_suffix().to_string()));
self.form
.push(("E-MAIL".into(), self.card.email().to_string()));
self.form.push(("URL".into(), self.card.url().to_string()));
self.form.push(("KEY".into(), self.card.key().to_string()));
for (k, v) in self.card.extra_properties() {
self.form.push((k.to_string().into(), v.to_string()));
}
}
pub fn set_parent_id(&mut self, new_val: ComponentId) {
self.parent_id = Some(new_val);
}
}
impl Component for ContactManager {
fn draw(&mut self, grid: &mut CellBuffer, area: Area, context: &mut Context) {
if !self.initialized {
self.initialize(context);
self.initialized = true;
}
let upper_left = area.upper_left();
let bottom_right = area.bottom_right();
if self.dirty {
let (width, _height) = self.content.size();
grid.clear_area(
(upper_left, set_y(bottom_right, get_y(upper_left) + 1)),
self.theme_default,
);
grid.copy_area(&self.content, area, ((0, 0), (width - 1, 0)));
self.dirty = false;
}
self.form.draw(
grid,
(set_y(upper_left, get_y(upper_left) + 2), bottom_right),
context,
);
if let ViewMode::Discard(ref mut selector) = self.mode {
/* Let user choose whether to quit with/without saving or cancel */
selector.draw(grid, area, context);
}
context.dirty_areas.push_back(area);
}
fn process_event(&mut self, event: &mut UIEvent, context: &mut Context) -> bool {
if let UIEvent::ConfigReload { old_settings: _ } = event {
self.theme_default = crate::conf::value(context, "theme_default");
self.content = CellBuffer::new_with_context(100, 1, None, context);
self.initialized = false;
self.set_dirty(true);
}
match self.mode {
ViewMode::Discard(ref mut selector) => {
if selector.process_event(event, context) {
self.set_dirty(true);
return true;
}
}
ViewMode::Edit => {
if let (Some(parent_id), &UIEvent::Input(Key::Esc)) = (self.parent_id, &event) {
if self.can_quit_cleanly(context) {
context
.replies
.push_back(UIEvent::Action(Tab(TabAction::Kill(parent_id))));
}
return true;
}
if self.form.process_event(event, context) {
match self.form.buttons_result() {
None => {}
Some(true) => {
let fields = std::mem::take(&mut self.form).collect().unwrap();
let fields: HashMap<String, String> = fields
.into_iter()
.map(|(s, v)| {
(
s.to_string(),
match v {
Field::Text(v) => v.as_str().to_string(),
Field::Choice(mut v, c, _) => v.remove(c).to_string(),
},
)
})
.collect();
let mut new_card = Card::from(fields);
new_card.set_id(*self.card.id());
context.accounts[self.account_pos]
.address_book
.add_card(new_card);
context.replies.push_back(UIEvent::StatusEvent(
StatusEvent::DisplayMessage("Saved.".into()),
));
self.unrealize(context);
}
Some(false) => {
self.unrealize(context);
}
}
self.set_dirty(true);
if let UIEvent::InsertInput(_) = event {
self.has_changes = true;
}
return true;
}
}
ViewMode::ReadOnly => {
if let &mut UIEvent::Input(Key::Esc) = event {
if self.can_quit_cleanly(context) {
self.unrealize(context);
}
return true;
}
}
}
false
}
fn is_dirty(&self) -> bool {
self.dirty
|| self.form.is_dirty()
|| if let ViewMode::Discard(ref selector) = self.mode {
selector.is_dirty()
} else {
false
}
}
fn set_dirty(&mut self, value: bool) {
self.dirty = value;
self.form.set_dirty(value);
if let ViewMode::Discard(ref mut selector) = self.mode {
selector.set_dirty(value);
}
}
fn id(&self) -> ComponentId {
self.id
}
fn can_quit_cleanly(&mut self, context: &Context) -> bool {
if !self.has_changes {
return true;
}
if let Some(parent_id) = self.parent_id {
/* Play it safe and ask user for confirmation */
self.mode = ViewMode::Discard(Box::new(UIDialog::new(
"this contact has unsaved changes",
vec![
('x', "quit without saving".to_string()),
('y', "save draft and quit".to_string()),
('n', "cancel".to_string()),
],
true,
Some(Box::new(move |_, results: &[char]| match results[0] {
'x' => Some(UIEvent::Action(Tab(TabAction::Kill(parent_id)))),
'n' => None,
'y' => None,
_ => None,
})),
context,
)));
self.set_dirty(true);
false
} else {
true
}
}
}