2 * This file is part of DisOrder
3 * Copyright (C) 2007, 2008 Richard Kettlewell
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.
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.
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
20 /** @file disobedience/login.c
21 * @brief Login box for Disobedience
23 * As of 2.1 we have only two buttons: Login and Cancel.
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
29 * It you hit Cancel then the window disappears without saving anything.
32 * - cancel/close should be consistent with properties
35 #include "disobedience.h"
39 #include <sys/types.h>
43 /** @brief One field in the login window */
44 struct login_window_item {
45 /** @brief Description label */
46 const char *description;
48 /** @brief Return the current value */
49 const char *(*get)(void);
51 /** @brief Set a new value */
52 void (*set)(struct config *c, const char *value);
56 * - @ref LWI_HIDDEN - this is a password
57 * - @ref LWI_REMOTE - this is for remote connections
63 /** @brief This is a password */
64 #define LWI_HIDDEN 0x0001
66 /** @brief This is for remote connections */
67 #define LWI_REMOTE 0x0002
69 /** @brief Current login window */
70 GtkWidget *login_window;
72 /** @brief Set connection defaults */
73 static void default_connect(void) {
74 if(!config->connect.n) {
75 config->connect.n = 2;
76 config->connect.s = xcalloc(2, sizeof (char *));
77 config->connect.s[0] = xstrdup("localhost");
78 config->connect.s[1] = xstrdup("9999"); /* whatever */
82 static const char *get_hostname(void) {
83 return config->connect.n >= 2 ? config->connect.s[0] : "";
86 static const char *get_service(void) {
87 return config->connect.n >= 2 ? config->connect.s[1] : "";
90 static const char *get_username(void) {
91 return config->username;
94 static const char *get_password(void) {
95 return config->password ? config->password : "";
98 static void set_hostname(struct config *c, const char *s) {
99 c->connect.s[0] = (char *)s;
102 static void set_service(struct config *c, const char *s) {
103 c->connect.s[1] = (char *)s;
106 static void set_username(struct config *c, const char *s) {
110 static void set_password(struct config *c, const char *s) {
114 /** @brief Table used to generate the form */
115 static const struct login_window_item lwis[] = {
116 { "Hostname", get_hostname, set_hostname, LWI_REMOTE },
117 { "Service", get_service, set_service, LWI_REMOTE },
118 { "User name", get_username, set_username, 0 },
119 { "Password", get_password, set_password, LWI_HIDDEN },
121 #define NLWIS (sizeof lwis / sizeof *lwis)
123 static GtkWidget *lwi_remote;
124 static GtkWidget *lwi_entry[NLWIS];
126 static void login_update_config(struct config *c) {
128 const gboolean remote = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(lwi_remote));
132 c->connect.s = xcalloc(2, sizeof (char *));
137 for(n = 0; n < NLWIS; ++n)
138 if(remote || !(lwis[n].flags & LWI_REMOTE))
139 lwis[n].set(c, xstrdup(gtk_entry_get_text(GTK_ENTRY(lwi_entry[n]))));
142 /** @brief Save current login details */
143 static void login_save_config(void) {
144 char *path = config_userconf(0, 0), *tmp;
147 byte_xasprintf(&tmp, "%s.tmp", path);
148 /* Make sure the directory exists; don't care if it already exists. */
149 mkdir(d_dirname(tmp), 02700);
150 /* Write out the file */
151 if(!(fp = fopen(tmp, "w"))) {
152 fpopup_msg(GTK_MESSAGE_ERROR, "error opening %s: %s",
153 tmp, strerror(errno));
156 int rc = fprintf(fp, "username %s\n"
158 quoteutf8(config->username),
159 quoteutf8(config->password));
160 if(rc >= 0 && config->connect.n)
161 rc = fprintf(fp, "connect %s %s\n",
162 quoteutf8(config->connect.s[0]),
163 quoteutf8(config->connect.s[1]));
165 fpopup_msg(GTK_MESSAGE_ERROR, "error writing to %s: %s",
166 tmp, strerror(errno));
171 fpopup_msg(GTK_MESSAGE_ERROR, "error closing %s: %s",
172 tmp, strerror(errno));
175 /* Rename into place */
176 if(rename(tmp, path) < 0) {
177 fpopup_msg(GTK_MESSAGE_ERROR, "error renaming %s: %s",
178 tmp, strerror(errno));
185 /** @brief User pressed OK in login window */
186 static void login_ok(GtkButton attribute((unused)) *button,
187 gpointer attribute((unused)) userdata) {
189 struct config *tmpconfig = xmalloc(sizeof *tmpconfig);
191 tmpconfig->home = xstrdup(pkgstatedir);
192 /* Copy the new config into @ref config */
193 login_update_config(tmpconfig);
194 /* Attempt a login with the new details */
196 if(!disorder_connect_generic(tmpconfig, c,
197 tmpconfig->username, tmpconfig->password,
199 /* Success; save the config and start using it */
200 login_update_config(config);
203 /* Pop down login window */
204 gtk_widget_destroy(login_window);
206 /* Failed to connect - report the error */
207 popup_msg(GTK_MESSAGE_ERROR, disorder_last(c));
209 disorder_close(c); /* no use for this any more */
212 /** @brief User pressed cancel in the login window */
213 static void login_cancel(GtkButton attribute((unused)) *button,
214 gpointer attribute((unused)) userdata) {
215 gtk_widget_destroy(login_window);
218 /** @brief Keypress handler */
219 static gboolean login_keypress(GtkWidget attribute((unused)) *widget,
221 gpointer attribute((unused)) user_data) {
224 switch(event->keyval) {
236 /* Buttons that appear at the bottom of the window */
237 static struct button buttons[] = {
241 "(Re-)connect using these settings",
247 "Discard changes and close window",
252 #define NBUTTONS (int)(sizeof buttons / sizeof *buttons)
254 /** @brief Called when the remote/local button is toggled (and initially)
256 * Sets the sensitivity of the host/port entries.
258 static void lwi_remote_toggled(GtkToggleButton *togglebutton,
259 gpointer attribute((unused)) user_data) {
260 const gboolean remote = gtk_toggle_button_get_active(togglebutton);
262 for(unsigned n = 0; n < NLWIS; ++n)
263 if(lwis[n].flags & LWI_REMOTE)
264 gtk_widget_set_sensitive(lwi_entry[n], remote);
267 /** @brief Pop up a login box */
268 void login_box(void) {
269 GtkWidget *table, *label, *entry, *buttonbox, *vbox;
272 /* If there's one already then bring it to the front */
274 gtk_window_present(GTK_WINDOW(login_window));
278 /* Create a new login window */
279 login_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
280 gtk_widget_set_style(login_window, tool_style);
281 g_signal_connect(login_window, "destroy",
282 G_CALLBACK(gtk_widget_destroyed), &login_window);
283 gtk_window_set_title(GTK_WINDOW(login_window), "Login Details");
284 /* Construct the form */
285 table = gtk_table_new(1 + NLWIS/*rows*/, 2/*columns*/, FALSE/*homogenous*/);
286 gtk_widget_set_style(table, tool_style);
287 label = gtk_label_new("Remote");
288 gtk_widget_set_style(label, tool_style);
289 gtk_misc_set_alignment(GTK_MISC(label), 1/*right*/, 0/*bottom*/);
290 gtk_table_attach(GTK_TABLE(table), label,
291 0, 1, /* left/right_attach */
292 0, 1, /* top/bottom_attach */
293 GTK_FILL, 0, /* x/yoptions */
294 1, 1); /* x/ypadding */
295 lwi_remote = gtk_check_button_new();
296 gtk_widget_set_style(lwi_remote, tool_style);
297 gtk_table_attach(GTK_TABLE(table), lwi_remote,
298 1, 2, /* left/right_attach */
299 0, 1, /* top/bottom_attach */
300 GTK_EXPAND|GTK_FILL, 0, /* x/yoptions */
301 1, 1); /* x/ypadding */
302 g_signal_connect(lwi_remote, "toggled", G_CALLBACK(lwi_remote_toggled), 0);
303 for(n = 0; n < NLWIS; ++n) {
304 label = gtk_label_new(lwis[n].description);
305 gtk_widget_set_style(label, tool_style);
306 gtk_misc_set_alignment(GTK_MISC(label), 1/*right*/, 0/*bottom*/);
307 gtk_table_attach(GTK_TABLE(table), label,
308 0, 1, /* left/right_attach */
309 n+1, n+2, /* top/bottom_attach */
310 GTK_FILL, 0, /* x/yoptions */
311 1, 1); /* x/ypadding */
312 entry = gtk_entry_new();
313 gtk_widget_set_style(entry, tool_style);
314 gtk_entry_set_visibility(GTK_ENTRY(entry),
315 lwis[n].flags & LWI_HIDDEN ? FALSE : TRUE);
316 gtk_entry_set_text(GTK_ENTRY(entry), lwis[n].get());
317 gtk_table_attach(GTK_TABLE(table), entry,
318 1, 2, /* left/right_attach */
319 n+1, n+2, /* top/bottom_attach */
320 GTK_EXPAND|GTK_FILL, 0, /* x/yoptions */
321 1, 1); /* x/ypadding */
322 lwi_entry[n] = entry;
324 /* Initial settings */
325 lwi_remote_toggled(GTK_TOGGLE_BUTTON(lwi_remote), 0);
326 gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(lwi_remote),
327 config->connect.n >= 2);
328 buttonbox = create_buttons(buttons, NBUTTONS);
329 vbox = gtk_vbox_new(FALSE, 1);
330 gtk_box_pack_start(GTK_BOX(vbox),
331 gtk_image_new_from_pixbuf(find_image("logo256.png")),
335 gtk_box_pack_start(GTK_BOX(vbox), table,
336 TRUE/*expand*/, TRUE/*fill*/, 1/*padding*/);
337 gtk_box_pack_start(GTK_BOX(vbox), buttonbox,
338 FALSE/*expand*/, FALSE/*fill*/, 1/*padding*/);
339 gtk_container_add(GTK_CONTAINER(login_window), frame_widget(vbox, NULL));
340 gtk_window_set_transient_for(GTK_WINDOW(login_window),
341 GTK_WINDOW(toplevel));
342 /* Keyboard shortcuts */
343 g_signal_connect(login_window, "key-press-event",
344 G_CALLBACK(login_keypress), 0);
345 gtk_widget_show_all(login_window);