using Gee;
using Gtk;
-class IdentityManagerView : Window {
+public class IdentityManagerView : Window {
private const int WINDOW_WIDTH = 400;
private const int WINDOW_HEIGHT = 500;
protected IdentityManagerApp parent_app;
private UIManager ui_manager = new UIManager();
private Entry search_entry;
private VBox vbox_right;
+ private VBox login_vbox;
+ private VBox services_vbox;
private CustomVBox custom_vbox;
private VBox services_internal_vbox;
+ private Entry issuer_entry;
private Entry username_entry;
private Entry password_entry;
+ private Label prompting_service;
+ private Label no_identity_title;
+ private CheckButton remember_checkbutton;
+ private Button update_password_button;
private ListStore* listmodel;
private TreeModelFilter filter;
public IdentityManagerModel identities_manager;
- private SList<IdCard> candidates;
+ private unowned SList<IdCard> candidates;
- private IdCard default_id_card;
public GLib.Queue<IdentityRequest> request_queue;
private HashTable<Gtk.Button, string> service_button_map;
identities_manager = parent_app.model;
request_queue = new GLib.Queue<IdentityRequest>();
service_button_map = new HashTable<Gtk.Button, string> (direct_hash, direct_equal);
- this.title = "Moonshoot";
+ this.title = "Moonshot Identity Selector";
this.set_position (WindowPosition.CENTER);
set_default_size (WINDOW_WIDTH, WINDOW_HEIGHT);
build_ui();
setup_list_model();
- load_id_cards();
-#if OS_MACOS
- osxApp = app.osxApp;
-#endif
+ load_id_cards();
connect_signals();
}
load_id_cards();
}
- public void add_candidate (IdCard idcard)
- {
- candidates.append (idcard);
- }
-
private bool visible_func (TreeModel model, TreeIter iter)
{
IdCard id_card;
return false;
}
+ private void update_password_cb()
+ {
+ if (this.custom_vbox.current_idcard != null) {
+ var identity = this.custom_vbox.current_idcard.id_card;
+ var dialog = new AddPasswordDialog(identity, null);
+ var result = dialog.run ();
+
+ switch (result) {
+ case ResponseType.OK:
+ identity.password = dialog.password;
+ identity.store_password = dialog.remember;
+ if (dialog.remember)
+ identity.temporary = false;
+ identity = identities_manager.update_card(identity);
+ break;
+ default:
+ break;
+ }
+ dialog.destroy ();
+ }
+ }
+
private void load_id_cards () {
+ string current_idcard_nai = null;
+ if (this.custom_vbox.current_idcard != null) {
+ current_idcard_nai = custom_vbox.current_idcard.id_card.nai;
+ custom_vbox.current_idcard = null;
+ }
var children = this.custom_vbox.get_children ();
foreach (var id_card_widget in children) {
remove_id_card_widget((IdCardWidget)id_card_widget);
}
-
- this.default_id_card = null;
+ this.listmodel->clear();
LinkedList<IdCard> card_list = identities_manager.get_card_list() ;
if (card_list == null) {
return;
foreach (IdCard id_card in card_list) {
add_id_card_data (id_card);
- add_id_card_widget (id_card);
- }
-
- if (card_list.size > 0){
- this.default_id_card = card_list.first();
+ IdCardWidget id_card_widget = add_id_card_widget (id_card);
+ if (id_card_widget.id_card.nai == current_idcard_nai) {
+ fill_details(id_card_widget);
+ id_card_widget.expand();
+ }
}
+ if (custom_vbox.current_idcard == null)
+ fill_details(null);
}
- private void fill_details (IdCardWidget id_card_widget)
+ private void fill_details (IdCardWidget? id_card_widget)
{
- var id_card = id_card_widget.id_card;
- this.username_entry.set_text (id_card.username);
- this.password_entry.set_text (id_card.password ?? "");
+ var vr_children = this.vbox_right.get_children();
+ foreach (var vr_child in vr_children)
+ this.vbox_right.remove(vr_child);
+ if (id_card_widget != null) {
+ var id_card = id_card_widget.id_card;
+ if (id_card.display_name == IdCard.NO_IDENTITY) {
+ this.vbox_right.pack_start(no_identity_title, false, true, 0);
+ } else {
+ this.issuer_entry.set_text (id_card.issuer);
+ this.username_entry.set_text (id_card.username);
+ this.password_entry.set_text (id_card.password ?? "");
+ this.vbox_right.pack_start(login_vbox, false, true, 0);
+ this.remember_checkbutton.active = id_card.store_password;
+ }
+ this.vbox_right.pack_start (services_vbox, false, true, 0);
- var children = this.services_internal_vbox.get_children ();
- foreach (var hbox in children)
- hbox.destroy();
- fill_services_vbox (id_card_widget.id_card);
-// identities_manager.store_id_cards();
+ var children = this.services_internal_vbox.get_children ();
+ foreach (var hbox in children)
+ services_internal_vbox.remove(hbox);
+ fill_services_vbox (id_card_widget.id_card);
+ }
}
private void show_details (IdCard id_card)
id_card.display_name = dialog.display_name;
id_card.issuer = dialog.issuer;
- if (id_card.issuer == "")
- id_card.issuer = "Issuer";
id_card.username = dialog.username;
id_card.password = dialog.password;
+ id_card.store_password = dialog.store_password;
id_card.services = {};
- id_card.set_data("pixbuf", find_icon ("avatar-default", 48));
return id_card;
}
TreeIter iter;
Gdk.Pixbuf pixbuf;
this.listmodel->append (out iter);
- pixbuf = id_card.get_data("pixbuf");
+ pixbuf = get_pixbuf(id_card);
listmodel->set (iter,
Columns.IDCARD_COL, id_card,
Columns.LOGO_COL, pixbuf,
}
}
- private void add_id_card_widget (IdCard id_card)
+ private IdCardWidget add_id_card_widget (IdCard id_card)
{
var id_card_widget = new IdCardWidget (id_card);
this.custom_vbox.add_id_card_widget (id_card_widget);
id_card_widget.send_id.connect ((w) => send_identity_cb (w.id_card));
id_card_widget.expanded.connect (this.custom_vbox.receive_expanded_event);
id_card_widget.expanded.connect (fill_details);
+ return id_card_widget;
}
- /* This method finds a valid display name */
- public bool display_name_is_valid (string name,
- out string? candidate)
- {
- foreach (IdCard id_card in identities_manager.get_card_list())
- {
- if (id_card.display_name == name)
- {
- if (&candidate != null)
- {
- for (int i=0; i<1000; i++)
- {
- string tmp = "%s %d".printf (name, i);
- if (display_name_is_valid (tmp, null))
- {
- candidate = tmp;
- break;
- }
- }
- }
- return false;
- }
- }
-
- return true;
- }
-
- public void insert_id_card (IdCard id_card)
- {
- string candidate;
-
- if (!display_name_is_valid (id_card.display_name, out candidate))
- {
- id_card.display_name = candidate;
- }
-
- this.identities_manager.add_card(id_card);
- }
-
- public bool add_identity (IdCard id_card)
+ public bool add_identity (IdCard id_card, bool force_flat_file_store)
{
#if OS_MACOS
/*
* so for now we will install silently
*/
var ret = Gtk.ResponseType.YES;
- public OSXApplication osxApp;
#else
-
- var dialog = new Gtk.MessageDialog (this,
+ Gtk.MessageDialog dialog;
+ IdCard? prev_id = identities_manager.find_id_card(id_card.nai, force_flat_file_store);
+ if (prev_id!=null) {
+ int flags = prev_id.Compare(id_card);
+ if (flags == 0) {
+ return false; // no changes, no need to update
+ } else if ((flags & (1<<IdCard.DiffFlags.DISPLAY_NAME)) != 0) {
+ dialog = new Gtk.MessageDialog (this,
+ Gtk.DialogFlags.DESTROY_WITH_PARENT,
+ Gtk.MessageType.QUESTION,
+ Gtk.ButtonsType.YES_NO,
+ _("Would you like to replace ID Card '%s' using nai '%s' with the new ID Card '%s'?"),
+ prev_id.display_name,
+ prev_id.nai,
+ id_card.display_name);
+ } else {
+ dialog = new Gtk.MessageDialog (this,
+ Gtk.DialogFlags.DESTROY_WITH_PARENT,
+ Gtk.MessageType.QUESTION,
+ Gtk.ButtonsType.YES_NO,
+ _("Would you like to update ID Card '%s' using nai '%s'?"),
+ id_card.display_name,
+ id_card.nai);
+ }
+ } else {
+ dialog = new Gtk.MessageDialog (this,
Gtk.DialogFlags.DESTROY_WITH_PARENT,
Gtk.MessageType.QUESTION,
Gtk.ButtonsType.YES_NO,
_("Would you like to add '%s' ID Card to the ID Card Organizer?"),
id_card.display_name);
-
+ }
var ret = dialog.run ();
dialog.destroy ();
#endif
if (ret == Gtk.ResponseType.YES) {
- id_card.set_data ("pixbuf", find_icon ("avatar-default", 48));
- this.insert_id_card (id_card);
+ this.identities_manager.add_card (id_card, force_flat_file_store);
return true;
}
private void add_identity_manual_cb ()
{
var dialog = new AddIdentityDialog ();
- var result = dialog.run ();
+ int result = ResponseType.CANCEL;
+ while (!dialog.complete)
+ result = dialog.run ();
switch (result) {
case ResponseType.OK:
- insert_id_card (get_id_card_data (dialog));
+ this.identities_manager.add_card (get_id_card_data (dialog), false);
break;
default:
break;
var children = this.custom_vbox.get_children ();
foreach (var id_card_widget in children)
- id_card_widget.destroy();
+ remove_id_card_widget((IdCardWidget )id_card_widget); //id_card_widget.destroy();
if (filter.get_iter_first (out iter))
{
dialog.destroy ();
}
- public void select_identity (IdentityRequest request)
+ public void set_prompting_service(string service)
{
- IdCard identity = null;
-
- this.request_queue.push_tail (request);
-
- if (custom_vbox.current_idcard != null &&
- custom_vbox.current_idcard.send_button != null)
- custom_vbox.current_idcard.send_button.set_sensitive (true);
+ prompting_service.set_label( _("Identity requested for service: %s").printf(service) );
+ }
- if (request.select_default)
- {
- identity = default_id_card;
+ public void queue_identity_request(IdentityRequest request)
+ {
+ if (this.request_queue.is_empty())
+ { /* setup widgets */
+ candidates = request.candidates;
+ filter.refilter();
+ redraw_id_card_widgets ();
+ set_prompting_service(request.service);
+ show ();
}
+ this.request_queue.push_tail (request);
+ }
- if (identity == null)
- {
- bool has_nai = request.nai != null && request.nai != "";
- bool has_srv = request.service != null && request.service != "";
- bool confirm = false;
- IdCard nai_provided = null;
-
- foreach (IdCard id in identities_manager.get_card_list())
- {
- /* If NAI matches we add id card to the candidate list */
- if (has_nai && request.nai == id.nai)
- {
- nai_provided = id;
- add_candidate (id);
- continue;
- }
-
- /* If any service matches we add id card to the candidate list */
- if (has_srv)
- {
- foreach (string srv in id.services)
- {
- if (request.service == srv)
- {
- add_candidate (id);
- continue;
- }
- }
- }
- }
-
- /* If more than one candidate we dissasociate service from all ids */
- if (has_srv && candidates.length() > 1)
- {
- foreach (IdCard id in candidates)
- {
- int i = 0;
- SList<string> services_list = null;
- bool has_service = false;
-
- foreach (string srv in id.services)
- {
- if (srv == request.service)
- {
- has_service = true;
- continue;
- }
- services_list.append (srv);
- }
-
- if (!has_service)
- continue;
-
- if (services_list.length () == 0)
- {
- id.services = {};
- continue;
- }
-
- string[] services = new string[services_list.length ()];
- foreach (string srv in services_list)
- {
- services[i] = srv;
- i++;
- }
-
- id.services = services;
- }
- }
-
-// identities_manager.store_id_cards ();
-
- /* If there are no candidates we use the service matching rules */
- if (candidates.length () == 0)
- {
- foreach (IdCard id in identities_manager.get_card_list())
- {
- foreach (Rule rule in id.rules)
- {
- if (!match_service_pattern (request.service, rule.pattern))
- continue;
-
- candidates.append (id);
-
- if (rule.always_confirm == "true")
- confirm = true;
- }
- }
- }
-
- if (candidates.length () > 1)
- {
- if (has_nai && nai_provided != null)
- {
- identity = nai_provided;
- confirm = false;
+ public IdCard check_add_password(IdCard identity, IdentityRequest request, IdentityManagerModel model)
+ {
+ IdCard retval = identity;
+ bool idcard_has_pw = (identity.password != null) && (identity.password != "");
+ bool request_has_pw = (request.password != null) && (request.password != "");
+ if ((!idcard_has_pw) && (!identity.IsNoIdentity())) {
+ if (request_has_pw) {
+ identity.password = request.password;
+ retval = model.update_card(identity);
+ } else {
+ var dialog = new AddPasswordDialog (identity, request);
+ var result = dialog.run ();
+
+ switch (result) {
+ case ResponseType.OK:
+ identity.password = dialog.password;
+ identity.store_password = dialog.remember;
+ if (dialog.remember)
+ identity.temporary = false;
+ retval = model.update_card(identity);
+ break;
+ default:
+ identity = null;
+ break;
}
- else
- confirm = true;
- }
- else
- identity = candidates.nth_data (0);
-
- /* TODO: If candidate list empty return fail */
-
- if (confirm)
- {
- filter.refilter();
- redraw_id_card_widgets ();
- show ();
- return;
+ dialog.destroy ();
}
}
- // Send back the identity (we can't directly run the
- // callback because we may be being called from a 'yield')
- Idle.add (() => { send_identity_cb (identity); return false; });
- return;
- }
-
- private bool match_service_pattern (string service, string pattern)
- {
- var pspec = new PatternSpec (pattern);
- return pspec.match_string (service);
+ return retval;
}
- public void send_identity_cb (IdCard identity)
+ public void send_identity_cb (IdCard id)
{
+ IdCard identity = id;
return_if_fail (request_queue.length > 0);
+ candidates = null;
var request = this.request_queue.pop_head ();
- bool reset_password = false;
-
- if (request.service != null && request.service != "")
- {
- bool duplicate_service = false;
-
- foreach (string service in identity.services)
- {
- if (service == request.service)
- duplicate_service = true;
- }
- if (duplicate_service == false)
- {
- string[] services = new string[identity.services.length + 1];
-
- for (int i = 0; i < identity.services.length; i++)
- services[i] = identity.services[i];
-
- services[identity.services.length] = request.service;
- identity.services = services;
-
- identities_manager.update_card (identity);
- }
- }
-
- if (identity.password == null)
+ identity = check_add_password(identity, request, identities_manager);
+ if (this.request_queue.is_empty())
{
- var dialog = new AddPasswordDialog ();
- var result = dialog.run ();
-
- switch (result) {
- case ResponseType.OK:
- identity.password = dialog.password;
- reset_password = ! dialog.remember;
- break;
- default:
- identity = null;
- break;
+ candidates = null;
+ prompting_service.set_label(_(""));
+ if (!parent_app.explicitly_launched) {
+// The following occasionally causes the app to exit without sending the dbus
+// reply, so for now we just don't exit
+// Gtk.main_quit ();
+// just hide instead
+ this.hide();
}
-
- dialog.destroy ();
+ } else {
+ IdentityRequest next = this.request_queue.peek_head();
+ candidates = next.candidates;
+ set_prompting_service(next.service);
}
+ filter.refilter();
+ redraw_id_card_widgets ();
- if (this.request_queue.is_empty())
- Gtk.main_quit ();
-
- if (identity != null)
- this.default_id_card = identity;
+ if ((identity != null) && (!identity.IsNoIdentity()))
+ parent_app.default_id_card = identity;
request.return_identity (identity);
-
- if (reset_password)
- identity.password = null;
-
- candidates = null;
}
private void label_make_bold (Label label)
remove_button.clicked.connect ((remove_button) =>
{
+ var candidate = service_button_map.lookup (remove_button);
+ if (candidate == null)
+ return;
var dialog = new Gtk.MessageDialog (this,
Gtk.DialogFlags.DESTROY_WITH_PARENT,
Gtk.MessageType.QUESTION,
Gtk.ButtonsType.YES_NO,
_("Are you sure you want to stop '%s' ID Card from being used with %s?"),
custom_vbox.current_idcard.id_card.display_name,
- _("this service"));
+ candidate);
var ret = dialog.run();
dialog.hide();
if (ret == Gtk.ResponseType.YES)
{
IdCard idcard = custom_vbox.current_idcard.id_card;
- var candidate = service_button_map.lookup (remove_button);
-
- SList<string> services = new SList<string>();
+ if (idcard != null) {
+ SList<string> services = new SList<string>();
- foreach (string srv in idcard.services)
- {
- if (srv == candidate)
- continue;
- services.append (srv);
- }
-
- idcard.services = new string[services.length()];
- for (int j=0; j<idcard.services.length; j++)
- {
- idcard.services[j] = services.nth_data(j);
- }
+ foreach (string srv in idcard.services)
+ {
+ if (srv == candidate)
+ continue;
+ services.append (srv);
+ }
- var children = services_internal_vbox.get_children ();
- foreach (var hbox in children)
- hbox.destroy();
+ idcard.services = new string[services.length()];
+ for (int j=0; j<idcard.services.length; j++)
+ {
+ idcard.services[j] = services.nth_data(j);
+ }
- fill_services_vbox (idcard);
- custom_vbox.current_idcard.update_id_card_label ();
+ identities_manager.update_card(idcard);
+ }
}
});
string copyright = "Copyright 2011 JANET";
- const string license =
+ string license =
"""
Copyright (c) 2011, JANET(UK)
All rights reserved.
scroll.set_policy (PolicyType.NEVER, PolicyType.AUTOMATIC);
scroll.set_shadow_type (ShadowType.IN);
scroll.add_with_viewport (viewport);
+ this.prompting_service = new Label (_(""));
+ // left-align
+ prompting_service.set_alignment(0, (float )0.5);
var vbox_left = new VBox (false, 0);
vbox_left.pack_start (search_entry, false, false, 6);
vbox_left.pack_start (scroll, true, true, 0);
+ vbox_left.pack_start (prompting_service, false, false, 6);
vbox_left.set_size_request (WINDOW_WIDTH, 0);
+ this.no_identity_title = new Label (_("No Identity: Send this identity to services which should not use Moonshot"));
+ no_identity_title.set_alignment(0, (float ) 0.5);
+ no_identity_title.set_line_wrap(true);
+ no_identity_title.show();
+
var login_vbox_title = new Label (_("Login: "));
label_make_bold (login_vbox_title);
login_vbox_title.set_alignment (0, (float) 0.5);
+ var issuer_label = new Label (_("Issuer:"));
+ issuer_label.set_alignment (1, (float) 0.5);
+ this.issuer_entry = new Entry ();
+ issuer_entry.set_can_focus (false);
var username_label = new Label (_("Username:"));
username_label.set_alignment (1, (float) 0.5);
this.username_entry = new Entry ();
+ username_entry.set_can_focus (false);
var password_label = new Label (_("Password:"));
password_label.set_alignment (1, (float) 0.5);
this.password_entry = new Entry ();
password_entry.set_invisible_char ('*');
password_entry.set_visibility (false);
- var remember_checkbutton = new CheckButton.with_label (_("Remember password"));
- var login_table = new Table (3, 3, false);
+ password_entry.set_sensitive (false);
+ this.remember_checkbutton = new CheckButton.with_label (_("Remember password"));
+ remember_checkbutton.set_sensitive(false);
+ this.update_password_button = new Button.with_label (_("Update Pasword"));
+ this.update_password_button.clicked.connect(update_password_cb);
+
+ set_atk_relation (issuer_label, issuer_entry, Atk.RelationType.LABEL_FOR);
+ set_atk_relation (username_label, username_entry, Atk.RelationType.LABEL_FOR);
+ set_atk_relation (password_entry, password_entry, Atk.RelationType.LABEL_FOR);
+
+ var login_table = new Table (5, 2, false);
login_table.set_col_spacings (10);
login_table.set_row_spacings (10);
- login_table.attach_defaults (username_label, 0, 1, 0, 1);
- login_table.attach_defaults (username_entry, 1, 2, 0, 1);
- login_table.attach_defaults (password_label, 0, 1, 1, 2);
- login_table.attach_defaults (password_entry, 1, 2, 1, 2);
- login_table.attach_defaults (remember_checkbutton, 1, 2, 2, 3);
+ login_table.attach_defaults (issuer_label, 0, 1, 0, 1);
+ login_table.attach_defaults (issuer_entry, 1, 2, 0, 1);
+ login_table.attach_defaults (username_label, 0, 1, 1, 2);
+ login_table.attach_defaults (username_entry, 1, 2, 1, 2);
+ login_table.attach_defaults (password_label, 0, 1, 2, 3);
+ login_table.attach_defaults (password_entry, 1, 2, 2, 3);
+ login_table.attach_defaults (remember_checkbutton, 1, 2, 3, 4);
+ login_table.attach_defaults (update_password_button, 0, 1, 4, 5);
var login_vbox_alignment = new Alignment (0, 0, 0, 0);
login_vbox_alignment.set_padding (0, 0, 12, 0);
login_vbox_alignment.add (login_table);
- var login_vbox = new VBox (false, 6);
+ this.login_vbox = new VBox (false, 6);
login_vbox.pack_start (login_vbox_title, false, true, 0);
login_vbox.pack_start (login_vbox_alignment, false, true, 0);
services_vbox_alignment.set_padding (0, 0, 12, 0);
this.services_internal_vbox = new VBox (true, 6);
services_vbox_alignment.add (services_internal_vbox);
- var services_vbox = new VBox (false, 6);
+ this.services_vbox = new VBox (false, 6);
services_vbox.pack_start (services_vbox_title, false, true, 0);
services_vbox.pack_start (services_vbox_alignment, false, true, 0);
vbox_right.pack_start (services_vbox, false, true, 0);
var hbox = new HBox (false, 12);
- hbox.pack_start (vbox_left, true, true, 0);
- hbox.pack_start (vbox_right, false, false, 0);
+ hbox.pack_start (vbox_left, false, false, 0);
+ hbox.pack_start (vbox_right, true, true, 0);
var main_vbox = new VBox (false, 0);
main_vbox.set_border_width (12);
this.destroy.connect (Gtk.main_quit);
this.identities_manager.card_list_changed.connect(this.on_card_list_changed);
}
+
+ private static void set_atk_relation (Widget widget, Widget target_widget, Atk.RelationType relationship)
+ {
+ var atk_widget = widget.get_accessible ();
+ var atk_target_widget = target_widget.get_accessible ();
+
+ atk_widget.add_relationship (relationship, atk_target_widget);
+ }
}