/* -*- Mode: Vala; indent-tabs-mode: nil; tab-width: 4 -*- * * Copyright (C) 2012 Canonical Ltd * Copyright (C) 2015-2017 Mike Gabriel * Copyright (C) 2023 Robert Tari * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License version 3 as * published by the Free Software Foundation. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . * * Authors: Robert Ancell * Michael Terry * Scott Sweeny * Mike Gabriel * Robert Tari */ private const int MAX_FIELD_SIZE = 200; public int _scale_factor = 1; private int get_grid_offset (int size) { return (int) (size % grid_size) / 2; } [DBus (name="com.lomiri.LomiriGreeter.List")] public class ListDBusInterface : Object { private GreeterList list; public ListDBusInterface (GreeterList list) { this.list = list; this.list.entry_selected.connect ((name) => { entry_selected (name); }); } public string get_active_entry () throws GLib.DBusError, GLib.IOError { return list.get_active_entry (); } public void set_active_entry (string entry_name) throws GLib.DBusError, GLib.IOError { list.set_active_entry (entry_name); } public signal void entry_selected (string entry_name); } public abstract class GreeterList : FadableBox { public Background background { get; construct; } public MenuBar menubar { get; construct; } public PromptBox? selected_entry { get; private set; default = null; } public bool start_scrolling { get; set; default = true; } protected string greeter_authenticating_user; protected ArcticaGreeter greeter; protected bool _always_show_manual = false; public bool always_show_manual { get { return _always_show_manual; } set { _always_show_manual = value; if (value) add_manual_entry (); else if (have_entries ()) remove_entry ("*other"); } } protected List entries = null; private ListDBusInterface dbus_object; private double scroll_target_location; private double scroll_start_location; private double scroll_location; private double scroll_direction; private AnimateTimer scroll_timer; private Gtk.Fixed fixed; public DashBox greeter_box; private int cached_box_height = -1; protected enum Mode { ENTRY, SCROLLING, } protected Mode mode = Mode.ENTRY; public const int BORDER = 4; public const int BOX_WIDTH = 9; /* in grid_size blocks */ public const int DEFAULT_BOX_HEIGHT = 3; /* in grid_size blocks */ private uint n_above = 4; private uint n_below = 4; private int box_x { get { return 0; } } private int box_y { get { /* First, get grid row number as if menubar weren't there */ var row = (ArcticaGreeter.MENUBAR_HEIGHT + get_allocated_height ()) / grid_size; row = row - DEFAULT_BOX_HEIGHT; /* and no default dash box */ row = row / 2; /* and in the middle */ /* Now calculate y pixel spot keeping in mind menubar's allocation */ return row * grid_size - ArcticaGreeter.MENUBAR_HEIGHT; } } public signal void entry_selected (string? name); public signal void entry_displayed_start (); public signal void entry_displayed_done (); protected virtual string? get_selected_id () { if (selected_entry == null) return null; return selected_entry.id; } private string? _manual_name = null; public string? manual_name { get { return _manual_name; } set { _manual_name = value; if (find_entry ("*other") != null) add_manual_entry (); } } private PromptBox _scrolling_entry = null; private PromptBox scrolling_entry { get { return _scrolling_entry; } set { /* When we swap out a scrolling entry, make sure to hide its * image button, else it will appear in the tab chain. */ if (_scrolling_entry != null) _scrolling_entry.set_options_image (null, null); _scrolling_entry = value; } } internal GreeterList (Background bg, MenuBar mb) { Object (background: bg, menubar: mb); } construct { can_focus = false; visible_window = false; _scale_factor = get_scale_factor (); fixed = new Gtk.Fixed (); fixed.show (); add (fixed); greeter_box = new DashBox (background); greeter_box.notify["base-alpha"].connect (() => { queue_draw (); }); greeter_box.show (); greeter_box.size_allocate.connect (greeter_box_size_allocate_cb); add_with_class (greeter_box); scroll_timer = new AnimateTimer (AnimateTimer.ease_out_quint, AnimateTimer.FAST); scroll_timer.animate.connect (animate_scrolling); greeter = new ArcticaGreeter(); if (!greeter.test_mode) { try { Bus.get.begin (BusType.SESSION, null, on_bus_acquired); } catch (IOError e) { debug ("Error getting session bus: %s", e.message); } } } private void on_bus_acquired (Object? obj, AsyncResult res) { try { var conn = Bus.get.end (res); this.dbus_object = new ListDBusInterface (this); conn.register_object ("/list", this.dbus_object); } catch (IOError e) { debug ("Error registering user list dbus object: %s", e.message); } } public enum ScrollTarget { START, END, UP, DOWN, } public override void get_preferred_width (out int min, out int nat) { min = BOX_WIDTH * grid_size; nat = BOX_WIDTH * grid_size; } public override void get_preferred_height (out int min, out int nat) { base.get_preferred_height (out min, out nat); min = 0; } public void cancel_authentication () { var greeter = new ArcticaGreeter (); greeter.cancel_authentication (); entry_selected (selected_entry.id); } public void scroll (ScrollTarget target) { if (!sensitive) return; switch (target) { case ScrollTarget.START: select_entry (entries.nth_data (0), -1.0); break; case ScrollTarget.END: select_entry (entries.nth_data (entries.length () - 1), 1.0); break; case ScrollTarget.UP: var index = entries.index (selected_entry) - 1; if (index < 0) index = 0; select_entry (entries.nth_data (index), -1.0); break; case ScrollTarget.DOWN: var index = entries.index (selected_entry) + 1; if (index >= (int) entries.length ()) index = (int) entries.length () - 1; select_entry (entries.nth_data (index), 1.0); break; } } protected void add_with_class (Gtk.Widget widget) { fixed.add (widget); ArcticaGreeter.add_style_class (widget); } protected void redraw_greeter_box () { Gtk.Allocation allocation; greeter_box.get_allocation (out allocation); queue_draw_area (allocation.x, allocation.y, allocation.width, allocation.height); } public void show_message (string text, bool is_error = false) { if (will_clear) { selected_entry.clear (); will_clear = false; } selected_entry.add_message (text, is_error); redraw_greeter_box (); } public DashEntry add_prompt (string text, bool secret = false) { if (will_clear) { selected_entry.clear (); will_clear = false; } string accessible_text = null; if (selected_entry != null && selected_entry.label != null) { if (secret) accessible_text = _("Enter password for %s").printf (selected_entry.label); else accessible_text = _("Enter your username"); } var prompt = selected_entry.add_prompt (text, accessible_text, secret); if (mode != Mode.SCROLLING) selected_entry.show_prompts (); focus_prompt (); redraw_greeter_box (); return prompt; } public Gtk.ComboBox add_combo (GenericArray texts, bool read_only) { if (will_clear) { selected_entry.clear (); will_clear = false; } var combo = selected_entry.add_combo (texts, read_only); focus_prompt (); redraw_greeter_box (); return combo; } public override void grab_focus () { focus_prompt (); } public virtual void focus_prompt () { selected_entry.sensitive = true; selected_entry.grab_focus (); } public abstract void show_authenticated (bool successful = true); protected PromptBox? find_entry (string id) { foreach (var entry in entries) { if (entry.id == id) return entry; } return null; } protected static int compare_entry (PromptBox a, PromptBox b) { if (a.id.has_prefix ("*") || b.id.has_prefix ("*")) { /* Special entries go after normal ones */ if (!a.id.has_prefix ("*")) return -1; if (!b.id.has_prefix ("*")) return 1; /* Manual comes before guest */ if (a.id == "*other") return -1; if (a.id == "*guest") return 1; } /* Alphabetical by label */ return a.label.ascii_casecmp (b.label); } protected bool have_entries () { foreach (var e in entries) { if (e.id != "*other") return true; } return false; } protected virtual void insert_entry (PromptBox entry) { entries.insert_sorted (entry, compare_entry); } protected abstract void add_manual_entry (); protected void add_entry (PromptBox entry) { entry.expand = true; entry.set_size_request (grid_size * BOX_WIDTH - BORDER * 2, -1); add_with_class (entry); insert_entry (entry); entry.name_clicked.connect (entry_clicked_cb); if (selected_entry == null) select_entry (entry, 1.0); else select_entry (selected_entry, 1.0); move_names (); } public string get_active_entry () { string entry = ""; if (selected_entry != null && selected_entry.id != null) entry = selected_entry.id; return entry; } public void set_active_entry (string ?name) { var e = find_entry (name); if (e != null) { var direction = 1.0; if (selected_entry != null && entries.index (selected_entry) > entries.index (e)) { direction = -1.0; } select_entry (e, direction); } } public void set_active_first_entry_with_prefix (string prefix) { foreach (var e in entries) { if (e.id.has_prefix (prefix)) { select_entry (e, 1.0); break; } } } public void remove_entry (string? name) { remove_entry_by_entry (find_entry (name)); } public void remove_entries_with_prefix (string prefix) { int i = 0; while (i < entries.length ()) { PromptBox e = entries.nth_data (i); if (e.id.has_prefix (prefix)) remove_entry_by_entry (e); else i++; } } public void remove_entry_by_entry (PromptBox? entry) { if (entry == null) return; var index = entries.index (entry); /* Select another entry if the selected one was removed */ if (entry == selected_entry) { if (index > 0) { index--; if (entries.nth_data (index) != null) select_entry (entries.nth_data (index), -1.0); } else if (index < entries.length () -1 ) { index++; if (entries.nth_data (index) != null) select_entry (entries.nth_data (index), +1.0); } } entries.remove (entry); entry.destroy (); /* Show a manual login if no users and no remote login entry */ var greeter = new ArcticaGreeter (); if (!have_entries () && !greeter.show_remote_login_hint ()) add_manual_entry (); queue_draw (); } protected int get_greeter_box_height () { int height; greeter_box.get_preferred_height (null, out height); return height; } protected int get_greeter_box_height_grids () { int height = get_greeter_box_height (); return height / grid_size + 1; /* +1 because we'll be slightly under due to BORDER */ } protected int get_greeter_box_x () { return box_x + BORDER; } protected int get_greeter_box_y () { return box_y + BORDER; } protected virtual int get_position_y (double position) { // Most position heights are just the grid height. Except for the // greeter box itself. int box_height = get_greeter_box_height_grids () * grid_size; double offset; if (position < 0) offset = position * grid_size; else if (position < 1) offset = position * box_height; else offset = (position - 1) * grid_size + box_height; return box_y + (int)Math.round(offset); } private void move_entry (PromptBox entry, double position) { var alpha = 1.0; if (position < 0) alpha = 1.0 + position / (n_above + 1); else alpha = 1.0 - position / (n_below + 1); entry.set_alpha (alpha); /* Some entry types may care where they are (e.g. wifi prompt) */ entry.position = position; Gtk.Allocation allocation; get_allocation (out allocation); var child_allocation = Gtk.Allocation (); child_allocation.width = grid_size * BOX_WIDTH - BORDER * 2; entry.get_preferred_height_for_width (child_allocation.width, null, out child_allocation.height); child_allocation.x = allocation.x + get_greeter_box_x (); child_allocation.y = allocation.y + get_position_y (position); fixed.move (entry, child_allocation.x, child_allocation.y); entry.size_allocate (child_allocation); } public void greeter_box_size_allocate_cb (Gtk.Allocation allocation) { /* If the greeter box allocation changes while not moving fix the entries position */ if (scrolling_entry == null && allocation.height != cached_box_height) { /* We run in idle because it's kind of a recursive loop and * ends up positioning the entries in the wrong place if we try * to do it during an existing allocation. */ Idle.add (() => { move_names (); return false; }); } cached_box_height = allocation.height; } public void move_names () { var index = 0; foreach (var entry in entries) { var position = index - scroll_location; /* Draw entries above, in and below the box */ if (position > -1 * (int)(n_above + 1) && position < n_below + 1) { move_entry (entry, position); // Sometimes we will be overlayed by another widget like the // session chooser. In such cases, don't try to show ourselves var is_hidden = (position == 0 && greeter_box.has_base && greeter_box.base_alpha == 0.0); if (!is_hidden) entry.show (); } else entry.hide (); index++; } queue_draw (); } private void animate_scrolling (double progress) { /* Total height of list */ var h = entries.length (); /* How far we have to go in total, either up or down with wrapping */ var distance = scroll_target_location - scroll_start_location; if (scroll_direction * distance < 0) distance += scroll_direction * h; /* How far we've gone so far */ distance *= progress; /* Go that far and wrap around */ scroll_location = scroll_start_location + distance; if (scroll_location > h) scroll_location -= h; if (scroll_location < 0) scroll_location += h; move_names (); if (progress >= 0.975 && !greeter_box.has_base) { setup_prompt_box (); entry_displayed_start (); } /* Stop when we get there */ if (progress >= 1.0) finished_scrolling (); } private void finished_scrolling () { scrolling_entry = null; selected_entry.show_prompts (); /* set prompts to be visible immediately */ focus_prompt (); entry_displayed_done (); mode = Mode.ENTRY; #if HAVE_GTK_3_20_0 queue_allocate (); #endif } protected void select_entry (PromptBox entry, double direction, bool do_scroll = true) { if (!get_realized ()) { /* Just note it for the future if we haven't been realized yet */ selected_entry = entry; return; } if (scroll_target_location != entries.index (entry)) { var new_target = entries.index (entry); var new_direction = direction; var new_start = scroll_location; if (scroll_location != new_target && do_scroll) { var new_distance = new_direction * (new_target - new_start); /* Base rate is 350 (250 + 100). If we find ourselves going further, slow down animation */ scroll_timer.reset (250 + int.min ((int)(100 * (Math.fabs (new_distance))), 500)); mode = Mode.SCROLLING; } scrolling_entry = selected_entry; scroll_target_location = new_target; scroll_direction = new_direction; scroll_start_location = new_start; } if (selected_entry != entry) { greeter_box.set_base (null); if (selected_entry != null) selected_entry.clear (); selected_entry = entry; entry_selected (selected_entry.id); if (mode == Mode.ENTRY) { /* don't need to move, but make sure we trigger the same side effects */ setup_prompt_box (); scroll_timer.reset (0); } } } protected virtual void setup_prompt_box (bool fade = true) { greeter_box.set_base (selected_entry); selected_entry.add_static_prompts (); if (fade) selected_entry.fade_in_prompts (); else selected_entry.show_prompts (); } public override void realize () { base.realize (); /* NOTE: This is going to cause the entry_selected signal to be emitted even if selected_entry has not changed */ var saved_entry = selected_entry; selected_entry = null; select_entry (saved_entry, 1, start_scrolling); move_names (); } private void allocate_greeter_box () { Gtk.Allocation allocation; get_allocation (out allocation); var child_allocation = Gtk.Allocation (); greeter_box.get_preferred_width (null, out child_allocation.width); greeter_box.get_preferred_height (null, out child_allocation.height); child_allocation.x = allocation.x + get_greeter_box_x (); child_allocation.y = allocation.y + get_greeter_box_y (); fixed.move (greeter_box, child_allocation.x, child_allocation.y); greeter_box.size_allocate (child_allocation); foreach (var entry in entries) { entry.set_zone (greeter_box); } } public override void size_allocate (Gtk.Allocation allocation) { base.size_allocate (allocation); if (!get_realized ()) return; allocate_greeter_box (); move_names (); } public override bool draw (Cairo.Context c) { c.push_group (); c.save (); fixed.propagate_draw (greeter_box, c); /* Always full alpha */ c.restore (); if (greeter_box.base_alpha != 0.0) { c.save (); c.push_group (); c.rectangle (get_greeter_box_x (), get_greeter_box_y () - n_above * grid_size, grid_size * BOX_WIDTH - BORDER * 2, grid_size * (n_above + n_below + get_greeter_box_height_grids ())); c.clip (); foreach (var child in fixed.get_children ()) { if (child != greeter_box) fixed.propagate_draw (child, c); } c.pop_group_to_source (); c.paint_with_alpha (greeter_box.base_alpha); c.restore (); } c.pop_group_to_source (); c.paint_with_alpha (fade_tracker.alpha); return false; } private void entry_clicked_cb (PromptBox entry) { if (mode != Mode.ENTRY) return; var index = entries.index (entry); var position = index - scroll_location; if (position < 0.0) select_entry (entry, -1.0); else if (position >= 1.0) select_entry (entry, 1.0); } /* Not all subclasses are going to be interested in talking to lightdm, but for those that are, make it easy. */ protected bool will_clear = false; protected bool prompted = false; protected bool unacknowledged_messages = false; protected void connect_to_lightdm () { var greeter = new ArcticaGreeter (); greeter.show_message.connect (show_message_cb); greeter.show_prompt.connect (show_prompt_cb); greeter.authentication_complete.connect (authentication_complete_cb); } protected void show_message_cb (string text, LightDM.MessageType type) { unacknowledged_messages = true; show_message (text, type == LightDM.MessageType.ERROR); } protected virtual void show_prompt_cb (string text, LightDM.PromptType type) { var show_label_if_configured = false; /* Notify the greeter on what user has been logged */ if (get_selected_id () == "*other" && manual_name == null) { var greeter = new ArcticaGreeter (); if (greeter.test_mode) manual_name = test_username; else manual_name = greeter.authentication_user(); } prompted = true; if (text == "Password: ") { text = _("Password:"); show_label_if_configured = true; } if (text == "login:") { text = _("Username:"); show_label_if_configured = true; } if ((AGSettings.get_boolean(AGSettings.KEY_SHOW_LOGIN_LABELS)) && (show_label_if_configured)) { show_message(text); } var entry = add_prompt (text, type == LightDM.PromptType.SECRET); /* Limit the number of characters in case a cat is sitting on the keyboard... */ entry.max_length = MAX_FIELD_SIZE; } protected virtual void authentication_complete_cb () { /* Not the best of the solutions but seems the asynchrony * when talking to lightdm process means that you can * go to the "Guest" account, start authenticating as guest * keep moving down to some of the remote servers * and the answer will come after that, and even calling * greeter.cancel_authentication won't help * so basically i'm just ignoring any authentication callback * if we are not in the same place in the list as we were * when we called greeter.authenticate* */ if (greeter_authenticating_user != selected_entry.id) return; bool is_authenticated; var greeter = new ArcticaGreeter (); if (greeter.test_mode) is_authenticated = test_is_authenticated; else is_authenticated = greeter.is_authenticated(); if (is_authenticated) { /* Login immediately if prompted and user has acknowledged all messages */ if (prompted && !unacknowledged_messages) { login_complete (); if (greeter.test_mode) start_session (); else { if (background.alpha == 1.0) start_session (); else background.notify["alpha"].connect (background_loaded_cb); } } else { prompted = true; show_authenticated (); } } else { if (prompted) { /* Show an error if one wasn't provided */ if (will_clear) show_message (_("Invalid username or password, please try again"), true); selected_entry.reset_spinners (); /* Restart authentication */ start_authentication (); } else { /* Show an error if one wasn't provided */ if (!selected_entry.has_errors) show_message (_("Failed to authenticate"), true); /* Stop authentication */ show_authenticated (false); } } } protected virtual void start_authentication () { prompted = false; unacknowledged_messages = false; /* Reset manual username */ manual_name = null; will_clear = false; greeter_authenticating_user = get_selected_id (); var greeter = new ArcticaGreeter (); if (greeter.test_mode) test_start_authentication (); else { if (get_selected_id () == "*other") greeter.authenticate (); else if (get_selected_id () == "*guest") greeter.authenticate_as_guest (); else greeter.authenticate (get_selected_id ()); } } private void background_loaded_cb (ParamSpec pspec) { if (background.alpha == 1.0) { background.notify["alpha"].disconnect (background_loaded_cb); start_session (); } } private void start_session () { var greeter = new ArcticaGreeter (); if (!greeter.start_session (get_lightdm_session (), background)) { show_message (_("Failed to start session"), true); start_authentication (); return; } /* Set the background */ background.queue_draw (); } public void login_complete () { sensitive = false; selected_entry.clear (); selected_entry.add_message (_("Logging in…"), false); redraw_greeter_box (); } protected virtual string get_lightdm_session () { greeter = new ArcticaGreeter(); return greeter.get_default_session (); } /* Testing code below this */ protected string? test_username = null; protected bool test_is_authenticated = false; protected virtual void test_start_authentication () { } }