From 228640a881aead53d307775cc2d3726fe64f3fc2 Mon Sep 17 00:00:00 2001 From: fiaxh Date: Sun, 15 Sep 2024 20:38:00 +0200 Subject: Code cleanup: Remove remnants of previous accounts dialog --- main/src/ui/manage_accounts/account_row.vala | 41 --- .../src/ui/manage_accounts/add_account_dialog.vala | 406 --------------------- main/src/ui/manage_accounts/dialog.vala | 241 ------------ .../account_preferences_subpage.vala | 2 +- .../preferences_window/add_account_dialog.vala | 406 +++++++++++++++++++++ .../encryption_preferences_page.vala | 1 - .../general_preferences_page.vala | 2 +- .../preferences_window/preferences_window.vala | 2 +- 8 files changed, 409 insertions(+), 692 deletions(-) delete mode 100644 main/src/ui/manage_accounts/account_row.vala delete mode 100644 main/src/ui/manage_accounts/add_account_dialog.vala delete mode 100644 main/src/ui/manage_accounts/dialog.vala create mode 100644 main/src/windows/preferences_window/add_account_dialog.vala (limited to 'main/src') diff --git a/main/src/ui/manage_accounts/account_row.vala b/main/src/ui/manage_accounts/account_row.vala deleted file mode 100644 index ae734b83..00000000 --- a/main/src/ui/manage_accounts/account_row.vala +++ /dev/null @@ -1,41 +0,0 @@ -using Gtk; - -using Dino.Entities; - -namespace Dino.Ui.ManageAccounts { - -[GtkTemplate (ui = "/im/dino/Dino/manage_accounts/account_row.ui")] -public class AccountRow : Gtk.ListBoxRow { - - [GtkChild] public unowned AvatarPicture picture; - [GtkChild] public unowned Label jid_label; - [GtkChild] public unowned Image icon; - - public Account account; - private StreamInteractor stream_interactor; - - public AccountRow(StreamInteractor stream_interactor, Account account) { - this.stream_interactor = stream_interactor; - this.account = account; - picture.model = new ViewModel.CompatAvatarPictureModel(stream_interactor).add_participant(new Conversation(account.bare_jid, account, Conversation.Type.CHAT), account.bare_jid); - jid_label.set_label(account.bare_jid.to_string()); - - stream_interactor.connection_manager.connection_error.connect((account, error) => { - if (account.equals(this.account)) { - update_warning_icon(); - } - }); - stream_interactor.connection_manager.connection_state_changed.connect((account, state) => { - if (account.equals(this.account)) { - update_warning_icon(); - } - }); - } - - private void update_warning_icon() { - ConnectionManager.ConnectionError? error = stream_interactor.connection_manager.get_error(account); - icon.visible = (error != null); - } -} - -} diff --git a/main/src/ui/manage_accounts/add_account_dialog.vala b/main/src/ui/manage_accounts/add_account_dialog.vala deleted file mode 100644 index 3700849f..00000000 --- a/main/src/ui/manage_accounts/add_account_dialog.vala +++ /dev/null @@ -1,406 +0,0 @@ -using Gee; -using Gtk; -using Pango; - -using Dino.Entities; -using Xmpp; - -namespace Dino.Ui.ManageAccounts { - -[GtkTemplate (ui = "/im/dino/Dino/manage_accounts/add_account_dialog.ui")] -public class AddAccountDialog : Adw.Window { - - public signal void added(Account account); - - enum Page { - SIGN_IN, - SIGN_IN_TLS_ERROR, - CREATE_ACCOUNT_SELECT_SERVER, - CREATE_ACCOUNT_REGISTER_FORM, - SUCCESS - } - - [GtkChild] private unowned Stack stack; - - [GtkChild] private unowned Revealer notification_revealer; - [GtkChild] private unowned Label notification_label; - - // Sign in - JID - [GtkChild] private unowned Box sign_in_box; - [GtkChild] private unowned Label sign_in_error_label; - [GtkChild] private unowned Adw.EntryRow jid_entry; - [GtkChild] private unowned Adw.PreferencesGroup password_group; - [GtkChild] private unowned Adw.PasswordEntryRow password_entry; - [GtkChild] private unowned Button sign_in_continue_button; - [GtkChild] private unowned Spinner sign_in_continue_spinner; - [GtkChild] private unowned Button sign_in_serverlist_button; - - // Sign in - TLS error - [GtkChild] private unowned Box sign_in_tls_box; - [GtkChild] private unowned Label sign_in_tls_label; - [GtkChild] private unowned Button sign_in_tls_back_button; - - // Select Server - [GtkChild] private unowned Box create_account_box; - [GtkChild] private unowned Button login_button; - [GtkChild] private unowned Spinner select_server_continue_spinner; - [GtkChild] private unowned Button select_server_continue; - [GtkChild] private unowned Label register_form_continue_label; - [GtkChild] private unowned ListBox server_list_box; - [GtkChild] private unowned Entry server_entry; - - // Register Form - [GtkChild] private unowned Button back_button; - [GtkChild] private unowned Box register_box; - [GtkChild] private unowned Box form_box; - [GtkChild] private unowned Spinner register_form_continue_spinner; - [GtkChild] private unowned Button register_form_continue; - - // Success - [GtkChild] private unowned Box success_box; - [GtkChild] private unowned Label success_description; - [GtkChild] private unowned Button success_continue_button; - - private static string[] server_list = new string[]{ - "5222.de", - "jabber.fr", - "movim.eu", - "yax.im" - }; - - private StreamInteractor stream_interactor; - private Database db; - private HashMap list_box_jids = new HashMap(); - private Jid? server_jid = null; - private Jid? login_jid = null; - private Xep.InBandRegistration.Form? form = null; - - public AddAccountDialog(StreamInteractor stream_interactor, Database db) { - this.stream_interactor = stream_interactor; - this.db = db; - this.title = _("Add Account"); - - // Sign in - Jid - jid_entry.changed.connect(on_jid_entry_changed); - sign_in_continue_button.clicked.connect(on_sign_in_continue_button_clicked); - sign_in_serverlist_button.clicked.connect(show_select_server); - - // Sign in - TLS error - sign_in_tls_back_button.clicked.connect(() => show_sign_in() ); - - // Select Server - server_entry.changed.connect(() => { - try { - Jid jid = new Jid(server_entry.text); - select_server_continue.sensitive = jid != null && jid.localpart == null && jid.resourcepart == null; - } catch (InvalidJidError e) { - select_server_continue.sensitive = false; - } - }); - select_server_continue.clicked.connect(on_select_server_continue); - login_button.clicked.connect(() => show_sign_in() ); - - foreach (string server in server_list) { - ListBoxRow list_box_row = new ListBoxRow(); - list_box_row.set_child(new Label(server) { xalign=0, margin_start=7, margin_end=7 }); - list_box_jids[list_box_row] = server; - server_list_box.append(list_box_row); - } - - // Register Form - register_form_continue.clicked.connect(on_register_form_continue_clicked); - back_button.clicked.connect(() => { - show_select_server(); - back_button.visible = false; - }); - - // Success - success_continue_button.clicked.connect(() => close()); - - show_sign_in(); - } - - private void show_sign_in(bool keep_jid = false) { - switch_stack_page(Page.SIGN_IN); - - this.title = _("Sign in"); - - set_default_widget(sign_in_continue_button); - sign_in_error_label.visible = false; - sign_in_continue_spinner.visible = false; - if (!keep_jid) { - jid_entry.text = ""; - jid_entry.grab_focus(); - } - password_entry.text = ""; - password_group.visible = false; - sign_in_serverlist_button.visible = true; - } - - private void show_tls_error(string domain, TlsCertificateFlags error_flags) { - switch_stack_page(Page.SIGN_IN_TLS_ERROR); - - string error_desc = _("The server could not prove that it is %s.").printf("" + domain + ""); - if (TlsCertificateFlags.UNKNOWN_CA in error_flags) { - error_desc += " " + _("Its security certificate is not trusted by your operating system."); - } else if (TlsCertificateFlags.BAD_IDENTITY in error_flags) { - error_desc += " " + _("Its security certificate is issued to another domain."); - } else if (TlsCertificateFlags.NOT_ACTIVATED in error_flags) { - error_desc += " " + _("Its security certificate will only become valid in the future."); - } else if (TlsCertificateFlags.EXPIRED in error_flags) { - error_desc += " " + _("Its security certificate is expired."); - } - sign_in_tls_label.label = error_desc; - } - - private void show_select_server() { - switch_stack_page(Page.CREATE_ACCOUNT_SELECT_SERVER); - - this.title = _("Create account"); - server_entry.text = ""; - server_entry.grab_focus(); - set_default_widget(select_server_continue); - - server_list_box.row_activated.disconnect(on_server_list_row_activated); - server_list_box.unselect_all(); - server_list_box.row_activated.connect(on_server_list_row_activated); - } - - private void show_register_form() { - switch_stack_page(Page.CREATE_ACCOUNT_REGISTER_FORM); - - set_default_widget(register_form_continue); - } - - private void show_success(Account account) { - switch_stack_page(Page.SUCCESS); - - success_description.label = _("You can now use the account %s.").printf("" + Markup.escape_text(account.bare_jid.to_string()) + ""); - - set_default_widget(success_continue_button); - } - - private void on_jid_entry_changed() { - try { - login_jid = new Jid(jid_entry.text); - if (login_jid.localpart != null && login_jid.resourcepart == null) { - sign_in_continue_button.sensitive = true; - } else { - sign_in_continue_button.sensitive = false; - } - } catch (InvalidJidError e) { - sign_in_continue_button.sensitive = false; - } - } - - private async void on_sign_in_continue_button_clicked() { - try { - login_jid = new Jid(jid_entry.text); - sign_in_tls_label.label = ""; - sign_in_error_label.visible = false; - sign_in_continue_button.sensitive = false; - sign_in_continue_spinner.visible = true; - - ulong jid_entry_changed_handler_id = -1; - jid_entry_changed_handler_id = jid_entry.changed.connect(() => { - jid_entry.disconnect(jid_entry_changed_handler_id); - show_sign_in(true); - return; - }); - - if (password_group.visible) { - // JID + Psw fields were visible: Try to log in - string password = password_entry.text; - Account account = new Account(login_jid, null, password, null); - - ConnectionManager.ConnectionError.Source? error = yield stream_interactor.get_module(Register.IDENTITY).add_check_account(account); - sign_in_continue_spinner.visible = false; - sign_in_continue_button.sensitive = true; - - if (error != null) { - sign_in_error_label.visible = true; - switch (error) { - case ConnectionManager.ConnectionError.Source.SASL: - sign_in_error_label.label = _("Wrong username or password"); - break; - default: - sign_in_error_label.label = _("Something went wrong"); - break; - } - } else { - add_activate_account(account); - show_success(account); - } - } else { - // Only JID field was visible: Check if server exists - Register.ServerAvailabilityReturn server_status = yield Register.check_server_availability(login_jid); - sign_in_continue_spinner.visible = false; - sign_in_continue_button.sensitive = true; - if (server_status.available) { - password_group.visible = true; - password_entry.grab_focus(); - sign_in_serverlist_button.visible = false; - } else { - if (server_status.error_flags != null) { - show_tls_error(login_jid.domainpart, server_status.error_flags); - } else { - sign_in_error_label.visible = true; - sign_in_error_label.label = _("Could not connect to %s").printf(login_jid.domainpart); - } - } - } - } catch (InvalidJidError e) { - warning("Invalid address from interface allowed login: %s", e.message); - sign_in_error_label.visible = true; - sign_in_error_label.label = _("Invalid address"); - } - } - - private void on_select_server_continue() { - try { - server_jid = new Jid(server_entry.text); - request_show_register_form.begin(server_jid); - } catch (InvalidJidError e) { - warning("Invalid address from interface allowed server: %s", e.message); - display_notification(_("Invalid address")); - } - } - - private void on_server_list_row_activated(ListBox box, ListBoxRow row) { - try { - server_jid = new Jid(list_box_jids[row]); - request_show_register_form.begin(server_jid); - } catch (InvalidJidError e) { - warning("Invalid address from selected server: %s", e.message); - display_notification(_("Invalid address")); - } - } - - private async void request_show_register_form(Jid server_jid) { - select_server_continue_spinner.visible = true; - Register.RegistrationFormReturn form_return = yield Register.get_registration_form(server_jid); - if (select_server_continue_spinner == null) { - return; - } - select_server_continue_spinner.visible = false; - if (form_return.form != null) { - form = form_return.form; - set_register_form(server_jid, form); - show_register_form(); - } else if (form_return.error_flags != null) { - show_tls_error(server_jid.domainpart, form_return.error_flags); - } else { - display_notification(_("No response from server")); - } - } - - private void set_register_form(Jid server, Xep.InBandRegistration.Form form) { - Widget widget = form_box.get_first_child(); - while (widget != null) { - form_box.remove(widget); - widget = form_box.get_first_child(); - } - - this.title = _("Register on %s").printf(server.to_string()); - - if (form.oob != null) { - form_box.append(new Label(_("The server requires to sign up through a website"))); - form_box.append(new Label(@"$(form.oob)") { use_markup=true }); - register_form_continue_label.label = _("Open website"); - register_form_continue.visible = true; - register_form_continue.grab_focus(); - } else if (form.fields.size > 0) { - if (form.instructions != null && form.instructions != "") { - string markup_instructions = Util.parse_add_markup(form.instructions, null, true, false); - form_box.append(new Label(markup_instructions) { use_markup=true, xalign=0, margin_top=7, - wrap=true, wrap_mode=Pango.WrapMode.WORD_CHAR }); - } - var form_preference_group = Util.rows_to_preference_group(Util.get_data_form_view_model(form), ""); - form_box.append(form_preference_group); - register_form_continue.visible = true; - register_form_continue_label.label = _("Register"); - } else { - form_box.append(new Label(_("Check %s for information on how to sign up").printf(@"$(server)")) { use_markup=true }); - register_form_continue.visible = false; - } - } - - private async void on_register_form_continue_clicked() { - notification_revealer.set_reveal_child(false); - // Button is opening a registration website - if (form.oob != null) { - try { - AppInfo.launch_default_for_uri(form.oob, null); - } catch (Error e) { } - show_sign_in(); - return; - } - - register_form_continue_spinner.visible = true; - string? error = yield Register.submit_form(server_jid, form); - if (register_form_continue_spinner == null) { - return; - } - register_form_continue_spinner.visible = false; - if (error == null) { - string? username = null, password = null; - foreach (Xep.DataForms.DataForm.Field field in form.fields) { - switch (field.var) { - case "username": username = field.get_value_string(); break; - case "password": password = field.get_value_string(); break; - } - } - try { - Account account = new Account(new Jid.components(username, server_jid.domainpart, null), null, password, null); - add_activate_account(account); - show_success(account); - } catch (InvalidJidError e) { - warning("Invalid address from components of registration: %s", e.message); - display_notification(_("Invalid address")); - } - } else { - display_notification(error); - } - } - - private void display_notification(string text) { - notification_label.label = text; - notification_revealer.set_reveal_child(true); - Timeout.add_seconds(5, () => { - notification_revealer.set_reveal_child(false); - return false; - }); - } - - private void add_activate_account(Account account) { - account.enabled = true; - account.persist(db); - stream_interactor.connect_account(account); - added(account); - } - - private void switch_stack_page(Page page) { - sign_in_box.visible = page == SIGN_IN; - sign_in_tls_box.visible = page == SIGN_IN_TLS_ERROR; - create_account_box.visible = page == CREATE_ACCOUNT_SELECT_SERVER; - register_box.visible = page == CREATE_ACCOUNT_REGISTER_FORM; - success_box.visible = page == SUCCESS; - - stack.visible_child_name = get_visible_stack_child_name(page); - - back_button.visible = page == CREATE_ACCOUNT_REGISTER_FORM; - } - - private string get_visible_stack_child_name(Page page) { - switch (page) { - case SIGN_IN: return "login_jid"; - case SIGN_IN_TLS_ERROR: return "tls_error"; - case CREATE_ACCOUNT_SELECT_SERVER: return "server"; - case CREATE_ACCOUNT_REGISTER_FORM: return "form"; - case SUCCESS: return "success"; - default: assert_not_reached(); - } - } -} - -} diff --git a/main/src/ui/manage_accounts/dialog.vala b/main/src/ui/manage_accounts/dialog.vala deleted file mode 100644 index b67f3a49..00000000 --- a/main/src/ui/manage_accounts/dialog.vala +++ /dev/null @@ -1,241 +0,0 @@ -using Gdk; -using Gee; -using Gtk; -using Markup; - -using Dino.Entities; -using Xmpp; - -namespace Dino.Ui.ManageAccounts { - -[GtkTemplate (ui = "/im/dino/Dino/manage_accounts/dialog.ui")] -public class Dialog : Gtk.Dialog { - - public signal void account_enabled(Account account); - public signal void account_disabled(Account account); - - [GtkChild] public unowned Stack main_stack; - [GtkChild] public unowned ListBox account_list; - [GtkChild] public unowned Button no_accounts_add; - [GtkChild] public unowned Button add_account_button; - [GtkChild] public unowned Button remove_account_button; - [GtkChild] public unowned AvatarPicture picture; - [GtkChild] public unowned Button image_button; - [GtkChild] public unowned Label jid_label; - [GtkChild] public unowned Label state_label; - [GtkChild] public unowned Switch active_switch; - [GtkChild] public unowned Util.EntryLabelHybrid password_hybrid; - [GtkChild] public unowned Util.EntryLabelHybrid alias_hybrid; - [GtkChild] public unowned Grid settings_list; - - private Database db; - private StreamInteractor stream_interactor; - private Account? selected_account; - - construct { - Util.force_error_color(state_label, ".is_error"); - account_list.row_selected.connect(on_account_list_row_selected); - add_account_button.clicked.connect(show_add_account_dialog); - no_accounts_add.clicked.connect(show_add_account_dialog); - remove_account_button.clicked.connect(() => { - AccountRow? account_row = account_list.get_selected_row() as AccountRow; - if (selected_account != null) remove_account(account_row); - }); - image_button.clicked.connect(show_select_avatar); - alias_hybrid.entry.changed.connect(() => { selected_account.alias = alias_hybrid.text; }); - password_hybrid.entry.changed.connect(() => { selected_account.password = password_hybrid.text; }); - - Util.LabelHybridGroup label_hybrid_group = new Util.LabelHybridGroup(); - label_hybrid_group.add(alias_hybrid); - label_hybrid_group.add(password_hybrid); - - main_stack.set_visible_child_name("no_accounts"); - - int row_index = 4; - int16 default_top_padding = new Gtk.Button().get_style_context().get_padding().top + 1; - Application app = GLib.Application.get_default() as Application; - foreach (Plugins.AccountSettingsEntry e in app.plugin_registry.account_settings_entries) { - Widget? widget = e.get_widget(Plugins.WidgetType.GTK4) as Widget; - if (widget == null) continue; - - Label label = new Label(e.name) { xalign=1, yalign=0 }; - label.add_css_class("dim-label"); - label.margin_top = e.label_top_padding == -1 ? default_top_padding : e.label_top_padding; - settings_list.attach(label, 0, row_index); - - settings_list.attach(widget, 1, row_index, 2); - row_index++; - } - } - - public Dialog(StreamInteractor stream_interactor, Database db) { - Object(use_header_bar : 1); - this.db = db; - this.stream_interactor = stream_interactor; - foreach (Account account in db.get_accounts()) { - add_account(account); - } - - stream_interactor.get_module(AvatarManager.IDENTITY).received_avatar.connect(on_received_avatar); - stream_interactor.connection_manager.connection_error.connect((account, error) => { - if (account.equals(selected_account)) { - update_status_label(account); - } - }); - stream_interactor.connection_manager.connection_state_changed.connect((account, state) => { - if (account.equals(selected_account)) { - update_status_label(account); - } - }); - - if (account_list.get_row_at_index(0) != null) account_list.select_row(account_list.get_row_at_index(0)); - } - - public AccountRow add_account(Account account) { - AccountRow account_item = new AccountRow (stream_interactor, account); - account_list.append(account_item); - main_stack.set_visible_child_name("accounts_exist"); - return account_item; - } - - private void show_add_account_dialog() { - AddAccountDialog add_account_dialog = new AddAccountDialog(stream_interactor, db); - add_account_dialog.set_transient_for(this); - add_account_dialog.added.connect((account) => { - AccountRow account_item = add_account(account); - account_list.select_row(account_item); - account_list.queue_draw(); - }); - add_account_dialog.present(); - } - - private void remove_account(AccountRow account_item) { - Gtk.MessageDialog msg = new Gtk.MessageDialog ( - this, Gtk.DialogFlags.DESTROY_WITH_PARENT | Gtk.DialogFlags.MODAL, - Gtk.MessageType.WARNING, Gtk.ButtonsType.OK_CANCEL, - _("Remove account %s?"), account_item.jid_label.get_text()); - msg.secondary_text = "You won't be able to access your conversation history anymore."; // TODO remove history! - Button ok_button = msg.get_widget_for_response(ResponseType.OK) as Button; - ok_button.label = _("Remove"); - ok_button.add_css_class("destructive-action"); - msg.response.connect((response) => { - if (response == ResponseType.OK) { - account_list.remove(account_item); - if (account_item.account.enabled) account_disabled(account_item.account); - account_item.account.remove(); - if (account_list.get_row_at_index(0) != null) { - account_list.select_row(account_list.get_row_at_index(0)); - } else { - main_stack.set_visible_child_name("no_accounts"); - } - } - msg.close(); - }); - msg.present(); - } - - private void on_account_list_row_selected(ListBoxRow? row) { - AccountRow? account_item = row as AccountRow; - if (account_item != null) { - selected_account = account_item.account; - populate_grid_data(account_item.account); - } - } - - private void show_select_avatar() { - FileChooserNative chooser = new FileChooserNative(_("Select avatar"), this, FileChooserAction.OPEN, _("Select"), _("Cancel")); - FileFilter filter = new FileFilter(); - foreach (PixbufFormat pixbuf_format in Pixbuf.get_formats()) { - foreach (string mime_type in pixbuf_format.get_mime_types()) { - filter.add_mime_type(mime_type); - } - } - filter.set_filter_name(_("Images")); - chooser.add_filter(filter); - - filter = new FileFilter(); - filter.set_filter_name(_("All files")); - filter.add_pattern("*"); - chooser.add_filter(filter); - - chooser.response.connect(() => { - string uri = chooser.get_file().get_path(); - stream_interactor.get_module(AvatarManager.IDENTITY).publish(selected_account, uri); - }); - - chooser.show(); - } - - private bool change_account_state(bool state) { - selected_account.enabled = state; - if (state) { - account_enabled(selected_account); - } else { - account_disabled(selected_account); - } - return false; - } - - private void on_received_avatar(Jid jid, Account account) { - if (selected_account.equals(account) && jid.equals(account.bare_jid)) { - picture.model = new ViewModel.CompatAvatarPictureModel(stream_interactor).add_participant(new Conversation(account.bare_jid, account, Conversation.Type.CHAT), account.bare_jid); - } - } - - private void populate_grid_data(Account account) { - active_switch.state_set.disconnect(change_account_state); - - picture.model = new ViewModel.CompatAvatarPictureModel(stream_interactor).add_participant(new Conversation(account.bare_jid, account, Conversation.Type.CHAT), account.bare_jid); - active_switch.set_active(account.enabled); - jid_label.label = account.bare_jid.to_string(); - - alias_hybrid.text = account.alias ?? ""; - password_hybrid.entry.input_purpose = InputPurpose.PASSWORD; - password_hybrid.text = account.password; - - update_status_label(account); - - active_switch.state_set.connect(change_account_state); - - Application app = GLib.Application.get_default() as Application; - foreach (Plugins.AccountSettingsEntry e in app.plugin_registry.account_settings_entries) { - e.set_account(account); - } - } - - private void update_status_label(Account account) { - state_label.label = ""; - ConnectionManager.ConnectionError? error = stream_interactor.connection_manager.get_error(account); - if (error != null) { - state_label.label = get_connection_error_description(error); - state_label.add_css_class("is_error"); - } else { - ConnectionManager.ConnectionState state = stream_interactor.connection_manager.get_state(account); - switch (state) { - case ConnectionManager.ConnectionState.CONNECTING: - state_label.label = _("Connecting…"); break; - case ConnectionManager.ConnectionState.CONNECTED: - state_label.label = _("Connected"); break; - case ConnectionManager.ConnectionState.DISCONNECTED: - state_label.label = _("Disconnected"); break; - } - state_label.remove_css_class("is_error"); - } - } - - private string get_connection_error_description(ConnectionManager.ConnectionError error) { - switch (error.source) { - case ConnectionManager.ConnectionError.Source.SASL: - return _("Wrong password"); - case ConnectionManager.ConnectionError.Source.TLS: - return _("Invalid TLS certificate"); - } - if (error.identifier != null) { - return _("Error") + ": " + error.identifier; - } else { - return _("Error"); - } - } -} - -} diff --git a/main/src/windows/preferences_window/account_preferences_subpage.vala b/main/src/windows/preferences_window/account_preferences_subpage.vala index a462666d..e0896158 100644 --- a/main/src/windows/preferences_window/account_preferences_subpage.vala +++ b/main/src/windows/preferences_window/account_preferences_subpage.vala @@ -5,7 +5,7 @@ using Gee; using Gtk; using Gdk; -[GtkTemplate (ui = "/im/dino/Dino/preferences_window_account.ui")] +[GtkTemplate (ui = "/im/dino/Dino/preferences_window/account_preferences_subpage.ui")] public class Dino.Ui.AccountPreferencesSubpage : Gtk.Box { [GtkChild] public unowned Adw.HeaderBar headerbar; diff --git a/main/src/windows/preferences_window/add_account_dialog.vala b/main/src/windows/preferences_window/add_account_dialog.vala new file mode 100644 index 00000000..00acb6aa --- /dev/null +++ b/main/src/windows/preferences_window/add_account_dialog.vala @@ -0,0 +1,406 @@ +using Gee; +using Gtk; +using Pango; + +using Dino.Entities; +using Xmpp; + +namespace Dino.Ui.ManageAccounts { + +[GtkTemplate (ui = "/im/dino/Dino/preferences_window/add_account_dialog.ui")] +public class AddAccountDialog : Adw.Window { + + public signal void added(Account account); + + enum Page { + SIGN_IN, + SIGN_IN_TLS_ERROR, + CREATE_ACCOUNT_SELECT_SERVER, + CREATE_ACCOUNT_REGISTER_FORM, + SUCCESS + } + + [GtkChild] private unowned Stack stack; + + [GtkChild] private unowned Revealer notification_revealer; + [GtkChild] private unowned Label notification_label; + + // Sign in - JID + [GtkChild] private unowned Box sign_in_box; + [GtkChild] private unowned Label sign_in_error_label; + [GtkChild] private unowned Adw.EntryRow jid_entry; + [GtkChild] private unowned Adw.PreferencesGroup password_group; + [GtkChild] private unowned Adw.PasswordEntryRow password_entry; + [GtkChild] private unowned Button sign_in_continue_button; + [GtkChild] private unowned Spinner sign_in_continue_spinner; + [GtkChild] private unowned Button sign_in_serverlist_button; + + // Sign in - TLS error + [GtkChild] private unowned Box sign_in_tls_box; + [GtkChild] private unowned Label sign_in_tls_label; + [GtkChild] private unowned Button sign_in_tls_back_button; + + // Select Server + [GtkChild] private unowned Box create_account_box; + [GtkChild] private unowned Button login_button; + [GtkChild] private unowned Spinner select_server_continue_spinner; + [GtkChild] private unowned Button select_server_continue; + [GtkChild] private unowned Label register_form_continue_label; + [GtkChild] private unowned ListBox server_list_box; + [GtkChild] private unowned Entry server_entry; + + // Register Form + [GtkChild] private unowned Button back_button; + [GtkChild] private unowned Box register_box; + [GtkChild] private unowned Box form_box; + [GtkChild] private unowned Spinner register_form_continue_spinner; + [GtkChild] private unowned Button register_form_continue; + + // Success + [GtkChild] private unowned Box success_box; + [GtkChild] private unowned Label success_description; + [GtkChild] private unowned Button success_continue_button; + + private static string[] server_list = new string[]{ + "5222.de", + "jabber.fr", + "movim.eu", + "yax.im" + }; + + private StreamInteractor stream_interactor; + private Database db; + private HashMap list_box_jids = new HashMap(); + private Jid? server_jid = null; + private Jid? login_jid = null; + private Xep.InBandRegistration.Form? form = null; + + public AddAccountDialog(StreamInteractor stream_interactor, Database db) { + this.stream_interactor = stream_interactor; + this.db = db; + this.title = _("Add Account"); + + // Sign in - Jid + jid_entry.changed.connect(on_jid_entry_changed); + sign_in_continue_button.clicked.connect(on_sign_in_continue_button_clicked); + sign_in_serverlist_button.clicked.connect(show_select_server); + + // Sign in - TLS error + sign_in_tls_back_button.clicked.connect(() => show_sign_in() ); + + // Select Server + server_entry.changed.connect(() => { + try { + Jid jid = new Jid(server_entry.text); + select_server_continue.sensitive = jid != null && jid.localpart == null && jid.resourcepart == null; + } catch (InvalidJidError e) { + select_server_continue.sensitive = false; + } + }); + select_server_continue.clicked.connect(on_select_server_continue); + login_button.clicked.connect(() => show_sign_in() ); + + foreach (string server in server_list) { + ListBoxRow list_box_row = new ListBoxRow(); + list_box_row.set_child(new Label(server) { xalign=0, margin_start=7, margin_end=7 }); + list_box_jids[list_box_row] = server; + server_list_box.append(list_box_row); + } + + // Register Form + register_form_continue.clicked.connect(on_register_form_continue_clicked); + back_button.clicked.connect(() => { + show_select_server(); + back_button.visible = false; + }); + + // Success + success_continue_button.clicked.connect(() => close()); + + show_sign_in(); + } + + private void show_sign_in(bool keep_jid = false) { + switch_stack_page(Page.SIGN_IN); + + this.title = _("Sign in"); + + set_default_widget(sign_in_continue_button); + sign_in_error_label.visible = false; + sign_in_continue_spinner.visible = false; + if (!keep_jid) { + jid_entry.text = ""; + jid_entry.grab_focus(); + } + password_entry.text = ""; + password_group.visible = false; + sign_in_serverlist_button.visible = true; + } + + private void show_tls_error(string domain, TlsCertificateFlags error_flags) { + switch_stack_page(Page.SIGN_IN_TLS_ERROR); + + string error_desc = _("The server could not prove that it is %s.").printf("" + domain + ""); + if (TlsCertificateFlags.UNKNOWN_CA in error_flags) { + error_desc += " " + _("Its security certificate is not trusted by your operating system."); + } else if (TlsCertificateFlags.BAD_IDENTITY in error_flags) { + error_desc += " " + _("Its security certificate is issued to another domain."); + } else if (TlsCertificateFlags.NOT_ACTIVATED in error_flags) { + error_desc += " " + _("Its security certificate will only become valid in the future."); + } else if (TlsCertificateFlags.EXPIRED in error_flags) { + error_desc += " " + _("Its security certificate is expired."); + } + sign_in_tls_label.label = error_desc; + } + + private void show_select_server() { + switch_stack_page(Page.CREATE_ACCOUNT_SELECT_SERVER); + + this.title = _("Create account"); + server_entry.text = ""; + server_entry.grab_focus(); + set_default_widget(select_server_continue); + + server_list_box.row_activated.disconnect(on_server_list_row_activated); + server_list_box.unselect_all(); + server_list_box.row_activated.connect(on_server_list_row_activated); + } + + private void show_register_form() { + switch_stack_page(Page.CREATE_ACCOUNT_REGISTER_FORM); + + set_default_widget(register_form_continue); + } + + private void show_success(Account account) { + switch_stack_page(Page.SUCCESS); + + success_description.label = _("You can now use the account %s.").printf("" + Markup.escape_text(account.bare_jid.to_string()) + ""); + + set_default_widget(success_continue_button); + } + + private void on_jid_entry_changed() { + try { + login_jid = new Jid(jid_entry.text); + if (login_jid.localpart != null && login_jid.resourcepart == null) { + sign_in_continue_button.sensitive = true; + } else { + sign_in_continue_button.sensitive = false; + } + } catch (InvalidJidError e) { + sign_in_continue_button.sensitive = false; + } + } + + private async void on_sign_in_continue_button_clicked() { + try { + login_jid = new Jid(jid_entry.text); + sign_in_tls_label.label = ""; + sign_in_error_label.visible = false; + sign_in_continue_button.sensitive = false; + sign_in_continue_spinner.visible = true; + + ulong jid_entry_changed_handler_id = -1; + jid_entry_changed_handler_id = jid_entry.changed.connect(() => { + jid_entry.disconnect(jid_entry_changed_handler_id); + show_sign_in(true); + return; + }); + + if (password_group.visible) { + // JID + Psw fields were visible: Try to log in + string password = password_entry.text; + Account account = new Account(login_jid, null, password, null); + + ConnectionManager.ConnectionError.Source? error = yield stream_interactor.get_module(Register.IDENTITY).add_check_account(account); + sign_in_continue_spinner.visible = false; + sign_in_continue_button.sensitive = true; + + if (error != null) { + sign_in_error_label.visible = true; + switch (error) { + case ConnectionManager.ConnectionError.Source.SASL: + sign_in_error_label.label = _("Wrong username or password"); + break; + default: + sign_in_error_label.label = _("Something went wrong"); + break; + } + } else { + add_activate_account(account); + show_success(account); + } + } else { + // Only JID field was visible: Check if server exists + Register.ServerAvailabilityReturn server_status = yield Register.check_server_availability(login_jid); + sign_in_continue_spinner.visible = false; + sign_in_continue_button.sensitive = true; + if (server_status.available) { + password_group.visible = true; + password_entry.grab_focus(); + sign_in_serverlist_button.visible = false; + } else { + if (server_status.error_flags != null) { + show_tls_error(login_jid.domainpart, server_status.error_flags); + } else { + sign_in_error_label.visible = true; + sign_in_error_label.label = _("Could not connect to %s").printf(login_jid.domainpart); + } + } + } + } catch (InvalidJidError e) { + warning("Invalid address from interface allowed login: %s", e.message); + sign_in_error_label.visible = true; + sign_in_error_label.label = _("Invalid address"); + } + } + + private void on_select_server_continue() { + try { + server_jid = new Jid(server_entry.text); + request_show_register_form.begin(server_jid); + } catch (InvalidJidError e) { + warning("Invalid address from interface allowed server: %s", e.message); + display_notification(_("Invalid address")); + } + } + + private void on_server_list_row_activated(ListBox box, ListBoxRow row) { + try { + server_jid = new Jid(list_box_jids[row]); + request_show_register_form.begin(server_jid); + } catch (InvalidJidError e) { + warning("Invalid address from selected server: %s", e.message); + display_notification(_("Invalid address")); + } + } + + private async void request_show_register_form(Jid server_jid) { + select_server_continue_spinner.visible = true; + Register.RegistrationFormReturn form_return = yield Register.get_registration_form(server_jid); + if (select_server_continue_spinner == null) { + return; + } + select_server_continue_spinner.visible = false; + if (form_return.form != null) { + form = form_return.form; + set_register_form(server_jid, form); + show_register_form(); + } else if (form_return.error_flags != null) { + show_tls_error(server_jid.domainpart, form_return.error_flags); + } else { + display_notification(_("No response from server")); + } + } + + private void set_register_form(Jid server, Xep.InBandRegistration.Form form) { + Widget widget = form_box.get_first_child(); + while (widget != null) { + form_box.remove(widget); + widget = form_box.get_first_child(); + } + + this.title = _("Register on %s").printf(server.to_string()); + + if (form.oob != null) { + form_box.append(new Label(_("The server requires to sign up through a website"))); + form_box.append(new Label(@"$(form.oob)") { use_markup=true }); + register_form_continue_label.label = _("Open website"); + register_form_continue.visible = true; + register_form_continue.grab_focus(); + } else if (form.fields.size > 0) { + if (form.instructions != null && form.instructions != "") { + string markup_instructions = Util.parse_add_markup(form.instructions, null, true, false); + form_box.append(new Label(markup_instructions) { use_markup=true, xalign=0, margin_top=7, + wrap=true, wrap_mode=Pango.WrapMode.WORD_CHAR }); + } + var form_preference_group = Util.rows_to_preference_group(Util.get_data_form_view_model(form), ""); + form_box.append(form_preference_group); + register_form_continue.visible = true; + register_form_continue_label.label = _("Register"); + } else { + form_box.append(new Label(_("Check %s for information on how to sign up").printf(@"$(server)")) { use_markup=true }); + register_form_continue.visible = false; + } + } + + private async void on_register_form_continue_clicked() { + notification_revealer.set_reveal_child(false); + // Button is opening a registration website + if (form.oob != null) { + try { + AppInfo.launch_default_for_uri(form.oob, null); + } catch (Error e) { } + show_sign_in(); + return; + } + + register_form_continue_spinner.visible = true; + string? error = yield Register.submit_form(server_jid, form); + if (register_form_continue_spinner == null) { + return; + } + register_form_continue_spinner.visible = false; + if (error == null) { + string? username = null, password = null; + foreach (Xep.DataForms.DataForm.Field field in form.fields) { + switch (field.var) { + case "username": username = field.get_value_string(); break; + case "password": password = field.get_value_string(); break; + } + } + try { + Account account = new Account(new Jid.components(username, server_jid.domainpart, null), null, password, null); + add_activate_account(account); + show_success(account); + } catch (InvalidJidError e) { + warning("Invalid address from components of registration: %s", e.message); + display_notification(_("Invalid address")); + } + } else { + display_notification(error); + } + } + + private void display_notification(string text) { + notification_label.label = text; + notification_revealer.set_reveal_child(true); + Timeout.add_seconds(5, () => { + notification_revealer.set_reveal_child(false); + return false; + }); + } + + private void add_activate_account(Account account) { + account.enabled = true; + account.persist(db); + stream_interactor.connect_account(account); + added(account); + } + + private void switch_stack_page(Page page) { + sign_in_box.visible = page == SIGN_IN; + sign_in_tls_box.visible = page == SIGN_IN_TLS_ERROR; + create_account_box.visible = page == CREATE_ACCOUNT_SELECT_SERVER; + register_box.visible = page == CREATE_ACCOUNT_REGISTER_FORM; + success_box.visible = page == SUCCESS; + + stack.visible_child_name = get_visible_stack_child_name(page); + + back_button.visible = page == CREATE_ACCOUNT_REGISTER_FORM; + } + + private string get_visible_stack_child_name(Page page) { + switch (page) { + case SIGN_IN: return "login_jid"; + case SIGN_IN_TLS_ERROR: return "tls_error"; + case CREATE_ACCOUNT_SELECT_SERVER: return "server"; + case CREATE_ACCOUNT_REGISTER_FORM: return "form"; + case SUCCESS: return "success"; + default: assert_not_reached(); + } + } +} + +} diff --git a/main/src/windows/preferences_window/encryption_preferences_page.vala b/main/src/windows/preferences_window/encryption_preferences_page.vala index 7477e6cd..eec38908 100644 --- a/main/src/windows/preferences_window/encryption_preferences_page.vala +++ b/main/src/windows/preferences_window/encryption_preferences_page.vala @@ -4,7 +4,6 @@ using Xmpp.Xep; using Gee; using Gtk; -//[GtkTemplate (ui = "/im/dino/Dino/preferences_window_encryption.ui")] public class Dino.Ui.PreferencesWindowEncryption : Adw.PreferencesPage { private DropDown drop_down = null; diff --git a/main/src/windows/preferences_window/general_preferences_page.vala b/main/src/windows/preferences_window/general_preferences_page.vala index 7aa6c2bd..6f8dd771 100644 --- a/main/src/windows/preferences_window/general_preferences_page.vala +++ b/main/src/windows/preferences_window/general_preferences_page.vala @@ -7,7 +7,7 @@ public class Dino.Ui.ViewModel.GeneralPreferencesPage : Object { public bool convert_emojis { get; set; } } -[GtkTemplate (ui = "/im/dino/Dino/preferences_window_general.ui")] +[GtkTemplate (ui = "/im/dino/Dino/preferences_window/general_preferences_page.ui")] public class Dino.Ui.GeneralPreferencesPage : Adw.PreferencesPage { [GtkChild] private unowned Switch typing_switch; [GtkChild] private unowned Switch marker_switch; diff --git a/main/src/windows/preferences_window/preferences_window.vala b/main/src/windows/preferences_window/preferences_window.vala index e34261e9..1a75aa0a 100644 --- a/main/src/windows/preferences_window/preferences_window.vala +++ b/main/src/windows/preferences_window/preferences_window.vala @@ -5,7 +5,7 @@ using Xmpp.Xep; using Gee; using Gtk; -[GtkTemplate (ui = "/im/dino/Dino/preferences_window.ui")] +[GtkTemplate (ui = "/im/dino/Dino/preferences_window/preferences_window.ui")] public class Dino.Ui.PreferencesWindow : Adw.PreferencesWindow { [GtkChild] public unowned Dino.Ui.PreferencesWindowAccounts accounts_page; [GtkChild] public unowned Dino.Ui.PreferencesWindowEncryption encryption_page; -- cgit v1.2.3-70-g09d2