From 7a32dd7a378a587a4084fa5843b9c1d0f10a34d2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?H=C3=A9ctor=20Ram=C3=B3n=20Jim=C3=A9nez?= Date: Sat, 20 Jul 2024 20:16:33 +0200 Subject: [PATCH] Add `Delete` chat interaction to `Conversation` screen --- fonts/icebreaker-icons.ttf | Bin 7416 -> 7788 bytes src/data/chat.rs | 43 ++++++++++++++++++++++++++++++++++++- src/icon.rs | 4 ++++ src/screen/conversation.rs | 28 +++++++++++++++++++++--- 4 files changed, 71 insertions(+), 4 deletions(-) diff --git a/fonts/icebreaker-icons.ttf b/fonts/icebreaker-icons.ttf index f070824fa830d23a6ddd15bfffa43901da33e1de..b3f05ae324ecdd760be4546b2c694aa583af0aa3 100644 GIT binary patch delta 822 zcmZ{h&rcIk5XWcUc7MG>%Su~gM4&CS{1QwlZIG&n_#;t+K@bQcEWcV3Dl0UC#@K^~ z#26J*58n6(N+Q;9_E$W4(3lv#8BQ2T^kPym6lZIU(Zt!jnfZR^y?MJk+2)PNf;agQ z0M7ux&`4Z2K6gY`Nk5|XY2|p^%G{pEM2nKn3_qv`*F!hFHlAG8^-jQaebv_1z^vc z``LaRPj%rTR8kMHp?VaMwg6lJA)nPzhiag>ES+0&Fh}I|M7(p$r-z4E@mC+I_N_DN?n2m!_hGz)n#w zSt^jD92Jn1zCtULER4}rFd+&IRTcZpi0UAysCM-Ac6WB1K7Q;-tIt#HGtFz0%4#dM zpsyA^l2dcIS}+g_Ysr7X-x?eJKEGFzMW;imMa?03-5#G`^Ll&-Fc@mV@IeeW23%!G z4okUGq42!B_b0u}_mB}DBnP<{|wa)c;XN~G#Qg|oc zGFfic^#93O;IN1&o~7PfU+UZ2>RECA&H-l$mO2K^0ntyUl`U@14ca5$z%HzrlURMW zwEjD8Z>4f-6Y^A$SA+!3#ANgE3hU5dz=qgsp++{A*6?{7W=`5S=vQgT^|0#F{iS`c`l$H29=E5k_x$(VgLUE1x!E!Ir+(nY<27hfcyv`UnDoNqJWW;Zvv2?1LP~@ zCFZ6^Cwi0v`8_~ObPMu}OZYZ^&tPES-2;@bE+|SZsM}_}je$WXX5x=%#{9`4jP{J| zlczB1YbG%;0+lc@#4s>}Svo+P3+T%v1{MYbpg0Q<0#&?VpZtbVaWg*?4lK(It E07WNS0RR91 diff --git a/src/data/chat.rs b/src/data/chat.rs index abef852..8f0f1e7 100644 --- a/src/data/chat.rs +++ b/src/data/chat.rs @@ -100,6 +100,30 @@ impl Chat { history: chat.history, }) } + + pub async fn delete(id: Id) -> Result<(), Error> { + fs::remove_file(Self::path(&id).await?).await?; + + let _ = List::remove(&id).await; + + match LastOpened::fetch().await { + Ok(LastOpened(last_opened)) if id == last_opened => { + let list = List::fetch().await.ok(); + + match list.as_ref().and_then(|list| list.entries.first()) { + Some(entry) => { + LastOpened::update(entry.id.clone()).await?; + } + None => { + LastOpened::delete().await?; + } + } + } + _ => {} + } + + Ok(()) + } } #[derive(Debug, Clone)] @@ -232,7 +256,18 @@ impl List { let mut list = Self::fetch().await.unwrap_or_default(); list.entries.insert(0, entry); - let json = task::spawn_blocking(move || serde_json::to_vec_pretty(&list)).await?; + list.save().await + } + + async fn remove(id: &Id) -> Result<(), Error> { + let mut list = List::fetch().await?; + list.entries.retain(|entry| &entry.id != id); + + list.save().await + } + + async fn save(self) -> Result<(), Error> { + let json = task::spawn_blocking(move || serde_json::to_vec_pretty(&self)).await?; fs::write(Self::path().await?, json?).await?; @@ -262,6 +297,12 @@ impl LastOpened { Ok(()) } + + async fn delete() -> Result<(), Error> { + fs::remove_file(Self::path().await?).await?; + + Ok(()) + } } async fn storage_dir() -> Result { diff --git a/src/icon.rs b/src/icon.rs index bee60e1..bcf7336 100644 --- a/src/icon.rs +++ b/src/icon.rs @@ -35,6 +35,10 @@ pub fn expand<'a>() -> Text<'a> { with_codepoint('\u{E807}') } +pub fn trash<'a>() -> Text<'a> { + with_codepoint('\u{E808}') +} + fn with_codepoint<'a>(codepoint: char) -> Text<'a> { const FONT: Font = Font::with_name("icebreaker-icons"); diff --git a/src/screen/conversation.rs b/src/screen/conversation.rs index 9b3e186..c6f7e9f 100644 --- a/src/screen/conversation.rs +++ b/src/screen/conversation.rs @@ -10,8 +10,8 @@ use iced::padding; use iced::task::{self, Task}; use iced::time::{self, Duration, Instant}; use iced::widget::{ - self, button, center, column, container, hover, progress_bar, row, scrollable, stack, text, - text_input, tooltip, value, + self, button, center, column, container, horizontal_space, hover, progress_bar, row, + scrollable, stack, text, text_input, tooltip, value, }; use iced::{Center, Element, Fill, Font, Left, Right, Subscription, Theme}; @@ -55,6 +55,7 @@ pub enum Message { Saved(Result), Open(chat::Id), ChatFetched(Result), + Delete, New, Search, ToggleSidebar, @@ -318,6 +319,18 @@ impl Conversation { Action::Run(widget::focus_next()) } + Message::Delete => { + if let Some(id) = self.id.clone() { + Action::Run(Task::future(Chat::delete(id)).and_then(|_| { + Task::batch([ + Task::perform(Chat::fetch_last_opened(), Message::ChatFetched), + Task::perform(Chat::list(), Message::ChatsListed), + ]) + })) + } else { + Action::None + } + } Message::Search => Action::Back, Message::ToggleSidebar => { self.sidebar_open = !self.sidebar_open; @@ -366,7 +379,16 @@ impl Conversation { tip::Position::Right, ); - let bar = stack![title, toggle_sidebar].into(); + let delete = tip( + button(icon::trash().style(text::danger)) + .padding(0) + .on_press(Message::Delete) + .style(button::text), + "Delete Chat", + tip::Position::Left, + ); + + let bar = stack![title, row![toggle_sidebar, horizontal_space(), delete]].into(); match &self.state { State::Booting {