feat: improve compose modal
This commit is contained in:
@@ -326,20 +326,26 @@ fn main() {
|
||||
while let Some(signal) = signal_rx.recv().await {
|
||||
match signal {
|
||||
Signal::Eose => {
|
||||
cx.update_window(*window.deref(), |_this, _window, cx| {
|
||||
cx.update_global::<ChatRegistry, _>(|this, cx| {
|
||||
this.load(cx);
|
||||
});
|
||||
})
|
||||
.unwrap();
|
||||
if let Err(e) =
|
||||
cx.update_window(*window.deref(), |_this, window, cx| {
|
||||
cx.update_global::<ChatRegistry, _>(|this, cx| {
|
||||
this.load(window, cx);
|
||||
});
|
||||
})
|
||||
{
|
||||
error!("Error: {}", e)
|
||||
}
|
||||
}
|
||||
Signal::Event(event) => {
|
||||
cx.update_window(*window.deref(), |_this, _window, cx| {
|
||||
cx.update_global::<ChatRegistry, _>(|this, cx| {
|
||||
this.new_room_message(event, cx);
|
||||
});
|
||||
})
|
||||
.unwrap();
|
||||
if let Err(e) =
|
||||
cx.update_window(*window.deref(), |_this, window, cx| {
|
||||
cx.update_global::<ChatRegistry, _>(|this, cx| {
|
||||
this.new_room_message(event, window, cx);
|
||||
});
|
||||
})
|
||||
{
|
||||
error!("Error: {}", e)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -191,7 +191,8 @@ impl AppView {
|
||||
PanelKind::Room(id) => {
|
||||
if let Some(weak_room) = cx.global::<ChatRegistry>().get_room(id, cx) {
|
||||
if let Some(room) = weak_room.upgrade() {
|
||||
let panel = Arc::new(chat::init(room, window, cx));
|
||||
let panel = Arc::new(chat::init(&room, window, cx));
|
||||
|
||||
self.dock.update(cx, |dock_area, cx| {
|
||||
dock_area.add_panel(panel, action.position, window, cx);
|
||||
});
|
||||
|
||||
@@ -10,7 +10,7 @@ use ui::{
|
||||
|
||||
#[derive(Clone, Debug, IntoElement)]
|
||||
pub struct Message {
|
||||
member: NostrProfile,
|
||||
profile: NostrProfile,
|
||||
content: SharedString,
|
||||
ago: SharedString,
|
||||
}
|
||||
@@ -18,7 +18,7 @@ pub struct Message {
|
||||
impl PartialEq for Message {
|
||||
fn eq(&self, other: &Self) -> bool {
|
||||
let content = self.content == other.content;
|
||||
let member = self.member == other.member;
|
||||
let member = self.profile == other.profile;
|
||||
let ago = self.ago == other.ago;
|
||||
|
||||
content && member && ago
|
||||
@@ -26,9 +26,9 @@ impl PartialEq for Message {
|
||||
}
|
||||
|
||||
impl Message {
|
||||
pub fn new(member: NostrProfile, content: SharedString, ago: SharedString) -> Self {
|
||||
pub fn new(profile: NostrProfile, content: SharedString, ago: SharedString) -> Self {
|
||||
Self {
|
||||
member,
|
||||
profile,
|
||||
content,
|
||||
ago,
|
||||
}
|
||||
@@ -58,7 +58,7 @@ impl RenderOnce for Message {
|
||||
}),
|
||||
)
|
||||
.child(
|
||||
img(self.member.avatar())
|
||||
img(self.profile.avatar())
|
||||
.size_8()
|
||||
.rounded_full()
|
||||
.flex_shrink_0(),
|
||||
@@ -75,7 +75,7 @@ impl RenderOnce for Message {
|
||||
.items_baseline()
|
||||
.gap_2()
|
||||
.text_xs()
|
||||
.child(div().font_semibold().child(self.member.name()))
|
||||
.child(div().font_semibold().child(self.profile.name()))
|
||||
.child(
|
||||
div()
|
||||
.child(self.ago)
|
||||
|
||||
@@ -2,13 +2,14 @@ use async_utility::task::spawn;
|
||||
use chat_state::room::Room;
|
||||
use common::{
|
||||
constants::IMAGE_SERVICE,
|
||||
profile::NostrProfile,
|
||||
utils::{compare, message_time, nip96_upload},
|
||||
};
|
||||
use gpui::{
|
||||
div, img, list, prelude::FluentBuilder, px, white, AnyElement, App, AppContext, Context,
|
||||
Entity, EventEmitter, Flatten, FocusHandle, Focusable, InteractiveElement, IntoElement,
|
||||
ListAlignment, ListState, ObjectFit, ParentElement, PathPromptOptions, Pixels, Render,
|
||||
SharedString, StatefulInteractiveElement, Styled, StyledImage, WeakEntity, Window,
|
||||
SharedString, StatefulInteractiveElement, Styled, StyledImage, Window,
|
||||
};
|
||||
use itertools::Itertools;
|
||||
use message::Message;
|
||||
@@ -27,7 +28,7 @@ use ui::{
|
||||
|
||||
mod message;
|
||||
|
||||
pub fn init(room: Entity<Room>, window: &mut Window, cx: &mut App) -> Entity<Chat> {
|
||||
pub fn init(room: &Entity<Room>, window: &mut Window, cx: &mut App) -> Entity<Chat> {
|
||||
Chat::new(room, window, cx)
|
||||
}
|
||||
|
||||
@@ -45,7 +46,8 @@ pub struct Chat {
|
||||
// Chat Room
|
||||
id: SharedString,
|
||||
name: SharedString,
|
||||
room: Entity<Room>,
|
||||
owner: NostrProfile,
|
||||
members: Vec<NostrProfile>,
|
||||
state: Entity<State>,
|
||||
list: ListState,
|
||||
// New Message
|
||||
@@ -56,12 +58,15 @@ pub struct Chat {
|
||||
}
|
||||
|
||||
impl Chat {
|
||||
pub fn new(model: Entity<Room>, window: &mut Window, cx: &mut App) -> Entity<Self> {
|
||||
pub fn new(model: &Entity<Room>, window: &mut Window, cx: &mut App) -> Entity<Self> {
|
||||
let room = model.read(cx);
|
||||
let id = room.id.to_string().into();
|
||||
let name = room.title.clone().unwrap_or("Untitled".into());
|
||||
let owner = room.owner.clone();
|
||||
let members = room.members.clone();
|
||||
|
||||
cx.new(|cx| {
|
||||
// Load all messages
|
||||
cx.observe_new::<Self>(|this, window, cx| {
|
||||
if let Some(window) = window {
|
||||
this.load_messages(window, cx);
|
||||
@@ -69,7 +74,13 @@ impl Chat {
|
||||
})
|
||||
.detach();
|
||||
|
||||
// Form
|
||||
// Observe and load new messages
|
||||
cx.observe_in(model, window, |this: &mut Chat, model, _, cx| {
|
||||
this.load_new_messages(&model, cx);
|
||||
})
|
||||
.detach();
|
||||
|
||||
// New message form
|
||||
let input = cx.new(|cx| {
|
||||
TextInput::new(window, cx)
|
||||
.appearance(false)
|
||||
@@ -77,60 +88,56 @@ impl Chat {
|
||||
.placeholder("Message...")
|
||||
});
|
||||
|
||||
// List
|
||||
let state = cx.new(|_| State {
|
||||
count: 0,
|
||||
items: vec![],
|
||||
});
|
||||
|
||||
// Send message when user presses enter
|
||||
cx.subscribe_in(
|
||||
&input,
|
||||
window,
|
||||
move |this: &mut Chat, view, input_event, window, cx| {
|
||||
move |this: &mut Chat, _, input_event, window, cx| {
|
||||
if let InputEvent::PressEnter = input_event {
|
||||
this.send_message(view.downgrade(), window, cx);
|
||||
this.send_message(window, cx);
|
||||
}
|
||||
},
|
||||
)
|
||||
.detach();
|
||||
|
||||
// List state model
|
||||
let state = cx.new(|_| State {
|
||||
count: 0,
|
||||
items: vec![],
|
||||
});
|
||||
|
||||
// Update list on every state changes
|
||||
cx.observe(&state, |this, model, cx| {
|
||||
let items = model.read(cx).items.clone();
|
||||
|
||||
this.list = ListState::new(
|
||||
items.len(),
|
||||
model.read(cx).items.len(),
|
||||
ListAlignment::Bottom,
|
||||
Pixels(256.),
|
||||
move |idx, _window, _cx| {
|
||||
let item = items.get(idx).unwrap().clone();
|
||||
div().child(item).into_any_element()
|
||||
Pixels(1024.),
|
||||
move |idx, _window, cx| {
|
||||
if let Some(message) = model.read(cx).items.get(idx) {
|
||||
div().child(message.clone()).into_any_element()
|
||||
} else {
|
||||
div().into_any_element()
|
||||
}
|
||||
},
|
||||
);
|
||||
|
||||
cx.notify();
|
||||
})
|
||||
.detach();
|
||||
|
||||
cx.observe_in(&model, window, |this, model, window, cx| {
|
||||
this.load_new_messages(model.downgrade(), window, cx);
|
||||
})
|
||||
.detach();
|
||||
|
||||
let attaches = cx.new(|_| None);
|
||||
|
||||
Self {
|
||||
closable: true,
|
||||
zoomable: true,
|
||||
focus_handle: cx.focus_handle(),
|
||||
room: model,
|
||||
list: ListState::new(0, ListAlignment::Bottom, Pixels(256.), move |_, _, _| {
|
||||
list: ListState::new(0, ListAlignment::Bottom, Pixels(1024.), move |_, _, _| {
|
||||
div().into_any_element()
|
||||
}),
|
||||
is_uploading: false,
|
||||
id,
|
||||
name,
|
||||
owner,
|
||||
members,
|
||||
input,
|
||||
state,
|
||||
attaches,
|
||||
@@ -138,150 +145,139 @@ impl Chat {
|
||||
})
|
||||
}
|
||||
|
||||
fn load_messages(&self, _window: &mut Window, cx: &mut Context<Self>) {
|
||||
let room = self.room.read(cx);
|
||||
let members = room.members.clone();
|
||||
let owner = room.owner.clone();
|
||||
// Get all public keys
|
||||
let all_keys = room.get_pubkeys();
|
||||
fn load_messages(&self, window: &mut Window, cx: &mut Context<Self>) {
|
||||
let window_handle = window.window_handle();
|
||||
// Get current user
|
||||
let author = self.owner.public_key();
|
||||
// Get other users in room
|
||||
let pubkeys = self
|
||||
.members
|
||||
.iter()
|
||||
.map(|m| m.public_key())
|
||||
.collect::<Vec<_>>();
|
||||
// Get all public keys for comparisation
|
||||
let mut all_keys = pubkeys.clone();
|
||||
all_keys.push(author);
|
||||
|
||||
// Async
|
||||
let async_state = self.state.clone();
|
||||
let mut async_cx = cx.to_async();
|
||||
cx.spawn(|this, mut cx| async move {
|
||||
let (tx, rx) = oneshot::channel::<Events>();
|
||||
|
||||
cx.foreground_executor()
|
||||
.spawn(async move {
|
||||
let events: anyhow::Result<Events, anyhow::Error> = async_cx
|
||||
.background_executor()
|
||||
.spawn({
|
||||
let client = get_client();
|
||||
let pubkeys = members.iter().map(|m| m.public_key()).collect::<Vec<_>>();
|
||||
cx.background_spawn({
|
||||
let client = get_client();
|
||||
|
||||
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.iter().copied());
|
||||
|
||||
let recv = Filter::new()
|
||||
.kind(Kind::PrivateDirectMessage)
|
||||
.author(author)
|
||||
.pubkeys(pubkeys.clone());
|
||||
let send = Filter::new()
|
||||
.kind(Kind::PrivateDirectMessage)
|
||||
.authors(pubkeys)
|
||||
.pubkey(author);
|
||||
|
||||
let send = Filter::new()
|
||||
.kind(Kind::PrivateDirectMessage)
|
||||
.authors(pubkeys)
|
||||
.pubkey(author);
|
||||
|
||||
// Get all DM events in database
|
||||
let recv_events = client.database().query(recv).await?;
|
||||
let send_events = client.database().query(send).await?;
|
||||
let events = recv_events.merge(send_events);
|
||||
|
||||
Ok(events)
|
||||
}
|
||||
})
|
||||
.await;
|
||||
|
||||
if let Ok(events) = events {
|
||||
let items: Vec<Message> = events
|
||||
.into_iter()
|
||||
.sorted_by_key(|ev| ev.created_at)
|
||||
.filter_map(|ev| {
|
||||
let mut pubkeys: Vec<_> = ev.tags.public_keys().copied().collect();
|
||||
pubkeys.push(ev.pubkey);
|
||||
|
||||
if compare(&pubkeys, &all_keys) {
|
||||
let member = if let Some(member) =
|
||||
members.iter().find(|&m| m.public_key() == ev.pubkey)
|
||||
{
|
||||
member.to_owned()
|
||||
} else {
|
||||
owner.clone()
|
||||
};
|
||||
|
||||
Some(Message::new(
|
||||
member,
|
||||
ev.content.into(),
|
||||
message_time(ev.created_at).into(),
|
||||
))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
})
|
||||
.collect();
|
||||
|
||||
let total = items.len();
|
||||
|
||||
_ = async_cx.update_entity(&async_state, |a, b| {
|
||||
a.items = items;
|
||||
a.count = total;
|
||||
b.notify();
|
||||
});
|
||||
// Get all DM events in database
|
||||
async move {
|
||||
let recv_events = client.database().query(recv).await.unwrap();
|
||||
let send_events = client.database().query(send).await.unwrap();
|
||||
let events = recv_events.merge(send_events);
|
||||
_ = tx.send(events);
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
|
||||
if let Ok(events) = rx.await {
|
||||
_ = cx.update_window(window_handle, |_, _, cx| {
|
||||
_ = this.update(cx, |this, cx| {
|
||||
let items: Vec<Message> = events
|
||||
.into_iter()
|
||||
.sorted_by_key(|ev| ev.created_at)
|
||||
.filter_map(|ev| {
|
||||
let mut pubkeys: Vec<_> = ev.tags.public_keys().copied().collect();
|
||||
pubkeys.push(ev.pubkey);
|
||||
|
||||
if compare(&pubkeys, &all_keys) {
|
||||
let member = if let Some(member) =
|
||||
this.members.iter().find(|&m| m.public_key() == ev.pubkey)
|
||||
{
|
||||
member.to_owned()
|
||||
} else {
|
||||
this.owner.clone()
|
||||
};
|
||||
|
||||
Some(Message::new(
|
||||
member,
|
||||
ev.content.into(),
|
||||
message_time(ev.created_at).into(),
|
||||
))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
})
|
||||
.collect();
|
||||
|
||||
cx.update_entity(&this.state, |this, cx| {
|
||||
this.count = items.len();
|
||||
this.items = items;
|
||||
cx.notify();
|
||||
});
|
||||
});
|
||||
});
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
}
|
||||
|
||||
fn load_new_messages(
|
||||
&self,
|
||||
model: WeakEntity<Room>,
|
||||
_window: &mut Window,
|
||||
cx: &mut Context<Self>,
|
||||
) {
|
||||
if let Some(model) = model.upgrade() {
|
||||
let room = model.read(cx);
|
||||
let items: Vec<Message> = room
|
||||
.new_messages
|
||||
.iter()
|
||||
.filter_map(|event| {
|
||||
room.member(&event.pubkey).map(|member| {
|
||||
Message::new(
|
||||
member,
|
||||
event.content.clone().into(),
|
||||
message_time(event.created_at).into(),
|
||||
)
|
||||
})
|
||||
fn load_new_messages(&self, model: &Entity<Room>, cx: &mut Context<Self>) {
|
||||
let room = model.read(cx);
|
||||
let items: Vec<Message> = room
|
||||
.new_messages
|
||||
.iter()
|
||||
.filter_map(|event| {
|
||||
room.member(&event.pubkey).map(|member| {
|
||||
Message::new(
|
||||
member,
|
||||
event.content.clone().into(),
|
||||
message_time(event.created_at).into(),
|
||||
)
|
||||
})
|
||||
})
|
||||
.collect();
|
||||
|
||||
cx.update_entity(&self.state, |this, cx| {
|
||||
let messages: Vec<Message> = items
|
||||
.into_iter()
|
||||
.filter_map(|new| {
|
||||
if !this.items.iter().any(|old| old == &new) {
|
||||
Some(new)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
})
|
||||
.collect();
|
||||
|
||||
cx.update_entity(&self.state, |model, cx| {
|
||||
let messages: Vec<Message> = items
|
||||
.into_iter()
|
||||
.filter_map(|new| {
|
||||
if !model.items.iter().any(|old| old == &new) {
|
||||
Some(new)
|
||||
} else {
|
||||
None
|
||||
}
|
||||
})
|
||||
.collect();
|
||||
|
||||
model.items.extend(messages);
|
||||
model.count = model.items.len();
|
||||
cx.notify();
|
||||
});
|
||||
}
|
||||
this.items.extend(messages);
|
||||
this.count = this.items.len();
|
||||
cx.notify();
|
||||
});
|
||||
}
|
||||
|
||||
fn send_message(
|
||||
&mut self,
|
||||
view: WeakEntity<TextInput>,
|
||||
window: &mut Window,
|
||||
cx: &mut Context<Self>,
|
||||
) {
|
||||
fn send_message(&mut self, window: &mut Window, cx: &mut Context<Self>) {
|
||||
let window_handle = window.window_handle();
|
||||
let room = self.room.read(cx);
|
||||
let owner = room.owner.clone();
|
||||
let mut members = room.members.to_vec();
|
||||
members.push(owner.clone());
|
||||
|
||||
// Get current user
|
||||
let author = self.owner.public_key();
|
||||
|
||||
// Get other users in room
|
||||
let mut pubkeys = self
|
||||
.members
|
||||
.iter()
|
||||
.map(|m| m.public_key())
|
||||
.collect::<Vec<_>>();
|
||||
pubkeys.push(author);
|
||||
|
||||
// Get message
|
||||
let mut content = self.input.read(cx).text().to_string();
|
||||
|
||||
if content.is_empty() {
|
||||
window.push_notification("Message cannot be empty", cx);
|
||||
return;
|
||||
}
|
||||
|
||||
// Get all attaches and merge with message
|
||||
if let Some(attaches) = self.attaches.read(cx).as_ref() {
|
||||
let merged = attaches
|
||||
@@ -293,75 +289,77 @@ impl Chat {
|
||||
content = format!("{}\n{}", content, merged)
|
||||
}
|
||||
|
||||
// Update input state
|
||||
if let Some(input) = view.upgrade() {
|
||||
cx.update_entity(&input, |input, cx| {
|
||||
input.set_loading(true, window, cx);
|
||||
input.set_disabled(true, window, cx);
|
||||
});
|
||||
if content.is_empty() {
|
||||
window.push_notification("Cannot send an empty message", cx);
|
||||
return;
|
||||
}
|
||||
|
||||
cx.spawn(|this, mut cx| async move {
|
||||
// Send message to all members
|
||||
cx.background_executor()
|
||||
.spawn({
|
||||
let client = get_client();
|
||||
let content = content.clone().to_string();
|
||||
let tags: Vec<Tag> = members
|
||||
.iter()
|
||||
.filter_map(|m| {
|
||||
if m.public_key() != owner.public_key() {
|
||||
Some(Tag::public_key(m.public_key()))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
})
|
||||
.collect();
|
||||
// Disable input when sending message
|
||||
self.input.update(cx, |this, cx| {
|
||||
this.set_loading(true, window, cx);
|
||||
this.set_disabled(true, window, cx);
|
||||
});
|
||||
|
||||
async move {
|
||||
// Send message to all members
|
||||
for member in members.iter() {
|
||||
_ = client
|
||||
.send_private_msg(member.public_key(), &content, tags.clone())
|
||||
.await
|
||||
cx.spawn(|this, mut cx| async move {
|
||||
cx.background_spawn({
|
||||
let client = get_client();
|
||||
let content = content.clone();
|
||||
let tags: Vec<Tag> = pubkeys
|
||||
.iter()
|
||||
.filter_map(|pubkey| {
|
||||
if pubkey != &author {
|
||||
Some(Tag::public_key(*pubkey))
|
||||
} else {
|
||||
None
|
||||
}
|
||||
})
|
||||
.collect();
|
||||
|
||||
async move {
|
||||
// Send message to all members
|
||||
for pubkey in pubkeys.iter() {
|
||||
if let Err(_e) = client
|
||||
.send_private_msg(*pubkey, &content, tags.clone())
|
||||
.await
|
||||
{
|
||||
// TODO: handle error
|
||||
}
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
|
||||
if let Some(view) = this.upgrade() {
|
||||
_ = cx.update_entity(&view, |this, cx| {
|
||||
cx.update_entity(&this.state, |model, cx| {
|
||||
let message = Message::new(
|
||||
owner,
|
||||
content.to_string().into(),
|
||||
message_time(Timestamp::now()).into(),
|
||||
);
|
||||
_ = cx.update_window(window_handle, |_, window, cx| {
|
||||
_ = this.update(cx, |this, cx| {
|
||||
let message = Message::new(
|
||||
this.owner.clone(),
|
||||
content.to_string().into(),
|
||||
message_time(Timestamp::now()).into(),
|
||||
);
|
||||
|
||||
model.items.extend(vec![message]);
|
||||
model.count = model.items.len();
|
||||
// Update message list
|
||||
cx.update_entity(&this.state, |this, cx| {
|
||||
this.items.extend(vec![message]);
|
||||
this.count = this.items.len();
|
||||
cx.notify();
|
||||
});
|
||||
cx.notify();
|
||||
});
|
||||
}
|
||||
|
||||
if let Some(input) = view.upgrade() {
|
||||
cx.update_window(window_handle, |_, window, cx| {
|
||||
cx.update_entity(&input, |input, cx| {
|
||||
input.set_loading(false, window, cx);
|
||||
input.set_disabled(false, window, cx);
|
||||
input.set_text("", window, cx);
|
||||
// Reset message input
|
||||
cx.update_entity(&this.input, |this, cx| {
|
||||
this.set_loading(false, window, cx);
|
||||
this.set_disabled(false, window, cx);
|
||||
this.set_text("", window, cx);
|
||||
cx.notify();
|
||||
});
|
||||
})
|
||||
.unwrap()
|
||||
}
|
||||
});
|
||||
});
|
||||
})
|
||||
.detach();
|
||||
}
|
||||
|
||||
fn upload(&mut self, _window: &mut Window, cx: &mut Context<Self>) {
|
||||
let attaches = self.attaches.clone();
|
||||
fn upload(&mut self, window: &mut Window, cx: &mut Context<Self>) {
|
||||
let window_handle = window.window_handle();
|
||||
|
||||
let paths = cx.prompt_for_paths(PathPromptOptions {
|
||||
files: true,
|
||||
directories: false,
|
||||
@@ -372,7 +370,7 @@ impl Chat {
|
||||
self.set_loading(true, cx);
|
||||
|
||||
// TODO: support multiple upload
|
||||
cx.spawn(move |this, mut async_cx| async move {
|
||||
cx.spawn(move |this, mut cx| async move {
|
||||
match Flatten::flatten(paths.await.map_err(|e| e.into())) {
|
||||
Ok(Some(mut paths)) => {
|
||||
let path = paths.pop().unwrap();
|
||||
@@ -382,33 +380,39 @@ impl Chat {
|
||||
|
||||
spawn(async move {
|
||||
let client = get_client();
|
||||
|
||||
if let Ok(url) = nip96_upload(client, file_data).await {
|
||||
_ = tx.send(url);
|
||||
}
|
||||
});
|
||||
|
||||
if let Ok(url) = rx.await {
|
||||
// Stop loading spinner
|
||||
if let Some(view) = this.upgrade() {
|
||||
_ = async_cx.update_entity(&view, |this, cx| {
|
||||
_ = cx.update_window(window_handle, |_, _, cx| {
|
||||
_ = this.update(cx, |this, cx| {
|
||||
// Stop loading spinner
|
||||
this.set_loading(false, cx);
|
||||
});
|
||||
}
|
||||
|
||||
// Update attaches model
|
||||
_ = async_cx.update_entity(&attaches, |model, cx| {
|
||||
if let Some(model) = model.as_mut() {
|
||||
model.push(url);
|
||||
} else {
|
||||
*model = Some(vec![url]);
|
||||
}
|
||||
cx.notify();
|
||||
this.attaches.update(cx, |this, cx| {
|
||||
if let Some(model) = this.as_mut() {
|
||||
model.push(url);
|
||||
} else {
|
||||
*this = Some(vec![url]);
|
||||
}
|
||||
cx.notify();
|
||||
});
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
}
|
||||
Ok(None) => {}
|
||||
Ok(None) => {
|
||||
// Stop loading spinner
|
||||
if let Some(view) = this.upgrade() {
|
||||
cx.update_entity(&view, |this, cx| {
|
||||
this.set_loading(false, cx);
|
||||
})
|
||||
.unwrap();
|
||||
}
|
||||
}
|
||||
Err(_) => {}
|
||||
}
|
||||
})
|
||||
@@ -436,15 +440,8 @@ impl Panel for Chat {
|
||||
self.id.clone()
|
||||
}
|
||||
|
||||
fn panel_facepile(&self, cx: &App) -> Option<Vec<String>> {
|
||||
Some(
|
||||
self.room
|
||||
.read(cx)
|
||||
.members
|
||||
.iter()
|
||||
.map(|member| member.avatar())
|
||||
.collect(),
|
||||
)
|
||||
fn panel_facepile(&self, _cx: &App) -> Option<Vec<String>> {
|
||||
Some(self.members.iter().map(|member| member.avatar()).collect())
|
||||
}
|
||||
|
||||
fn title(&self, _cx: &App) -> AnyElement {
|
||||
@@ -561,11 +558,7 @@ impl Render for Chat {
|
||||
.rounded(ButtonRounded::Medium)
|
||||
.label("SEND")
|
||||
.on_click(cx.listener(|this, _, window, cx| {
|
||||
this.send_message(
|
||||
this.input.downgrade(),
|
||||
window,
|
||||
cx,
|
||||
)
|
||||
this.send_message(window, cx)
|
||||
})),
|
||||
),
|
||||
),
|
||||
|
||||
@@ -1,14 +1,14 @@
|
||||
use std::str::FromStr;
|
||||
|
||||
use async_utility::task::spawn;
|
||||
use common::{constants::IMAGE_SERVICE, profile::NostrProfile, utils::nip96_upload};
|
||||
use gpui::{
|
||||
div, img, AnyElement, App, AppContext, Context, Entity, EventEmitter, Flatten, FocusHandle,
|
||||
Focusable, IntoElement, ParentElement, PathPromptOptions, Render, SharedString, Styled, Window,
|
||||
div, img, prelude::FluentBuilder, AnyElement, App, AppContext, Context, Entity, EventEmitter,
|
||||
Flatten, FocusHandle, Focusable, IntoElement, ParentElement, PathPromptOptions, Render,
|
||||
SharedString, Styled, Window,
|
||||
};
|
||||
use nostr_sdk::prelude::*;
|
||||
use smol::fs;
|
||||
use state::get_client;
|
||||
use std::str::FromStr;
|
||||
use tokio::sync::oneshot;
|
||||
use ui::{
|
||||
button::{Button, ButtonVariants},
|
||||
@@ -112,34 +112,39 @@ impl Profile {
|
||||
|
||||
spawn(async move {
|
||||
let client = get_client();
|
||||
|
||||
if let Ok(url) = nip96_upload(client, file_data).await {
|
||||
_ = tx.send(url);
|
||||
}
|
||||
});
|
||||
|
||||
if let Ok(url) = rx.await {
|
||||
// Stop loading spinner
|
||||
if let Some(view) = this.upgrade() {
|
||||
cx.update_entity(&view, |this, cx| {
|
||||
cx.update_window(window_handle, |_, window, cx| {
|
||||
// Stop loading spinner
|
||||
this.update(cx, |this, cx| {
|
||||
this.set_loading(false, cx);
|
||||
})
|
||||
.unwrap();
|
||||
}
|
||||
|
||||
// Update avatar input
|
||||
if let Some(input) = avatar_input.upgrade() {
|
||||
cx.update_window(window_handle, |_, window, cx| {
|
||||
cx.update_entity(&input, |this, cx| {
|
||||
// Set avatar input
|
||||
avatar_input
|
||||
.update(cx, |this, cx| {
|
||||
this.set_text(url.to_string(), window, cx);
|
||||
});
|
||||
})
|
||||
.unwrap();
|
||||
}
|
||||
})
|
||||
.unwrap();
|
||||
})
|
||||
.unwrap();
|
||||
}
|
||||
}
|
||||
}
|
||||
Ok(None) => {}
|
||||
Ok(None) => {
|
||||
// Stop loading spinner
|
||||
if let Some(view) = this.upgrade() {
|
||||
cx.update_entity(&view, |this, cx| {
|
||||
this.set_loading(false, cx);
|
||||
})
|
||||
.unwrap();
|
||||
}
|
||||
}
|
||||
Err(_) => {}
|
||||
}
|
||||
})
|
||||
@@ -189,18 +194,14 @@ impl Profile {
|
||||
.detach();
|
||||
|
||||
if rx.await.is_ok() {
|
||||
if let Some(profile) = this.upgrade() {
|
||||
cx.update_window(window_handle, |_, window, cx| {
|
||||
cx.update_entity(&profile, |this, cx| {
|
||||
this.set_submitting(false, cx);
|
||||
window.push_notification(
|
||||
"Your profile has been updated successfully",
|
||||
cx,
|
||||
);
|
||||
})
|
||||
cx.update_window(window_handle, |_, window, cx| {
|
||||
this.update(cx, |this, cx| {
|
||||
this.set_submitting(false, cx);
|
||||
window.push_notification("Your profile has been updated successfully", cx);
|
||||
})
|
||||
.unwrap();
|
||||
}
|
||||
.unwrap()
|
||||
})
|
||||
.unwrap();
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
@@ -263,16 +264,29 @@ impl Render for Profile {
|
||||
.gap_2()
|
||||
.w_full()
|
||||
.h_24()
|
||||
.child(
|
||||
img(format!(
|
||||
"{}/?url={}&w=100&h=100&fit=cover&mask=circle&n=-1",
|
||||
IMAGE_SERVICE,
|
||||
self.avatar_input.read(cx).text()
|
||||
))
|
||||
.size_10()
|
||||
.rounded_full()
|
||||
.flex_shrink_0(),
|
||||
)
|
||||
.map(|this| {
|
||||
let picture = self.avatar_input.read(cx).text();
|
||||
|
||||
if picture.is_empty() {
|
||||
this.child(
|
||||
img("brand/avatar.png")
|
||||
.size_10()
|
||||
.rounded_full()
|
||||
.flex_shrink_0(),
|
||||
)
|
||||
} else {
|
||||
this.child(
|
||||
img(format!(
|
||||
"{}/?url={}&w=100&h=100&fit=cover&mask=circle&n=-1",
|
||||
IMAGE_SERVICE,
|
||||
self.avatar_input.read(cx).text()
|
||||
))
|
||||
.size_10()
|
||||
.rounded_full()
|
||||
.flex_shrink_0(),
|
||||
)
|
||||
}
|
||||
})
|
||||
.child(
|
||||
div()
|
||||
.flex()
|
||||
|
||||
@@ -1,25 +1,25 @@
|
||||
use app_state::registry::AppRegistry;
|
||||
use chat_state::room::Room;
|
||||
use chat_state::registry::ChatRegistry;
|
||||
use common::{
|
||||
constants::FAKE_SIG,
|
||||
profile::NostrProfile,
|
||||
utils::{random_name, room_hash},
|
||||
utils::{random_name, signer_public_key},
|
||||
};
|
||||
use gpui::{
|
||||
div, img, impl_internal_actions, prelude::FluentBuilder, px, relative, uniform_list, App,
|
||||
AppContext, Context, Entity, FocusHandle, InteractiveElement, IntoElement, ParentElement,
|
||||
Render, SharedString, StatefulInteractiveElement, Styled, TextAlign, Window,
|
||||
AppContext, BorrowAppContext, Context, Entity, FocusHandle, InteractiveElement, IntoElement,
|
||||
ParentElement, Render, SharedString, StatefulInteractiveElement, Styled, TextAlign, Window,
|
||||
};
|
||||
use nostr_sdk::prelude::*;
|
||||
use serde::Deserialize;
|
||||
use state::get_client;
|
||||
use std::{collections::HashSet, time::Duration};
|
||||
use std::{collections::HashSet, str::FromStr, time::Duration};
|
||||
use tokio::sync::oneshot;
|
||||
use ui::{
|
||||
button::{Button, ButtonRounded},
|
||||
indicator::Indicator,
|
||||
input::{InputEvent, TextInput},
|
||||
theme::{scale::ColorScaleStep, ActiveTheme},
|
||||
Icon, IconName, Sizable, Size, StyledExt,
|
||||
ContextModal, Icon, IconName, Sizable, Size, StyledExt,
|
||||
};
|
||||
|
||||
#[derive(Clone, PartialEq, Eq, Deserialize)]
|
||||
@@ -31,15 +31,16 @@ pub struct Compose {
|
||||
title_input: Entity<TextInput>,
|
||||
message_input: Entity<TextInput>,
|
||||
user_input: Entity<TextInput>,
|
||||
contacts: Entity<Option<Vec<NostrProfile>>>,
|
||||
contacts: Entity<Vec<NostrProfile>>,
|
||||
selected: Entity<HashSet<PublicKey>>,
|
||||
focus_handle: FocusHandle,
|
||||
is_loading: bool,
|
||||
is_submitting: bool,
|
||||
}
|
||||
|
||||
impl Compose {
|
||||
pub fn new(window: &mut Window, cx: &mut Context<'_, Self>) -> Self {
|
||||
let contacts = cx.new(|_| None);
|
||||
let contacts = cx.new(|_| Vec::with_capacity(200));
|
||||
let selected = cx.new(|_| HashSet::new());
|
||||
|
||||
let user_input = cx.new(|cx| {
|
||||
@@ -64,9 +65,10 @@ impl Compose {
|
||||
TextInput::new(window, cx)
|
||||
.appearance(false)
|
||||
.text_size(Size::XSmall)
|
||||
.placeholder("Hello... (Optional)")
|
||||
.placeholder("Hello...")
|
||||
});
|
||||
|
||||
// Handle Enter event for message input
|
||||
cx.subscribe_in(
|
||||
&user_input,
|
||||
window,
|
||||
@@ -79,23 +81,22 @@ impl Compose {
|
||||
.detach();
|
||||
|
||||
cx.spawn(|this, mut cx| async move {
|
||||
let client = get_client();
|
||||
let (tx, rx) = oneshot::channel::<Vec<NostrProfile>>();
|
||||
|
||||
cx.background_executor()
|
||||
.spawn(async move {
|
||||
let signer = client.signer().await.unwrap();
|
||||
let public_key = signer.get_public_key().await.unwrap();
|
||||
let client = get_client();
|
||||
if let Ok(public_key) = signer_public_key(client).await {
|
||||
if let Ok(profiles) = client.database().contacts(public_key).await {
|
||||
let members: Vec<NostrProfile> = profiles
|
||||
.into_iter()
|
||||
.map(|profile| {
|
||||
NostrProfile::new(profile.public_key(), profile.metadata())
|
||||
})
|
||||
.collect();
|
||||
|
||||
if let Ok(profiles) = client.database().contacts(public_key).await {
|
||||
let members: Vec<NostrProfile> = profiles
|
||||
.into_iter()
|
||||
.map(|profile| {
|
||||
NostrProfile::new(profile.public_key(), profile.metadata())
|
||||
})
|
||||
.collect();
|
||||
|
||||
_ = tx.send(members);
|
||||
_ = tx.send(members);
|
||||
}
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
@@ -104,7 +105,7 @@ impl Compose {
|
||||
if let Some(view) = this.upgrade() {
|
||||
_ = cx.update_entity(&view, |this, cx| {
|
||||
this.contacts.update(cx, |this, cx| {
|
||||
*this = Some(contacts);
|
||||
this.extend(contacts);
|
||||
cx.notify();
|
||||
});
|
||||
cx.notify();
|
||||
@@ -121,43 +122,116 @@ impl Compose {
|
||||
contacts,
|
||||
selected,
|
||||
is_loading: false,
|
||||
is_submitting: false,
|
||||
focus_handle: cx.focus_handle(),
|
||||
}
|
||||
}
|
||||
|
||||
pub fn room(&self, _window: &Window, cx: &App) -> Option<Room> {
|
||||
if let Some(current_user) = cx.global::<AppRegistry>().user() {
|
||||
// Convert selected pubkeys into nostr tags
|
||||
let tags: Vec<Tag> = self
|
||||
.selected
|
||||
.read(cx)
|
||||
.iter()
|
||||
.map(|pk| Tag::public_key(*pk))
|
||||
.collect();
|
||||
let tags = Tags::new(tags);
|
||||
pub fn compose(&mut self, window: &mut Window, cx: &mut Context<Self>) {
|
||||
let selected = self.selected.read(cx).to_owned();
|
||||
let message = self.message_input.read(cx).text();
|
||||
|
||||
// Convert selected pubkeys into members
|
||||
let members: Vec<NostrProfile> = self
|
||||
.selected
|
||||
.read(cx)
|
||||
.clone()
|
||||
.into_iter()
|
||||
.map(|pk| NostrProfile::new(pk, Metadata::new()))
|
||||
.collect();
|
||||
|
||||
// Get room's id
|
||||
let id = room_hash(&tags);
|
||||
|
||||
// Get room's owner (current user)
|
||||
let owner = NostrProfile::new(current_user.public_key(), Metadata::new());
|
||||
|
||||
// Get room's title
|
||||
let title = self.title_input.read(cx).text().to_string().into();
|
||||
|
||||
Some(Room::new(id, owner, members, Some(title), Timestamp::now()))
|
||||
} else {
|
||||
None
|
||||
if selected.is_empty() {
|
||||
window.push_notification("You need to add at least 1 receiver", cx);
|
||||
return;
|
||||
}
|
||||
|
||||
if message.is_empty() {
|
||||
window.push_notification("Message is required", cx);
|
||||
return;
|
||||
}
|
||||
|
||||
let current_user = if let Some(profile) = cx.global::<AppRegistry>().user() {
|
||||
profile
|
||||
} else {
|
||||
return;
|
||||
};
|
||||
|
||||
// Show loading spinner
|
||||
self.set_submitting(true, cx);
|
||||
|
||||
// Get nostr client
|
||||
let client = get_client();
|
||||
|
||||
// Get message from user's input
|
||||
let content = message.to_string();
|
||||
|
||||
// Get room title from user's input
|
||||
let title = Tag::title(self.title_input.read(cx).text().to_string());
|
||||
|
||||
// Get all pubkeys
|
||||
let current_user = current_user.public_key();
|
||||
let mut pubkeys: Vec<PublicKey> = selected.iter().copied().collect();
|
||||
pubkeys.push(current_user);
|
||||
|
||||
// Convert selected pubkeys into Nostr tags
|
||||
let mut tag_list: Vec<Tag> = selected.iter().map(|pk| Tag::public_key(*pk)).collect();
|
||||
tag_list.push(title);
|
||||
|
||||
let tags = Tags::new(tag_list);
|
||||
let window_handle = window.window_handle();
|
||||
|
||||
cx.spawn(|this, mut cx| async move {
|
||||
let (tx, rx) = oneshot::channel::<Event>();
|
||||
|
||||
cx.background_spawn(async move {
|
||||
let mut event: Option<Event> = None;
|
||||
|
||||
for pubkey in pubkeys.iter() {
|
||||
if let Ok(output) = client
|
||||
.send_private_msg(*pubkey, &content, tags.clone())
|
||||
.await
|
||||
{
|
||||
if pubkey == ¤t_user && event.is_none() {
|
||||
if let Ok(Some(ev)) = client.database().event_by_id(&output.val).await {
|
||||
if let Ok(UnwrappedGift { mut rumor, .. }) =
|
||||
client.unwrap_gift_wrap(&ev).await
|
||||
{
|
||||
// Compute event id if not exist
|
||||
rumor.ensure_id();
|
||||
|
||||
if let Some(id) = rumor.id {
|
||||
let ev = Event::new(
|
||||
id,
|
||||
rumor.pubkey,
|
||||
rumor.created_at,
|
||||
rumor.kind,
|
||||
rumor.tags,
|
||||
rumor.content,
|
||||
Signature::from_str(FAKE_SIG).unwrap(),
|
||||
);
|
||||
|
||||
event = Some(ev);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if let Some(event) = event {
|
||||
_ = tx.send(event);
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
|
||||
if let Ok(event) = rx.await {
|
||||
_ = cx.update_window(window_handle, |_, window, cx| {
|
||||
cx.update_global::<ChatRegistry, _>(|this, cx| {
|
||||
this.new_room_message(event, window, cx);
|
||||
});
|
||||
|
||||
// Stop loading spinner
|
||||
_ = this.update(cx, |this, cx| {
|
||||
this.set_submitting(false, cx);
|
||||
});
|
||||
|
||||
// Close modal
|
||||
window.close_modal(cx);
|
||||
});
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
}
|
||||
|
||||
pub fn label(&self, _window: &Window, cx: &App) -> SharedString {
|
||||
@@ -168,35 +242,47 @@ impl Compose {
|
||||
}
|
||||
}
|
||||
|
||||
pub fn is_submitting(&self) -> bool {
|
||||
self.is_submitting
|
||||
}
|
||||
|
||||
fn add(&mut self, window: &mut Window, cx: &mut Context<Self>) {
|
||||
let window_handle = window.window_handle();
|
||||
let content = self.user_input.read(cx).text().to_string();
|
||||
let input = self.user_input.downgrade();
|
||||
|
||||
// Show loading spinner
|
||||
self.set_loading(true, cx);
|
||||
|
||||
if let Ok(public_key) = PublicKey::parse(&content) {
|
||||
cx.spawn(|this, mut async_cx| async move {
|
||||
let query: anyhow::Result<Metadata, anyhow::Error> = async_cx
|
||||
.background_executor()
|
||||
.spawn(async move {
|
||||
let client = get_client();
|
||||
let metadata = client
|
||||
.fetch_metadata(public_key, Duration::from_secs(3))
|
||||
.await?;
|
||||
if self
|
||||
.contacts
|
||||
.read(cx)
|
||||
.iter()
|
||||
.any(|c| c.public_key() == public_key)
|
||||
{
|
||||
self.set_loading(false, cx);
|
||||
return;
|
||||
};
|
||||
|
||||
Ok(metadata)
|
||||
})
|
||||
.await;
|
||||
cx.spawn(|this, mut cx| async move {
|
||||
let (tx, rx) = oneshot::channel::<Metadata>();
|
||||
|
||||
if let Ok(metadata) = query {
|
||||
if let Some(view) = this.upgrade() {
|
||||
_ = async_cx.update_entity(&view, |this, cx| {
|
||||
cx.background_spawn(async move {
|
||||
let client = get_client();
|
||||
let metadata = (client
|
||||
.fetch_metadata(public_key, Duration::from_secs(3))
|
||||
.await)
|
||||
.unwrap_or_default();
|
||||
|
||||
_ = tx.send(metadata);
|
||||
})
|
||||
.detach();
|
||||
|
||||
if let Ok(metadata) = rx.await {
|
||||
_ = cx.update_window(window_handle, |_, window, cx| {
|
||||
_ = this.update(cx, |this, cx| {
|
||||
this.contacts.update(cx, |this, cx| {
|
||||
if let Some(members) = this {
|
||||
members.insert(0, NostrProfile::new(public_key, metadata));
|
||||
}
|
||||
this.insert(0, NostrProfile::new(public_key, metadata));
|
||||
cx.notify();
|
||||
});
|
||||
|
||||
@@ -205,22 +291,22 @@ impl Compose {
|
||||
cx.notify();
|
||||
});
|
||||
|
||||
// Stop loading indicator
|
||||
this.set_loading(false, cx);
|
||||
});
|
||||
}
|
||||
|
||||
if let Some(input) = input.upgrade() {
|
||||
_ = async_cx.update_window(window_handle, |_, window, cx| {
|
||||
cx.update_entity(&input, |this, cx| {
|
||||
// Clear input
|
||||
this.user_input.update(cx, |this, cx| {
|
||||
this.set_text("", window, cx);
|
||||
})
|
||||
cx.notify();
|
||||
});
|
||||
});
|
||||
}
|
||||
});
|
||||
}
|
||||
})
|
||||
.detach();
|
||||
} else {
|
||||
// Handle error
|
||||
self.set_loading(false, cx);
|
||||
window.push_notification("Public Key is not valid", cx);
|
||||
}
|
||||
}
|
||||
|
||||
@@ -229,6 +315,11 @@ impl Compose {
|
||||
cx.notify();
|
||||
}
|
||||
|
||||
fn set_submitting(&mut self, status: bool, cx: &mut Context<Self>) {
|
||||
self.is_submitting = status;
|
||||
cx.notify();
|
||||
}
|
||||
|
||||
fn on_action_select(
|
||||
&mut self,
|
||||
action: &SelectContact,
|
||||
@@ -318,114 +409,103 @@ impl Render for Compose {
|
||||
.child(self.user_input.clone()),
|
||||
)
|
||||
.map(|this| {
|
||||
if let Some(contacts) = self.contacts.read(cx).clone() {
|
||||
let view = cx.entity();
|
||||
let total = contacts.len();
|
||||
let contacts = self.contacts.read(cx).clone();
|
||||
let view = cx.entity();
|
||||
|
||||
if total != 0 {
|
||||
this.child(
|
||||
div()
|
||||
.w_full()
|
||||
.h_24()
|
||||
.flex()
|
||||
.flex_col()
|
||||
.items_center()
|
||||
.justify_center()
|
||||
.text_align(TextAlign::Center)
|
||||
.child(
|
||||
div()
|
||||
.text_xs()
|
||||
.font_semibold()
|
||||
.line_height(relative(1.2))
|
||||
.child("No contacts"),
|
||||
)
|
||||
.child(
|
||||
div()
|
||||
.text_xs()
|
||||
.text_color(
|
||||
cx.theme()
|
||||
.base
|
||||
.step(cx, ColorScaleStep::ELEVEN),
|
||||
)
|
||||
.child("Your recently contacts will appear here."),
|
||||
),
|
||||
)
|
||||
} else {
|
||||
this.child(
|
||||
uniform_list(
|
||||
view,
|
||||
"contacts",
|
||||
total,
|
||||
move |this, range, _window, cx| {
|
||||
let selected = this.selected.read(cx);
|
||||
let mut items = Vec::new();
|
||||
|
||||
for ix in range {
|
||||
let item = contacts.get(ix).unwrap().clone();
|
||||
let is_select =
|
||||
selected.contains(&item.public_key());
|
||||
|
||||
items.push(
|
||||
div()
|
||||
.id(ix)
|
||||
.w_full()
|
||||
.h_9()
|
||||
.px_2()
|
||||
.flex()
|
||||
.items_center()
|
||||
.justify_between()
|
||||
.child(
|
||||
div()
|
||||
.flex()
|
||||
.items_center()
|
||||
.gap_2()
|
||||
.text_xs()
|
||||
.child(div().flex_shrink_0().child(
|
||||
img(item.avatar()).size_6(),
|
||||
))
|
||||
.child(item.name()),
|
||||
)
|
||||
.when(is_select, |this| {
|
||||
this.child(
|
||||
Icon::new(IconName::CircleCheck)
|
||||
.size_3()
|
||||
.text_color(
|
||||
cx.theme().base.step(
|
||||
cx,
|
||||
ColorScaleStep::TWELVE,
|
||||
),
|
||||
),
|
||||
)
|
||||
})
|
||||
.hover(|this| {
|
||||
this.bg(cx
|
||||
.theme()
|
||||
.base
|
||||
.step(cx, ColorScaleStep::THREE))
|
||||
})
|
||||
.on_click(move |_, window, cx| {
|
||||
window.dispatch_action(
|
||||
Box::new(SelectContact(
|
||||
item.public_key(),
|
||||
)),
|
||||
cx,
|
||||
);
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
items
|
||||
},
|
||||
if contacts.is_empty() {
|
||||
this.child(
|
||||
div()
|
||||
.w_full()
|
||||
.h_24()
|
||||
.flex()
|
||||
.flex_col()
|
||||
.items_center()
|
||||
.justify_center()
|
||||
.text_align(TextAlign::Center)
|
||||
.child(
|
||||
div()
|
||||
.text_xs()
|
||||
.font_semibold()
|
||||
.line_height(relative(1.2))
|
||||
.child("No contacts"),
|
||||
)
|
||||
.min_h(px(300.)),
|
||||
)
|
||||
}
|
||||
.child(
|
||||
div()
|
||||
.text_xs()
|
||||
.text_color(
|
||||
cx.theme().base.step(cx, ColorScaleStep::ELEVEN),
|
||||
)
|
||||
.child("Your recently contacts will appear here."),
|
||||
),
|
||||
)
|
||||
} else {
|
||||
this.flex()
|
||||
.items_center()
|
||||
.justify_center()
|
||||
.h_16()
|
||||
.child(Indicator::new().small())
|
||||
this.child(
|
||||
uniform_list(
|
||||
view,
|
||||
"contacts",
|
||||
contacts.len(),
|
||||
move |this, range, _window, cx| {
|
||||
let selected = this.selected.read(cx);
|
||||
let mut items = Vec::new();
|
||||
|
||||
for ix in range {
|
||||
let item = contacts.get(ix).unwrap().clone();
|
||||
let is_select = selected.contains(&item.public_key());
|
||||
|
||||
items.push(
|
||||
div()
|
||||
.id(ix)
|
||||
.w_full()
|
||||
.h_9()
|
||||
.px_2()
|
||||
.flex()
|
||||
.items_center()
|
||||
.justify_between()
|
||||
.child(
|
||||
div()
|
||||
.flex()
|
||||
.items_center()
|
||||
.gap_2()
|
||||
.text_xs()
|
||||
.child(
|
||||
div().flex_shrink_0().child(
|
||||
img(item.avatar()).size_6(),
|
||||
),
|
||||
)
|
||||
.child(item.name()),
|
||||
)
|
||||
.when(is_select, |this| {
|
||||
this.child(
|
||||
Icon::new(IconName::CircleCheck)
|
||||
.size_3()
|
||||
.text_color(cx.theme().base.step(
|
||||
cx,
|
||||
ColorScaleStep::TWELVE,
|
||||
)),
|
||||
)
|
||||
})
|
||||
.hover(|this| {
|
||||
this.bg(cx
|
||||
.theme()
|
||||
.base
|
||||
.step(cx, ColorScaleStep::THREE))
|
||||
})
|
||||
.on_click(move |_, window, cx| {
|
||||
window.dispatch_action(
|
||||
Box::new(SelectContact(
|
||||
item.public_key(),
|
||||
)),
|
||||
cx,
|
||||
);
|
||||
}),
|
||||
);
|
||||
}
|
||||
|
||||
items
|
||||
},
|
||||
)
|
||||
.min_h(px(250.)),
|
||||
)
|
||||
}
|
||||
}),
|
||||
)
|
||||
|
||||
@@ -89,7 +89,7 @@ impl Inbox {
|
||||
.text_xs()
|
||||
.rounded(px(cx.theme().radius))
|
||||
.hover(|this| this.bg(cx.theme().base.step(cx, ColorScaleStep::FOUR)))
|
||||
.child(div().font_medium().map(|this| {
|
||||
.child(div().flex_1().truncate().font_medium().map(|this| {
|
||||
if room.is_group {
|
||||
this.flex()
|
||||
.items_center()
|
||||
@@ -113,6 +113,7 @@ impl Inbox {
|
||||
}))
|
||||
.child(
|
||||
div()
|
||||
.flex_shrink_0()
|
||||
.text_color(cx.theme().base.step(cx, ColorScaleStep::ELEVEN))
|
||||
.child(ago),
|
||||
)
|
||||
|
||||
@@ -1,9 +1,8 @@
|
||||
use crate::views::sidebar::inbox::Inbox;
|
||||
use chat_state::registry::ChatRegistry;
|
||||
use compose::Compose;
|
||||
use gpui::{
|
||||
div, px, AnyElement, App, AppContext, BorrowAppContext, Context, Entity, EventEmitter,
|
||||
FocusHandle, Focusable, InteractiveElement, IntoElement, ParentElement, Render, SharedString,
|
||||
div, px, AnyElement, App, AppContext, Context, Entity, EventEmitter, FocusHandle, Focusable,
|
||||
InteractiveElement, IntoElement, ParentElement, Render, SharedString,
|
||||
StatefulInteractiveElement, Styled, Window,
|
||||
};
|
||||
use ui::{
|
||||
@@ -11,7 +10,7 @@ use ui::{
|
||||
dock_area::panel::{Panel, PanelEvent},
|
||||
popup_menu::PopupMenu,
|
||||
theme::{scale::ColorScaleStep, ActiveTheme},
|
||||
v_flex, ContextModal, Icon, IconName, Sizable, StyledExt,
|
||||
v_flex, ContextModal, Disableable, Icon, IconName, Sizable, StyledExt,
|
||||
};
|
||||
|
||||
mod compose;
|
||||
@@ -53,6 +52,7 @@ impl Sidebar {
|
||||
|
||||
window.open_modal(cx, move |modal, window, cx| {
|
||||
let label = compose.read(cx).label(window, cx);
|
||||
let is_submitting = compose.read(cx).is_submitting();
|
||||
|
||||
modal
|
||||
.title("Direct Messages")
|
||||
@@ -70,14 +70,10 @@ impl Sidebar {
|
||||
.bold()
|
||||
.rounded(ButtonRounded::Large)
|
||||
.w_full()
|
||||
.loading(is_submitting)
|
||||
.disabled(is_submitting)
|
||||
.on_click(window.listener_for(&compose, |this, _, window, cx| {
|
||||
if let Some(room) = this.room(window, cx) {
|
||||
cx.update_global::<ChatRegistry, _>(|this, cx| {
|
||||
this.new_room(room, cx);
|
||||
});
|
||||
|
||||
window.close_modal(cx);
|
||||
}
|
||||
this.compose(window, cx)
|
||||
})),
|
||||
),
|
||||
)
|
||||
|
||||
Reference in New Issue
Block a user