chiark / gitweb /
Disobedience login window now only remembers password etc if they
[disorder] / disobedience / login.c
1 /*
2  * This file is part of DisOrder
3  * Copyright (C) 2007, 2008 Richard Kettlewell
4  *
5  * This program is free software; you can redistribute it and/or modify
6  * it under the terms of the GNU General Public License as published by
7  * the Free Software Foundation; either version 2 of the License, or
8  * (at your option) any later version.
9  *
10  * This program is distributed in the hope that it will be useful, but
11  * WITHOUT ANY WARRANTY; without even the implied warranty of
12  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13  * General Public License for more details.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program; if not, write to the Free Software
17  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
18  * USA
19  */
20 /** @file disobedience/login.c
21  * @brief Login box for Disobedience
22  *
23  * As of 2.1 we have only two buttons: Login and Cancel.
24  *
25  * If you hit Login then a login is attempted.  If it works the window
26  * disappears and the settings are saved, otherwise they are NOT saved and the
27  * window remains.
28  *
29  * It you hit Cancel then the window disappears without saving anything.
30  *
31  * TODO
32  * - cancel/close should be consistent with properties
33  */
34
35 #include "disobedience.h"
36 #include "split.h"
37 #include "filepart.h"
38 #include "client.h"
39 #include <sys/types.h>
40 #include <sys/stat.h>
41 #include <unistd.h>
42
43 /** @brief One field in the login window */
44 struct login_window_item {
45   /** @brief Description label */
46   const char *description;
47
48   /** @brief Return the current value */
49   const char *(*get)(void);
50
51   /** @brief Set a new value */
52   void (*set)(struct config *c, const char *value);
53
54   /** @brief Flags
55    * 
56    * - @ref LWI_HIDDEN - this is a password
57    */
58   unsigned flags;
59
60 };
61
62 /** @brief This is a password */
63 #define LWI_HIDDEN 0x0001
64
65 /** @brief Current login window */
66 GtkWidget *login_window;
67
68 /** @brief Set connection defaults */
69 static void default_connect(void) {
70   if(!config->connect.n) {
71     config->connect.n = 2;
72     config->connect.s = xcalloc(2, sizeof (char *));
73     config->connect.s[0] = xstrdup("localhost");
74     config->connect.s[1] = xstrdup("9999"); /* whatever */
75   }
76 }
77
78 static const char *get_hostname(void) { return config->connect.s[0]; }
79 static const char *get_service(void) { return config->connect.s[1]; }
80 static const char *get_username(void) { return config->username; }
81 static const char *get_password(void) { return config->password ? config->password : ""; }
82
83 static void set_hostname(struct config *c, const char *s) {
84   c->connect.s[0] = (char *)s;
85 }
86
87 static void set_service(struct config *c, const char *s) {
88   c->connect.s[1] = (char *)s;
89 }
90
91 static void set_username(struct config *c, const char *s) {
92   c->username = s;
93 }
94
95 static void set_password(struct config *c, const char *s) {
96   c->password = s;
97 }
98
99 /** @brief Table used to generate the form */
100 static const struct login_window_item lwis[] = {
101   { "Hostname", get_hostname, set_hostname, 0 },
102   { "Service", get_service, set_service, 0 },
103   { "User name", get_username, set_username, 0 },
104   { "Password", get_password, set_password, LWI_HIDDEN },
105 };
106 #define NLWIS (sizeof lwis / sizeof *lwis)
107
108 static GtkWidget *lwi_entry[NLWIS];
109
110 static void login_update_config(struct config *c) {
111   size_t n;
112
113   if(c->connect.n < 2) {
114     c->connect.n = 2;
115     c->connect.s = xcalloc(2, sizeof (char *));
116   }
117   for(n = 0; n < NLWIS; ++n)
118     lwis[n].set(c, xstrdup(gtk_entry_get_text(GTK_ENTRY(lwi_entry[n]))));
119 }
120
121 /** @brief Save current login details */
122 static void login_save_config(void) {
123   char *path = config_userconf(0, 0), *tmp;
124   FILE *fp;
125
126   byte_xasprintf(&tmp, "%s.tmp", path);
127   /* Make sure the directory exists; don't care if it already exists. */
128   mkdir(d_dirname(tmp), 02700);
129   /* Write out the file */
130   if(!(fp = fopen(tmp, "w"))) {
131     fpopup_msg(GTK_MESSAGE_ERROR, "error opening %s: %s",
132                tmp, strerror(errno));
133     goto done;
134   }
135   if(fprintf(fp, "username %s\n"
136              "password %s\n"
137              "connect %s %s\n",
138              quoteutf8(config->username),
139              quoteutf8(config->password),
140              quoteutf8(config->connect.s[0]),
141              quoteutf8(config->connect.s[1])) < 0) {
142     fpopup_msg(GTK_MESSAGE_ERROR, "error writing to %s: %s",
143                tmp, strerror(errno));
144     fclose(fp);
145     goto done;
146   }
147   if(fclose(fp) < 0) {
148     fpopup_msg(GTK_MESSAGE_ERROR, "error closing %s: %s",
149                tmp, strerror(errno));
150     goto done;
151   }
152   /* Rename into place */
153   if(rename(tmp, path) < 0) {
154     fpopup_msg(GTK_MESSAGE_ERROR, "error renaming %s: %s",
155                tmp, strerror(errno));
156     goto done;
157   }
158 done:
159   ;
160 }
161
162 /** @brief User pressed OK in login window */
163 static void login_ok(GtkButton attribute((unused)) *button,
164                      gpointer attribute((unused)) userdata) {
165   disorder_client *c;
166   struct config *tmpconfig = xmalloc(sizeof *tmpconfig);
167   
168   /* Copy the new config into @ref config */
169   login_update_config(tmpconfig);
170   /* Attempt a login with the new details */
171   c = disorder_new(0);
172   if(!disorder_connect_generic(tmpconfig, c,
173                                tmpconfig->username, tmpconfig->password,
174                                NULL/*cookie*/)) {
175     /* Success; save the config and start using it */
176     login_update_config(config);
177     login_save_config();
178     logged_in();
179     /* Pop down login window */
180     gtk_widget_destroy(login_window);
181   } else {
182     /* Failed to connect - report the error */
183     popup_msg(GTK_MESSAGE_ERROR, disorder_last(c));
184   }
185   disorder_close(c);                    /* no use for this any more */
186 }
187
188 /** @brief User pressed cancel in the login window */
189 static void login_cancel(GtkButton attribute((unused)) *button,
190                          gpointer attribute((unused)) userdata) {
191   gtk_widget_destroy(login_window);
192 }
193
194 /** @brief Keypress handler */
195 static gboolean login_keypress(GtkWidget attribute((unused)) *widget,
196                                GdkEventKey *event,
197                                gpointer attribute((unused)) user_data) {
198   if(event->state)
199     return FALSE;
200   switch(event->keyval) {
201   case GDK_Return:
202     login_ok(0, 0);
203     return TRUE;
204   case GDK_Escape:
205     login_cancel(0, 0);
206     return TRUE;
207   default:
208     return FALSE;
209   }
210 }
211
212 /* Buttons that appear at the bottom of the window */
213 static struct button buttons[] = {
214   {
215     "Login",
216     login_ok,
217     "(Re-)connect using these settings",
218     0
219   },
220   {
221     GTK_STOCK_CLOSE,
222     login_cancel,
223     "Discard changes and close window",
224     0
225   },
226 };
227
228 #define NBUTTONS (int)(sizeof buttons / sizeof *buttons)
229
230 /** @brief Pop up a login box */
231 void login_box(void) {
232   GtkWidget *table, *label, *entry,  *buttonbox, *vbox;
233   size_t n;
234
235   /* If there's one already then bring it to the front */
236   if(login_window) {
237     gtk_window_present(GTK_WINDOW(login_window));
238     return;
239   }
240   default_connect();
241   /* Create a new login window */
242   login_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
243   gtk_widget_set_style(login_window, tool_style);
244   g_signal_connect(login_window, "destroy",
245                    G_CALLBACK(gtk_widget_destroyed), &login_window);
246   gtk_window_set_title(GTK_WINDOW(login_window), "Login Details");
247   /* Construct the form */
248   table = gtk_table_new(NLWIS/*rows*/, 2/*columns*/, FALSE/*homogenous*/);
249   gtk_widget_set_style(table, tool_style);
250   for(n = 0; n < NLWIS; ++n) {
251     label = gtk_label_new(lwis[n].description);
252     gtk_widget_set_style(label, tool_style);
253     gtk_misc_set_alignment(GTK_MISC(label), 1/*right*/, 0/*bottom*/);
254     gtk_table_attach(GTK_TABLE(table), label,
255                      0, 1,              /* left/right_attach */
256                      n, n+1,            /* top/bottom_attach */
257                      GTK_FILL, 0,       /* x/yoptions */
258                      1, 1);             /* x/ypadding */
259     entry = gtk_entry_new();
260     gtk_widget_set_style(entry, tool_style);
261     gtk_entry_set_visibility(GTK_ENTRY(entry),
262                              lwis[n].flags & LWI_HIDDEN ? FALSE : TRUE);
263     gtk_entry_set_text(GTK_ENTRY(entry), lwis[n].get());
264     gtk_table_attach(GTK_TABLE(table), entry,
265                      1, 2,              /* left/right_attach */
266                      n, n+1,            /* top/bottom_attach */
267                      GTK_EXPAND|GTK_FILL, 0, /* x/yoptions */
268                      1, 1);             /* x/ypadding */
269     lwi_entry[n] = entry;
270   }
271   buttonbox = create_buttons(buttons, NBUTTONS);
272   vbox = gtk_vbox_new(FALSE, 1);
273   gtk_box_pack_start(GTK_BOX(vbox),
274                      gtk_image_new_from_pixbuf(find_image("logo256.png")),
275                      TRUE/*expand*/,
276                      TRUE/*fill*/,
277                      4/*padding*/);
278   gtk_box_pack_start(GTK_BOX(vbox), table, 
279                      TRUE/*expand*/, TRUE/*fill*/, 1/*padding*/);
280   gtk_box_pack_start(GTK_BOX(vbox), buttonbox,
281                      FALSE/*expand*/, FALSE/*fill*/, 1/*padding*/);
282   gtk_container_add(GTK_CONTAINER(login_window), frame_widget(vbox, NULL));
283   gtk_window_set_transient_for(GTK_WINDOW(login_window),
284                                GTK_WINDOW(toplevel));
285   /* Keyboard shortcuts */
286   g_signal_connect(login_window, "key-press-event",
287                    G_CALLBACK(login_keypress), 0);
288   gtk_widget_show_all(login_window);
289 }
290
291 /*
292 Local Variables:
293 c-basic-offset:2
294 comment-column:40
295 fill-column:79
296 indent-tabs-mode:nil
297 End:
298 */