Compare commits
3 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| a1df66e176 | |||
|
|
78d913ae38 | ||
| b4691aa689 |
26
Cargo.lock
generated
26
Cargo.lock
generated
@@ -184,7 +184,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "assets"
|
name = "assets"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"gpui",
|
"gpui",
|
||||||
@@ -425,7 +425,7 @@ checksum = "1505bd5d3d116872e7271a6d4e16d81d0c8570876c8de68093a09ac269d8aac0"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "auto_update"
|
name = "auto_update"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"cargo-packager-updater",
|
"cargo-packager-updater",
|
||||||
@@ -1029,7 +1029,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "client_keys"
|
name = "client_keys"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"global",
|
"global",
|
||||||
@@ -1166,7 +1166,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "common"
|
name = "common"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
@@ -1242,7 +1242,7 @@ checksum = "6245d59a3e82a7fc217c5828a6692dbc6dfb63a0c8c90495621f7b9d79704a0e"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "coop"
|
name = "coop"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"assets",
|
"assets",
|
||||||
@@ -2371,7 +2371,7 @@ checksum = "0cc23270f6e1808e30a928bdc84dea0b9b4136a8bc82338574f23baf47bbd280"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "global"
|
name = "global"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"dirs 5.0.1",
|
"dirs 5.0.1",
|
||||||
@@ -2919,7 +2919,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "i18n"
|
name = "i18n"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"rust-i18n",
|
"rust-i18n",
|
||||||
]
|
]
|
||||||
@@ -5054,7 +5054,7 @@ checksum = "caf4aa5b0f434c91fe5c7f1ecb6a5ece2130b02ad2a590589dda5146df959001"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "registry"
|
name = "registry"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"common",
|
"common",
|
||||||
@@ -5843,7 +5843,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "settings"
|
name = "settings"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"global",
|
"global",
|
||||||
@@ -5910,7 +5910,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "signer_proxy"
|
name = "signer_proxy"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"atomic-destructor",
|
"atomic-destructor",
|
||||||
@@ -6452,7 +6452,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "theme"
|
name = "theme"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"gpui",
|
"gpui",
|
||||||
@@ -6612,7 +6612,7 @@ checksum = "1f3ccbac311fea05f86f61904b462b55fb3df8837a366dfc601a0161d0532f20"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "title_bar"
|
name = "title_bar"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"common",
|
"common",
|
||||||
@@ -6983,7 +6983,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "ui"
|
name = "ui"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"common",
|
"common",
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ members = ["crates/*"]
|
|||||||
default-members = ["crates/coop"]
|
default-members = ["crates/coop"]
|
||||||
|
|
||||||
[workspace.package]
|
[workspace.package]
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
publish = false
|
publish = false
|
||||||
|
|
||||||
|
|||||||
@@ -14,7 +14,7 @@ product-name = "Coop"
|
|||||||
description = "Chat Freely, Stay Private on Nostr"
|
description = "Chat Freely, Stay Private on Nostr"
|
||||||
identifier = "su.reya.coop"
|
identifier = "su.reya.coop"
|
||||||
category = "SocialNetworking"
|
category = "SocialNetworking"
|
||||||
version = "0.2.4"
|
version = "0.2.5"
|
||||||
out-dir = "../../dist"
|
out-dir = "../../dist"
|
||||||
before-packaging-command = "cargo build --release"
|
before-packaging-command = "cargo build --release"
|
||||||
resources = ["Cargo.toml", "src"]
|
resources = ["Cargo.toml", "src"]
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
use std::collections::{BTreeSet, HashMap};
|
use std::collections::{HashMap, HashSet};
|
||||||
|
|
||||||
use anyhow::anyhow;
|
use anyhow::anyhow;
|
||||||
use common::display::{ReadableProfile, ReadableTimestamp};
|
use common::display::{ReadableProfile, ReadableTimestamp};
|
||||||
@@ -7,16 +7,16 @@ use global::{nostr_client, sent_ids};
|
|||||||
use gpui::prelude::FluentBuilder;
|
use gpui::prelude::FluentBuilder;
|
||||||
use gpui::{
|
use gpui::{
|
||||||
div, img, list, px, red, relative, rems, svg, white, Action, AnyElement, App, AppContext,
|
div, img, list, px, red, relative, rems, svg, white, Action, AnyElement, App, AppContext,
|
||||||
ClipboardItem, Context, Div, Element, Entity, EventEmitter, Flatten, FocusHandle, Focusable,
|
ClipboardItem, Context, Element, Entity, EventEmitter, Flatten, FocusHandle, Focusable,
|
||||||
InteractiveElement, IntoElement, ListAlignment, ListState, MouseButton, ObjectFit,
|
InteractiveElement, IntoElement, ListAlignment, ListState, MouseButton, ObjectFit,
|
||||||
ParentElement, PathPromptOptions, Render, RetainAllImageCache, SharedString, Stateful,
|
ParentElement, PathPromptOptions, Render, RetainAllImageCache, SharedString,
|
||||||
StatefulInteractiveElement, Styled, StyledImage, Subscription, Task, Window,
|
StatefulInteractiveElement, Styled, StyledImage, Subscription, Task, Window,
|
||||||
};
|
};
|
||||||
use gpui_tokio::Tokio;
|
use gpui_tokio::Tokio;
|
||||||
use i18n::{shared_t, t};
|
use i18n::{shared_t, t};
|
||||||
use itertools::Itertools;
|
use itertools::Itertools;
|
||||||
use nostr_sdk::prelude::*;
|
use nostr_sdk::prelude::*;
|
||||||
use registry::message::RenderedMessage;
|
use registry::message::{Message, RenderedMessage};
|
||||||
use registry::room::{Room, RoomKind, RoomSignal, SendReport};
|
use registry::room::{Room, RoomKind, RoomSignal, SendReport};
|
||||||
use registry::Registry;
|
use registry::Registry;
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
@@ -48,23 +48,25 @@ pub fn init(room: Entity<Room>, window: &mut Window, cx: &mut App) -> Entity<Cha
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct Chat {
|
pub struct Chat {
|
||||||
// Panel
|
|
||||||
id: SharedString,
|
|
||||||
focus_handle: FocusHandle,
|
|
||||||
// Chat Room
|
// Chat Room
|
||||||
room: Entity<Room>,
|
room: Entity<Room>,
|
||||||
list_state: ListState,
|
list_state: ListState,
|
||||||
messages: BTreeSet<RenderedMessage>,
|
messages: Vec<Message>,
|
||||||
rendered_texts_by_id: HashMap<EventId, RenderedText>,
|
rendered_texts_by_id: HashMap<EventId, RenderedText>,
|
||||||
reports_by_id: HashMap<EventId, Vec<SendReport>>,
|
reports_by_id: HashMap<EventId, Vec<SendReport>>,
|
||||||
|
|
||||||
// New Message
|
// New Message
|
||||||
input: Entity<InputState>,
|
input: Entity<InputState>,
|
||||||
replies_to: Entity<Vec<EventId>>,
|
replies_to: Entity<Vec<EventId>>,
|
||||||
sending: bool,
|
sending: bool,
|
||||||
|
|
||||||
// Media Attachment
|
// Media Attachment
|
||||||
attachments: Entity<Vec<Url>>,
|
attachments: Entity<Vec<Url>>,
|
||||||
uploading: bool,
|
uploading: bool,
|
||||||
// System
|
|
||||||
|
// Panel
|
||||||
|
id: SharedString,
|
||||||
|
focus_handle: FocusHandle,
|
||||||
image_cache: Entity<RetainAllImageCache>,
|
image_cache: Entity<RetainAllImageCache>,
|
||||||
|
|
||||||
_subscriptions: SmallVec<[Subscription; 2]>,
|
_subscriptions: SmallVec<[Subscription; 2]>,
|
||||||
@@ -73,10 +75,7 @@ pub struct Chat {
|
|||||||
|
|
||||||
impl Chat {
|
impl Chat {
|
||||||
pub fn new(room: Entity<Room>, window: &mut Window, cx: &mut Context<Self>) -> Self {
|
pub fn new(room: Entity<Room>, window: &mut Window, cx: &mut Context<Self>) -> Self {
|
||||||
let attachments = cx.new(|_| vec![]);
|
|
||||||
let replies_to = cx.new(|_| vec![]);
|
|
||||||
let list_state = ListState::new(1, ListAlignment::Bottom, px(1024.));
|
let list_state = ListState::new(1, ListAlignment::Bottom, px(1024.));
|
||||||
|
|
||||||
let input = cx.new(|cx| {
|
let input = cx.new(|cx| {
|
||||||
InputState::new(window, cx)
|
InputState::new(window, cx)
|
||||||
.placeholder(t!("chat.placeholder"))
|
.placeholder(t!("chat.placeholder"))
|
||||||
@@ -88,6 +87,8 @@ impl Chat {
|
|||||||
.clean_on_escape()
|
.clean_on_escape()
|
||||||
});
|
});
|
||||||
|
|
||||||
|
let attachments = cx.new(|_| vec![]);
|
||||||
|
let replies_to = cx.new(|_| vec![]);
|
||||||
let load_messages = room.read(cx).load_messages(cx);
|
let load_messages = room.read(cx).load_messages(cx);
|
||||||
|
|
||||||
let mut subscriptions = smallvec![];
|
let mut subscriptions = smallvec![];
|
||||||
@@ -154,7 +155,7 @@ impl Chat {
|
|||||||
focus_handle: cx.focus_handle(),
|
focus_handle: cx.focus_handle(),
|
||||||
uploading: false,
|
uploading: false,
|
||||||
sending: false,
|
sending: false,
|
||||||
messages: BTreeSet::new(),
|
messages: vec![Message::System],
|
||||||
rendered_texts_by_id: HashMap::new(),
|
rendered_texts_by_id: HashMap::new(),
|
||||||
reports_by_id: HashMap::new(),
|
reports_by_id: HashMap::new(),
|
||||||
room,
|
room,
|
||||||
@@ -329,9 +330,17 @@ impl Chat {
|
|||||||
|
|
||||||
/// Get a message by its ID
|
/// Get a message by its ID
|
||||||
fn message(&self, id: &EventId) -> Option<&RenderedMessage> {
|
fn message(&self, id: &EventId) -> Option<&RenderedMessage> {
|
||||||
self.messages.iter().find(|m| m.id == *id)
|
self.messages.iter().find_map(|msg| {
|
||||||
|
if let Message::User(rendered) = msg {
|
||||||
|
if &rendered.id == id {
|
||||||
|
return Some(rendered);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
None
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Convert and insert a nostr event into the chat panel
|
||||||
fn insert_message<E>(&mut self, event: E, cx: &mut Context<Self>)
|
fn insert_message<E>(&mut self, event: E, cx: &mut Context<Self>)
|
||||||
where
|
where
|
||||||
E: Into<RenderedMessage>,
|
E: Into<RenderedMessage>,
|
||||||
@@ -340,7 +349,7 @@ impl Chat {
|
|||||||
let new_len = 1;
|
let new_len = 1;
|
||||||
|
|
||||||
// Extend the messages list with the new events
|
// Extend the messages list with the new events
|
||||||
self.messages.insert(event.into());
|
self.messages.push(Message::user(event));
|
||||||
|
|
||||||
// Update list state with the new messages
|
// Update list state with the new messages
|
||||||
self.list_state.splice(old_len..old_len, new_len);
|
self.list_state.splice(old_len..old_len, new_len);
|
||||||
@@ -348,17 +357,42 @@ impl Chat {
|
|||||||
cx.notify();
|
cx.notify();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Convert and insert bulk nostr events into the chat panel
|
||||||
fn insert_messages<E>(&mut self, events: E, cx: &mut Context<Self>)
|
fn insert_messages<E>(&mut self, events: E, cx: &mut Context<Self>)
|
||||||
where
|
where
|
||||||
E: IntoIterator,
|
E: IntoIterator,
|
||||||
E::Item: Into<RenderedMessage>,
|
E::Item: Into<RenderedMessage>,
|
||||||
{
|
{
|
||||||
|
let old_events: HashSet<EventId> = self
|
||||||
|
.messages
|
||||||
|
.iter()
|
||||||
|
.filter_map(|msg| {
|
||||||
|
if let Message::User(rendered) = msg {
|
||||||
|
Some(rendered.id)
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let events: Vec<Message> = events
|
||||||
|
.into_iter()
|
||||||
|
.map(|ev| ev.into())
|
||||||
|
.filter(|msg: &RenderedMessage| !old_events.contains(&msg.id))
|
||||||
|
.map(Message::User)
|
||||||
|
.collect();
|
||||||
|
|
||||||
let old_len = self.messages.len();
|
let old_len = self.messages.len();
|
||||||
let events: Vec<RenderedMessage> = events.into_iter().map(Into::into).collect();
|
|
||||||
let new_len = events.len();
|
let new_len = events.len();
|
||||||
|
|
||||||
// Extend the messages list with the new events
|
// Extend the messages list with the new events
|
||||||
self.messages.extend(events);
|
self.messages.extend(events);
|
||||||
|
self.messages.sort_by(|a, b| match (a, b) {
|
||||||
|
(Message::System, Message::System) => std::cmp::Ordering::Equal,
|
||||||
|
(Message::System, Message::User(_)) => std::cmp::Ordering::Less,
|
||||||
|
(Message::User(_), Message::System) => std::cmp::Ordering::Greater,
|
||||||
|
(Message::User(a_msg), Message::User(b_msg)) => a_msg.created_at.cmp(&b_msg.created_at),
|
||||||
|
});
|
||||||
|
|
||||||
// Update list state with the new messages
|
// Update list state with the new messages
|
||||||
self.list_state.splice(old_len..old_len, new_len);
|
self.list_state.splice(old_len..old_len, new_len);
|
||||||
@@ -372,7 +406,13 @@ impl Chat {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn scroll_to(&self, id: EventId) {
|
fn scroll_to(&self, id: EventId) {
|
||||||
if let Some(ix) = self.messages.iter().position(|m| m.id == id) {
|
if let Some(ix) = self.messages.iter().position(|m| {
|
||||||
|
if let Message::User(msg) = m {
|
||||||
|
msg.id == id
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}) {
|
||||||
self.list_state.scroll_to_reveal_item(ix);
|
self.list_state.scroll_to_reveal_item(ix);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -495,7 +535,7 @@ impl Chat {
|
|||||||
cx.notify();
|
cx.notify();
|
||||||
}
|
}
|
||||||
|
|
||||||
fn render_announcement(&mut self, ix: usize, cx: &mut Context<Self>) -> Stateful<Div> {
|
fn render_announcement(&mut self, ix: usize, cx: &mut Context<Self>) -> AnyElement {
|
||||||
v_flex()
|
v_flex()
|
||||||
.id(ix)
|
.id(ix)
|
||||||
.group("")
|
.group("")
|
||||||
@@ -518,18 +558,30 @@ impl Chat {
|
|||||||
.text_color(cx.theme().elevated_surface_background),
|
.text_color(cx.theme().elevated_surface_background),
|
||||||
)
|
)
|
||||||
.child(shared_t!("chat.notice"))
|
.child(shared_t!("chat.notice"))
|
||||||
|
.into_any_element()
|
||||||
|
}
|
||||||
|
|
||||||
|
fn render_message_not_found(&self, cx: &Context<Self>) -> AnyElement {
|
||||||
|
div()
|
||||||
|
.w_full()
|
||||||
|
.py_1()
|
||||||
|
.px_3()
|
||||||
|
.child(
|
||||||
|
div()
|
||||||
|
.text_xs()
|
||||||
|
.text_color(cx.theme().danger_foreground)
|
||||||
|
.child(shared_t!("chat.not_found")),
|
||||||
|
)
|
||||||
|
.into_any_element()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn render_message(
|
fn render_message(
|
||||||
&mut self,
|
&self,
|
||||||
ix: usize,
|
ix: usize,
|
||||||
window: &mut Window,
|
message: &RenderedMessage,
|
||||||
cx: &mut Context<Self>,
|
text: AnyElement,
|
||||||
) -> Stateful<Div> {
|
cx: &Context<Self>,
|
||||||
let Some(message) = self.messages.iter().nth(ix) else {
|
) -> AnyElement {
|
||||||
return div().id(ix);
|
|
||||||
};
|
|
||||||
|
|
||||||
let proxy = AppSettings::get_proxy_user_avatars(cx);
|
let proxy = AppSettings::get_proxy_user_avatars(cx);
|
||||||
let hide_avatar = AppSettings::get_hide_user_avatars(cx);
|
let hide_avatar = AppSettings::get_hide_user_avatars(cx);
|
||||||
|
|
||||||
@@ -545,13 +597,6 @@ impl Chat {
|
|||||||
// Check if message is sent successfully
|
// Check if message is sent successfully
|
||||||
let is_sent_success = self.is_sent_success(&id);
|
let is_sent_success = self.is_sent_success(&id);
|
||||||
|
|
||||||
// Get or insert rendered text
|
|
||||||
let rendered_text = self
|
|
||||||
.rendered_texts_by_id
|
|
||||||
.entry(id)
|
|
||||||
.or_insert_with(|| RenderedText::new(&message.content, cx))
|
|
||||||
.element(ix.into(), window, cx);
|
|
||||||
|
|
||||||
div()
|
div()
|
||||||
.id(ix)
|
.id(ix)
|
||||||
.group("")
|
.group("")
|
||||||
@@ -595,7 +640,7 @@ impl Chat {
|
|||||||
.when(has_replies, |this| {
|
.when(has_replies, |this| {
|
||||||
this.children(self.render_message_replies(replies, cx))
|
this.children(self.render_message_replies(replies, cx))
|
||||||
})
|
})
|
||||||
.child(rendered_text)
|
.child(text)
|
||||||
.when(is_sent_failed, |this| {
|
.when(is_sent_failed, |this| {
|
||||||
this.child(self.render_message_reports(&id, cx))
|
this.child(self.render_message_reports(&id, cx))
|
||||||
}),
|
}),
|
||||||
@@ -615,6 +660,7 @@ impl Chat {
|
|||||||
}
|
}
|
||||||
}))
|
}))
|
||||||
.hover(|this| this.bg(cx.theme().surface_background))
|
.hover(|this| this.bg(cx.theme().surface_background))
|
||||||
|
.into_any_element()
|
||||||
}
|
}
|
||||||
|
|
||||||
fn render_message_replies(
|
fn render_message_replies(
|
||||||
@@ -1126,11 +1172,22 @@ impl Render for Chat {
|
|||||||
.child(
|
.child(
|
||||||
list(
|
list(
|
||||||
self.list_state.clone(),
|
self.list_state.clone(),
|
||||||
cx.processor(move |this, ix, window, cx| {
|
cx.processor(move |this, ix: usize, window, cx| {
|
||||||
if ix == 0 {
|
if let Some(message) = this.messages.get(ix) {
|
||||||
this.render_announcement(ix, cx).into_any_element()
|
match message {
|
||||||
|
Message::User(rendered) => {
|
||||||
|
let text = this
|
||||||
|
.rendered_texts_by_id
|
||||||
|
.entry(rendered.id)
|
||||||
|
.or_insert_with(|| RenderedText::new(&rendered.content, cx))
|
||||||
|
.element(ix.into(), window, cx);
|
||||||
|
|
||||||
|
this.render_message(ix, rendered, text, cx)
|
||||||
|
}
|
||||||
|
Message::System => this.render_announcement(ix, cx),
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
this.render_message(ix, window, cx).into_any_element()
|
this.render_message_not_found(cx)
|
||||||
}
|
}
|
||||||
}),
|
}),
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -15,7 +15,6 @@ use ui::actions::OpenProfile;
|
|||||||
use ui::avatar::Avatar;
|
use ui::avatar::Avatar;
|
||||||
use ui::context_menu::ContextMenuExt;
|
use ui::context_menu::ContextMenuExt;
|
||||||
use ui::modal::ModalButtonProps;
|
use ui::modal::ModalButtonProps;
|
||||||
use ui::skeleton::Skeleton;
|
|
||||||
use ui::{h_flex, ContextModal, StyledExt};
|
use ui::{h_flex, ContextModal, StyledExt};
|
||||||
|
|
||||||
use crate::views::screening;
|
use crate::views::screening;
|
||||||
@@ -109,21 +108,7 @@ impl RenderOnce for RoomListItem {
|
|||||||
self.handler,
|
self.handler,
|
||||||
)
|
)
|
||||||
else {
|
else {
|
||||||
return h_flex()
|
return div().id(self.ix);
|
||||||
.id(self.ix)
|
|
||||||
.h_9()
|
|
||||||
.w_full()
|
|
||||||
.px_1p5()
|
|
||||||
.gap_2()
|
|
||||||
.child(Skeleton::new().flex_shrink_0().size_6().rounded_full())
|
|
||||||
.child(
|
|
||||||
div()
|
|
||||||
.flex_1()
|
|
||||||
.flex()
|
|
||||||
.justify_between()
|
|
||||||
.child(Skeleton::new().w_32().h_2p5().rounded_sm())
|
|
||||||
.child(Skeleton::new().w_6().h_2p5().rounded_sm()),
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
h_flex()
|
h_flex()
|
||||||
|
|||||||
@@ -33,7 +33,6 @@ mod list_item;
|
|||||||
|
|
||||||
const FIND_DELAY: u64 = 600;
|
const FIND_DELAY: u64 = 600;
|
||||||
const FIND_LIMIT: usize = 10;
|
const FIND_LIMIT: usize = 10;
|
||||||
const TOTAL_SKELETONS: usize = 3;
|
|
||||||
|
|
||||||
pub fn init(window: &mut Window, cx: &mut App) -> Entity<Sidebar> {
|
pub fn init(window: &mut Window, cx: &mut App) -> Entity<Sidebar> {
|
||||||
Sidebar::new(window, cx)
|
Sidebar::new(window, cx)
|
||||||
@@ -595,7 +594,6 @@ impl Focusable for Sidebar {
|
|||||||
impl Render for Sidebar {
|
impl Render for Sidebar {
|
||||||
fn render(&mut self, _window: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
fn render(&mut self, _window: &mut Window, cx: &mut Context<Self>) -> impl IntoElement {
|
||||||
let registry = Registry::read_global(cx);
|
let registry = Registry::read_global(cx);
|
||||||
let loading = registry.loading;
|
|
||||||
|
|
||||||
// Get rooms from either search results or the chat registry
|
// Get rooms from either search results or the chat registry
|
||||||
let rooms = if let Some(results) = self.local_result.read(cx).as_ref() {
|
let rooms = if let Some(results) = self.local_result.read(cx).as_ref() {
|
||||||
@@ -611,15 +609,6 @@ impl Render for Sidebar {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Get total rooms count
|
|
||||||
let mut total_rooms = rooms.len();
|
|
||||||
|
|
||||||
// If loading in progress
|
|
||||||
// Add 3 skeletons to the room list
|
|
||||||
if loading {
|
|
||||||
total_rooms += TOTAL_SKELETONS;
|
|
||||||
}
|
|
||||||
|
|
||||||
v_flex()
|
v_flex()
|
||||||
.image_cache(self.image_cache.clone())
|
.image_cache(self.image_cache.clone())
|
||||||
.size_full()
|
.size_full()
|
||||||
@@ -707,7 +696,7 @@ impl Render for Sidebar {
|
|||||||
.child(
|
.child(
|
||||||
uniform_list(
|
uniform_list(
|
||||||
"rooms",
|
"rooms",
|
||||||
total_rooms,
|
rooms.len(),
|
||||||
cx.processor(move |this, range, _window, cx| {
|
cx.processor(move |this, range, _window, cx| {
|
||||||
this.list_items(&rooms, range, cx)
|
this.list_items(&rooms, range, cx)
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -2,6 +2,18 @@ use std::hash::Hash;
|
|||||||
|
|
||||||
use nostr_sdk::prelude::*;
|
use nostr_sdk::prelude::*;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Hash, PartialEq, Eq, PartialOrd, Ord)]
|
||||||
|
pub enum Message {
|
||||||
|
User(RenderedMessage),
|
||||||
|
System,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Message {
|
||||||
|
pub fn user(user: impl Into<RenderedMessage>) -> Self {
|
||||||
|
Self::User(user.into())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct RenderedMessage {
|
pub struct RenderedMessage {
|
||||||
pub id: EventId,
|
pub id: EventId,
|
||||||
|
|||||||
@@ -325,6 +325,8 @@ chat:
|
|||||||
en: "This conversation is private. Only members can see each other's messages."
|
en: "This conversation is private. Only members can see each other's messages."
|
||||||
placeholder:
|
placeholder:
|
||||||
en: "Message..."
|
en: "Message..."
|
||||||
|
not_found:
|
||||||
|
en: "Something is wrong. Coop cannot display this message"
|
||||||
empty_message_error:
|
empty_message_error:
|
||||||
en: "Cannot send an empty message"
|
en: "Cannot send an empty message"
|
||||||
copy_message_button:
|
copy_message_button:
|
||||||
|
|||||||
Reference in New Issue
Block a user