mirror of
				https://github.com/johrpan/musicus.git
				synced 2025-10-26 11:47:25 +01:00 
			
		
		
		
	Modernize login dialog
This commit is contained in:
		
							parent
							
								
									7bf82eefbc
								
							
						
					
					
						commit
						c9d9c1bc24
					
				
					 4 changed files with 145 additions and 126 deletions
				
			
		|  | @ -2,85 +2,87 @@ | |||
| <interface> | ||||
|   <requires lib="gtk" version="4.0"/> | ||||
|   <requires lib="libadwaita" version="1.0"/> | ||||
|   <object class="AdwWindow" id="window"> | ||||
|     <property name="modal">True</property> | ||||
|     <property name="default-width">350</property> | ||||
|   <object class="GtkStack" id="widget"> | ||||
|     <property name="transition-type">crossfade</property> | ||||
|     <child> | ||||
|       <object class="GtkStack" id="stack"> | ||||
|         <property name="transition-type">crossfade</property> | ||||
|         <child> | ||||
|           <object class="GtkStackPage"> | ||||
|             <property name="name">content</property> | ||||
|             <property name="child"> | ||||
|               <object class="GtkBox"> | ||||
|                 <property name="orientation">vertical</property> | ||||
|       <object class="GtkStackPage"> | ||||
|         <property name="name">content</property> | ||||
|         <property name="child"> | ||||
|           <object class="GtkBox"> | ||||
|             <property name="orientation">vertical</property> | ||||
|             <child> | ||||
|               <object class="AdwHeaderBar"> | ||||
|                 <property name="show-start-title-buttons">false</property> | ||||
|                 <property name="show-end-title-buttons">false</property> | ||||
|                 <property name="title-widget"> | ||||
|                   <object class="GtkLabel"> | ||||
|                   </object> | ||||
|                 </property> | ||||
|                 <child> | ||||
|                   <object class="AdwHeaderBar"> | ||||
|                     <property name="show-start-title-buttons">false</property> | ||||
|                     <property name="show-end-title-buttons">false</property> | ||||
|                     <property name="title-widget"> | ||||
|                       <object class="GtkLabel"> | ||||
|                         <property name="label" translatable="yes">Login</property> | ||||
|                         <style> | ||||
|                           <class name="title"/> | ||||
|                         </style> | ||||
|                       </object> | ||||
|                     </property> | ||||
|                     <child> | ||||
|                       <object class="GtkButton" id="cancel_button"> | ||||
|                         <property name="label" translatable="yes">Cancel</property> | ||||
|                       </object> | ||||
|                     </child> | ||||
|                     <child type="end"> | ||||
|                       <object class="GtkButton" id="login_button"> | ||||
|                         <property name="label" translatable="yes">Login</property> | ||||
|                         <property name="has-default">True</property> | ||||
|                         <style> | ||||
|                           <class name="suggested-action"/> | ||||
|                         </style> | ||||
|                       </object> | ||||
|                     </child> | ||||
|                   <object class="GtkButton" id="cancel_button"> | ||||
|                     <property name="label" translatable="yes">Cancel</property> | ||||
|                   </object> | ||||
|                 </child> | ||||
|                 <child> | ||||
|                   <object class="GtkInfoBar" id="info_bar"> | ||||
|                     <property name="message-type">error</property> | ||||
|                     <property name="revealed">False</property> | ||||
|                     <child> | ||||
|                       <object class="GtkLabel"> | ||||
|                         <property name="label" translatable="yes">The login credentials were wrong!</property> | ||||
|                       </object> | ||||
|                     </child> | ||||
|                 <child type="end"> | ||||
|                   <object class="GtkButton" id="login_button"> | ||||
|                     <property name="label" translatable="yes">Login</property> | ||||
|                     <style> | ||||
|                       <class name="suggested-action"/> | ||||
|                     </style> | ||||
|                   </object> | ||||
|                 </child> | ||||
|               </object> | ||||
|             </child> | ||||
|             <child> | ||||
|               <object class="GtkInfoBar" id="info_bar"> | ||||
|                 <property name="revealed">False</property> | ||||
|               </object> | ||||
|             </child> | ||||
|             <child> | ||||
|               <object class="GtkScrolledWindow"> | ||||
|                 <property name="vexpand">true</property> | ||||
|                 <child> | ||||
|                   <object class="GtkListBox"> | ||||
|                     <property name="selection-mode">none</property> | ||||
|                   <object class="AdwClamp"> | ||||
|                     <property name="margin-start">12</property> | ||||
|                     <property name="margin-end">12</property> | ||||
|                     <property name="margin-top">18</property> | ||||
|                     <property name="margin-bottom">12</property> | ||||
|                     <property name="maximum-size">500</property> | ||||
|                     <property name="tightening-threshold">300</property> | ||||
|                     <child> | ||||
|                       <object class="AdwActionRow"> | ||||
|                         <property name="activatable">True</property> | ||||
|                         <property name="title" translatable="yes">Username</property> | ||||
|                         <property name="activatable-widget">username_entry</property> | ||||
|                       <object class="GtkFrame"> | ||||
|                         <property name="valign">start</property> | ||||
|                         <child> | ||||
|                           <object class="GtkEntry" id="username_entry"> | ||||
|                             <property name="valign">center</property> | ||||
|                             <property name="hexpand">True</property> | ||||
|                           </object> | ||||
|                         </child> | ||||
|                       </object> | ||||
|                     </child> | ||||
|                     <child> | ||||
|                       <object class="AdwActionRow"> | ||||
|                         <property name="activatable">True</property> | ||||
|                         <property name="title" translatable="yes">Password</property> | ||||
|                         <property name="activatable-widget">password_entry</property> | ||||
|                         <child> | ||||
|                           <object class="GtkEntry" id="password_entry"> | ||||
|                             <property name="valign">center</property> | ||||
|                             <property name="hexpand">True</property> | ||||
|                             <property name="visibility">False</property> | ||||
|                             <property name="activates-default">True</property> | ||||
|                             <property name="input-purpose">password</property> | ||||
|                           <object class="GtkListBox"> | ||||
|                             <property name="selection-mode">none</property> | ||||
|                             <child> | ||||
|                               <object class="AdwActionRow"> | ||||
|                                 <property name="activatable">True</property> | ||||
|                                 <property name="title" translatable="yes">Username</property> | ||||
|                                 <property name="activatable-widget">username_entry</property> | ||||
|                                 <child> | ||||
|                                   <object class="GtkEntry" id="username_entry"> | ||||
|                                     <property name="valign">center</property> | ||||
|                                     <property name="hexpand">True</property> | ||||
|                                   </object> | ||||
|                                 </child> | ||||
|                               </object> | ||||
|                             </child> | ||||
|                             <child> | ||||
|                               <object class="AdwActionRow"> | ||||
|                                 <property name="activatable">True</property> | ||||
|                                 <property name="title" translatable="yes">Password</property> | ||||
|                                 <property name="activatable-widget">password_entry</property> | ||||
|                                 <child> | ||||
|                                   <object class="GtkEntry" id="password_entry"> | ||||
|                                     <property name="valign">center</property> | ||||
|                                     <property name="hexpand">True</property> | ||||
|                                     <property name="visibility">False</property> | ||||
|                                     <property name="input-purpose">password</property> | ||||
|                                   </object> | ||||
|                                 </child> | ||||
|                               </object> | ||||
|                             </child> | ||||
|                           </object> | ||||
|                         </child> | ||||
|                       </object> | ||||
|  | @ -88,49 +90,43 @@ | |||
|                   </object> | ||||
|                 </child> | ||||
|               </object> | ||||
|             </property> | ||||
|             </child> | ||||
|           </object> | ||||
|         </child> | ||||
|         <child> | ||||
|           <object class="GtkStackPage"> | ||||
|             <property name="name">loading</property> | ||||
|             <property name="child"> | ||||
|               <object class="GtkBox"> | ||||
|                 <property name="orientation">vertical</property> | ||||
|                 <child> | ||||
|                   <object class="AdwHeaderBar"> | ||||
|                     <property name="show-start-title-buttons">false</property> | ||||
|                     <property name="show-end-title-buttons">false</property> | ||||
|                     <property name="title-widget"> | ||||
|                       <object class="GtkLabel"> | ||||
|                         <property name="label" translatable="yes">Login</property> | ||||
|                         <style> | ||||
|                           <class name="title"/> | ||||
|                         </style> | ||||
|                       </object> | ||||
|                     </property> | ||||
|         </property> | ||||
|       </object> | ||||
|     </child> | ||||
|     <child> | ||||
|       <object class="GtkStackPage"> | ||||
|         <property name="name">loading</property> | ||||
|         <property name="child"> | ||||
|           <object class="GtkBox"> | ||||
|             <property name="orientation">vertical</property> | ||||
|             <child> | ||||
|               <object class="AdwHeaderBar"> | ||||
|                 <property name="show-start-title-buttons">false</property> | ||||
|                 <property name="show-end-title-buttons">false</property> | ||||
|                 <property name="title-widget"> | ||||
|                   <object class="GtkLabel"> | ||||
|                     <property name="label" translatable="yes">Login</property> | ||||
|                     <style> | ||||
|                       <class name="title"/> | ||||
|                     </style> | ||||
|                   </object> | ||||
|                 </child> | ||||
|                 <child> | ||||
|                   <object class="GtkSpinner"> | ||||
|                     <property name="spinning">True</property> | ||||
|                     <property name="hexpand">true</property> | ||||
|                     <property name="vexpand">true</property> | ||||
|                     <property name="halign">center</property> | ||||
|                     <property name="valign">center</property> | ||||
|                   </object> | ||||
|                 </child> | ||||
|                 </property> | ||||
|               </object> | ||||
|             </property> | ||||
|             </child> | ||||
|             <child> | ||||
|               <object class="GtkSpinner"> | ||||
|                 <property name="spinning">true</property> | ||||
|                 <property name="hexpand">true</property> | ||||
|                 <property name="vexpand">true</property> | ||||
|                 <property name="halign">center</property> | ||||
|                 <property name="valign">center</property> | ||||
|               </object> | ||||
|             </child> | ||||
|           </object> | ||||
|         </child> | ||||
|         </property> | ||||
|       </object> | ||||
|     </child> | ||||
|   </object> | ||||
|   <object class="GtkSizeGroup"> | ||||
|     <widgets> | ||||
|       <widget name="username_entry"/> | ||||
|       <widget name="password_entry"/> | ||||
|     </widgets> | ||||
|   </object> | ||||
| </interface> | ||||
|  |  | |||
|  | @ -1,4 +1,5 @@ | |||
| use crate::backend::{Backend, LoginData}; | ||||
| use crate::widgets::{Navigator, NavigatorScreen}; | ||||
| use glib::clone; | ||||
| use gtk::prelude::*; | ||||
| use gtk_macros::get_widget; | ||||
|  | @ -8,48 +9,48 @@ use std::rc::Rc; | |||
| /// A dialog for entering login credentials.
 | ||||
