wip: refactor
This commit is contained in:
@@ -1,8 +1,8 @@
|
|||||||
use crate::get_client;
|
use crate::{get_client, utils::room_hash};
|
||||||
use crate::utils::get_room_id;
|
|
||||||
use gpui::{AppContext, Context, Global, Model, SharedString, WeakModel};
|
use gpui::{AppContext, Context, Global, Model, SharedString, WeakModel};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use nostr_sdk::prelude::*;
|
use nostr_sdk::prelude::*;
|
||||||
|
use profile::cut_public_key;
|
||||||
use rnglib::{Language, RNG};
|
use rnglib::{Language, RNG};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::{
|
use std::{
|
||||||
@@ -11,56 +11,79 @@ use std::{
|
|||||||
sync::{Arc, RwLock},
|
sync::{Arc, RwLock},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
#[derive(Clone, PartialEq, Eq, Deserialize)]
|
||||||
|
pub struct Member {
|
||||||
|
public_key: PublicKey,
|
||||||
|
metadata: Metadata,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Member {
|
||||||
|
pub fn new(public_key: PublicKey, metadata: Metadata) -> Self {
|
||||||
|
Self {
|
||||||
|
public_key,
|
||||||
|
metadata,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn public_key(&self) -> PublicKey {
|
||||||
|
self.public_key
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn metadata(&self) -> Metadata {
|
||||||
|
self.metadata.clone()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn name(&self) -> String {
|
||||||
|
if let Some(display_name) = &self.metadata.display_name {
|
||||||
|
if !display_name.is_empty() {
|
||||||
|
return display_name.clone();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if let Some(name) = &self.metadata.name {
|
||||||
|
if !name.is_empty() {
|
||||||
|
return name.clone();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
cut_public_key(self.public_key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Clone, PartialEq, Eq, Deserialize)]
|
#[derive(Clone, PartialEq, Eq, Deserialize)]
|
||||||
pub struct Room {
|
pub struct Room {
|
||||||
pub id: SharedString,
|
pub id: SharedString,
|
||||||
pub owner: PublicKey,
|
pub owner: PublicKey,
|
||||||
pub members: Vec<PublicKey>,
|
pub members: Vec<Member>,
|
||||||
pub last_seen: Timestamp,
|
pub last_seen: Timestamp,
|
||||||
pub title: Option<SharedString>,
|
pub title: Option<SharedString>,
|
||||||
pub metadata: Option<Metadata>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Room {
|
impl Room {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
id: SharedString,
|
id: SharedString,
|
||||||
owner: PublicKey,
|
owner: PublicKey,
|
||||||
members: Vec<PublicKey>,
|
|
||||||
last_seen: Timestamp,
|
last_seen: Timestamp,
|
||||||
title: Option<SharedString>,
|
title: Option<SharedString>,
|
||||||
metadata: Option<Metadata>,
|
members: Vec<Member>,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
|
let title = if title.is_none() {
|
||||||
|
let rng = RNG::from(&Language::Roman);
|
||||||
|
let name = rng.generate_names(2, true).join("-").to_lowercase();
|
||||||
|
|
||||||
|
Some(name.into())
|
||||||
|
} else {
|
||||||
|
title
|
||||||
|
};
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
id,
|
id,
|
||||||
title,
|
title,
|
||||||
members,
|
members,
|
||||||
last_seen,
|
last_seen,
|
||||||
owner,
|
owner,
|
||||||
metadata,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn parse(event: &Event, metadata: Option<Metadata>) -> Self {
|
|
||||||
let owner = event.pubkey;
|
|
||||||
let last_seen = event.created_at;
|
|
||||||
let id = SharedString::from(get_room_id(&owner, &event.tags));
|
|
||||||
|
|
||||||
// Get all members from event's tag
|
|
||||||
let mut members: Vec<PublicKey> = event.tags.public_keys().copied().collect();
|
|
||||||
members.push(owner);
|
|
||||||
|
|
||||||
// Get title from event's tag
|
|
||||||
let title = if let Some(tag) = event.tags.find(TagKind::Title) {
|
|
||||||
tag.content().map(|s| s.to_owned().into())
|
|
||||||
} else {
|
|
||||||
let rng = RNG::from(&Language::Roman);
|
|
||||||
let name = rng.generate_names(2, true).join("-").to_lowercase();
|
|
||||||
|
|
||||||
Some(name.into())
|
|
||||||
};
|
|
||||||
|
|
||||||
Self::new(id, owner, members, last_seen, title, metadata)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
@@ -76,32 +99,32 @@ impl Message {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type Inbox = Vec<Event>;
|
||||||
type Messages = RwLock<HashMap<SharedString, Arc<RwLock<Vec<Message>>>>>;
|
type Messages = RwLock<HashMap<SharedString, Arc<RwLock<Vec<Message>>>>>;
|
||||||
|
|
||||||
pub struct ChatRegistry {
|
pub struct ChatRegistry {
|
||||||
messages: Model<Messages>,
|
messages: Model<Messages>,
|
||||||
rooms: Model<Vec<Event>>,
|
inbox: Model<Inbox>,
|
||||||
}
|
}
|
||||||
|
|
||||||
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 rooms = cx.new_model(|_| Vec::new());
|
let inbox = cx.new_model(|_| Vec::new());
|
||||||
let messages = cx.new_model(|_| RwLock::new(HashMap::new()));
|
let messages = cx.new_model(|_| RwLock::new(HashMap::new()));
|
||||||
|
|
||||||
cx.set_global(Self { messages, rooms });
|
cx.set_global(Self { inbox, messages });
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn init(&mut self, cx: &mut AppContext) {
|
pub fn init(&mut self, cx: &mut AppContext) {
|
||||||
let async_cx = cx.to_async();
|
let async_cx = cx.to_async();
|
||||||
|
// Get all current room's hashes
|
||||||
// Get all current room's ids
|
let hashes: Vec<u64> = self
|
||||||
let ids: Vec<String> = self
|
.inbox
|
||||||
.rooms
|
|
||||||
.read(cx)
|
.read(cx)
|
||||||
.iter()
|
.iter()
|
||||||
.map(|ev| get_room_id(&ev.pubkey, &ev.tags))
|
.map(|ev| room_hash(&ev.tags))
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
cx.foreground_executor()
|
cx.foreground_executor()
|
||||||
@@ -115,18 +138,19 @@ impl ChatRegistry {
|
|||||||
|
|
||||||
let filter = Filter::new()
|
let filter = Filter::new()
|
||||||
.kind(Kind::PrivateDirectMessage)
|
.kind(Kind::PrivateDirectMessage)
|
||||||
.pubkey(public_key);
|
.author(public_key);
|
||||||
|
|
||||||
|
// Get all DM events from database
|
||||||
let events = client.database().query(vec![filter]).await?;
|
let events = client.database().query(vec![filter]).await?;
|
||||||
|
|
||||||
|
// Filter result
|
||||||
|
// 1. Only new rooms
|
||||||
|
// 2. Only unique rooms
|
||||||
|
// 3. Sorted by created_at
|
||||||
let result = events
|
let result = events
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.filter(|ev| ev.pubkey != public_key)
|
.filter(|ev| !hashes.iter().any(|h| h == &room_hash(&ev.tags)))
|
||||||
.filter(|ev| {
|
.unique_by(|ev| room_hash(&ev.tags))
|
||||||
let new_id = get_room_id(&ev.pubkey, &ev.tags);
|
|
||||||
// Get new events only
|
|
||||||
!ids.iter().any(|id| id == &new_id)
|
|
||||||
}) // Filter all messages from current user
|
|
||||||
.unique_by(|ev| ev.pubkey)
|
|
||||||
.sorted_by_key(|ev| Reverse(ev.created_at))
|
.sorted_by_key(|ev| Reverse(ev.created_at))
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
|
|
||||||
@@ -136,7 +160,7 @@ impl ChatRegistry {
|
|||||||
|
|
||||||
if let Ok(events) = query {
|
if let Ok(events) = query {
|
||||||
_ = async_cx.update_global::<Self, _>(|state, cx| {
|
_ = async_cx.update_global::<Self, _>(|state, cx| {
|
||||||
state.rooms.update(cx, |model, cx| {
|
state.inbox.update(cx, |model, cx| {
|
||||||
model.extend(events);
|
model.extend(events);
|
||||||
cx.notify();
|
cx.notify();
|
||||||
});
|
});
|
||||||
@@ -148,7 +172,7 @@ impl ChatRegistry {
|
|||||||
|
|
||||||
pub fn new_message(&mut self, event: Event, metadata: Option<Metadata>, cx: &mut AppContext) {
|
pub fn new_message(&mut self, event: Event, metadata: Option<Metadata>, cx: &mut AppContext) {
|
||||||
// Get room id
|
// Get room id
|
||||||
let room_id = SharedString::from(get_room_id(&event.pubkey, &event.tags));
|
let room_id = SharedString::from(room_hash(&event.tags).to_string());
|
||||||
// Create message
|
// Create message
|
||||||
let message = Message::new(event, metadata);
|
let message = Message::new(event, metadata);
|
||||||
|
|
||||||
@@ -169,7 +193,7 @@ impl ChatRegistry {
|
|||||||
self.messages.downgrade()
|
self.messages.downgrade()
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn rooms(&self) -> WeakModel<Vec<Event>> {
|
pub fn inbox(&self) -> WeakModel<Inbox> {
|
||||||
self.rooms.downgrade()
|
self.inbox.downgrade()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,23 +1,14 @@
|
|||||||
use chrono::{Duration, Local, TimeZone};
|
use chrono::{Duration, Local, TimeZone};
|
||||||
use nostr_sdk::prelude::*;
|
use nostr_sdk::prelude::*;
|
||||||
|
use std::hash::{DefaultHasher, Hash, Hasher};
|
||||||
|
|
||||||
pub fn get_room_id(author: &PublicKey, tags: &Tags) -> String {
|
pub fn room_hash(tags: &Tags) -> u64 {
|
||||||
// Get all public keys
|
let pubkeys: Vec<PublicKey> = tags.public_keys().copied().collect();
|
||||||
let mut pubkeys: Vec<PublicKey> = tags.public_keys().copied().collect();
|
let mut hasher = DefaultHasher::new();
|
||||||
// Add author to public keys list
|
// Generate unique hash
|
||||||
pubkeys.insert(0, *author);
|
pubkeys.hash(&mut hasher);
|
||||||
|
|
||||||
let hex: Vec<String> = pubkeys
|
hasher.finish()
|
||||||
.iter()
|
|
||||||
.map(|m| {
|
|
||||||
let hex = m.to_hex();
|
|
||||||
let split = &hex[..6];
|
|
||||||
|
|
||||||
split.to_owned()
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
|
|
||||||
hex.join("-")
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn show_npub(public_key: PublicKey, len: usize) -> String {
|
pub fn show_npub(public_key: PublicKey, len: usize) -> String {
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ pub struct RoomMessage {
|
|||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
author: PublicKey,
|
author: PublicKey,
|
||||||
fallback: SharedString,
|
fallback: SharedString,
|
||||||
metadata: Option<Metadata>,
|
metadata: Metadata,
|
||||||
content: SharedString,
|
content: SharedString,
|
||||||
created_at: SharedString,
|
created_at: SharedString,
|
||||||
}
|
}
|
||||||
@@ -23,7 +23,7 @@ pub struct RoomMessage {
|
|||||||
impl RoomMessage {
|
impl RoomMessage {
|
||||||
pub fn new(
|
pub fn new(
|
||||||
author: PublicKey,
|
author: PublicKey,
|
||||||
metadata: Option<Metadata>,
|
metadata: Metadata,
|
||||||
content: String,
|
content: String,
|
||||||
created_at: Timestamp,
|
created_at: Timestamp,
|
||||||
) -> Self {
|
) -> Self {
|
||||||
@@ -55,18 +55,14 @@ impl RenderOnce for RoomMessage {
|
|||||||
.text_color(cx.theme().muted_foreground)
|
.text_color(cx.theme().muted_foreground)
|
||||||
})
|
})
|
||||||
.child(div().flex_shrink_0().map(|this| {
|
.child(div().flex_shrink_0().map(|this| {
|
||||||
if let Some(metadata) = self.metadata.clone() {
|
if let Some(picture) = self.metadata.picture {
|
||||||
if let Some(picture) = metadata.picture {
|
this.child(
|
||||||
this.child(
|
img(format!(
|
||||||
img(format!(
|
"{}/?url={}&w=100&h=100&fit=cover&mask=circle&n=-1",
|
||||||
"{}/?url={}&w=100&h=100&fit=cover&mask=circle&n=-1",
|
IMAGE_SERVICE, picture
|
||||||
IMAGE_SERVICE, picture
|
))
|
||||||
))
|
.size_8(),
|
||||||
.size_8(),
|
)
|
||||||
)
|
|
||||||
} else {
|
|
||||||
this.child(img("brand/avatar.png").size_8().rounded_full())
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
this.child(img("brand/avatar.png").size_8().rounded_full())
|
this.child(img("brand/avatar.png").size_8().rounded_full())
|
||||||
}
|
}
|
||||||
@@ -84,12 +80,8 @@ impl RenderOnce for RoomMessage {
|
|||||||
.gap_2()
|
.gap_2()
|
||||||
.text_xs()
|
.text_xs()
|
||||||
.child(div().font_semibold().map(|this| {
|
.child(div().font_semibold().map(|this| {
|
||||||
if let Some(metadata) = self.metadata {
|
if let Some(display_name) = self.metadata.display_name {
|
||||||
if let Some(display_name) = metadata.display_name {
|
this.child(display_name)
|
||||||
this.child(display_name)
|
|
||||||
} else {
|
|
||||||
this.child(self.fallback)
|
|
||||||
}
|
|
||||||
} else {
|
} else {
|
||||||
this.child(self.fallback)
|
this.child(self.fallback)
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,20 +1,29 @@
|
|||||||
|
use crate::{get_client, states::chat::Room, utils::room_hash};
|
||||||
use gpui::{
|
use gpui::{
|
||||||
div, AnyElement, AppContext, EventEmitter, FocusHandle, FocusableView, IntoElement,
|
div, list, px, AnyElement, AppContext, Context, EventEmitter, Flatten, FocusHandle,
|
||||||
ParentElement, Render, SharedString, Styled, View, VisualContext, WindowContext,
|
FocusableView, IntoElement, ListAlignment, ListState, Model, ParentElement, PathPromptOptions,
|
||||||
|
Pixels, Render, SharedString, Styled, View, ViewContext, VisualContext, WindowContext,
|
||||||
};
|
};
|
||||||
|
use itertools::Itertools;
|
||||||
|
use message::RoomMessage;
|
||||||
use nostr_sdk::prelude::*;
|
use nostr_sdk::prelude::*;
|
||||||
use room::RoomPanel;
|
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use ui::{
|
use ui::{
|
||||||
button::Button,
|
button::{Button, ButtonVariants},
|
||||||
dock::{Panel, PanelEvent, PanelState},
|
dock::{Panel, PanelEvent, PanelState},
|
||||||
|
input::{InputEvent, TextInput},
|
||||||
popup_menu::PopupMenu,
|
popup_menu::PopupMenu,
|
||||||
|
theme::ActiveTheme,
|
||||||
|
v_flex, Icon, IconName,
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::states::chat::Room;
|
|
||||||
|
|
||||||
mod message;
|
mod message;
|
||||||
mod room;
|
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct State {
|
||||||
|
count: usize,
|
||||||
|
items: Vec<RoomMessage>,
|
||||||
|
}
|
||||||
|
|
||||||
pub struct ChatPanel {
|
pub struct ChatPanel {
|
||||||
// Panel
|
// Panel
|
||||||
@@ -22,38 +31,148 @@ pub struct ChatPanel {
|
|||||||
closeable: bool,
|
closeable: bool,
|
||||||
zoomable: bool,
|
zoomable: bool,
|
||||||
focus_handle: FocusHandle,
|
focus_handle: FocusHandle,
|
||||||
// Room
|
// Chat Room
|
||||||
id: SharedString,
|
id: SharedString,
|
||||||
room: View<RoomPanel>,
|
room: Arc<Room>,
|
||||||
metadata: Option<Metadata>,
|
input: View<TextInput>,
|
||||||
|
list: ListState,
|
||||||
|
state: Model<State>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ChatPanel {
|
impl ChatPanel {
|
||||||
pub fn new(room: &Arc<Room>, cx: &mut WindowContext) -> View<Self> {
|
pub fn new(room: &Arc<Room>, cx: &mut WindowContext) -> View<Self> {
|
||||||
|
let room = Arc::clone(room);
|
||||||
let id = room.id.clone();
|
let id = room.id.clone();
|
||||||
let title = room.title.clone();
|
let name = room.title.clone().unwrap_or("Untitled".into());
|
||||||
let metadata = room.metadata.clone();
|
|
||||||
|
|
||||||
let room = cx.new_view(|cx| {
|
cx.observe_new_views::<Self>(|this, cx| {
|
||||||
let view = RoomPanel::new(room, cx);
|
this.load_messages(cx);
|
||||||
// Load messages
|
})
|
||||||
view.load(cx);
|
.detach();
|
||||||
// Subscribe for new messages
|
|
||||||
view.subscribe(cx);
|
|
||||||
|
|
||||||
view
|
cx.new_view(|cx| {
|
||||||
});
|
// Form
|
||||||
|
let input = cx.new_view(|cx| {
|
||||||
|
TextInput::new(cx)
|
||||||
|
.appearance(false)
|
||||||
|
.text_size(ui::Size::Small)
|
||||||
|
.placeholder("Message...")
|
||||||
|
.cleanable()
|
||||||
|
});
|
||||||
|
|
||||||
cx.new_view(|cx| Self {
|
// Send message when user presses enter on form.
|
||||||
name: title.unwrap_or("Untitled".into()),
|
cx.subscribe(&input, move |this: &mut ChatPanel, _, input_event, cx| {
|
||||||
closeable: true,
|
if let InputEvent::PressEnter = input_event {
|
||||||
zoomable: true,
|
this.send_message(cx);
|
||||||
focus_handle: cx.focus_handle(),
|
}
|
||||||
id,
|
})
|
||||||
room,
|
.detach();
|
||||||
metadata,
|
|
||||||
|
let state = cx.new_model(|_| State {
|
||||||
|
count: 0,
|
||||||
|
items: vec![],
|
||||||
|
});
|
||||||
|
|
||||||
|
cx.observe(&state, |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 {
|
||||||
|
closeable: true,
|
||||||
|
zoomable: true,
|
||||||
|
focus_handle: cx.focus_handle(),
|
||||||
|
id,
|
||||||
|
name,
|
||||||
|
room,
|
||||||
|
input,
|
||||||
|
list,
|
||||||
|
state,
|
||||||
|
}
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn load_messages(&self, cx: &mut ViewContext<Self>) {
|
||||||
|
let members = self.room.members.clone();
|
||||||
|
let async_state = self.state.clone();
|
||||||
|
let id = self.room.id.to_string();
|
||||||
|
|
||||||
|
let client = get_client();
|
||||||
|
let mut async_cx = cx.to_async();
|
||||||
|
|
||||||
|
cx.foreground_executor()
|
||||||
|
.spawn(async move {
|
||||||
|
let events: anyhow::Result<Events, anyhow::Error> = async_cx
|
||||||
|
.background_executor()
|
||||||
|
.spawn({
|
||||||
|
let pubkeys = members.iter().map(|m| m.public_key()).collect::<Vec<_>>();
|
||||||
|
|
||||||
|
async move {
|
||||||
|
let signer = client.signer().await?;
|
||||||
|
let author = signer.get_public_key().await?;
|
||||||
|
|
||||||
|
let recv = Filter::new()
|
||||||
|
.kind(Kind::PrivateDirectMessage)
|
||||||
|
.author(author)
|
||||||
|
.pubkeys(pubkeys.clone());
|
||||||
|
|
||||||
|
let send = Filter::new()
|
||||||
|
.kind(Kind::PrivateDirectMessage)
|
||||||
|
.authors(pubkeys)
|
||||||
|
.pubkey(author);
|
||||||
|
|
||||||
|
// Get all DM events in database
|
||||||
|
let query = client.database().query(vec![recv, send]).await?;
|
||||||
|
|
||||||
|
Ok(query)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
if let Ok(events) = events {
|
||||||
|
let items: Vec<RoomMessage> = events
|
||||||
|
.into_iter()
|
||||||
|
.sorted_by_key(|ev| ev.created_at)
|
||||||
|
.map(|ev| {
|
||||||
|
let metadata = members
|
||||||
|
.iter()
|
||||||
|
.find(|&m| m.public_key() == ev.pubkey)
|
||||||
|
.unwrap()
|
||||||
|
.metadata();
|
||||||
|
|
||||||
|
RoomMessage::new(ev.pubkey, metadata, ev.content, ev.created_at)
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let total = items.len();
|
||||||
|
|
||||||
|
_ = async_cx.update_model(&async_state, |a, b| {
|
||||||
|
a.items = items;
|
||||||
|
a.count = total;
|
||||||
|
b.notify();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.detach();
|
||||||
|
}
|
||||||
|
|
||||||
|
fn send_message(&mut self, cx: &mut ViewContext<Self>) {}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Panel for ChatPanel {
|
impl Panel for ChatPanel {
|
||||||
@@ -62,7 +181,7 @@ impl Panel for ChatPanel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn panel_metadata(&self) -> Option<Metadata> {
|
fn panel_metadata(&self) -> Option<Metadata> {
|
||||||
self.metadata.clone()
|
None
|
||||||
}
|
}
|
||||||
|
|
||||||
fn title(&self, _cx: &WindowContext) -> AnyElement {
|
fn title(&self, _cx: &WindowContext) -> AnyElement {
|
||||||
@@ -99,7 +218,52 @@ impl FocusableView for ChatPanel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
impl Render for ChatPanel {
|
impl Render for ChatPanel {
|
||||||
fn render(&mut self, _cx: &mut gpui::ViewContext<Self>) -> impl IntoElement {
|
fn render(&mut self, cx: &mut ViewContext<Self>) -> impl IntoElement {
|
||||||
div().size_full().child(self.room.clone())
|
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()),
|
||||||
|
),
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ use ui::{
|
|||||||
use super::message::RoomMessage;
|
use super::message::RoomMessage;
|
||||||
use crate::{
|
use crate::{
|
||||||
get_client,
|
get_client,
|
||||||
states::chat::{ChatRegistry, Room},
|
states::chat::{ChatRegistry, Member, Room},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Clone)]
|
||||||
@@ -27,7 +27,7 @@ pub struct Messages {
|
|||||||
pub struct RoomPanel {
|
pub struct RoomPanel {
|
||||||
id: SharedString,
|
id: SharedString,
|
||||||
owner: PublicKey,
|
owner: PublicKey,
|
||||||
members: Arc<[PublicKey]>,
|
members: Vec<Member>,
|
||||||
// Form
|
// Form
|
||||||
input: View<TextInput>,
|
input: View<TextInput>,
|
||||||
// Messages
|
// Messages
|
||||||
@@ -38,7 +38,7 @@ pub struct RoomPanel {
|
|||||||
impl RoomPanel {
|
impl RoomPanel {
|
||||||
pub fn new(room: &Arc<Room>, cx: &mut ViewContext<'_, Self>) -> Self {
|
pub fn new(room: &Arc<Room>, cx: &mut ViewContext<'_, Self>) -> Self {
|
||||||
let id = room.id.clone();
|
let id = room.id.clone();
|
||||||
let members: Arc<[PublicKey]> = room.members.clone().into();
|
let members = room.members.clone();
|
||||||
let owner = room.owner;
|
let owner = room.owner;
|
||||||
|
|
||||||
// Form
|
// Form
|
||||||
@@ -98,20 +98,21 @@ impl RoomPanel {
|
|||||||
let async_messages = self.messages.clone();
|
let async_messages = self.messages.clone();
|
||||||
let mut async_cx = cx.to_async();
|
let mut async_cx = cx.to_async();
|
||||||
|
|
||||||
|
let public_keys: Vec<PublicKey> = self.members.iter().map(|m| m.public_key()).collect();
|
||||||
|
|
||||||
cx.foreground_executor()
|
cx.foreground_executor()
|
||||||
.spawn({
|
.spawn({
|
||||||
let client = get_client();
|
let client = get_client();
|
||||||
let owner = self.owner;
|
let owner = self.owner;
|
||||||
let members = self.members.to_vec();
|
|
||||||
|
|
||||||
let recv = Filter::new()
|
let recv = Filter::new()
|
||||||
.kind(Kind::PrivateDirectMessage)
|
.kind(Kind::PrivateDirectMessage)
|
||||||
.author(owner)
|
.author(owner)
|
||||||
.pubkeys(members.clone());
|
.pubkeys(public_keys.clone());
|
||||||
|
|
||||||
let send = Filter::new()
|
let send = Filter::new()
|
||||||
.kind(Kind::PrivateDirectMessage)
|
.kind(Kind::PrivateDirectMessage)
|
||||||
.authors(members)
|
.authors(public_keys)
|
||||||
.pubkey(owner);
|
.pubkey(owner);
|
||||||
|
|
||||||
async move {
|
async move {
|
||||||
@@ -200,8 +201,9 @@ impl RoomPanel {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn send_message(&mut self, cx: &mut ViewContext<Self>) {
|
fn send_message(&mut self, cx: &mut ViewContext<Self>) {
|
||||||
let members = self.members.clone();
|
let members: Vec<PublicKey> = self.members.iter().map(|m| m.public_key()).collect();
|
||||||
let members2 = members.clone();
|
let members2 = members.clone();
|
||||||
|
|
||||||
let content = self.input.read(cx).text().to_string();
|
let content = self.input.read(cx).text().to_string();
|
||||||
let content2 = content.clone();
|
let content2 = content.clone();
|
||||||
let content3 = content2.clone();
|
let content3 = content2.clone();
|
||||||
@@ -227,19 +229,21 @@ impl RoomPanel {
|
|||||||
async_cx
|
async_cx
|
||||||
.background_executor()
|
.background_executor()
|
||||||
.spawn(async move {
|
.spawn(async move {
|
||||||
for member in members.iter() {
|
let extra_tags: Vec<Tag> = members
|
||||||
let tags: Vec<Tag> = members
|
.iter()
|
||||||
.iter()
|
.filter_map(|m| {
|
||||||
.filter_map(|public_key| {
|
if m != ¤t_user {
|
||||||
if public_key != member {
|
Some(Tag::public_key(*m))
|
||||||
Some(Tag::public_key(*public_key))
|
} else {
|
||||||
} else {
|
None
|
||||||
None
|
}
|
||||||
}
|
})
|
||||||
})
|
.collect();
|
||||||
.collect();
|
|
||||||
|
|
||||||
_ = client.send_private_msg(*member, &content, tags).await;
|
for member in members.iter() {
|
||||||
|
_ = client
|
||||||
|
.send_private_msg(*member, &content, extra_tags.clone())
|
||||||
|
.await
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.detach();
|
.detach();
|
||||||
@@ -248,18 +252,20 @@ impl RoomPanel {
|
|||||||
async_cx
|
async_cx
|
||||||
.background_executor()
|
.background_executor()
|
||||||
.spawn(async move {
|
.spawn(async move {
|
||||||
let tags: Vec<Tag> = members2
|
let extra_tags: Vec<Tag> = members2
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|public_key| {
|
.filter_map(|m| {
|
||||||
if public_key != ¤t_user {
|
if m != ¤t_user {
|
||||||
Some(Tag::public_key(*public_key))
|
Some(Tag::public_key(*m))
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
_ = client.send_private_msg(current_user, content2, tags).await;
|
_ = client
|
||||||
|
.send_private_msg(current_user, content2, extra_tags)
|
||||||
|
.await;
|
||||||
})
|
})
|
||||||
.detach();
|
.detach();
|
||||||
|
|
||||||
|
|||||||
@@ -3,9 +3,9 @@ use crate::{
|
|||||||
get_client,
|
get_client,
|
||||||
states::{
|
states::{
|
||||||
app::AppRegistry,
|
app::AppRegistry,
|
||||||
chat::{ChatRegistry, Room},
|
chat::{ChatRegistry, Member, Room},
|
||||||
},
|
},
|
||||||
utils::{ago, get_room_id, show_npub},
|
utils::{ago, room_hash},
|
||||||
views::app::{AddPanel, PanelKind},
|
views::app::{AddPanel, PanelKind},
|
||||||
};
|
};
|
||||||
use gpui::prelude::FluentBuilder;
|
use gpui::prelude::FluentBuilder;
|
||||||
@@ -18,143 +18,6 @@ use nostr_sdk::prelude::*;
|
|||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
use ui::{skeleton::Skeleton, theme::ActiveTheme, v_flex, Collapsible, Icon, IconName, StyledExt};
|
use ui::{skeleton::Skeleton, theme::ActiveTheme, v_flex, Collapsible, Icon, IconName, StyledExt};
|
||||||
|
|
||||||
struct InboxListItem {
|
|
||||||
id: SharedString,
|
|
||||||
event: Event,
|
|
||||||
metadata: Model<Option<Metadata>>,
|
|
||||||
}
|
|
||||||
|
|
||||||
impl InboxListItem {
|
|
||||||
pub fn new(event: Event, metadata: Option<Metadata>, cx: &mut ViewContext<'_, Self>) -> Self {
|
|
||||||
let id = SharedString::from(get_room_id(&event.pubkey, &event.tags));
|
|
||||||
let metadata = cx.new_model(|_| metadata);
|
|
||||||
let refreshs = cx.global_mut::<AppRegistry>().refreshs();
|
|
||||||
|
|
||||||
if let Some(refreshs) = refreshs.upgrade() {
|
|
||||||
cx.observe(&refreshs, |this, _, cx| {
|
|
||||||
this.load_metadata(cx);
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}
|
|
||||||
|
|
||||||
Self {
|
|
||||||
id,
|
|
||||||
event,
|
|
||||||
metadata,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn load_metadata(&self, cx: &mut ViewContext<Self>) {
|
|
||||||
let mut async_cx = cx.to_async();
|
|
||||||
let async_metadata = self.metadata.clone();
|
|
||||||
|
|
||||||
cx.foreground_executor()
|
|
||||||
.spawn({
|
|
||||||
let client = get_client();
|
|
||||||
let public_key = self.event.pubkey;
|
|
||||||
|
|
||||||
async move {
|
|
||||||
let metadata = async_cx
|
|
||||||
.background_executor()
|
|
||||||
.spawn(async move { client.database().metadata(public_key).await })
|
|
||||||
.await;
|
|
||||||
|
|
||||||
if let Ok(metadata) = metadata {
|
|
||||||
_ = async_cx.update_model(&async_metadata, |model, cx| {
|
|
||||||
*model = metadata;
|
|
||||||
cx.notify();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn action(&self, cx: &mut WindowContext<'_>) {
|
|
||||||
let metadata = self.metadata.read(cx).clone();
|
|
||||||
let room = Arc::new(Room::parse(&self.event, metadata));
|
|
||||||
|
|
||||||
cx.dispatch_action(Box::new(AddPanel {
|
|
||||||
panel: PanelKind::Room(room),
|
|
||||||
position: ui::dock::DockPlacement::Center,
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
impl Render for InboxListItem {
|
|
||||||
fn render(&mut self, cx: &mut ViewContext<Self>) -> impl IntoElement {
|
|
||||||
let ago = ago(self.event.created_at.as_u64());
|
|
||||||
let fallback_name = show_npub(self.event.pubkey, 16);
|
|
||||||
|
|
||||||
let mut content = div()
|
|
||||||
.font_medium()
|
|
||||||
.text_color(cx.theme().sidebar_accent_foreground);
|
|
||||||
|
|
||||||
if let Some(metadata) = self.metadata.read(cx).as_ref() {
|
|
||||||
content = content
|
|
||||||
.flex()
|
|
||||||
.items_center()
|
|
||||||
.gap_2()
|
|
||||||
.map(|this| {
|
|
||||||
if let Some(picture) = metadata.picture.clone() {
|
|
||||||
this.flex_shrink_0().child(
|
|
||||||
img(format!(
|
|
||||||
"{}/?url={}&w=72&h=72&fit=cover&mask=circle&n=-1",
|
|
||||||
IMAGE_SERVICE, picture
|
|
||||||
))
|
|
||||||
.size_6(),
|
|
||||||
)
|
|
||||||
} else {
|
|
||||||
this.flex_shrink_0()
|
|
||||||
.child(img("brand/avatar.png").size_6().rounded_full())
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.map(|this| {
|
|
||||||
if let Some(display_name) = metadata.display_name.clone() {
|
|
||||||
this.child(display_name)
|
|
||||||
} else {
|
|
||||||
this.child(fallback_name)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
} else {
|
|
||||||
content = content
|
|
||||||
.flex()
|
|
||||||
.items_center()
|
|
||||||
.gap_2()
|
|
||||||
.child(
|
|
||||||
img("brand/avatar.png")
|
|
||||||
.flex_shrink_0()
|
|
||||||
.size_6()
|
|
||||||
.rounded_full(),
|
|
||||||
)
|
|
||||||
.child(fallback_name)
|
|
||||||
}
|
|
||||||
|
|
||||||
div()
|
|
||||||
.id(self.id.clone())
|
|
||||||
.h_8()
|
|
||||||
.px_1()
|
|
||||||
.flex()
|
|
||||||
.items_center()
|
|
||||||
.justify_between()
|
|
||||||
.text_xs()
|
|
||||||
.rounded_md()
|
|
||||||
.hover(|this| {
|
|
||||||
this.bg(cx.theme().sidebar_accent)
|
|
||||||
.text_color(cx.theme().sidebar_accent_foreground)
|
|
||||||
})
|
|
||||||
.child(content)
|
|
||||||
.child(
|
|
||||||
div()
|
|
||||||
.child(ago)
|
|
||||||
.text_color(cx.theme().sidebar_accent_foreground.opacity(0.7)),
|
|
||||||
)
|
|
||||||
.on_click(cx.listener(|this, _, cx| {
|
|
||||||
this.action(cx);
|
|
||||||
}))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub struct Inbox {
|
pub struct Inbox {
|
||||||
label: SharedString,
|
label: SharedString,
|
||||||
items: Model<Option<Vec<View<InboxListItem>>>>,
|
items: Model<Option<Vec<View<InboxListItem>>>>,
|
||||||
@@ -165,10 +28,10 @@ pub struct Inbox {
|
|||||||
impl Inbox {
|
impl Inbox {
|
||||||
pub fn new(cx: &mut ViewContext<'_, Self>) -> Self {
|
pub fn new(cx: &mut ViewContext<'_, Self>) -> Self {
|
||||||
let items = cx.new_model(|_| None);
|
let items = cx.new_model(|_| None);
|
||||||
let events = cx.global::<ChatRegistry>().rooms();
|
let inbox = cx.global::<ChatRegistry>().inbox();
|
||||||
|
|
||||||
if let Some(events) = events.upgrade() {
|
if let Some(inbox) = inbox.upgrade() {
|
||||||
cx.observe(&events, |this, model, cx| {
|
cx.observe(&inbox, |this, model, cx| {
|
||||||
this.load(model, cx);
|
this.load(model, cx);
|
||||||
})
|
})
|
||||||
.detach();
|
.detach();
|
||||||
@@ -184,41 +47,26 @@ impl Inbox {
|
|||||||
|
|
||||||
pub fn load(&mut self, model: Model<Vec<Event>>, cx: &mut ViewContext<Self>) {
|
pub fn load(&mut self, model: Model<Vec<Event>>, cx: &mut ViewContext<Self>) {
|
||||||
let events = model.read(cx).clone();
|
let events = model.read(cx).clone();
|
||||||
|
let views: Vec<View<InboxListItem>> = events
|
||||||
|
.into_iter()
|
||||||
|
.map(|event| {
|
||||||
|
cx.new_view(|cx| {
|
||||||
|
let view = InboxListItem::new(event, cx);
|
||||||
|
// Initial metadata
|
||||||
|
view.load_metadata(cx);
|
||||||
|
|
||||||
cx.spawn(|view, mut async_cx| async move {
|
view
|
||||||
let client = get_client();
|
})
|
||||||
let mut views = Vec::new();
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
for event in events.into_iter() {
|
self.items.update(cx, |model, cx| {
|
||||||
let metadata = async_cx
|
*model = Some(views);
|
||||||
.background_executor()
|
cx.notify();
|
||||||
.spawn(async move { client.database().metadata(event.pubkey).await })
|
});
|
||||||
.await;
|
|
||||||
|
|
||||||
let item = async_cx
|
self.is_loading = false;
|
||||||
.new_view(|cx| {
|
cx.notify();
|
||||||
if let Ok(metadata) = metadata {
|
|
||||||
InboxListItem::new(event, metadata, cx)
|
|
||||||
} else {
|
|
||||||
InboxListItem::new(event, None, cx)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
.unwrap();
|
|
||||||
|
|
||||||
views.push(item);
|
|
||||||
}
|
|
||||||
|
|
||||||
_ = view.update(&mut async_cx, |this, cx| {
|
|
||||||
this.items.update(cx, |model, cx| {
|
|
||||||
*model = Some(views);
|
|
||||||
cx.notify()
|
|
||||||
});
|
|
||||||
|
|
||||||
this.is_loading = false;
|
|
||||||
cx.notify();
|
|
||||||
});
|
|
||||||
})
|
|
||||||
.detach();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -285,3 +133,197 @@ impl Render for Inbox {
|
|||||||
.when(!self.is_collapsed, |this| this.child(content))
|
.when(!self.is_collapsed, |this| this.child(content))
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
struct InboxListItem {
|
||||||
|
id: SharedString,
|
||||||
|
created_at: Timestamp,
|
||||||
|
owner: PublicKey,
|
||||||
|
pubkeys: Vec<PublicKey>,
|
||||||
|
members: Model<Vec<Member>>,
|
||||||
|
is_group: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl InboxListItem {
|
||||||
|
pub fn new(event: Event, cx: &mut ViewContext<'_, Self>) -> Self {
|
||||||
|
let id = room_hash(&event.tags).to_string().into();
|
||||||
|
let created_at = event.created_at;
|
||||||
|
let owner = event.pubkey;
|
||||||
|
|
||||||
|
let pubkeys: Vec<PublicKey> = event.tags.public_keys().copied().collect();
|
||||||
|
let is_group = pubkeys.len() > 1;
|
||||||
|
|
||||||
|
let members = cx.new_model(|_| Vec::new());
|
||||||
|
let refreshs = cx.global_mut::<AppRegistry>().refreshs();
|
||||||
|
|
||||||
|
if let Some(refreshs) = refreshs.upgrade() {
|
||||||
|
cx.observe(&refreshs, |this, _, cx| {
|
||||||
|
this.load_metadata(cx);
|
||||||
|
})
|
||||||
|
.detach();
|
||||||
|
}
|
||||||
|
|
||||||
|
Self {
|
||||||
|
id,
|
||||||
|
created_at,
|
||||||
|
owner,
|
||||||
|
pubkeys,
|
||||||
|
members,
|
||||||
|
is_group,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn load_metadata(&self, cx: &mut ViewContext<Self>) {
|
||||||
|
let owner = self.owner;
|
||||||
|
let public_keys = self.pubkeys.clone();
|
||||||
|
let async_members = self.members.clone();
|
||||||
|
|
||||||
|
let mut async_cx = cx.to_async();
|
||||||
|
|
||||||
|
cx.foreground_executor()
|
||||||
|
.spawn({
|
||||||
|
let client = get_client();
|
||||||
|
|
||||||
|
async move {
|
||||||
|
let metadata: anyhow::Result<Vec<Member>, anyhow::Error> = async_cx
|
||||||
|
.background_executor()
|
||||||
|
.spawn(async move {
|
||||||
|
let mut result = Vec::new();
|
||||||
|
|
||||||
|
for public_key in public_keys.into_iter() {
|
||||||
|
let metadata = client.database().metadata(public_key).await?;
|
||||||
|
let profile = Member::new(public_key, metadata.unwrap_or_default());
|
||||||
|
|
||||||
|
result.push(profile);
|
||||||
|
}
|
||||||
|
|
||||||
|
let metadata = client.database().metadata(owner).await?;
|
||||||
|
let profile = Member::new(owner, metadata.unwrap_or_default());
|
||||||
|
|
||||||
|
result.push(profile);
|
||||||
|
|
||||||
|
Ok(result)
|
||||||
|
})
|
||||||
|
.await;
|
||||||
|
|
||||||
|
if let Ok(metadata) = metadata {
|
||||||
|
_ = async_cx.update_model(&async_members, |model, cx| {
|
||||||
|
*model = metadata;
|
||||||
|
cx.notify();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.detach();
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn action(&self, cx: &mut WindowContext<'_>) {
|
||||||
|
let members = self.members.read(cx).clone();
|
||||||
|
let room = Arc::new(Room::new(
|
||||||
|
self.id.clone(),
|
||||||
|
self.owner,
|
||||||
|
self.created_at,
|
||||||
|
None,
|
||||||
|
members,
|
||||||
|
));
|
||||||
|
|
||||||
|
cx.dispatch_action(Box::new(AddPanel {
|
||||||
|
panel: PanelKind::Room(room),
|
||||||
|
position: ui::dock::DockPlacement::Center,
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Render for InboxListItem {
|
||||||
|
fn render(&mut self, cx: &mut ViewContext<Self>) -> impl IntoElement {
|
||||||
|
let ago = ago(self.created_at.as_u64());
|
||||||
|
let members = self.members.read(cx);
|
||||||
|
|
||||||
|
let mut content = div()
|
||||||
|
.font_medium()
|
||||||
|
.text_color(cx.theme().sidebar_accent_foreground);
|
||||||
|
|
||||||
|
if self.is_group {
|
||||||
|
content = content
|
||||||
|
.flex()
|
||||||
|
.items_center()
|
||||||
|
.gap_2()
|
||||||
|
.child(img("brand/avatar.png").size_6().rounded_full())
|
||||||
|
.map(|this| {
|
||||||
|
let names: Vec<String> = members
|
||||||
|
.iter()
|
||||||
|
.filter_map(|m| {
|
||||||
|
if m.public_key() != self.owner {
|
||||||
|
Some(m.name())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
this.child(names.join(", "))
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
content = content.flex().items_center().gap_2().map(|this| {
|
||||||
|
if let Some(member) = members.first() {
|
||||||
|
let mut child = this;
|
||||||
|
|
||||||
|
// Avatar
|
||||||
|
if let Some(picture) = member.metadata().picture.clone() {
|
||||||
|
child = child.child(
|
||||||
|
img(format!(
|
||||||
|
"{}/?url={}&w=72&h=72&fit=cover&mask=circle&n=-1",
|
||||||
|
IMAGE_SERVICE, picture
|
||||||
|
))
|
||||||
|
.flex_shrink_0()
|
||||||
|
.size_6()
|
||||||
|
.rounded_full(),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
child = child.child(
|
||||||
|
img("brand/avatar.png")
|
||||||
|
.flex_shrink_0()
|
||||||
|
.size_6()
|
||||||
|
.rounded_full(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Display name
|
||||||
|
child = child.child(member.name());
|
||||||
|
|
||||||
|
child
|
||||||
|
} else {
|
||||||
|
this.child(
|
||||||
|
img("brand/avatar.png")
|
||||||
|
.flex_shrink_0()
|
||||||
|
.size_6()
|
||||||
|
.rounded_full(),
|
||||||
|
)
|
||||||
|
.child("Unknown")
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
div()
|
||||||
|
.id(self.id.clone())
|
||||||
|
.h_8()
|
||||||
|
.px_1()
|
||||||
|
.flex()
|
||||||
|
.items_center()
|
||||||
|
.justify_between()
|
||||||
|
.text_xs()
|
||||||
|
.rounded_md()
|
||||||
|
.hover(|this| {
|
||||||
|
this.bg(cx.theme().sidebar_accent)
|
||||||
|
.text_color(cx.theme().sidebar_accent_foreground)
|
||||||
|
})
|
||||||
|
.child(content)
|
||||||
|
.child(
|
||||||
|
div()
|
||||||
|
.child(ago)
|
||||||
|
.text_color(cx.theme().sidebar_accent_foreground.opacity(0.7)),
|
||||||
|
)
|
||||||
|
.on_click(cx.listener(|this, _, cx| {
|
||||||
|
this.action(cx);
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user