mirror of
https://github.com/aljazceru/notedeck.git
synced 2026-01-15 06:14:20 +01:00
This adds local nostrdb thread subscriptions. When navigating to a thread, we first check to see if we have any active nostrdb subscriptions for that thread. If not, we create a new subscription. If we do, we re-use that subscription. This works by storing thread state in the Threads struct in the Damus application state. When we pop a route, we check to see if its a thread route. If it is, then we try to unsubscribe, but only if that is the last remaining subscriber for that thread, as there could be more than one. Signed-off-by: William Casarin <jb55@jb55.com>
125 lines
4.2 KiB
Rust
125 lines
4.2 KiB
Rust
use crate::{timeline::TimelineSource, ui, Damus};
|
|
use nostrdb::{NoteKey, Transaction};
|
|
use std::collections::HashSet;
|
|
use tracing::warn;
|
|
|
|
pub struct ThreadView<'a> {
|
|
app: &'a mut Damus,
|
|
timeline: usize,
|
|
selected_note_id: &'a [u8; 32],
|
|
}
|
|
|
|
impl<'a> ThreadView<'a> {
|
|
pub fn new(app: &'a mut Damus, timeline: usize, selected_note_id: &'a [u8; 32]) -> Self {
|
|
ThreadView {
|
|
app,
|
|
timeline,
|
|
selected_note_id,
|
|
}
|
|
}
|
|
|
|
pub fn ui(&mut self, ui: &mut egui::Ui) {
|
|
let txn = Transaction::new(&self.app.ndb).expect("txn");
|
|
|
|
let selected_note_key = if let Ok(key) = self
|
|
.app
|
|
.ndb
|
|
.get_notekey_by_id(&txn, self.selected_note_id)
|
|
.map(NoteKey::new)
|
|
{
|
|
key
|
|
} else {
|
|
// TODO: render 404 ?
|
|
return;
|
|
};
|
|
|
|
let scroll_id = egui::Id::new((
|
|
"threadscroll",
|
|
self.app.timelines[self.timeline].selected_view,
|
|
self.timeline,
|
|
selected_note_key,
|
|
));
|
|
|
|
ui.label(
|
|
egui::RichText::new("Threads ALPHA! It's not done. Things will be broken.")
|
|
.color(egui::Color32::RED),
|
|
);
|
|
|
|
egui::ScrollArea::vertical()
|
|
.id_source(scroll_id)
|
|
.animated(false)
|
|
.auto_shrink([false, false])
|
|
.scroll_bar_visibility(egui::scroll_area::ScrollBarVisibility::AlwaysVisible)
|
|
.show(ui, |ui| {
|
|
let note = if let Ok(note) = self.app.ndb.get_note_by_key(&txn, selected_note_key) {
|
|
note
|
|
} else {
|
|
return;
|
|
};
|
|
|
|
let root_id = {
|
|
let cached_note = self
|
|
.app
|
|
.note_cache_mut()
|
|
.cached_note_or_insert(selected_note_key, ¬e);
|
|
|
|
cached_note
|
|
.reply
|
|
.borrow(note.tags())
|
|
.root()
|
|
.map_or_else(|| self.selected_note_id, |nr| nr.id)
|
|
};
|
|
|
|
// poll for new notes and insert them into our existing notes
|
|
{
|
|
let mut ids = HashSet::new();
|
|
let _ = TimelineSource::Thread(root_id)
|
|
.poll_notes_into_view(self.app, &txn, &mut ids);
|
|
// TODO: do something with unknown ids
|
|
}
|
|
|
|
let (len, list) = {
|
|
let thread = self.app.threads.thread_mut(&self.app.ndb, &txn, root_id);
|
|
|
|
let len = thread.view.notes.len();
|
|
(len, &mut thread.view.list)
|
|
};
|
|
|
|
list.clone()
|
|
.borrow_mut()
|
|
.ui_custom_layout(ui, len, |ui, start_index| {
|
|
ui.spacing_mut().item_spacing.y = 0.0;
|
|
ui.spacing_mut().item_spacing.x = 4.0;
|
|
|
|
let note_key = {
|
|
let thread = self.app.threads.thread_mut(&self.app.ndb, &txn, root_id);
|
|
thread.view.notes[start_index].key
|
|
};
|
|
|
|
let note = if let Ok(note) = self.app.ndb.get_note_by_key(&txn, note_key) {
|
|
note
|
|
} else {
|
|
warn!("failed to query note {:?}", note_key);
|
|
return 0;
|
|
};
|
|
|
|
ui::padding(8.0, ui, |ui| {
|
|
let textmode = self.app.textmode;
|
|
let resp = ui::NoteView::new(self.app, ¬e)
|
|
.note_previews(!textmode)
|
|
.show(ui);
|
|
|
|
if let Some(action) = resp.action {
|
|
action.execute(self.app, self.timeline, note.id(), &txn);
|
|
}
|
|
});
|
|
|
|
ui::hline(ui);
|
|
//ui.add(egui::Separator::default().spacing(0.0));
|
|
|
|
1
|
|
});
|
|
});
|
|
}
|
|
}
|