Merge is_following fixes from kernel

kernelkind (4):
      add `Accounts` to `NoteContext`
      remove `MuteFun` prop
      make `Contacts::is_following` use bytes instead of `Pubkey`
      migrate to check following through `Contacts::is_following`
This commit is contained in:
William Casarin
2025-07-16 08:50:42 -07:00
16 changed files with 84 additions and 268 deletions

View File

@@ -49,14 +49,14 @@ impl Contacts {
update_state(&mut self.state, &res.note, res.note_key); update_state(&mut self.state, &res.note, res.note_key);
} }
pub fn is_following(&self, other: &Pubkey) -> IsFollowing { pub fn is_following(&self, other_pubkey: &[u8; 32]) -> IsFollowing {
match &self.state { match &self.state {
ContactState::Unreceived => IsFollowing::Unknown, ContactState::Unreceived => IsFollowing::Unknown,
ContactState::Received { ContactState::Received {
contacts, contacts,
note_key: _, note_key: _,
} => { } => {
if contacts.contains(other) { if contacts.contains(other_pubkey) {
IsFollowing::Yes IsFollowing::Yes
} else { } else {
IsFollowing::No IsFollowing::No

View File

@@ -4,6 +4,7 @@ mod context;
pub use action::{MediaAction, NoteAction, ScrollInfo, ZapAction, ZapTargetAmount}; pub use action::{MediaAction, NoteAction, ScrollInfo, ZapAction, ZapTargetAmount};
pub use context::{BroadcastContext, ContextSelection, NoteContextSelection}; pub use context::{BroadcastContext, ContextSelection, NoteContextSelection};
use crate::Accounts;
use crate::JobPool; use crate::JobPool;
use crate::UnknownIds; use crate::UnknownIds;
use crate::{notecache::NoteCache, zaps::Zaps, Images}; use crate::{notecache::NoteCache, zaps::Zaps, Images};
@@ -17,6 +18,7 @@ use std::fmt;
/// passed to inner UI elements, minimizing prop drilling. /// passed to inner UI elements, minimizing prop drilling.
pub struct NoteContext<'d> { pub struct NoteContext<'d> {
pub ndb: &'d Ndb, pub ndb: &'d Ndb,
pub accounts: &'d Accounts,
pub img_cache: &'d mut Images, pub img_cache: &'d mut Images,
pub note_cache: &'d mut NoteCache, pub note_cache: &'d mut NoteCache,
pub zaps: &'d mut Zaps, pub zaps: &'d mut Zaps,

View File

@@ -1,4 +1,4 @@
use enostr::{Keypair, KeypairUnowned, Pubkey}; use enostr::{Keypair, KeypairUnowned};
use tokenator::{ParseError, TokenParser, TokenSerializable}; use tokenator::{ParseError, TokenParser, TokenSerializable};
use crate::{ use crate::{
@@ -33,8 +33,8 @@ impl UserAccount {
self self
} }
pub fn is_following(&self, pk: &Pubkey) -> IsFollowing { pub fn is_following(&self, other_pubkey: &[u8; 32]) -> IsFollowing {
self.data.contacts.is_following(pk) self.data.contacts.is_following(other_pubkey)
} }
} }

View File

@@ -485,8 +485,10 @@ fn render_nav_body(
col: usize, col: usize,
inner_rect: egui::Rect, inner_rect: egui::Rect,
) -> Option<RenderNavAction> { ) -> Option<RenderNavAction> {
let current_account_has_wallet = get_current_wallet(ctx.accounts, ctx.global_wallet).is_some();
let mut note_context = NoteContext { let mut note_context = NoteContext {
ndb: ctx.ndb, ndb: ctx.ndb,
accounts: ctx.accounts,
img_cache: ctx.img_cache, img_cache: ctx.img_cache,
note_cache: ctx.note_cache, note_cache: ctx.note_cache,
zaps: ctx.zaps, zaps: ctx.zaps,
@@ -494,7 +496,7 @@ fn render_nav_body(
job_pool: ctx.job_pool, job_pool: ctx.job_pool,
unknown_ids: ctx.unknown_ids, unknown_ids: ctx.unknown_ids,
clipboard: ctx.clipboard, clipboard: ctx.clipboard,
current_account_has_wallet: get_current_wallet(ctx.accounts, ctx.global_wallet).is_some(), current_account_has_wallet,
}; };
match top { match top {
Route::Timeline(kind) => { Route::Timeline(kind) => {
@@ -507,7 +509,6 @@ fn render_nav_body(
let nav_action = render_timeline_route( let nav_action = render_timeline_route(
&mut app.timeline_cache, &mut app.timeline_cache,
ctx.accounts,
kind, kind,
col, col,
app.note_options, app.note_options,
@@ -527,7 +528,6 @@ fn render_nav_body(
} }
Route::Thread(selection) => render_thread_route( Route::Thread(selection) => render_thread_route(
&mut app.threads, &mut app.threads,
ctx.accounts,
selection, selection,
col, col,
app.note_options, app.note_options,
@@ -676,11 +676,9 @@ fn render_nav_body(
SearchView::new( SearchView::new(
&txn, &txn,
&ctx.accounts.mutefun(),
app.note_options, app.note_options,
search_buffer, search_buffer,
&mut note_context, &mut note_context,
&(&ctx.accounts.get_selected_account().key).into(),
&mut app.jobs, &mut app.jobs,
) )
.show(ui) .show(ui)

View File

@@ -6,13 +6,12 @@ use crate::{
}; };
use enostr::Pubkey; use enostr::Pubkey;
use notedeck::{Accounts, MuteFun, NoteContext}; use notedeck::NoteContext;
use notedeck_ui::{jobs::JobsCache, NoteOptions}; use notedeck_ui::{jobs::JobsCache, NoteOptions};
#[allow(clippy::too_many_arguments)] #[allow(clippy::too_many_arguments)]
pub fn render_timeline_route( pub fn render_timeline_route(
timeline_cache: &mut TimelineCache, timeline_cache: &mut TimelineCache,
accounts: &mut Accounts,
kind: &TimelineKind, kind: &TimelineKind,
col: usize, col: usize,
note_options: NoteOptions, note_options: NoteOptions,
@@ -30,18 +29,10 @@ pub fn render_timeline_route(
| TimelineKind::Universe | TimelineKind::Universe
| TimelineKind::Hashtag(_) | TimelineKind::Hashtag(_)
| TimelineKind::Generic(_) => { | TimelineKind::Generic(_) => {
let note_action = ui::TimelineView::new( let note_action =
kind, ui::TimelineView::new(kind, timeline_cache, note_context, note_options, jobs, col)
timeline_cache, .scroll_to_top(scroll_to_top)
&accounts.mutefun(), .ui(ui);
note_context,
note_options,
&(&accounts.get_selected_account().key).into(),
jobs,
col,
)
.scroll_to_top(scroll_to_top)
.ui(ui);
note_action.map(RenderNavAction::NoteAction) note_action.map(RenderNavAction::NoteAction)
} }
@@ -50,11 +41,9 @@ pub fn render_timeline_route(
if depth > 1 { if depth > 1 {
render_profile_route( render_profile_route(
pubkey, pubkey,
accounts,
timeline_cache, timeline_cache,
col, col,
ui, ui,
&accounts.mutefun(),
note_options, note_options,
note_context, note_context,
jobs, jobs,
@@ -64,10 +53,8 @@ pub fn render_timeline_route(
let note_action = ui::TimelineView::new( let note_action = ui::TimelineView::new(
kind, kind,
timeline_cache, timeline_cache,
&accounts.mutefun(),
note_context, note_context,
note_options, note_options,
&(&accounts.get_selected_account().key).into(),
jobs, jobs,
col, col,
) )
@@ -83,7 +70,6 @@ pub fn render_timeline_route(
#[allow(clippy::too_many_arguments)] #[allow(clippy::too_many_arguments)]
pub fn render_thread_route( pub fn render_thread_route(
threads: &mut Threads, threads: &mut Threads,
accounts: &mut Accounts,
selection: &ThreadSelection, selection: &ThreadSelection,
col: usize, col: usize,
mut note_options: NoteOptions, mut note_options: NoteOptions,
@@ -99,9 +85,7 @@ pub fn render_thread_route(
threads, threads,
selection.selected_or_root(), selection.selected_or_root(),
note_options, note_options,
&accounts.mutefun(),
note_context, note_context,
&(&accounts.get_selected_account().key).into(),
jobs, jobs,
) )
.id_source(col) .id_source(col)
@@ -112,22 +96,18 @@ pub fn render_thread_route(
#[allow(clippy::too_many_arguments)] #[allow(clippy::too_many_arguments)]
pub fn render_profile_route( pub fn render_profile_route(
pubkey: &Pubkey, pubkey: &Pubkey,
accounts: &Accounts,
timeline_cache: &mut TimelineCache, timeline_cache: &mut TimelineCache,
col: usize, col: usize,
ui: &mut egui::Ui, ui: &mut egui::Ui,
is_muted: &MuteFun,
note_options: NoteOptions, note_options: NoteOptions,
note_context: &mut NoteContext, note_context: &mut NoteContext,
jobs: &mut JobsCache, jobs: &mut JobsCache,
) -> Option<RenderNavAction> { ) -> Option<RenderNavAction> {
let profile_view = ProfileView::new( let profile_view = ProfileView::new(
pubkey, pubkey,
accounts,
col, col,
timeline_cache, timeline_cache,
note_options, note_options,
is_muted,
note_context, note_context,
jobs, jobs,
) )
@@ -135,7 +115,8 @@ pub fn render_profile_route(
if let Some(action) = profile_view { if let Some(action) = profile_view {
match action { match action {
ui::profile::ProfileViewAction::EditProfile => accounts ui::profile::ProfileViewAction::EditProfile => note_context
.accounts
.get_full(pubkey) .get_full(pubkey)
.map(|kp| RenderNavAction::ProfileAction(ProfileAction::Edit(kp.to_full()))), .map(|kp| RenderNavAction::ProfileAction(ProfileAction::Edit(kp.to_full()))),
ui::profile::ProfileViewAction::Note(note_action) => { ui::profile::ProfileViewAction::Note(note_action) => {

View File

@@ -12,7 +12,7 @@ use egui::{
widgets::text_edit::TextEdit, widgets::text_edit::TextEdit,
Frame, Layout, Margin, Pos2, ScrollArea, Sense, TextBuffer, Frame, Layout, Margin, Pos2, ScrollArea, Sense, TextBuffer,
}; };
use enostr::{FilledKeypair, FullKeypair, KeypairUnowned, NoteId, Pubkey, RelayPool}; use enostr::{FilledKeypair, FullKeypair, NoteId, Pubkey, RelayPool};
use nostrdb::{Ndb, Transaction}; use nostrdb::{Ndb, Transaction};
use notedeck_ui::{ use notedeck_ui::{
app_images, app_images,
@@ -352,15 +352,9 @@ impl<'a, 'd> PostView<'a, 'd> {
ui.vertical(|ui| { ui.vertical(|ui| {
ui.set_max_width(avail_size.x * 0.8); ui.set_max_width(avail_size.x * 0.8);
let zapping_acc = self
.note_context
.current_account_has_wallet
.then(|| KeypairUnowned::from(&self.poster));
render_note_preview( render_note_preview(
ui, ui,
self.note_context, self.note_context,
zapping_acc.as_ref(),
txn, txn,
id.bytes(), id.bytes(),
nostrdb::NoteKey::new(0), nostrdb::NoteKey::new(0),
@@ -793,6 +787,7 @@ mod preview {
let txn = Transaction::new(app.ndb).expect("txn"); let txn = Transaction::new(app.ndb).expect("txn");
let mut note_context = NoteContext { let mut note_context = NoteContext {
ndb: app.ndb, ndb: app.ndb,
accounts: app.accounts,
img_cache: app.img_cache, img_cache: app.img_cache,
note_cache: app.note_cache, note_cache: app.note_cache,
zaps: app.zaps, zaps: app.zaps,

View File

@@ -4,7 +4,7 @@ use crate::ui::{
note::{PostAction, PostResponse, PostType}, note::{PostAction, PostResponse, PostType},
}; };
use enostr::{FilledKeypair, KeypairUnowned, NoteId}; use enostr::{FilledKeypair, NoteId};
use notedeck::NoteContext; use notedeck::NoteContext;
use notedeck_ui::jobs::JobsCache; use notedeck_ui::jobs::JobsCache;
use notedeck_ui::{NoteOptions, NoteView, ProfilePic}; use notedeck_ui::{NoteOptions, NoteView, ProfilePic};
@@ -67,27 +67,16 @@ impl<'a, 'd> PostReplyView<'a, 'd> {
let note_offset: i8 = let note_offset: i8 =
pfp_offset - ProfilePic::medium_size() / 2 - NoteView::expand_size() / 2; pfp_offset - ProfilePic::medium_size() / 2 - NoteView::expand_size() / 2;
let zapping_acc = self
.note_context
.current_account_has_wallet
.then(|| KeypairUnowned::from(&self.poster));
let quoted_note = egui::Frame::NONE let quoted_note = egui::Frame::NONE
.outer_margin(egui::Margin::same(note_offset)) .outer_margin(egui::Margin::same(note_offset))
.show(ui, |ui| { .show(ui, |ui| {
NoteView::new( NoteView::new(self.note_context, self.note, self.note_options, self.jobs)
self.note_context, .truncate(false)
zapping_acc.as_ref(), .selectable_text(true)
self.note, .actionbar(false)
self.note_options, .medium_pfp(true)
self.jobs, .options_button(true)
) .show(ui)
.truncate(false)
.selectable_text(true)
.actionbar(false)
.medium_pfp(true)
.options_button(true)
.show(ui)
}) })
.inner; .inner;

View File

@@ -12,8 +12,8 @@ use crate::{
ui::timeline::{tabs_ui, TimelineTabView}, ui::timeline::{tabs_ui, TimelineTabView},
}; };
use notedeck::{ use notedeck::{
name::get_display_name, profile::get_profile_url, Accounts, IsFollowing, MuteFun, NoteAction, name::get_display_name, profile::get_profile_url, IsFollowing, NoteAction, NoteContext,
NoteContext, NotedeckTextStyle, NotedeckTextStyle,
}; };
use notedeck_ui::{ use notedeck_ui::{
app_images, app_images,
@@ -24,11 +24,9 @@ use notedeck_ui::{
pub struct ProfileView<'a, 'd> { pub struct ProfileView<'a, 'd> {
pubkey: &'a Pubkey, pubkey: &'a Pubkey,
accounts: &'a Accounts,
col_id: usize, col_id: usize,
timeline_cache: &'a mut TimelineCache, timeline_cache: &'a mut TimelineCache,
note_options: NoteOptions, note_options: NoteOptions,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
} }
@@ -44,21 +42,17 @@ impl<'a, 'd> ProfileView<'a, 'd> {
#[allow(clippy::too_many_arguments)] #[allow(clippy::too_many_arguments)]
pub fn new( pub fn new(
pubkey: &'a Pubkey, pubkey: &'a Pubkey,
accounts: &'a Accounts,
col_id: usize, col_id: usize,
timeline_cache: &'a mut TimelineCache, timeline_cache: &'a mut TimelineCache,
note_options: NoteOptions, note_options: NoteOptions,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
) -> Self { ) -> Self {
ProfileView { ProfileView {
pubkey, pubkey,
accounts,
col_id, col_id,
timeline_cache, timeline_cache,
note_options, note_options,
is_muted,
note_context, note_context,
jobs, jobs,
} }
@@ -116,9 +110,7 @@ impl<'a, 'd> ProfileView<'a, 'd> {
reversed, reversed,
self.note_options, self.note_options,
&txn, &txn,
self.is_muted,
self.note_context, self.note_context,
&(&self.accounts.get_selected_account().key).into(),
self.jobs, self.jobs,
) )
.show(ui) .show(ui)
@@ -179,14 +171,14 @@ impl<'a, 'd> ProfileView<'a, 'd> {
ui.add_space(24.0); ui.add_space(24.0);
let target_key = self.pubkey; let target_key = self.pubkey;
let selected = self.accounts.get_selected_account(); let selected = self.note_context.accounts.get_selected_account();
let profile_type = if selected.key.secret_key.is_none() { let profile_type = if selected.key.secret_key.is_none() {
ProfileType::ReadOnly ProfileType::ReadOnly
} else if &selected.key.pubkey == self.pubkey { } else if &selected.key.pubkey == self.pubkey {
ProfileType::MyProfile ProfileType::MyProfile
} else { } else {
ProfileType::Followable(selected.is_following(target_key)) ProfileType::Followable(selected.is_following(target_key.bytes()))
}; };
match profile_type { match profile_type {

View File

@@ -1,11 +1,11 @@
use egui::{vec2, Align, Color32, CornerRadius, RichText, Stroke, TextEdit}; use egui::{vec2, Align, Color32, CornerRadius, RichText, Stroke, TextEdit};
use enostr::{KeypairUnowned, NoteId, Pubkey}; use enostr::{NoteId, Pubkey};
use state::TypingType; use state::TypingType;
use crate::{timeline::TimelineTab, ui::timeline::TimelineTabView}; use crate::{timeline::TimelineTab, ui::timeline::TimelineTabView};
use egui_winit::clipboard::Clipboard; use egui_winit::clipboard::Clipboard;
use nostrdb::{Filter, Ndb, Transaction}; use nostrdb::{Filter, Ndb, Transaction};
use notedeck::{MuteFun, NoteAction, NoteContext, NoteRef}; use notedeck::{NoteAction, NoteContext, NoteRef};
use notedeck_ui::{ use notedeck_ui::{
context_menu::{input_context, PasteBehavior}, context_menu::{input_context, PasteBehavior},
icons::search_icon, icons::search_icon,
@@ -25,29 +25,23 @@ pub struct SearchView<'a, 'd> {
query: &'a mut SearchQueryState, query: &'a mut SearchQueryState,
note_options: NoteOptions, note_options: NoteOptions,
txn: &'a Transaction, txn: &'a Transaction,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
} }
impl<'a, 'd> SearchView<'a, 'd> { impl<'a, 'd> SearchView<'a, 'd> {
pub fn new( pub fn new(
txn: &'a Transaction, txn: &'a Transaction,
is_muted: &'a MuteFun,
note_options: NoteOptions, note_options: NoteOptions,
query: &'a mut SearchQueryState, query: &'a mut SearchQueryState,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
) -> Self { ) -> Self {
Self { Self {
txn, txn,
is_muted,
query, query,
note_options, note_options,
note_context, note_context,
cur_acc,
jobs, jobs,
} }
} }
@@ -155,9 +149,7 @@ impl<'a, 'd> SearchView<'a, 'd> {
reversed, reversed,
self.note_options, self.note_options,
self.txn, self.txn,
self.is_muted,
self.note_context, self.note_context,
self.cur_acc,
self.jobs, self.jobs,
) )
.show(ui) .show(ui)

View File

@@ -1,9 +1,8 @@
use egui::InnerResponse; use egui::InnerResponse;
use egui_virtual_list::VirtualList; use egui_virtual_list::VirtualList;
use enostr::KeypairUnowned;
use nostrdb::{Note, Transaction}; use nostrdb::{Note, Transaction};
use notedeck::note::root_note_id_from_selected_id; use notedeck::note::root_note_id_from_selected_id;
use notedeck::{MuteFun, NoteAction, NoteContext}; use notedeck::{NoteAction, NoteContext};
use notedeck_ui::jobs::JobsCache; use notedeck_ui::jobs::JobsCache;
use notedeck_ui::note::NoteResponse; use notedeck_ui::note::NoteResponse;
use notedeck_ui::{NoteOptions, NoteView}; use notedeck_ui::{NoteOptions, NoteView};
@@ -16,9 +15,7 @@ pub struct ThreadView<'a, 'd> {
note_options: NoteOptions, note_options: NoteOptions,
col: usize, col: usize,
id_source: egui::Id, id_source: egui::Id,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
} }
@@ -28,9 +25,7 @@ impl<'a, 'd> ThreadView<'a, 'd> {
threads: &'a mut Threads, threads: &'a mut Threads,
selected_note_id: &'a [u8; 32], selected_note_id: &'a [u8; 32],
note_options: NoteOptions, note_options: NoteOptions,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
) -> Self { ) -> Self {
let id_source = egui::Id::new("threadscroll_threadview"); let id_source = egui::Id::new("threadscroll_threadview");
@@ -39,9 +34,7 @@ impl<'a, 'd> ThreadView<'a, 'd> {
selected_note_id, selected_note_id,
note_options, note_options,
id_source, id_source,
is_muted,
note_context, note_context,
cur_acc,
jobs, jobs,
col: 0, col: 0,
} }
@@ -134,21 +127,14 @@ impl<'a, 'd> ThreadView<'a, 'd> {
ui.colored_label(ui.visuals().error_fg_color, "LOADING NOTES"); ui.colored_label(ui.visuals().error_fg_color, "LOADING NOTES");
} }
let zapping_acc = self
.note_context
.current_account_has_wallet
.then_some(self.cur_acc);
show_notes( show_notes(
ui, ui,
list, list,
&notes, &notes,
self.note_context, self.note_context,
zapping_acc,
self.note_options, self.note_options,
self.jobs, self.jobs,
txn, txn,
self.is_muted,
) )
} }
} }
@@ -159,11 +145,9 @@ fn show_notes(
list: &mut VirtualList, list: &mut VirtualList,
thread_notes: &ThreadNotes, thread_notes: &ThreadNotes,
note_context: &mut NoteContext<'_>, note_context: &mut NoteContext<'_>,
zapping_acc: Option<&KeypairUnowned<'_>>,
flags: NoteOptions, flags: NoteOptions,
jobs: &mut JobsCache, jobs: &mut JobsCache,
txn: &Transaction, txn: &Transaction,
is_muted: &MuteFun,
) -> Option<NoteAction> { ) -> Option<NoteAction> {
let mut action = None; let mut action = None;
@@ -173,6 +157,8 @@ fn show_notes(
let selected_note_index = thread_notes.selected_index; let selected_note_index = thread_notes.selected_index;
let notes = &thread_notes.notes; let notes = &thread_notes.notes;
let is_muted = note_context.accounts.mutefun();
list.ui_custom_layout(ui, notes.len(), |ui, cur_index| { list.ui_custom_layout(ui, notes.len(), |ui, cur_index| {
let note = &notes[cur_index]; let note = &notes[cur_index];
@@ -190,7 +176,7 @@ fn show_notes(
return 1; return 1;
} }
let resp = note.show(note_context, zapping_acc, flags, jobs, ui); let resp = note.show(note_context, flags, jobs, ui);
action = if cur_index == selected_note_index { action = if cur_index == selected_note_index {
resp.action.and_then(strip_note_action) resp.action.and_then(strip_note_action)
@@ -313,21 +299,14 @@ impl<'a> ThreadNote<'a> {
fn show( fn show(
&self, &self,
note_context: &'a mut NoteContext<'_>, note_context: &'a mut NoteContext<'_>,
zapping_acc: Option<&'a KeypairUnowned<'a>>,
flags: NoteOptions, flags: NoteOptions,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
ui: &mut egui::Ui, ui: &mut egui::Ui,
) -> NoteResponse { ) -> NoteResponse {
let inner = notedeck_ui::padding(8.0, ui, |ui| { let inner = notedeck_ui::padding(8.0, ui, |ui| {
NoteView::new( NoteView::new(note_context, &self.note, self.options(flags), jobs)
note_context, .unread_indicator(self.unread_and_have_replies)
zapping_acc, .show(ui)
&self.note,
self.options(flags),
jobs,
)
.unread_indicator(self.unread_and_have_replies)
.show(ui)
}); });
match self.note_type { match self.note_type {

View File

@@ -1,7 +1,6 @@
use egui::containers::scroll_area::ScrollBarVisibility; use egui::containers::scroll_area::ScrollBarVisibility;
use egui::{vec2, Direction, Layout, Pos2, Stroke}; use egui::{vec2, Direction, Layout, Pos2, Stroke};
use egui_tabs::TabColor; use egui_tabs::TabColor;
use enostr::KeypairUnowned;
use nostrdb::Transaction; use nostrdb::Transaction;
use notedeck::ui::is_narrow; use notedeck::ui::is_narrow;
use notedeck_ui::jobs::JobsCache; use notedeck_ui::jobs::JobsCache;
@@ -9,7 +8,7 @@ use std::f32::consts::PI;
use tracing::{error, warn}; use tracing::{error, warn};
use crate::timeline::{TimelineCache, TimelineKind, TimelineTab, ViewFilter}; use crate::timeline::{TimelineCache, TimelineKind, TimelineTab, ViewFilter};
use notedeck::{note::root_note_id_from_selected_id, MuteFun, NoteAction, NoteContext, ScrollInfo}; use notedeck::{note::root_note_id_from_selected_id, NoteAction, NoteContext, ScrollInfo};
use notedeck_ui::{ use notedeck_ui::{
anim::{AnimationHelper, ICON_EXPANSION_MULTIPLE}, anim::{AnimationHelper, ICON_EXPANSION_MULTIPLE},
show_pointer, NoteOptions, NoteView, show_pointer, NoteOptions, NoteView,
@@ -20,9 +19,7 @@ pub struct TimelineView<'a, 'd> {
timeline_cache: &'a mut TimelineCache, timeline_cache: &'a mut TimelineCache,
note_options: NoteOptions, note_options: NoteOptions,
reverse: bool, reverse: bool,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
col: usize, col: usize,
scroll_to_top: bool, scroll_to_top: bool,
@@ -33,10 +30,8 @@ impl<'a, 'd> TimelineView<'a, 'd> {
pub fn new( pub fn new(
timeline_id: &'a TimelineKind, timeline_id: &'a TimelineKind,
timeline_cache: &'a mut TimelineCache, timeline_cache: &'a mut TimelineCache,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
note_options: NoteOptions, note_options: NoteOptions,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
col: usize, col: usize,
) -> Self { ) -> Self {
@@ -47,9 +42,7 @@ impl<'a, 'd> TimelineView<'a, 'd> {
timeline_cache, timeline_cache,
note_options, note_options,
reverse, reverse,
is_muted,
note_context, note_context,
cur_acc,
jobs, jobs,
col, col,
scroll_to_top, scroll_to_top,
@@ -63,9 +56,7 @@ impl<'a, 'd> TimelineView<'a, 'd> {
self.timeline_cache, self.timeline_cache,
self.reverse, self.reverse,
self.note_options, self.note_options,
self.is_muted,
self.note_context, self.note_context,
self.cur_acc,
self.jobs, self.jobs,
self.col, self.col,
self.scroll_to_top, self.scroll_to_top,
@@ -90,9 +81,7 @@ fn timeline_ui(
timeline_cache: &mut TimelineCache, timeline_cache: &mut TimelineCache,
reversed: bool, reversed: bool,
note_options: NoteOptions, note_options: NoteOptions,
is_muted: &MuteFun,
note_context: &mut NoteContext, note_context: &mut NoteContext,
cur_acc: &KeypairUnowned,
jobs: &mut JobsCache, jobs: &mut JobsCache,
col: usize, col: usize,
scroll_to_top: bool, scroll_to_top: bool,
@@ -187,9 +176,7 @@ fn timeline_ui(
reversed, reversed,
note_options, note_options,
&txn, &txn,
is_muted,
note_context, note_context,
cur_acc,
jobs, jobs,
) )
.show(ui) .show(ui)
@@ -372,9 +359,7 @@ pub struct TimelineTabView<'a, 'd> {
reversed: bool, reversed: bool,
note_options: NoteOptions, note_options: NoteOptions,
txn: &'a Transaction, txn: &'a Transaction,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
} }
@@ -385,9 +370,7 @@ impl<'a, 'd> TimelineTabView<'a, 'd> {
reversed: bool, reversed: bool,
note_options: NoteOptions, note_options: NoteOptions,
txn: &'a Transaction, txn: &'a Transaction,
is_muted: &'a MuteFun,
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: &'a KeypairUnowned<'a>,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
) -> Self { ) -> Self {
Self { Self {
@@ -395,9 +378,7 @@ impl<'a, 'd> TimelineTabView<'a, 'd> {
reversed, reversed,
note_options, note_options,
txn, txn,
is_muted,
note_context, note_context,
cur_acc,
jobs, jobs,
} }
} }
@@ -406,7 +387,7 @@ impl<'a, 'd> TimelineTabView<'a, 'd> {
let mut action: Option<NoteAction> = None; let mut action: Option<NoteAction> = None;
let len = self.tab.notes.len(); let len = self.tab.notes.len();
let is_muted = self.is_muted; let is_muted = self.note_context.accounts.mutefun();
self.tab self.tab
.list .list
@@ -444,21 +425,10 @@ impl<'a, 'd> TimelineTabView<'a, 'd> {
}; };
if !muted { if !muted {
let zapping_acc = if self.note_context.current_account_has_wallet {
Some(self.cur_acc)
} else {
None
};
notedeck_ui::padding(8.0, ui, |ui| { notedeck_ui::padding(8.0, ui, |ui| {
let resp = NoteView::new( let resp =
self.note_context, NoteView::new(self.note_context, &note, self.note_options, self.jobs)
zapping_acc, .show(ui);
&note,
self.note_options,
self.jobs,
)
.show(ui);
if let Some(note_action) = resp.action { if let Some(note_action) = resp.action {
action = Some(note_action) action = Some(note_action)

View File

@@ -211,6 +211,7 @@ impl<'a> DaveUi<'a> {
) -> Option<NoteAction> { ) -> Option<NoteAction> {
let mut note_context = NoteContext { let mut note_context = NoteContext {
ndb: ctx.ndb, ndb: ctx.ndb,
accounts: ctx.accounts,
img_cache: ctx.img_cache, img_cache: ctx.img_cache,
note_cache: ctx.note_cache, note_cache: ctx.note_cache,
zaps: ctx.zaps, zaps: ctx.zaps,
@@ -243,7 +244,6 @@ impl<'a> DaveUi<'a> {
|ui| { |ui| {
notedeck_ui::NoteView::new( notedeck_ui::NoteView::new(
&mut note_context, &mut note_context,
None,
&note, &note,
NoteOptions::default(), NoteOptions::default(),
jobs, jobs,

View File

@@ -1,58 +1,5 @@
use nostrdb::{Filter, Ndb, Note, Transaction}; use nostrdb::Filter;
fn pk1_is_following_pk2(
ndb: &Ndb,
txn: &Transaction,
pk1: &[u8; 32],
pk2: &[u8; 32],
) -> Option<bool> {
let note = get_contacts_note(ndb, txn, pk1)?;
Some(note_follows(note, pk2))
}
pub fn trust_media_from_pk2(
ndb: &Ndb,
txn: &Transaction,
pk1: Option<&[u8; 32]>,
pk2: &[u8; 32],
) -> bool {
pk1.map(|pk| pk == pk2 || pk1_is_following_pk2(ndb, txn, pk, pk2).unwrap_or(false))
.unwrap_or(false)
}
fn get_contacts_note<'a>(ndb: &'a Ndb, txn: &'a Transaction, user: &[u8; 32]) -> Option<Note<'a>> {
Some(
ndb.query(txn, &[contacts_filter(user)], 1)
.ok()?
.first()?
.note
.clone(),
)
}
pub fn contacts_filter(pk: &[u8; 32]) -> Filter { pub fn contacts_filter(pk: &[u8; 32]) -> Filter {
Filter::new().authors([pk]).kinds([3]).limit(1).build() Filter::new().authors([pk]).kinds([3]).limit(1).build()
} }
fn note_follows(contacts_note: Note<'_>, pk: &[u8; 32]) -> bool {
for tag in contacts_note.tags() {
if tag.count() < 2 {
continue;
}
let Some("p") = tag.get_str(0) else {
continue;
};
let Some(author) = tag.get_id(1) else {
continue;
};
if pk == author {
return true;
}
}
false
}

View File

@@ -2,23 +2,20 @@ use std::cell::OnceCell;
use crate::{ use crate::{
blur::imeta_blurhashes, blur::imeta_blurhashes,
contacts::trust_media_from_pk2,
jobs::JobsCache, jobs::JobsCache,
note::{NoteAction, NoteOptions, NoteResponse, NoteView}, note::{NoteAction, NoteOptions, NoteResponse, NoteView},
}; };
use egui::{Color32, Hyperlink, RichText}; use egui::{Color32, Hyperlink, RichText};
use enostr::KeypairUnowned;
use nostrdb::{BlockType, Mention, Note, NoteKey, Transaction}; use nostrdb::{BlockType, Mention, Note, NoteKey, Transaction};
use tracing::warn; use tracing::warn;
use notedeck::NoteContext; use notedeck::{IsFollowing, NoteContext};
use super::media::{find_renderable_media, image_carousel, RenderableMedia}; use super::media::{find_renderable_media, image_carousel, RenderableMedia};
pub struct NoteContents<'a, 'd> { pub struct NoteContents<'a, 'd> {
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: Option<&'a KeypairUnowned<'a>>,
txn: &'a Transaction, txn: &'a Transaction,
note: &'a Note<'a>, note: &'a Note<'a>,
options: NoteOptions, options: NoteOptions,
@@ -30,7 +27,6 @@ impl<'a, 'd> NoteContents<'a, 'd> {
#[allow(clippy::too_many_arguments)] #[allow(clippy::too_many_arguments)]
pub fn new( pub fn new(
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
cur_acc: Option<&'a KeypairUnowned<'a>>,
txn: &'a Transaction, txn: &'a Transaction,
note: &'a Note, note: &'a Note,
options: NoteOptions, options: NoteOptions,
@@ -38,7 +34,6 @@ impl<'a, 'd> NoteContents<'a, 'd> {
) -> Self { ) -> Self {
NoteContents { NoteContents {
note_context, note_context,
cur_acc,
txn, txn,
note, note,
options, options,
@@ -53,7 +48,6 @@ impl egui::Widget for &mut NoteContents<'_, '_> {
let result = render_note_contents( let result = render_note_contents(
ui, ui,
self.note_context, self.note_context,
self.cur_acc,
self.txn, self.txn,
self.note, self.note,
self.options, self.options,
@@ -71,7 +65,6 @@ impl egui::Widget for &mut NoteContents<'_, '_> {
pub fn render_note_preview( pub fn render_note_preview(
ui: &mut egui::Ui, ui: &mut egui::Ui,
note_context: &mut NoteContext, note_context: &mut NoteContext,
cur_acc: Option<&KeypairUnowned>,
txn: &Transaction, txn: &Transaction,
id: &[u8; 32], id: &[u8; 32],
parent: NoteKey, parent: NoteKey,
@@ -105,7 +98,7 @@ pub fn render_note_preview(
*/ */
}; };
NoteView::new(note_context, cur_acc, &note, note_options, jobs) NoteView::new(note_context, &note, note_options, jobs)
.preview_style() .preview_style()
.parent(parent) .parent(parent)
.show(ui) .show(ui)
@@ -116,7 +109,6 @@ pub fn render_note_preview(
pub fn render_note_contents( pub fn render_note_contents(
ui: &mut egui::Ui, ui: &mut egui::Ui,
note_context: &mut NoteContext, note_context: &mut NoteContext,
cur_acc: Option<&KeypairUnowned>,
txn: &Transaction, txn: &Transaction,
note: &Note, note: &Note,
options: NoteOptions, options: NoteOptions,
@@ -275,7 +267,7 @@ pub fn render_note_contents(
}); });
let preview_note_action = inline_note.and_then(|(id, _)| { let preview_note_action = inline_note.and_then(|(id, _)| {
render_note_preview(ui, note_context, cur_acc, txn, id, note_key, options, jobs) render_note_preview(ui, note_context, txn, id, note_key, options, jobs)
.action .action
.map(|a| match a { .map(|a| match a {
NoteAction::Note { note_id, .. } => NoteAction::Note { NoteAction::Note { note_id, .. } => NoteAction::Note {
@@ -291,12 +283,11 @@ pub fn render_note_contents(
ui.add_space(2.0); ui.add_space(2.0);
let carousel_id = egui::Id::new(("carousel", note.key().expect("expected tx note"))); let carousel_id = egui::Id::new(("carousel", note.key().expect("expected tx note")));
let trusted_media = trust_media_from_pk2( let trusted_media = note_context
note_context.ndb, .accounts
txn, .get_selected_account()
cur_acc.as_ref().map(|k| k.pubkey.bytes()), .is_following(note.pubkey())
note.pubkey(), == IsFollowing::Yes;
);
media_action = image_carousel( media_action = image_carousel(
ui, ui,

View File

@@ -33,7 +33,6 @@ use notedeck::{
pub struct NoteView<'a, 'd> { pub struct NoteView<'a, 'd> {
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
zapping_acc: Option<&'a KeypairUnowned<'a>>,
parent: Option<NoteKey>, parent: Option<NoteKey>,
note: &'a nostrdb::Note<'a>, note: &'a nostrdb::Note<'a>,
framed: bool, framed: bool,
@@ -85,7 +84,6 @@ impl egui::Widget for &mut NoteView<'_, '_> {
impl<'a, 'd> NoteView<'a, 'd> { impl<'a, 'd> NoteView<'a, 'd> {
pub fn new( pub fn new(
note_context: &'a mut NoteContext<'d>, note_context: &'a mut NoteContext<'d>,
zapping_acc: Option<&'a KeypairUnowned<'a>>,
note: &'a nostrdb::Note<'a>, note: &'a nostrdb::Note<'a>,
mut flags: NoteOptions, mut flags: NoteOptions,
jobs: &'a mut JobsCache, jobs: &'a mut JobsCache,
@@ -98,7 +96,6 @@ impl<'a, 'd> NoteView<'a, 'd> {
Self { Self {
note_context, note_context,
zapping_acc,
parent, parent,
note, note,
flags, flags,
@@ -231,7 +228,6 @@ impl<'a, 'd> NoteView<'a, 'd> {
ui.add(&mut NoteContents::new( ui.add(&mut NoteContents::new(
self.note_context, self.note_context,
self.zapping_acc,
txn, txn,
self.note, self.note,
self.flags, self.flags,
@@ -317,14 +313,7 @@ impl<'a, 'd> NoteView<'a, 'd> {
.text_style(style.text_style()), .text_style(style.text_style()),
); );
}); });
NoteView::new( NoteView::new(self.note_context, &note_to_repost, self.flags, self.jobs).show(ui)
self.note_context,
self.zapping_acc,
&note_to_repost,
self.flags,
self.jobs,
)
.show(ui)
} }
pub fn show_impl(&mut self, ui: &mut egui::Ui) -> NoteResponse { pub fn show_impl(&mut self, ui: &mut egui::Ui) -> NoteResponse {
@@ -440,7 +429,6 @@ impl<'a, 'd> NoteView<'a, 'd> {
ui.horizontal(|ui| { ui.horizontal(|ui| {
note_action = reply_desc( note_action = reply_desc(
ui, ui,
self.zapping_acc,
txn, txn,
&note_reply, &note_reply,
self.note_context, self.note_context,
@@ -455,32 +443,27 @@ impl<'a, 'd> NoteView<'a, 'd> {
}) })
.inner; .inner;
let mut contents = NoteContents::new( let mut contents =
self.note_context, NoteContents::new(self.note_context, txn, self.note, self.flags, self.jobs);
self.zapping_acc,
txn,
self.note,
self.flags,
self.jobs,
);
ui.add(&mut contents); ui.add(&mut contents);
note_action = contents.action.or(note_action); note_action = contents.action.or(note_action);
if self.options().contains(NoteOptions::ActionBar) { if self.options().contains(NoteOptions::ActionBar) {
note_action = render_note_actionbar( let zapper = {
ui, let cur_acc = self.note_context.accounts.get_selected_account();
self.zapping_acc.as_ref().map(|c| Zapper { let has_wallet = cur_acc.wallet.is_some();
has_wallet.then_some(Zapper {
zaps: self.note_context.zaps, zaps: self.note_context.zaps,
cur_acc: c, cur_acc: cur_acc.keypair(),
}), })
self.note.id(), };
self.note.pubkey(), note_action =
note_key, render_note_actionbar(ui, zapper, self.note.id(), self.note.pubkey(), note_key)
) .inner
.inner .or(note_action);
.or(note_action);
} }
NoteUiResponse { NoteUiResponse {
@@ -527,7 +510,6 @@ impl<'a, 'd> NoteView<'a, 'd> {
note_action = reply_desc( note_action = reply_desc(
ui, ui,
self.zapping_acc,
txn, txn,
&note_reply, &note_reply,
self.note_context, self.note_context,
@@ -537,25 +519,25 @@ impl<'a, 'd> NoteView<'a, 'd> {
.or(note_action.take()); .or(note_action.take());
}); });
let mut contents = NoteContents::new( let mut contents =
self.note_context, NoteContents::new(self.note_context, txn, self.note, self.flags, self.jobs);
self.zapping_acc,
txn,
self.note,
self.flags,
self.jobs,
);
ui.add(&mut contents); ui.add(&mut contents);
note_action = contents.action.or(note_action); note_action = contents.action.or(note_action);
let zapper = {
let cur_acc = self.note_context.accounts.get_selected_account();
let has_wallet = cur_acc.wallet.is_some();
has_wallet.then_some(Zapper {
zaps: self.note_context.zaps,
cur_acc: cur_acc.keypair(),
})
};
if self.options().contains(NoteOptions::ActionBar) { if self.options().contains(NoteOptions::ActionBar) {
note_action = render_note_actionbar( note_action = render_note_actionbar(
ui, ui,
self.zapping_acc.as_ref().map(|c| Zapper { zapper,
zaps: self.note_context.zaps,
cur_acc: c,
}),
self.note.id(), self.note.id(),
self.note.pubkey(), self.note.pubkey(),
note_key, note_key,
@@ -774,7 +756,7 @@ fn note_hitbox_clicked(
struct Zapper<'a> { struct Zapper<'a> {
zaps: &'a Zaps, zaps: &'a Zaps,
cur_acc: &'a KeypairUnowned<'a>, cur_acc: KeypairUnowned<'a>,
} }
#[profiling::function] #[profiling::function]

View File

@@ -3,14 +3,12 @@ use nostrdb::{Note, NoteReply, Transaction};
use super::NoteOptions; use super::NoteOptions;
use crate::{jobs::JobsCache, note::NoteView, Mention}; use crate::{jobs::JobsCache, note::NoteView, Mention};
use enostr::KeypairUnowned;
use notedeck::{NoteAction, NoteContext}; use notedeck::{NoteAction, NoteContext};
#[must_use = "Please handle the resulting note action"] #[must_use = "Please handle the resulting note action"]
#[profiling::function] #[profiling::function]
pub fn reply_desc( pub fn reply_desc(
ui: &mut egui::Ui, ui: &mut egui::Ui,
cur_acc: Option<&KeypairUnowned>,
txn: &Transaction, txn: &Transaction,
note_reply: &NoteReply, note_reply: &NoteReply,
note_context: &mut NoteContext, note_context: &mut NoteContext,
@@ -43,7 +41,7 @@ pub fn reply_desc(
if r.hovered() { if r.hovered() {
r.on_hover_ui_at_pointer(|ui| { r.on_hover_ui_at_pointer(|ui| {
ui.set_max_width(400.0); ui.set_max_width(400.0);
NoteView::new(note_context, cur_acc, note, note_options, jobs) NoteView::new(note_context, note, note_options, jobs)
.actionbar(false) .actionbar(false)
.wide(true) .wide(true)
.show(ui); .show(ui);