| pub struct LoginDialog { | ||||
|     backend: Rc<Backend>, | ||||
|     window: libadwaita::Window, | ||||
|     stack: gtk::Stack, | ||||
|     widget: gtk::Stack, | ||||
|     info_bar: gtk::InfoBar, | ||||
|     username_entry: gtk::Entry, | ||||
|     password_entry: gtk::Entry, | ||||
|     selected_cb: RefCell<Option<Box<dyn Fn(LoginData) -> ()>>>, | ||||
|     navigator: RefCell<Option<Rc<Navigator>>>, | ||||
| } | ||||
| 
 | ||||
| impl LoginDialog { | ||||
|     /// Create a new login dialog.
 | ||||
|     pub fn new<P: IsA<gtk::Window>>(backend: Rc<Backend>, parent: &P) -> Rc<Self> { | ||||
|     pub fn new(backend: Rc<Backend>) -> Rc<Self> { | ||||
|         // Create UI
 | ||||
|         let builder = gtk::Builder::from_resource("/de/johrpan/musicus/ui/login_dialog.ui"); | ||||
| 
 | ||||
|         get_widget!(builder, libadwaita::Window, window); | ||||
|         get_widget!(builder, gtk::Stack, stack); | ||||
|         get_widget!(builder, gtk::Stack, widget); | ||||
|         get_widget!(builder, gtk::InfoBar, info_bar); | ||||
|         get_widget!(builder, gtk::Button, cancel_button); | ||||
|         get_widget!(builder, gtk::Button, login_button); | ||||
|         get_widget!(builder, gtk::Entry, username_entry); | ||||
|         get_widget!(builder, gtk::Entry, password_entry); | ||||
| 
 | ||||
|         window.set_transient_for(Some(parent)); | ||||
| 
 | ||||
|         let this = Rc::new(Self { | ||||
|             backend, | ||||
|             window, | ||||
|             stack, | ||||
|             widget, | ||||
|             info_bar, | ||||
|             username_entry, | ||||
|             password_entry, | ||||
|             selected_cb: RefCell::new(None), | ||||
|             navigator: RefCell::new(None), | ||||
|         }); | ||||
| 
 | ||||
|         // Connect signals and callbacks
 | ||||
| 
 | ||||
|         cancel_button.connect_clicked(clone!(@strong this => move |_| { | ||||
|             this.window.close(); | ||||
|             let navigator = this.navigator.borrow().clone(); | ||||
|             if let Some(navigator) = navigator { | ||||
|                 navigator.pop(); | ||||
|             } | ||||
|         })); | ||||
| 
 | ||||
|         login_button.connect_clicked(clone!(@strong this => move |_| { | ||||
|             this.stack.set_visible_child_name("loading"); | ||||
|             this.widget.set_visible_child_name("loading"); | ||||
| 
 | ||||
|             let data = LoginData { | ||||
|                 username: this.username_entry.get_text().unwrap().to_string(), | ||||
|  | @ -65,9 +66,12 @@ impl LoginDialog { | |||
|                         cb(data); | ||||
|                     } | ||||
| 
 | ||||
|                     clone.window.close(); | ||||
|                     let navigator = clone.navigator.borrow().clone(); | ||||
|                     if let Some(navigator) = navigator { | ||||
|                         navigator.pop(); | ||||
|                     } | ||||
|                 } else { | ||||
|                     clone.stack.set_visible_child_name("content"); | ||||
|                     clone.widget.set_visible_child_name("content"); | ||||
|                     clone.info_bar.set_revealed(true); | ||||
|                 } | ||||
|             }); | ||||
|  | @ -80,9 +84,18 @@ impl LoginDialog { | |||
|     pub fn set_selected_cb<F: Fn(LoginData) -> () + 'static>(&self, cb: F) { | ||||
|         self.selected_cb.replace(Some(Box::new(cb))); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|     /// Show the login dialog.
 | ||||
|     pub fn show(&self) { | ||||
|         self.window.show(); | ||||
| impl NavigatorScreen for LoginDialog { | ||||
|     fn attach_navigator(&self, navigator: Rc<Navigator>) { | ||||
|         self.navigator.replace(Some(navigator)); | ||||
|     } | ||||
| 
 | ||||
|     fn get_widget(&self) -> gtk::Widget { | ||||
|         self.widget.clone().upcast() | ||||
|     } | ||||
| 
 | ||||
|     fn detach_navigator(&self) { | ||||
|         self.navigator.replace(None); | ||||
|     } | ||||
| } | ||||
|  |  | |||
|  | @ -1,5 +1,6 @@ | |||
| use super::{LoginDialog, ServerDialog}; | ||||
| use crate::backend::Backend; | ||||
| use crate::widgets::NavigatorWindow; | ||||
| use gettextrs::gettext; | ||||
| use glib::clone; | ||||
| use gtk::prelude::*; | ||||
|  | @ -84,13 +85,16 @@ impl Preferences { | |||
|         })); | ||||
| 
 | ||||
|         login_button.connect_clicked(clone!(@strong this => move |_| { | ||||
|             let dialog = LoginDialog::new(this.backend.clone(), &this.window); | ||||
|             let dialog = LoginDialog::new(this.backend.clone()); | ||||
| 
 | ||||
|             dialog.set_selected_cb(clone!(@strong this => move |data| { | ||||
|                 this.login_row.set_subtitle(Some(&data.username)); | ||||
|             })); | ||||
| 
 | ||||
|             dialog.show(); | ||||
| 
 | ||||
|             let window = NavigatorWindow::new(dialog); | ||||
|             window.set_transient_for(&this.window); | ||||
|             window.show(); | ||||
|         })); | ||||
| 
 | ||||
|         // Initialize
 | ||||
|  |  | |||
|  | @ -35,6 +35,12 @@ impl NavigatorWindow { | |||
|         this | ||||
|     } | ||||
| 
 | ||||
|     /// Make the wrapped window transient. This will make the window modal.
 | ||||
|     pub fn set_transient_for<W: IsA<gtk::Window>>(&self, window: &W) { | ||||
|         self.window.set_modal(true); | ||||
|         self.window.set_transient_for(Some(window)); | ||||
|     } | ||||
| 
 | ||||
|     /// Show the navigator window.
 | ||||
|     pub fn show(&self) { | ||||
|         self.window.show(); | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Elias Projahn
						Elias Projahn