Compare commits
4 Commits
950571be81
...
5824275f06
Author | SHA1 | Date | |
---|---|---|---|
|
5824275f06 | ||
|
690e7113ab | ||
|
f09a1cff2d | ||
|
7c85e6e006 |
|
@ -15,4 +15,6 @@ hsluv = "0.3.1"
|
||||||
sha1 = "0.10.6"
|
sha1 = "0.10.6"
|
||||||
tokio = { version = "1", features = [ "rt" ] }
|
tokio = { version = "1", features = [ "rt" ] }
|
||||||
xmpp = { git = "https://gitlab.com/xmpp-rs/xmpp-rs" }
|
xmpp = { git = "https://gitlab.com/xmpp-rs/xmpp-rs" }
|
||||||
|
tokio-xmpp = { git = "https://gitlab.com/xmpp-rs/xmpp-rs", features = ["syntax-highlighting"] }
|
||||||
async-channel = "2.3.1"
|
async-channel = "2.3.1"
|
||||||
|
env_logger = { version = "0.11.3", default-features = false, features = ["color", "auto-color", "humantime"] }
|
||||||
|
|
48
src/main.rs
48
src/main.rs
|
@ -40,27 +40,25 @@ fn xep_0392(input: &str) -> String {
|
||||||
format!("#{:02x}{:02x}{:02x}", r, g, b)
|
format!("#{:02x}{:02x}{:02x}", r, g, b)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn main() {
|
fn on_login_pressed(win: &window::MainWindow) {
|
||||||
let mut args = std::env::args();
|
let jid = win.jid().text();
|
||||||
let _ = args.next().unwrap();
|
let password = win.password().text();
|
||||||
let username = args.next().expect("Please give username argument 1");
|
println!("jid={jid:?} password={password:?}");
|
||||||
let password = args.next().expect("Please give password argument 2");
|
|
||||||
|
|
||||||
let app = adw::Application::builder()
|
if !jid.is_empty() && !password.is_empty() {
|
||||||
.application_id("fr.linkmauve.XmppClient")
|
win.stack().set_visible_child(win.spinner());
|
||||||
.flags(gio::ApplicationFlags::HANDLES_COMMAND_LINE)
|
|
||||||
.build();
|
|
||||||
|
|
||||||
let tabs_store = gio::ListStore::new::<Tab>();
|
let (xmpp_receiver, cmd_sender) = xmpp_client::client(&jid, &password);
|
||||||
|
let win2 = win.clone();
|
||||||
let (xmpp_receiver, cmd_sender) = xmpp_client::client(&username, &password);
|
|
||||||
let tabs_store_copy = tabs_store.clone();
|
|
||||||
glib::spawn_future_local(async move {
|
glib::spawn_future_local(async move {
|
||||||
while let Ok(event) = xmpp_receiver.recv().await {
|
while let Ok(event) = xmpp_receiver.recv().await {
|
||||||
match event {
|
match event {
|
||||||
|
xmpp::Event::Online => {
|
||||||
|
win2.stack().set_visible_child(win2.split_view());
|
||||||
|
}
|
||||||
xmpp::Event::ContactAdded(jid) => {
|
xmpp::Event::ContactAdded(jid) => {
|
||||||
let tab = Tab::new(jid.jid.as_str(), jid.jid.as_str());
|
let tab = Tab::new(jid.jid.as_str(), jid.jid.as_str());
|
||||||
tabs_store_copy.append(&tab);
|
win2.tabs_store().append(&tab);
|
||||||
}
|
}
|
||||||
xmpp::Event::ChatMessage(_id, _from, _body, _time) => {
|
xmpp::Event::ChatMessage(_id, _from, _body, _time) => {
|
||||||
// TODO: Insert message into tab history
|
// TODO: Insert message into tab history
|
||||||
|
@ -70,8 +68,17 @@ fn main() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
app.connect_startup(move |app| {
|
fn main() {
|
||||||
|
env_logger::init();
|
||||||
|
|
||||||
|
let app = adw::Application::builder()
|
||||||
|
.application_id("fr.linkmauve.XmppClient")
|
||||||
|
.build();
|
||||||
|
|
||||||
|
app.connect_activate(move |app| {
|
||||||
let win = window::MainWindow::new(app);
|
let win = window::MainWindow::new(app);
|
||||||
|
|
||||||
let action_close = gio::ActionEntry::builder("close")
|
let action_close = gio::ActionEntry::builder("close")
|
||||||
|
@ -82,6 +89,11 @@ fn main() {
|
||||||
win.add_action_entries([action_close]);
|
win.add_action_entries([action_close]);
|
||||||
app.set_accels_for_action("win.close", &["<Ctrl>Q"]);
|
app.set_accels_for_action("win.close", &["<Ctrl>Q"]);
|
||||||
|
|
||||||
|
let win2 = win.clone();
|
||||||
|
win.login().connect_clicked(move |_| {
|
||||||
|
on_login_pressed(&win2);
|
||||||
|
});
|
||||||
|
|
||||||
assert!(Tab::static_type().is_valid());
|
assert!(Tab::static_type().is_valid());
|
||||||
assert!(Message::static_type().is_valid());
|
assert!(Message::static_type().is_valid());
|
||||||
|
|
||||||
|
@ -134,7 +146,7 @@ fn main() {
|
||||||
});
|
});
|
||||||
|
|
||||||
win.tabs().set_factory(Some(&tabs_factory));
|
win.tabs().set_factory(Some(&tabs_factory));
|
||||||
win.tabs_selection().set_model(Some(&tabs_store));
|
win.tabs_selection().set_model(Some(win.tabs_store()));
|
||||||
|
|
||||||
let win2 = win.clone();
|
let win2 = win.clone();
|
||||||
win.tabs_selection()
|
win.tabs_selection()
|
||||||
|
@ -155,7 +167,7 @@ fn main() {
|
||||||
});
|
});
|
||||||
|
|
||||||
let win2 = win.clone();
|
let win2 = win.clone();
|
||||||
let cmd_sender2 = cmd_sender.clone();
|
//let cmd_sender2 = cmd_sender.clone();
|
||||||
win.entry().connect_activate(move |entry| {
|
win.entry().connect_activate(move |entry| {
|
||||||
let text = entry.text();
|
let text = entry.text();
|
||||||
entry.set_text("");
|
entry.set_text("");
|
||||||
|
@ -165,12 +177,14 @@ fn main() {
|
||||||
.selected_item()
|
.selected_item()
|
||||||
.and_downcast()
|
.and_downcast()
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
/*
|
||||||
cmd_sender2
|
cmd_sender2
|
||||||
.send_blocking(xmpp_client::XMPPCommand::SendPM(
|
.send_blocking(xmpp_client::XMPPCommand::SendPM(
|
||||||
xmpp::BareJid::new(¤t_tab.jid()).unwrap(),
|
xmpp::BareJid::new(¤t_tab.jid()).unwrap(),
|
||||||
text.as_str().to_string(),
|
text.as_str().to_string(),
|
||||||
))
|
))
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
*/
|
||||||
let message = Message::now(&get_own_nick(), &text);
|
let message = Message::now(&get_own_nick(), &text);
|
||||||
let selection = win2.selection();
|
let selection = win2.selection();
|
||||||
let store = selection
|
let store = selection
|
||||||
|
|
|
@ -15,12 +15,33 @@
|
||||||
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
|
||||||
use adw::subclass::prelude::*;
|
use adw::subclass::prelude::*;
|
||||||
use gtk::glib;
|
use gtk::{gio, glib};
|
||||||
|
|
||||||
/// The private struct, which can hold widgets and other data.
|
/// The private struct, which can hold widgets and other data.
|
||||||
#[derive(Debug, Default, gtk::CompositeTemplate)]
|
#[derive(Debug, Default, gtk::CompositeTemplate)]
|
||||||
#[template(file = "window.ui")]
|
#[template(file = "window.ui")]
|
||||||
pub struct MainWindow {
|
pub struct MainWindow {
|
||||||
|
#[template_child]
|
||||||
|
pub stack: TemplateChild<gtk::Stack>,
|
||||||
|
|
||||||
|
// For the login page
|
||||||
|
|
||||||
|
#[template_child]
|
||||||
|
pub jid: TemplateChild<adw::EntryRow>,
|
||||||
|
#[template_child]
|
||||||
|
pub password: TemplateChild<adw::PasswordEntryRow>,
|
||||||
|
#[template_child]
|
||||||
|
pub login: TemplateChild<gtk::Button>,
|
||||||
|
|
||||||
|
// For the spinner
|
||||||
|
|
||||||
|
#[template_child]
|
||||||
|
pub spinner: TemplateChild<gtk::Box>,
|
||||||
|
|
||||||
|
// For the chats page
|
||||||
|
|
||||||
|
#[template_child]
|
||||||
|
pub tabs_store: TemplateChild<gio::ListStore>,
|
||||||
#[template_child]
|
#[template_child]
|
||||||
pub split_view: TemplateChild<adw::NavigationSplitView>,
|
pub split_view: TemplateChild<adw::NavigationSplitView>,
|
||||||
#[template_child]
|
#[template_child]
|
||||||
|
|
|
@ -30,6 +30,30 @@ impl MainWindow {
|
||||||
glib::Object::builder().property("application", app).build()
|
glib::Object::builder().property("application", app).build()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn stack(&self) -> >k::Stack {
|
||||||
|
&self.imp().stack
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn jid(&self) -> &adw::EntryRow {
|
||||||
|
&self.imp().jid
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn password(&self) -> &adw::PasswordEntryRow {
|
||||||
|
&self.imp().password
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn login(&self) -> >k::Button {
|
||||||
|
&self.imp().login
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn spinner(&self) -> >k::Box {
|
||||||
|
&self.imp().spinner
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn tabs_store(&self) -> &gio::ListStore {
|
||||||
|
&self.imp().tabs_store
|
||||||
|
}
|
||||||
|
|
||||||
pub fn split_view(&self) -> &adw::NavigationSplitView {
|
pub fn split_view(&self) -> &adw::NavigationSplitView {
|
||||||
&self.imp().split_view
|
&self.imp().split_view
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,6 +13,68 @@
|
||||||
</object>
|
</object>
|
||||||
</child>
|
</child>
|
||||||
<property name="content">
|
<property name="content">
|
||||||
|
<object class="GtkStack" id="stack">
|
||||||
|
<child>
|
||||||
|
<object class="GtkBox">
|
||||||
|
<property name="orientation">vertical</property>
|
||||||
|
<child>
|
||||||
|
<object class="AdwHeaderBar"/>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="AdwClamp">
|
||||||
|
<child>
|
||||||
|
<object class="GtkBox">
|
||||||
|
<property name="orientation">vertical</property>
|
||||||
|
<child>
|
||||||
|
<object class="AdwPreferencesGroup">
|
||||||
|
<property name="title" translatable="yes">Enter your XMPP credentials</property>
|
||||||
|
<child>
|
||||||
|
<object class="AdwEntryRow" id="jid">
|
||||||
|
<property name="title" translatable="yes">JID</property>
|
||||||
|
<property name="input-purpose">email</property>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="AdwPasswordEntryRow" id="password">
|
||||||
|
<property name="title" translatable="yes">Password</property>
|
||||||
|
<property name="input-purpose">password</property>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkButton" id="login">
|
||||||
|
<property name="label" translatable="yes">Login</property>
|
||||||
|
<property name="css-classes">suggested-action</property>
|
||||||
|
<property name="margin-top">8</property>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkBox" id="spinner">
|
||||||
|
<property name="orientation">vertical</property>
|
||||||
|
<child>
|
||||||
|
<object class="AdwHeaderBar"/>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel">
|
||||||
|
<property name="label">Connecting…</property>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkSpinner">
|
||||||
|
<property name="spinning">yes</property>
|
||||||
|
<property name="vexpand">no</property>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
<object class="AdwNavigationSplitView" id="split_view">
|
<object class="AdwNavigationSplitView" id="split_view">
|
||||||
<property name="sidebar">
|
<property name="sidebar">
|
||||||
<object class="AdwNavigationPage">
|
<object class="AdwNavigationPage">
|
||||||
|
@ -68,6 +130,7 @@
|
||||||
<object class="GtkEntry" id="entry">
|
<object class="GtkEntry" id="entry">
|
||||||
<property name="hexpand">yes</property>
|
<property name="hexpand">yes</property>
|
||||||
<property name="placeholder-text" translatable="yes">Send a message</property>
|
<property name="placeholder-text" translatable="yes">Send a message</property>
|
||||||
|
<property name="input-purpose">free-form</property>
|
||||||
</object>
|
</object>
|
||||||
</child>
|
</child>
|
||||||
<!--
|
<!--
|
||||||
|
@ -86,8 +149,11 @@
|
||||||
</object>
|
</object>
|
||||||
</property>
|
</property>
|
||||||
</object>
|
</object>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
</property>
|
</property>
|
||||||
</template>
|
</template>
|
||||||
|
<object class="GListStore" id="tabs_store"/>
|
||||||
<object class="GtkSingleSelection" id="tabs_selection"/>
|
<object class="GtkSingleSelection" id="tabs_selection"/>
|
||||||
<object class="GtkNoSelection" id="selection"/>
|
<object class="GtkNoSelection" id="selection"/>
|
||||||
</interface>
|
</interface>
|
||||||
|
|
|
@ -32,6 +32,9 @@ pub(crate) fn client(jid: &str, password: &str) -> (Receiver<xmpp::Event>, Sende
|
||||||
Some(events) = client.wait_for_events() => {
|
Some(events) = client.wait_for_events() => {
|
||||||
for event in events {
|
for event in events {
|
||||||
match event {
|
match event {
|
||||||
|
Event::Online => {
|
||||||
|
event_sender.send(event).await.expect("BOOOOOOHOOOO");
|
||||||
|
}
|
||||||
Event::ContactAdded(_) => {
|
Event::ContactAdded(_) => {
|
||||||
event_sender.send(event).await.expect("BOOOOOOHOOOO");
|
event_sender.send(event).await.expect("BOOOOOOHOOOO");
|
||||||
}
|
}
|
||||||
|
@ -45,8 +48,8 @@ pub(crate) fn client(jid: &str, password: &str) -> (Receiver<xmpp::Event>, Sende
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
command = cmd_receiver.recv() => {
|
Ok(command) = cmd_receiver.recv() => {
|
||||||
match command.unwrap() {
|
match command {
|
||||||
XMPPCommand::SendPM(jid, content) => {
|
XMPPCommand::SendPM(jid, content) => {
|
||||||
client.send_message(jid.into(), xmpp::parsers::message::MessageType::Chat, "en", &content).await;
|
client.send_message(jid.into(), xmpp::parsers::message::MessageType::Chat, "en", &content).await;
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in New Issue
Block a user