feat: handle new message
This commit is contained in:
@@ -287,21 +287,8 @@ async fn main() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
Signal::Event(event) => {
|
Signal::Event(event) => {
|
||||||
let metadata = async_cx
|
_ = async_cx.update_global::<ChatRegistry, _>(|state, cx| {
|
||||||
.background_executor()
|
state.receive(event, cx)
|
||||||
.spawn(async move {
|
|
||||||
if let Ok(metadata) =
|
|
||||||
client.database().metadata(event.pubkey).await
|
|
||||||
{
|
|
||||||
metadata.unwrap_or_default()
|
|
||||||
} else {
|
|
||||||
Metadata::new()
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.await;
|
|
||||||
|
|
||||||
_ = async_cx.update_global::<ChatRegistry, _>(|chat, cx| {
|
|
||||||
chat.receive(event, metadata, cx)
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,42 +1,38 @@
|
|||||||
use crate::{get_client, utils::room_hash};
|
use crate::{
|
||||||
use gpui::{AppContext, Context, Global, Model, SharedString, WeakModel};
|
get_client,
|
||||||
|
utils::{compare, room_hash},
|
||||||
|
};
|
||||||
|
use gpui::{AppContext, Context, Global, Model, WeakModel};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use nostr_sdk::prelude::*;
|
use nostr_sdk::prelude::*;
|
||||||
use room::Room;
|
use room::Room;
|
||||||
use std::{
|
use std::cmp::Reverse;
|
||||||
cmp::Reverse,
|
|
||||||
collections::HashMap,
|
|
||||||
sync::{Arc, RwLock},
|
|
||||||
};
|
|
||||||
|
|
||||||
pub mod room;
|
pub mod room;
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
pub struct Inbox {
|
||||||
pub struct NewMessage {
|
pub(crate) rooms: Vec<Model<Room>>,
|
||||||
pub event: Event,
|
pub(crate) is_loading: bool,
|
||||||
pub metadata: Metadata,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl NewMessage {
|
impl Inbox {
|
||||||
pub fn new(event: Event, metadata: Metadata) -> Self {
|
pub fn new() -> Self {
|
||||||
// TODO: parse event's content
|
Self {
|
||||||
Self { event, metadata }
|
rooms: vec![],
|
||||||
|
is_loading: true,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
type NewMessages = RwLock<HashMap<SharedString, Arc<RwLock<Vec<NewMessage>>>>>;
|
|
||||||
|
|
||||||
pub struct ChatRegistry {
|
pub struct ChatRegistry {
|
||||||
inbox: Model<Vec<Model<Room>>>,
|
inbox: Model<Inbox>,
|
||||||
new_messages: Model<NewMessages>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Global for ChatRegistry {}
|
impl Global for ChatRegistry {}
|
||||||
|
|
||||||
impl ChatRegistry {
|
impl ChatRegistry {
|
||||||
pub fn set_global(cx: &mut AppContext) {
|
pub fn set_global(cx: &mut AppContext) {
|
||||||
let inbox = cx.new_model(|_| Vec::new());
|
let inbox = cx.new_model(|_| Inbox::new());
|
||||||
let new_messages = cx.new_model(|_| RwLock::new(HashMap::new()));
|
|
||||||
|
|
||||||
cx.observe_new_models::<Room>(|this, cx| {
|
cx.observe_new_models::<Room>(|this, cx| {
|
||||||
// Get all pubkeys to load metadata
|
// Get all pubkeys to load metadata
|
||||||
@@ -76,10 +72,7 @@ impl ChatRegistry {
|
|||||||
})
|
})
|
||||||
.detach();
|
.detach();
|
||||||
|
|
||||||
cx.set_global(Self {
|
cx.set_global(Self { inbox });
|
||||||
inbox,
|
|
||||||
new_messages,
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn init(&mut self, cx: &mut AppContext) {
|
pub fn init(&mut self, cx: &mut AppContext) {
|
||||||
@@ -90,6 +83,7 @@ impl ChatRegistry {
|
|||||||
let hashes: Vec<u64> = self
|
let hashes: Vec<u64> = self
|
||||||
.inbox
|
.inbox
|
||||||
.read(cx)
|
.read(cx)
|
||||||
|
.rooms
|
||||||
.iter()
|
.iter()
|
||||||
.map(|room| room.read(cx).id)
|
.map(|room| room.read(cx).id)
|
||||||
.collect();
|
.collect();
|
||||||
@@ -139,7 +133,9 @@ impl ChatRegistry {
|
|||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
model.extend(items);
|
model.rooms.extend(items);
|
||||||
|
model.is_loading = false;
|
||||||
|
|
||||||
cx.notify();
|
cx.notify();
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -147,36 +143,45 @@ impl ChatRegistry {
|
|||||||
.detach();
|
.detach();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn inbox(&self) -> WeakModel<Vec<Model<Room>>> {
|
pub fn inbox(&self) -> WeakModel<Inbox> {
|
||||||
self.inbox.downgrade()
|
self.inbox.downgrade()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn room(&self, id: &u64, cx: &AppContext) -> Option<WeakModel<Room>> {
|
pub fn room(&self, id: &u64, cx: &AppContext) -> Option<WeakModel<Room>> {
|
||||||
self.inbox
|
self.inbox
|
||||||
.read(cx)
|
.read(cx)
|
||||||
|
.rooms
|
||||||
.iter()
|
.iter()
|
||||||
.find(|model| &model.read(cx).id == id)
|
.find(|model| &model.read(cx).id == id)
|
||||||
.map(|model| model.downgrade())
|
.map(|model| model.downgrade())
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn new_messages(&self) -> WeakModel<NewMessages> {
|
pub fn receive(&mut self, event: Event, cx: &mut AppContext) {
|
||||||
self.new_messages.downgrade()
|
let mut pubkeys: Vec<_> = event.tags.public_keys().copied().collect();
|
||||||
}
|
pubkeys.push(event.pubkey);
|
||||||
|
|
||||||
pub fn receive(&mut self, event: Event, metadata: Metadata, cx: &mut AppContext) {
|
self.inbox.update(cx, |this, cx| {
|
||||||
let entry = room_hash(&event.tags).to_string().into();
|
if let Some(room) = this.rooms.iter().find(|room| {
|
||||||
let message = NewMessage::new(event, metadata);
|
let room = room.read(cx);
|
||||||
|
let mut all_keys: Vec<_> = room.members.iter().map(|m| m.public_key()).collect();
|
||||||
|
all_keys.push(room.owner.public_key());
|
||||||
|
|
||||||
self.new_messages.update(cx, |this, cx| {
|
compare(&all_keys, &pubkeys)
|
||||||
this.write()
|
}) {
|
||||||
.unwrap()
|
room.update(cx, |this, cx| {
|
||||||
.entry(entry)
|
this.new_messages.push(event);
|
||||||
.or_insert(Arc::new(RwLock::new(Vec::new())))
|
cx.notify();
|
||||||
.write()
|
})
|
||||||
.unwrap()
|
} else {
|
||||||
.push(message);
|
let room = cx.new_model(|_| Room::new(&event));
|
||||||
|
|
||||||
|
self.inbox.update(cx, |this, cx| {
|
||||||
|
this.rooms.insert(0, room);
|
||||||
cx.notify();
|
cx.notify();
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// cx.notify();
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -54,6 +54,7 @@ pub struct Room {
|
|||||||
pub members: Vec<Member>, // Extract from event's tags
|
pub members: Vec<Member>, // Extract from event's tags
|
||||||
pub last_seen: Timestamp,
|
pub last_seen: Timestamp,
|
||||||
pub is_group: bool,
|
pub is_group: bool,
|
||||||
|
pub new_messages: Vec<Event>, // Hold all new messages
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Room {
|
impl Room {
|
||||||
@@ -87,6 +88,15 @@ impl Room {
|
|||||||
title,
|
title,
|
||||||
last_seen,
|
last_seen,
|
||||||
is_group,
|
is_group,
|
||||||
|
new_messages: vec![],
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn metadata(&self, public_key: PublicKey) -> Metadata {
|
||||||
|
if let Some(member) = self.members.iter().find(|m| m.public_key == public_key) {
|
||||||
|
member.metadata()
|
||||||
|
} else {
|
||||||
|
Metadata::default()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -12,8 +12,9 @@ use std::sync::Arc;
|
|||||||
use ui::{
|
use ui::{
|
||||||
dock::{DockArea, DockItem, DockPlacement},
|
dock::{DockArea, DockItem, DockPlacement},
|
||||||
indicator::Indicator,
|
indicator::Indicator,
|
||||||
|
notification::NotificationType,
|
||||||
theme::Theme,
|
theme::Theme,
|
||||||
Root, Sizable, TitleBar,
|
ContextModal, Root, Sizable, TitleBar,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone, PartialEq, Eq, Deserialize)]
|
#[derive(Clone, PartialEq, Eq, Deserialize)]
|
||||||
@@ -141,6 +142,11 @@ impl AppView {
|
|||||||
self.dock.update(cx, |dock_area, cx| {
|
self.dock.update(cx, |dock_area, cx| {
|
||||||
dock_area.add_panel(panel, action.position, cx);
|
dock_area.add_panel(panel, action.position, cx);
|
||||||
});
|
});
|
||||||
|
} else {
|
||||||
|
cx.push_notification((
|
||||||
|
NotificationType::Error,
|
||||||
|
"System error. Cannot open this chat room.",
|
||||||
|
));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,9 @@
|
|||||||
use crate::{
|
use crate::{get_client, states::chat::room::Room, utils::compare};
|
||||||
get_client,
|
|
||||||
states::chat::room::{Member, Room},
|
|
||||||
utils::compare,
|
|
||||||
};
|
|
||||||
use gpui::{
|
use gpui::{
|
||||||
div, list, px, AnyElement, AppContext, Context, EventEmitter, Flatten, FocusHandle,
|
div, list, px, AnyElement, AppContext, Context, EventEmitter, Flatten, FocusHandle,
|
||||||
FocusableView, IntoElement, ListAlignment, ListState, Model, ParentElement, PathPromptOptions,
|
FocusableView, IntoElement, ListAlignment, ListState, Model, ParentElement, PathPromptOptions,
|
||||||
Pixels, Render, SharedString, Styled, View, ViewContext, VisualContext, WindowContext,
|
Pixels, Render, SharedString, Styled, View, ViewContext, VisualContext, WeakModel,
|
||||||
|
WindowContext,
|
||||||
};
|
};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use message::RoomMessage;
|
use message::RoomMessage;
|
||||||
@@ -31,25 +28,22 @@ pub struct State {
|
|||||||
|
|
||||||
pub struct ChatPanel {
|
pub struct ChatPanel {
|
||||||
// Panel
|
// Panel
|
||||||
name: SharedString,
|
|
||||||
closeable: bool,
|
closeable: bool,
|
||||||
zoomable: bool,
|
zoomable: bool,
|
||||||
focus_handle: FocusHandle,
|
focus_handle: FocusHandle,
|
||||||
// Chat Room
|
// Chat Room
|
||||||
id: SharedString,
|
id: SharedString,
|
||||||
owner: Member,
|
name: SharedString,
|
||||||
members: Arc<[Member]>,
|
room: Model<Room>,
|
||||||
input: View<TextInput>,
|
|
||||||
list: ListState,
|
|
||||||
state: Model<State>,
|
state: Model<State>,
|
||||||
|
list: ListState,
|
||||||
|
input: View<TextInput>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ChatPanel {
|
impl ChatPanel {
|
||||||
pub fn new(room: Model<Room>, cx: &mut WindowContext) -> View<Self> {
|
pub fn new(model: Model<Room>, cx: &mut WindowContext) -> View<Self> {
|
||||||
let room = room.read(cx);
|
let room = model.read(cx);
|
||||||
let id = room.id.to_string().into();
|
let id = room.id.to_string().into();
|
||||||
let owner = room.owner.clone();
|
|
||||||
let members = room.members.clone().into();
|
|
||||||
let name = room.title.clone().unwrap_or("Untitled".into());
|
let name = room.title.clone().unwrap_or("Untitled".into());
|
||||||
|
|
||||||
cx.observe_new_views::<Self>(|this, cx| {
|
cx.observe_new_views::<Self>(|this, cx| {
|
||||||
@@ -67,7 +61,7 @@ impl ChatPanel {
|
|||||||
.cleanable()
|
.cleanable()
|
||||||
});
|
});
|
||||||
|
|
||||||
// Send message when user presses enter on form.
|
// Send message when user presses enter
|
||||||
cx.subscribe(&input, move |this: &mut ChatPanel, _, input_event, cx| {
|
cx.subscribe(&input, move |this: &mut ChatPanel, _, input_event, cx| {
|
||||||
if let InputEvent::PressEnter = input_event {
|
if let InputEvent::PressEnter = input_event {
|
||||||
this.send_message(cx);
|
this.send_message(cx);
|
||||||
@@ -80,6 +74,7 @@ impl ChatPanel {
|
|||||||
items: vec![],
|
items: vec![],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Update list on every state changes
|
||||||
cx.observe(&state, |this, model, cx| {
|
cx.observe(&state, |this, model, cx| {
|
||||||
let items = model.read(cx).items.clone();
|
let items = model.read(cx).items.clone();
|
||||||
|
|
||||||
@@ -97,32 +92,35 @@ impl ChatPanel {
|
|||||||
})
|
})
|
||||||
.detach();
|
.detach();
|
||||||
|
|
||||||
let list = ListState::new(0, ListAlignment::Bottom, Pixels(256.), move |_, _| {
|
cx.observe(&model, |this, model, cx| {
|
||||||
div().into_any_element()
|
this.load_new_messages(model.downgrade(), cx);
|
||||||
});
|
})
|
||||||
|
.detach();
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
closeable: true,
|
closeable: true,
|
||||||
zoomable: true,
|
zoomable: true,
|
||||||
focus_handle: cx.focus_handle(),
|
focus_handle: cx.focus_handle(),
|
||||||
|
room: model,
|
||||||
|
list: ListState::new(0, ListAlignment::Bottom, Pixels(256.), move |_, _| {
|
||||||
|
div().into_any_element()
|
||||||
|
}),
|
||||||
id,
|
id,
|
||||||
owner,
|
|
||||||
members,
|
|
||||||
name,
|
name,
|
||||||
input,
|
input,
|
||||||
list,
|
|
||||||
state,
|
state,
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
fn load_messages(&self, cx: &mut ViewContext<Self>) {
|
fn load_messages(&self, cx: &mut ViewContext<Self>) {
|
||||||
let members = Arc::clone(&self.members);
|
let room = self.room.read(cx);
|
||||||
let owner = self.owner.clone();
|
let members = room.members.clone();
|
||||||
|
let owner = room.owner.clone();
|
||||||
|
|
||||||
// Get all public keys
|
// Get all public keys
|
||||||
let mut all_keys: Vec<_> = self.members.iter().map(|m| m.public_key()).collect();
|
let mut all_keys: Vec<_> = room.members.iter().map(|m| m.public_key()).collect();
|
||||||
all_keys.push(self.owner.public_key());
|
all_keys.push(room.owner.public_key());
|
||||||
|
|
||||||
// Async
|
// Async
|
||||||
let async_state = self.state.clone();
|
let async_state = self.state.clone();
|
||||||
@@ -201,11 +199,38 @@ impl ChatPanel {
|
|||||||
.detach();
|
.detach();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn send_message(&mut self, cx: &mut ViewContext<Self>) {
|
fn load_new_messages(&self, model: WeakModel<Room>, cx: &mut ViewContext<Self>) {
|
||||||
let content = Arc::new(self.input.read(cx).text().to_string());
|
if let Some(model) = model.upgrade() {
|
||||||
let owner = self.owner.clone();
|
let room = model.read(cx);
|
||||||
|
let items: Vec<RoomMessage> = room
|
||||||
|
.new_messages
|
||||||
|
.iter()
|
||||||
|
.map(|event| {
|
||||||
|
let metadata = room.metadata(event.pubkey);
|
||||||
|
|
||||||
let mut members = self.members.to_vec();
|
RoomMessage::new(
|
||||||
|
event.pubkey,
|
||||||
|
metadata,
|
||||||
|
event.content.clone(),
|
||||||
|
event.created_at,
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
cx.update_model(&self.state, |model, cx| {
|
||||||
|
model.items.extend(items);
|
||||||
|
model.count = model.items.len();
|
||||||
|
cx.notify();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn send_message(&mut self, cx: &mut ViewContext<Self>) {
|
||||||
|
let room = self.room.read(cx);
|
||||||
|
let content = Arc::new(self.input.read(cx).text().to_string());
|
||||||
|
let owner = room.owner.clone();
|
||||||
|
|
||||||
|
let mut members = room.members.to_vec();
|
||||||
members.push(owner.clone());
|
members.push(owner.clone());
|
||||||
|
|
||||||
// Async
|
// Async
|
||||||
|
|||||||
@@ -1,351 +0,0 @@
|
|||||||
use gpui::{
|
|
||||||
div, list, px, Context, Flatten, IntoElement, ListAlignment, ListState, Model, ParentElement,
|
|
||||||
PathPromptOptions, Pixels, Render, SharedString, Styled, View, ViewContext, VisualContext,
|
|
||||||
};
|
|
||||||
use itertools::Itertools;
|
|
||||||
use nostr_sdk::prelude::*;
|
|
||||||
use std::sync::Arc;
|
|
||||||
use ui::{
|
|
||||||
button::{Button, ButtonVariants},
|
|
||||||
input::{InputEvent, TextInput},
|
|
||||||
theme::ActiveTheme,
|
|
||||||
v_flex, Icon, IconName,
|
|
||||||
};
|
|
||||||
|
|
||||||
use super::message::RoomMessage;
|
|
||||||
use crate::{
|
|
||||||
get_client,
|
|
||||||
states::chat::{ChatRegistry, Member, Room},
|
|
||||||
};
|
|
||||||
|
|
||||||
#[derive(Clone)]
|
|
||||||
pub struct Messages {
|
|
||||||
count: usize,
|
|
||||||
items: Vec<RoomMessage>,
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct RoomPanel {
|
|
||||||
id: SharedString,
|
|
||||||
owner: PublicKey,
|
|
||||||
members: Vec<Member>,
|
|
||||||
// Form
|
|
||||||
input: View<TextInput>,
|
|
||||||
// Messages
|
|
||||||
list: ListState,
|
|
||||||
messages: Model<Messages>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl RoomPanel {
|
|
||||||
pub fn new(room: &Arc<Room>, cx: &mut ViewContext<'_, Self>) -> Self {
|
|
||||||
let id = room.id.clone();
|
|
||||||
let members = room.members.clone();
|
|
||||||
let owner = room.owner;
|
|
||||||
|
|
||||||
// Form
|
|
||||||
let input = cx.new_view(|cx| {
|
|
||||||
TextInput::new(cx)
|
|
||||||
.appearance(false)
|
|
||||||
.text_size(ui::Size::Small)
|
|
||||||
.placeholder("Message...")
|
|
||||||
.cleanable()
|
|
||||||
});
|
|
||||||
|
|
||||||
// Send message when user presses enter on form.
|
|
||||||
cx.subscribe(&input, move |this, _, input_event, cx| {
|
|
||||||
if let InputEvent::PressEnter = input_event {
|
|
||||||
this.send_message(cx);
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
|
|
||||||
let messages = cx.new_model(|_| Messages {
|
|
||||||
count: 0,
|
|
||||||
items: vec![],
|
|
||||||
});
|
|
||||||
|
|
||||||
cx.observe(&messages, |this, model, cx| {
|
|
||||||
let items = model.read(cx).items.clone();
|
|
||||||
|
|
||||||
this.list = ListState::new(
|
|
||||||
items.len(),
|
|
||||||
ListAlignment::Bottom,
|
|
||||||
Pixels(256.),
|
|
||||||
move |idx, _cx| {
|
|
||||||
let item = items.get(idx).unwrap().clone();
|
|
||||||
div().child(item).into_any_element()
|
|
||||||
},
|
|
||||||
);
|
|
||||||
|
|
||||||
cx.notify();
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
|
|
||||||
let list = ListState::new(0, ListAlignment::Bottom, Pixels(256.), move |_, _| {
|
|
||||||
div().into_any_element()
|
|
||||||
});
|
|
||||||
|
|
||||||
Self {
|
|
||||||
id,
|
|
||||||
owner,
|
|
||||||
members,
|
|
||||||
input,
|
|
||||||
list,
|
|
||||||
messages,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn load(&self, cx: &mut ViewContext<Self>) {
|
|
||||||
let async_messages = self.messages.clone();
|
|
||||||
let mut async_cx = cx.to_async();
|
|
||||||
|
|
||||||
let public_keys: Vec<PublicKey> = self.members.iter().map(|m| m.public_key()).collect();
|
|
||||||
|
|
||||||
cx.foreground_executor()
|
|
||||||
.spawn({
|
|
||||||
let client = get_client();
|
|
||||||
let owner = self.owner;
|
|
||||||
|
|
||||||
let recv = Filter::new()
|
|
||||||
.kind(Kind::PrivateDirectMessage)
|
|
||||||
.author(owner)
|
|
||||||
.pubkeys(public_keys.clone());
|
|
||||||
|
|
||||||
let send = Filter::new()
|
|
||||||
.kind(Kind::PrivateDirectMessage)
|
|
||||||
.authors(public_keys)
|
|
||||||
.pubkey(owner);
|
|
||||||
|
|
||||||
async move {
|
|
||||||
let events = async_cx
|
|
||||||
.background_executor()
|
|
||||||
.spawn(async move { client.database().query(vec![recv, send]).await })
|
|
||||||
.await;
|
|
||||||
|
|
||||||
if let Ok(events) = events {
|
|
||||||
let mut items: Vec<RoomMessage> = Vec::new();
|
|
||||||
|
|
||||||
for event in events.into_iter().sorted_by_key(|ev| ev.created_at) {
|
|
||||||
let metadata = async_cx
|
|
||||||
.background_executor()
|
|
||||||
.spawn(
|
|
||||||
async move { client.database().metadata(event.pubkey).await },
|
|
||||||
)
|
|
||||||
.await;
|
|
||||||
|
|
||||||
let message = if let Ok(metadata) = metadata {
|
|
||||||
RoomMessage::new(
|
|
||||||
event.pubkey,
|
|
||||||
metadata,
|
|
||||||
event.content,
|
|
||||||
event.created_at,
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
RoomMessage::new(
|
|
||||||
event.pubkey,
|
|
||||||
None,
|
|
||||||
event.content,
|
|
||||||
event.created_at,
|
|
||||||
)
|
|
||||||
};
|
|
||||||
|
|
||||||
items.push(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
let total = items.len();
|
|
||||||
|
|
||||||
_ = async_cx.update_model(&async_messages, |a, b| {
|
|
||||||
a.items = items;
|
|
||||||
a.count = total;
|
|
||||||
b.notify();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn subscribe(&self, cx: &mut ViewContext<Self>) {
|
|
||||||
let room_id = self.id.clone();
|
|
||||||
let messages = self.messages.clone();
|
|
||||||
let state = cx.global::<ChatRegistry>().messages();
|
|
||||||
|
|
||||||
if let Some(state) = state.upgrade() {
|
|
||||||
cx.observe(&state, move |_, model, cx| {
|
|
||||||
let new_messages = model.read(cx).read().unwrap().get(&room_id).cloned();
|
|
||||||
|
|
||||||
if let Some(new_messages) = new_messages {
|
|
||||||
let items: Vec<RoomMessage> = new_messages
|
|
||||||
.read()
|
|
||||||
.unwrap()
|
|
||||||
.clone()
|
|
||||||
.into_iter()
|
|
||||||
.map(|m| {
|
|
||||||
RoomMessage::new(
|
|
||||||
m.event.pubkey,
|
|
||||||
m.metadata,
|
|
||||||
m.event.content,
|
|
||||||
m.event.created_at,
|
|
||||||
)
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
cx.update_model(&messages, |model, cx| {
|
|
||||||
model.items.extend(items);
|
|
||||||
model.count = model.items.len();
|
|
||||||
cx.notify();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn send_message(&mut self, cx: &mut ViewContext<Self>) {
|
|
||||||
let members: Vec<PublicKey> = self.members.iter().map(|m| m.public_key()).collect();
|
|
||||||
let members2 = members.clone();
|
|
||||||
|
|
||||||
let content = self.input.read(cx).text().to_string();
|
|
||||||
let content2 = content.clone();
|
|
||||||
let content3 = content2.clone();
|
|
||||||
|
|
||||||
let async_input = self.input.clone();
|
|
||||||
let async_messages = self.messages.clone();
|
|
||||||
let mut async_cx = cx.to_async();
|
|
||||||
|
|
||||||
cx.foreground_executor()
|
|
||||||
.spawn({
|
|
||||||
let client = get_client();
|
|
||||||
|
|
||||||
async move {
|
|
||||||
let current_user = async_cx
|
|
||||||
.background_executor()
|
|
||||||
.spawn(async move {
|
|
||||||
let signer = client.signer().await.unwrap();
|
|
||||||
signer.get_public_key().await.unwrap()
|
|
||||||
})
|
|
||||||
.await;
|
|
||||||
|
|
||||||
// Send message to all members
|
|
||||||
async_cx
|
|
||||||
.background_executor()
|
|
||||||
.spawn(async move {
|
|
||||||
let extra_tags: Vec<Tag> = members
|
|
||||||
.iter()
|
|
||||||
.filter_map(|m| {
|
|
||||||
if m != ¤t_user {
|
|
||||||
Some(Tag::public_key(*m))
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
for member in members.iter() {
|
|
||||||
_ = client
|
|
||||||
.send_private_msg(*member, &content, extra_tags.clone())
|
|
||||||
.await
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
|
|
||||||
// Send a copy to yourself
|
|
||||||
async_cx
|
|
||||||
.background_executor()
|
|
||||||
.spawn(async move {
|
|
||||||
let extra_tags: Vec<Tag> = members2
|
|
||||||
.iter()
|
|
||||||
.filter_map(|m| {
|
|
||||||
if m != ¤t_user {
|
|
||||||
Some(Tag::public_key(*m))
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
_ = client
|
|
||||||
.send_private_msg(current_user, content2, extra_tags)
|
|
||||||
.await;
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
|
|
||||||
// Create a new room message
|
|
||||||
let new_message: anyhow::Result<RoomMessage, anyhow::Error> = async_cx
|
|
||||||
.background_executor()
|
|
||||||
.spawn(async move {
|
|
||||||
let metadata = client.database().metadata(current_user).await?;
|
|
||||||
let created_at = Timestamp::now();
|
|
||||||
let message =
|
|
||||||
RoomMessage::new(current_user, metadata, content3, created_at);
|
|
||||||
|
|
||||||
Ok(message)
|
|
||||||
})
|
|
||||||
.await;
|
|
||||||
|
|
||||||
if let Ok(message) = new_message {
|
|
||||||
_ = async_cx.update_model(&async_messages, |model, cx| {
|
|
||||||
model.items.extend(vec![message]);
|
|
||||||
model.count = model.items.len();
|
|
||||||
cx.notify();
|
|
||||||
});
|
|
||||||
|
|
||||||
_ = async_cx.update_view(&async_input, |input, cx| {
|
|
||||||
input.set_text("", cx);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Render for RoomPanel {
|
|
||||||
fn render(&mut self, cx: &mut ViewContext<Self>) -> impl IntoElement {
|
|
||||||
v_flex()
|
|
||||||
.size_full()
|
|
||||||
.child(list(self.list.clone()).flex_1())
|
|
||||||
.child(
|
|
||||||
div()
|
|
||||||
.flex_shrink_0()
|
|
||||||
.w_full()
|
|
||||||
.h_12()
|
|
||||||
.flex()
|
|
||||||
.items_center()
|
|
||||||
.gap_2()
|
|
||||||
.px_2()
|
|
||||||
.child(
|
|
||||||
Button::new("upload")
|
|
||||||
.icon(Icon::new(IconName::Upload))
|
|
||||||
.ghost()
|
|
||||||
.on_click(|_, cx| {
|
|
||||||
let paths = cx.prompt_for_paths(PathPromptOptions {
|
|
||||||
files: true,
|
|
||||||
directories: false,
|
|
||||||
multiple: false,
|
|
||||||
});
|
|
||||||
|
|
||||||
cx.spawn(move |_async_cx| async move {
|
|
||||||
match Flatten::flatten(paths.await.map_err(|e| e.into())) {
|
|
||||||
Ok(Some(paths)) => {
|
|
||||||
// TODO: upload file to blossom server
|
|
||||||
println!("Paths: {:?}", paths)
|
|
||||||
}
|
|
||||||
Ok(None) => {}
|
|
||||||
Err(_) => {}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}),
|
|
||||||
)
|
|
||||||
.child(
|
|
||||||
div()
|
|
||||||
.flex_1()
|
|
||||||
.flex()
|
|
||||||
.bg(cx.theme().muted)
|
|
||||||
.rounded(px(cx.theme().radius))
|
|
||||||
.px_2()
|
|
||||||
.child(self.input.clone()),
|
|
||||||
),
|
|
||||||
)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -40,7 +40,13 @@ impl Inbox {
|
|||||||
let weak_model = cx.global::<ChatRegistry>().inbox();
|
let weak_model = cx.global::<ChatRegistry>().inbox();
|
||||||
|
|
||||||
if let Some(model) = weak_model.upgrade() {
|
if let Some(model) = weak_model.upgrade() {
|
||||||
div().children(model.read(cx).iter().map(|model| {
|
div().map(|this| {
|
||||||
|
let inbox = model.read(cx);
|
||||||
|
|
||||||
|
if inbox.is_loading {
|
||||||
|
this.children(self.render_skeleton(5))
|
||||||
|
} else {
|
||||||
|
this.children(inbox.rooms.iter().map(|model| {
|
||||||
let room = model.read(cx);
|
let room = model.read(cx);
|
||||||
let id = room.id;
|
let id = room.id;
|
||||||
let room_id: SharedString = id.to_string().into();
|
let room_id: SharedString = id.to_string().into();
|
||||||
@@ -79,7 +85,9 @@ impl Inbox {
|
|||||||
this.flex()
|
this.flex()
|
||||||
.items_center()
|
.items_center()
|
||||||
.gap_2()
|
.gap_2()
|
||||||
.child(img("brand/avatar.png").size_6().rounded_full())
|
.child(
|
||||||
|
img("brand/avatar.png").size_6().rounded_full(),
|
||||||
|
)
|
||||||
.child(name)
|
.child(name)
|
||||||
} else {
|
} else {
|
||||||
this.flex()
|
this.flex()
|
||||||
@@ -111,6 +119,8 @@ impl Inbox {
|
|||||||
this.action(id, cx);
|
this.action(id, cx);
|
||||||
}))
|
}))
|
||||||
}))
|
}))
|
||||||
|
}
|
||||||
|
})
|
||||||
} else {
|
} else {
|
||||||
div().children(self.render_skeleton(5))
|
div().children(self.render_skeleton(5))
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user