mirror of
https://github.com/aljazceru/notedeck.git
synced 2025-12-23 11:14:19 +01:00
Preview: https://cdn.jb55.com/s/bef26a2caf09e952.png Demo: https://cdn.jb55.com/s/hover-preview-2.mp4 Changelog-Added: Show note previews when hovering reply descriptions Signed-off-by: William Casarin <jb55@jb55.com>
786 lines
25 KiB
Rust
786 lines
25 KiB
Rust
pub mod contents;
|
|
pub mod context;
|
|
pub mod options;
|
|
pub mod post;
|
|
pub mod quote_repost;
|
|
pub mod reply;
|
|
|
|
pub use contents::NoteContents;
|
|
pub use context::{NoteContextButton, NoteContextSelection};
|
|
pub use options::NoteOptions;
|
|
pub use post::{PostAction, PostResponse, PostType, PostView};
|
|
pub use quote_repost::QuoteRepostView;
|
|
pub use reply::PostReplyView;
|
|
|
|
use crate::{
|
|
actionbar::NoteAction,
|
|
ui::{self, View},
|
|
};
|
|
|
|
use egui::emath::{pos2, Vec2};
|
|
use egui::{Id, Label, Pos2, Rect, Response, RichText, Sense};
|
|
use enostr::{NoteId, Pubkey};
|
|
use nostrdb::{Ndb, Note, NoteKey, NoteReply, Transaction};
|
|
use notedeck::{CachedNote, ImageCache, NoteCache, NotedeckTextStyle};
|
|
|
|
use super::profile::preview::{get_display_name, one_line_display_name_widget};
|
|
|
|
pub struct NoteView<'a> {
|
|
ndb: &'a Ndb,
|
|
note_cache: &'a mut NoteCache,
|
|
img_cache: &'a mut ImageCache,
|
|
parent: Option<NoteKey>,
|
|
note: &'a nostrdb::Note<'a>,
|
|
flags: NoteOptions,
|
|
}
|
|
|
|
pub struct NoteResponse {
|
|
pub response: egui::Response,
|
|
pub context_selection: Option<NoteContextSelection>,
|
|
pub action: Option<NoteAction>,
|
|
}
|
|
|
|
impl NoteResponse {
|
|
pub fn new(response: egui::Response) -> Self {
|
|
Self {
|
|
response,
|
|
context_selection: None,
|
|
action: None,
|
|
}
|
|
}
|
|
|
|
pub fn with_action(mut self, action: Option<NoteAction>) -> Self {
|
|
self.action = action;
|
|
self
|
|
}
|
|
|
|
pub fn select_option(mut self, context_selection: Option<NoteContextSelection>) -> Self {
|
|
self.context_selection = context_selection;
|
|
self
|
|
}
|
|
}
|
|
|
|
impl View for NoteView<'_> {
|
|
fn ui(&mut self, ui: &mut egui::Ui) {
|
|
self.show(ui);
|
|
}
|
|
}
|
|
|
|
fn reply_desc(
|
|
ui: &mut egui::Ui,
|
|
txn: &Transaction,
|
|
note_reply: &NoteReply,
|
|
ndb: &Ndb,
|
|
img_cache: &mut ImageCache,
|
|
note_cache: &mut NoteCache,
|
|
) -> Option<NoteAction> {
|
|
#[cfg(feature = "profiling")]
|
|
puffin::profile_function!();
|
|
|
|
let mut note_action: Option<NoteAction> = None;
|
|
let size = 10.0;
|
|
let selectable = false;
|
|
let visuals = ui.visuals();
|
|
let color = visuals.noninteractive().fg_stroke.color;
|
|
let link_color = visuals.hyperlink_color;
|
|
|
|
// note link renderer helper
|
|
let note_link = |ui: &mut egui::Ui,
|
|
note_cache: &mut NoteCache,
|
|
img_cache: &mut ImageCache,
|
|
text: &str,
|
|
note: &Note<'_>| {
|
|
let r = ui.add(
|
|
Label::new(RichText::new(text).size(size).color(link_color))
|
|
.sense(Sense::click())
|
|
.selectable(selectable),
|
|
);
|
|
|
|
if r.clicked() {
|
|
// TODO: jump to note
|
|
}
|
|
|
|
if r.hovered() {
|
|
r.on_hover_ui_at_pointer(|ui| {
|
|
ui.set_max_width(400.0);
|
|
ui::NoteView::new(ndb, note_cache, img_cache, note)
|
|
.actionbar(false)
|
|
.wide(true)
|
|
.show(ui);
|
|
});
|
|
}
|
|
};
|
|
|
|
ui.add(Label::new(RichText::new("replying to").size(size).color(color)).selectable(selectable));
|
|
|
|
let reply = note_reply.reply()?;
|
|
|
|
let reply_note = if let Ok(reply_note) = ndb.get_note_by_id(txn, reply.id) {
|
|
reply_note
|
|
} else {
|
|
ui.add(Label::new(RichText::new("a note").size(size).color(color)).selectable(selectable));
|
|
return None;
|
|
};
|
|
|
|
if note_reply.is_reply_to_root() {
|
|
// We're replying to the root, let's show this
|
|
ui.add(
|
|
ui::Mention::new(ndb, img_cache, txn, reply_note.pubkey())
|
|
.size(size)
|
|
.selectable(selectable),
|
|
);
|
|
ui.add(Label::new(RichText::new("'s").size(size).color(color)).selectable(selectable));
|
|
note_link(ui, note_cache, img_cache, "thread", &reply_note);
|
|
} else if let Some(root) = note_reply.root() {
|
|
// replying to another post in a thread, not the root
|
|
|
|
if let Ok(root_note) = ndb.get_note_by_id(txn, root.id) {
|
|
if root_note.pubkey() == reply_note.pubkey() {
|
|
// simply "replying to bob's note" when replying to bob in his thread
|
|
let action = ui::Mention::new(ndb, img_cache, txn, reply_note.pubkey())
|
|
.size(size)
|
|
.selectable(selectable)
|
|
.show(ui)
|
|
.inner;
|
|
|
|
if action.is_some() {
|
|
note_action = action;
|
|
}
|
|
|
|
ui.add(
|
|
Label::new(RichText::new("'s").size(size).color(color)).selectable(selectable),
|
|
);
|
|
|
|
note_link(ui, note_cache, img_cache, "note", &reply_note);
|
|
} else {
|
|
// replying to bob in alice's thread
|
|
|
|
ui.add(
|
|
ui::Mention::new(ndb, img_cache, txn, reply_note.pubkey())
|
|
.size(size)
|
|
.selectable(selectable),
|
|
);
|
|
ui.add(
|
|
Label::new(RichText::new("'s").size(size).color(color)).selectable(selectable),
|
|
);
|
|
note_link(ui, note_cache, img_cache, "note", &reply_note);
|
|
ui.add(
|
|
Label::new(RichText::new("in").size(size).color(color)).selectable(selectable),
|
|
);
|
|
ui.add(
|
|
ui::Mention::new(ndb, img_cache, txn, root_note.pubkey())
|
|
.size(size)
|
|
.selectable(selectable),
|
|
);
|
|
ui.add(
|
|
Label::new(RichText::new("'s").size(size).color(color)).selectable(selectable),
|
|
);
|
|
note_link(ui, note_cache, img_cache, "thread", &root_note);
|
|
}
|
|
} else {
|
|
ui.add(
|
|
ui::Mention::new(ndb, img_cache, txn, reply_note.pubkey())
|
|
.size(size)
|
|
.selectable(selectable),
|
|
);
|
|
ui.add(
|
|
Label::new(RichText::new("in someone's thread").size(size).color(color))
|
|
.selectable(selectable),
|
|
);
|
|
}
|
|
}
|
|
|
|
note_action
|
|
}
|
|
|
|
impl<'a> NoteView<'a> {
|
|
pub fn new(
|
|
ndb: &'a Ndb,
|
|
note_cache: &'a mut NoteCache,
|
|
img_cache: &'a mut ImageCache,
|
|
note: &'a nostrdb::Note<'a>,
|
|
) -> Self {
|
|
let flags = NoteOptions::actionbar | NoteOptions::note_previews;
|
|
let parent: Option<NoteKey> = None;
|
|
Self {
|
|
ndb,
|
|
note_cache,
|
|
img_cache,
|
|
parent,
|
|
note,
|
|
flags,
|
|
}
|
|
}
|
|
|
|
pub fn note_options(mut self, options: NoteOptions) -> Self {
|
|
*self.options_mut() = options;
|
|
self
|
|
}
|
|
|
|
pub fn textmode(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_textmode(enable);
|
|
self
|
|
}
|
|
|
|
pub fn actionbar(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_actionbar(enable);
|
|
self
|
|
}
|
|
|
|
pub fn small_pfp(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_small_pfp(enable);
|
|
self
|
|
}
|
|
|
|
pub fn medium_pfp(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_medium_pfp(enable);
|
|
self
|
|
}
|
|
|
|
pub fn note_previews(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_note_previews(enable);
|
|
self
|
|
}
|
|
|
|
pub fn selectable_text(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_selectable_text(enable);
|
|
self
|
|
}
|
|
|
|
pub fn wide(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_wide(enable);
|
|
self
|
|
}
|
|
|
|
pub fn options_button(mut self, enable: bool) -> Self {
|
|
self.options_mut().set_options_button(enable);
|
|
self
|
|
}
|
|
|
|
pub fn options(&self) -> NoteOptions {
|
|
self.flags
|
|
}
|
|
|
|
pub fn options_mut(&mut self) -> &mut NoteOptions {
|
|
&mut self.flags
|
|
}
|
|
|
|
pub fn parent(mut self, parent: NoteKey) -> Self {
|
|
self.parent = Some(parent);
|
|
self
|
|
}
|
|
|
|
fn textmode_ui(&mut self, ui: &mut egui::Ui) -> egui::Response {
|
|
let note_key = self.note.key().expect("todo: implement non-db notes");
|
|
let txn = self.note.txn().expect("todo: implement non-db notes");
|
|
|
|
ui.with_layout(egui::Layout::left_to_right(egui::Align::TOP), |ui| {
|
|
let profile = self.ndb.get_profile_by_pubkey(txn, self.note.pubkey());
|
|
|
|
//ui.horizontal(|ui| {
|
|
ui.spacing_mut().item_spacing.x = 2.0;
|
|
|
|
let cached_note = self
|
|
.note_cache
|
|
.cached_note_or_insert_mut(note_key, self.note);
|
|
|
|
let (_id, rect) = ui.allocate_space(egui::vec2(50.0, 20.0));
|
|
ui.allocate_rect(rect, Sense::hover());
|
|
ui.put(rect, |ui: &mut egui::Ui| {
|
|
render_reltime(ui, cached_note, false).response
|
|
});
|
|
let (_id, rect) = ui.allocate_space(egui::vec2(150.0, 20.0));
|
|
ui.allocate_rect(rect, Sense::hover());
|
|
ui.put(rect, |ui: &mut egui::Ui| {
|
|
ui.add(
|
|
ui::Username::new(profile.as_ref().ok(), self.note.pubkey())
|
|
.abbreviated(6)
|
|
.pk_colored(true),
|
|
)
|
|
});
|
|
|
|
ui.add(&mut NoteContents::new(
|
|
self.ndb,
|
|
self.img_cache,
|
|
self.note_cache,
|
|
txn,
|
|
self.note,
|
|
note_key,
|
|
self.flags,
|
|
));
|
|
//});
|
|
})
|
|
.response
|
|
}
|
|
|
|
pub fn expand_size() -> f32 {
|
|
5.0
|
|
}
|
|
|
|
fn pfp(
|
|
&mut self,
|
|
note_key: NoteKey,
|
|
profile: &Result<nostrdb::ProfileRecord<'_>, nostrdb::Error>,
|
|
ui: &mut egui::Ui,
|
|
) -> egui::Response {
|
|
if !self.options().has_wide() {
|
|
ui.spacing_mut().item_spacing.x = 16.0;
|
|
} else {
|
|
ui.spacing_mut().item_spacing.x = 4.0;
|
|
}
|
|
|
|
let pfp_size = self.options().pfp_size();
|
|
|
|
let sense = Sense::click();
|
|
match profile
|
|
.as_ref()
|
|
.ok()
|
|
.and_then(|p| p.record().profile()?.picture())
|
|
{
|
|
// these have different lifetimes and types,
|
|
// so the calls must be separate
|
|
Some(pic) => {
|
|
let anim_speed = 0.05;
|
|
let profile_key = profile.as_ref().unwrap().record().note_key();
|
|
let note_key = note_key.as_u64();
|
|
|
|
let (rect, size, resp) = ui::anim::hover_expand(
|
|
ui,
|
|
egui::Id::new((profile_key, note_key)),
|
|
pfp_size,
|
|
ui::NoteView::expand_size(),
|
|
anim_speed,
|
|
);
|
|
|
|
ui.put(rect, ui::ProfilePic::new(self.img_cache, pic).size(size))
|
|
.on_hover_ui_at_pointer(|ui| {
|
|
ui.set_max_width(300.0);
|
|
ui.add(ui::ProfilePreview::new(
|
|
profile.as_ref().unwrap(),
|
|
self.img_cache,
|
|
));
|
|
});
|
|
|
|
if resp.hovered() || resp.clicked() {
|
|
ui::show_pointer(ui);
|
|
}
|
|
|
|
resp
|
|
}
|
|
None => ui
|
|
.add(
|
|
ui::ProfilePic::new(self.img_cache, ui::ProfilePic::no_pfp_url())
|
|
.size(pfp_size),
|
|
)
|
|
.interact(sense),
|
|
}
|
|
}
|
|
|
|
pub fn show(&mut self, ui: &mut egui::Ui) -> NoteResponse {
|
|
if self.options().has_textmode() {
|
|
NoteResponse::new(self.textmode_ui(ui))
|
|
} else {
|
|
let txn = self.note.txn().expect("txn");
|
|
if let Some(note_to_repost) = get_reposted_note(self.ndb, txn, self.note) {
|
|
let profile = self.ndb.get_profile_by_pubkey(txn, self.note.pubkey());
|
|
|
|
let style = NotedeckTextStyle::Small;
|
|
ui.horizontal(|ui| {
|
|
ui.vertical(|ui| {
|
|
ui.add_space(2.0);
|
|
ui.add_sized([20.0, 20.0], repost_icon(ui.visuals().dark_mode));
|
|
});
|
|
ui.add_space(6.0);
|
|
let resp = ui.add(one_line_display_name_widget(
|
|
ui.visuals(),
|
|
get_display_name(profile.as_ref().ok()),
|
|
style,
|
|
));
|
|
if let Ok(rec) = &profile {
|
|
resp.on_hover_ui_at_pointer(|ui| {
|
|
ui.set_max_width(300.0);
|
|
ui.add(ui::ProfilePreview::new(rec, self.img_cache));
|
|
});
|
|
}
|
|
let color = ui.style().visuals.noninteractive().fg_stroke.color;
|
|
ui.add_space(4.0);
|
|
ui.label(
|
|
RichText::new("Reposted")
|
|
.color(color)
|
|
.text_style(style.text_style()),
|
|
);
|
|
});
|
|
NoteView::new(self.ndb, self.note_cache, self.img_cache, ¬e_to_repost).show(ui)
|
|
} else {
|
|
self.show_standard(ui)
|
|
}
|
|
}
|
|
}
|
|
|
|
fn note_header(
|
|
ui: &mut egui::Ui,
|
|
note_cache: &mut NoteCache,
|
|
note: &Note,
|
|
profile: &Result<nostrdb::ProfileRecord<'_>, nostrdb::Error>,
|
|
options: NoteOptions,
|
|
container_right: Pos2,
|
|
) -> NoteResponse {
|
|
#[cfg(feature = "profiling")]
|
|
puffin::profile_function!();
|
|
|
|
let note_key = note.key().unwrap();
|
|
|
|
let inner_response = ui.horizontal(|ui| {
|
|
ui.spacing_mut().item_spacing.x = 2.0;
|
|
ui.add(ui::Username::new(profile.as_ref().ok(), note.pubkey()).abbreviated(20));
|
|
|
|
let cached_note = note_cache.cached_note_or_insert_mut(note_key, note);
|
|
render_reltime(ui, cached_note, true);
|
|
|
|
if options.has_options_button() {
|
|
let context_pos = {
|
|
let size = NoteContextButton::max_width();
|
|
let min = Pos2::new(container_right.x - size, container_right.y);
|
|
Rect::from_min_size(min, egui::vec2(size, size))
|
|
};
|
|
|
|
let resp = ui.add(NoteContextButton::new(note_key).place_at(context_pos));
|
|
NoteContextButton::menu(ui, resp.clone())
|
|
} else {
|
|
None
|
|
}
|
|
});
|
|
|
|
NoteResponse::new(inner_response.response).select_option(inner_response.inner)
|
|
}
|
|
|
|
fn show_standard(&mut self, ui: &mut egui::Ui) -> NoteResponse {
|
|
#[cfg(feature = "profiling")]
|
|
puffin::profile_function!();
|
|
let note_key = self.note.key().expect("todo: support non-db notes");
|
|
let txn = self.note.txn().expect("todo: support non-db notes");
|
|
|
|
let mut note_action: Option<NoteAction> = None;
|
|
let mut selected_option: Option<NoteContextSelection> = None;
|
|
|
|
let hitbox_id = note_hitbox_id(note_key, self.options(), self.parent);
|
|
let profile = self.ndb.get_profile_by_pubkey(txn, self.note.pubkey());
|
|
let maybe_hitbox = maybe_note_hitbox(ui, hitbox_id);
|
|
let container_right = {
|
|
let r = ui.available_rect_before_wrap();
|
|
let x = r.max.x;
|
|
let y = r.min.y;
|
|
Pos2::new(x, y)
|
|
};
|
|
|
|
// wide design
|
|
let response = if self.options().has_wide() {
|
|
ui.vertical(|ui| {
|
|
ui.horizontal(|ui| {
|
|
if self.pfp(note_key, &profile, ui).clicked() {
|
|
note_action =
|
|
Some(NoteAction::OpenProfile(Pubkey::new(*self.note.pubkey())));
|
|
};
|
|
|
|
let size = ui.available_size();
|
|
ui.vertical(|ui| {
|
|
ui.add_sized([size.x, self.options().pfp_size()], |ui: &mut egui::Ui| {
|
|
ui.horizontal_centered(|ui| {
|
|
selected_option = NoteView::note_header(
|
|
ui,
|
|
self.note_cache,
|
|
self.note,
|
|
&profile,
|
|
self.options(),
|
|
container_right,
|
|
)
|
|
.context_selection;
|
|
})
|
|
.response
|
|
});
|
|
|
|
let note_reply = self
|
|
.note_cache
|
|
.cached_note_or_insert_mut(note_key, self.note)
|
|
.reply
|
|
.borrow(self.note.tags());
|
|
|
|
if note_reply.reply().is_some() {
|
|
ui.horizontal(|ui| {
|
|
reply_desc(
|
|
ui,
|
|
txn,
|
|
¬e_reply,
|
|
self.ndb,
|
|
self.img_cache,
|
|
self.note_cache,
|
|
);
|
|
});
|
|
}
|
|
});
|
|
});
|
|
|
|
let mut contents = NoteContents::new(
|
|
self.ndb,
|
|
self.img_cache,
|
|
self.note_cache,
|
|
txn,
|
|
self.note,
|
|
note_key,
|
|
self.options(),
|
|
);
|
|
|
|
ui.add(&mut contents);
|
|
|
|
if let Some(action) = contents.action() {
|
|
note_action = Some(*action);
|
|
}
|
|
|
|
if self.options().has_actionbar() {
|
|
if let Some(action) = render_note_actionbar(ui, self.note.id(), note_key).inner
|
|
{
|
|
note_action = Some(action);
|
|
}
|
|
}
|
|
})
|
|
.response
|
|
} else {
|
|
// main design
|
|
ui.with_layout(egui::Layout::left_to_right(egui::Align::TOP), |ui| {
|
|
if self.pfp(note_key, &profile, ui).clicked() {
|
|
note_action = Some(NoteAction::OpenProfile(Pubkey::new(*self.note.pubkey())));
|
|
};
|
|
|
|
ui.with_layout(egui::Layout::top_down(egui::Align::LEFT), |ui| {
|
|
selected_option = NoteView::note_header(
|
|
ui,
|
|
self.note_cache,
|
|
self.note,
|
|
&profile,
|
|
self.options(),
|
|
container_right,
|
|
)
|
|
.context_selection;
|
|
ui.horizontal(|ui| {
|
|
ui.spacing_mut().item_spacing.x = 2.0;
|
|
|
|
let note_reply = self
|
|
.note_cache
|
|
.cached_note_or_insert_mut(note_key, self.note)
|
|
.reply
|
|
.borrow(self.note.tags());
|
|
|
|
if note_reply.reply().is_some() {
|
|
reply_desc(
|
|
ui,
|
|
txn,
|
|
¬e_reply,
|
|
self.ndb,
|
|
self.img_cache,
|
|
self.note_cache,
|
|
);
|
|
}
|
|
});
|
|
|
|
let mut contents = NoteContents::new(
|
|
self.ndb,
|
|
self.img_cache,
|
|
self.note_cache,
|
|
txn,
|
|
self.note,
|
|
note_key,
|
|
self.options(),
|
|
);
|
|
ui.add(&mut contents);
|
|
|
|
if let Some(action) = contents.action() {
|
|
note_action = Some(*action);
|
|
}
|
|
|
|
if self.options().has_actionbar() {
|
|
if let Some(action) =
|
|
render_note_actionbar(ui, self.note.id(), note_key).inner
|
|
{
|
|
note_action = Some(action);
|
|
}
|
|
}
|
|
});
|
|
})
|
|
.response
|
|
};
|
|
|
|
let note_action = if note_hitbox_clicked(ui, hitbox_id, &response.rect, maybe_hitbox) {
|
|
Some(NoteAction::OpenThread(NoteId::new(*self.note.id())))
|
|
} else {
|
|
note_action
|
|
};
|
|
|
|
NoteResponse::new(response)
|
|
.with_action(note_action)
|
|
.select_option(selected_option)
|
|
}
|
|
}
|
|
|
|
fn get_reposted_note<'a>(ndb: &Ndb, txn: &'a Transaction, note: &Note) -> Option<Note<'a>> {
|
|
let new_note_id: &[u8; 32] = if note.kind() == 6 {
|
|
let mut res = None;
|
|
for tag in note.tags().iter() {
|
|
if tag.count() == 0 {
|
|
continue;
|
|
}
|
|
|
|
if let Some("e") = tag.get(0).and_then(|t| t.variant().str()) {
|
|
if let Some(note_id) = tag.get(1).and_then(|f| f.variant().id()) {
|
|
res = Some(note_id);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
res?
|
|
} else {
|
|
return None;
|
|
};
|
|
|
|
let note = ndb.get_note_by_id(txn, new_note_id).ok();
|
|
note.filter(|note| note.kind() == 1)
|
|
}
|
|
|
|
fn note_hitbox_id(
|
|
note_key: NoteKey,
|
|
note_options: NoteOptions,
|
|
parent: Option<NoteKey>,
|
|
) -> egui::Id {
|
|
Id::new(("note_size", note_key, note_options, parent))
|
|
}
|
|
|
|
fn maybe_note_hitbox(ui: &mut egui::Ui, hitbox_id: egui::Id) -> Option<Response> {
|
|
ui.ctx()
|
|
.data_mut(|d| d.get_persisted(hitbox_id))
|
|
.map(|note_size: Vec2| {
|
|
// The hitbox should extend the entire width of the
|
|
// container. The hitbox height was cached last layout.
|
|
let container_rect = ui.max_rect();
|
|
let rect = Rect {
|
|
min: pos2(container_rect.min.x, container_rect.min.y),
|
|
max: pos2(container_rect.max.x, container_rect.min.y + note_size.y),
|
|
};
|
|
|
|
let response = ui.interact(rect, ui.id().with(hitbox_id), egui::Sense::click());
|
|
|
|
response
|
|
.widget_info(|| egui::WidgetInfo::labeled(egui::WidgetType::Other, true, "hitbox"));
|
|
|
|
response
|
|
})
|
|
}
|
|
|
|
fn note_hitbox_clicked(
|
|
ui: &mut egui::Ui,
|
|
hitbox_id: egui::Id,
|
|
note_rect: &Rect,
|
|
maybe_hitbox: Option<Response>,
|
|
) -> bool {
|
|
// Stash the dimensions of the note content so we can render the
|
|
// hitbox in the next frame
|
|
ui.ctx().data_mut(|d| {
|
|
d.insert_persisted(hitbox_id, note_rect.size());
|
|
});
|
|
|
|
// If there was an hitbox and it was clicked open the thread
|
|
match maybe_hitbox {
|
|
Some(hitbox) => hitbox.clicked(),
|
|
_ => false,
|
|
}
|
|
}
|
|
|
|
fn render_note_actionbar(
|
|
ui: &mut egui::Ui,
|
|
note_id: &[u8; 32],
|
|
note_key: NoteKey,
|
|
) -> egui::InnerResponse<Option<NoteAction>> {
|
|
#[cfg(feature = "profiling")]
|
|
puffin::profile_function!();
|
|
|
|
ui.horizontal(|ui| {
|
|
let reply_resp = reply_button(ui, note_key);
|
|
let quote_resp = quote_repost_button(ui, note_key);
|
|
|
|
if reply_resp.clicked() {
|
|
Some(NoteAction::Reply(NoteId::new(*note_id)))
|
|
} else if quote_resp.clicked() {
|
|
Some(NoteAction::Quote(NoteId::new(*note_id)))
|
|
} else {
|
|
None
|
|
}
|
|
})
|
|
}
|
|
|
|
fn secondary_label(ui: &mut egui::Ui, s: impl Into<String>) {
|
|
let color = ui.style().visuals.noninteractive().fg_stroke.color;
|
|
ui.add(Label::new(RichText::new(s).size(10.0).color(color)));
|
|
}
|
|
|
|
fn render_reltime(
|
|
ui: &mut egui::Ui,
|
|
note_cache: &mut CachedNote,
|
|
before: bool,
|
|
) -> egui::InnerResponse<()> {
|
|
#[cfg(feature = "profiling")]
|
|
puffin::profile_function!();
|
|
|
|
ui.horizontal(|ui| {
|
|
if before {
|
|
secondary_label(ui, "⋅");
|
|
}
|
|
|
|
secondary_label(ui, note_cache.reltime_str_mut());
|
|
|
|
if !before {
|
|
secondary_label(ui, "⋅");
|
|
}
|
|
})
|
|
}
|
|
|
|
fn reply_button(ui: &mut egui::Ui, note_key: NoteKey) -> egui::Response {
|
|
let img_data = if ui.style().visuals.dark_mode {
|
|
egui::include_image!("../../../../../assets/icons/reply.png")
|
|
} else {
|
|
egui::include_image!("../../../../../assets/icons/reply-dark.png")
|
|
};
|
|
|
|
let (rect, size, resp) =
|
|
ui::anim::hover_expand_small(ui, ui.id().with(("reply_anim", note_key)));
|
|
|
|
// align rect to note contents
|
|
let expand_size = 5.0; // from hover_expand_small
|
|
let rect = rect.translate(egui::vec2(-(expand_size / 2.0), 0.0));
|
|
|
|
let put_resp = ui.put(rect, egui::Image::new(img_data).max_width(size));
|
|
|
|
resp.union(put_resp)
|
|
}
|
|
|
|
fn repost_icon(dark_mode: bool) -> egui::Image<'static> {
|
|
let img_data = if dark_mode {
|
|
egui::include_image!("../../../../../assets/icons/repost_icon_4x.png")
|
|
} else {
|
|
egui::include_image!("../../../../../assets/icons/repost_light_4x.png")
|
|
};
|
|
egui::Image::new(img_data)
|
|
}
|
|
|
|
fn quote_repost_button(ui: &mut egui::Ui, note_key: NoteKey) -> egui::Response {
|
|
let size = 14.0;
|
|
let expand_size = 5.0;
|
|
let anim_speed = 0.05;
|
|
let id = ui.id().with(("repost_anim", note_key));
|
|
|
|
let (rect, size, resp) = ui::anim::hover_expand(ui, id, size, expand_size, anim_speed);
|
|
|
|
let rect = rect.translate(egui::vec2(-(expand_size / 2.0), -1.0));
|
|
|
|
let put_resp = ui.put(rect, repost_icon(ui.visuals().dark_mode).max_width(size));
|
|
|
|
resp.union(put_resp)
|
|
}
|