nautilus-application.c 29.8 KB
Newer Older
Gene Z. Ragan's avatar
CVS:  
Gene Z. Ragan committed
1
/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
2 3 4 5 6

/*
 *  Nautilus
 *
 *  Copyright (C) 1999, 2000 Red Hat, Inc.
7
 *  Copyright (C) 2000, 2001 Eazel, Inc.
8
 *
9
 *  Nautilus is free software; you can redistribute it and/or
10 11 12 13
 *  modify it under the terms of the GNU General Public License as
 *  published by the Free Software Foundation; either version 2 of the
 *  License, or (at your option) any later version.
 *
14
 *  Nautilus is distributed in the hope that it will be useful,
15 16 17 18 19
 *  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
20
 *  along with this program; if not, write to the Free Software
21 22
 *  Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
 *
23 24
 *  Authors: Elliot Lee <sopwith@redhat.com>,
 *           Darin Adler <darin@eazel.com>
25 26 27
 *
 */

28
#include <config.h>
29
#include "nautilus-application.h"
30

31
#include "file-manager/fm-desktop-icon-view.h"
32
#include "file-manager/fm-icon-view.h"
33
#include "file-manager/fm-list-view.h"
34
#include "file-manager/fm-search-list-view.h"
35 36
#include "nautilus-desktop-window.h"
#include "nautilus-first-time-druid.h"
37
#include "nautilus-main.h"
38 39 40 41
#include "nautilus-shell-interface.h"
#include "nautilus-shell.h"
#include <bonobo/bonobo-main.h>
#include <bonobo/bonobo-object.h>
42
#include <dirent.h>
Ramiro Estrugo's avatar
Ramiro Estrugo committed
43 44 45 46 47
#include <eel/eel-gtk-macros.h>
#include <eel/eel-stock-dialogs.h>
#include <eel/eel-string-list.h>
#include <eel/eel-string.h>
#include <eel/eel-vfs-extensions.h>
48
#include <gtk/gtksignal.h>
49
#include <libgnome/gnome-config.h>
50
#include <libgnome/gnome-i18n.h>
51
#include <libgnome/gnome-metadata.h>
52
#include <libgnome/gnome-util.h>
53
#include <libgnomeui/gnome-client.h>
Ramiro Estrugo's avatar
Ramiro Estrugo committed
54
#include <libgnomeui/gnome-messagebox.h>
55
#include <libgnomeui/gnome-stock.h>
56
#include <libgnomevfs/gnome-vfs-mime-handlers.h>
57
#include <libgnomevfs/gnome-vfs-ops.h>
Ramiro Estrugo's avatar
Ramiro Estrugo committed
58
#include <libgnomevfs/gnome-vfs-utils.h>
59
#include <libnautilus-extensions/nautilus-file-utilities.h>
60
#include <libnautilus-extensions/nautilus-global-preferences.h>
61
#include <libnautilus-extensions/nautilus-icon-factory.h>
Ramiro Estrugo's avatar
Ramiro Estrugo committed
62
#include <libnautilus-extensions/nautilus-metafile-factory.h>
63
#include <libnautilus-extensions/nautilus-sound.h>
64
#include <libnautilus-extensions/nautilus-undo-manager.h>
65
#include <libnautilus-extensions/nautilus-volume-monitor.h>
66
#include <liboaf/liboaf.h>
67

68
#define FACTORY_IID	     "OAFIID:nautilus_factory:bd1e1862-92d7-4391-963e-37583f0daef3"
69
#define SEARCH_LIST_VIEW_IID "OAFIID:nautilus_file_manager_search_list_view:b186e381-198e-43cf-9c46-60b6bb35db0b"
70
#define SHELL_IID	     "OAFIID:nautilus_shell:cd5183b2-3913-4b74-9b8e-10528b0de08d"
71

72 73 74 75 76 77
/* Keeps track of the one and only desktop window. */
static NautilusDesktopWindow *nautilus_application_desktop_window;

/* Keeps track of all the nautilus windows. */
static GList *nautilus_application_window_list;

78 79 80 81 82 83 84 85 86 87 88 89 90
static CORBA_boolean manufactures                          (PortableServer_Servant    servant,
							    const CORBA_char         *iid,
							    CORBA_Environment        *ev);
static CORBA_Object  create_object                         (PortableServer_Servant    servant,
							    const CORBA_char         *iid,
							    const GNOME_stringlist   *params,
							    CORBA_Environment        *ev);
static void          nautilus_application_initialize       (NautilusApplication      *application);
static void          nautilus_application_initialize_class (NautilusApplicationClass *klass);
static void          nautilus_application_destroy          (GtkObject                *object);
static gboolean      confirm_ok_to_run_as_root             (void);
static gboolean      need_to_show_first_time_druid         (void);
static void          desktop_changed_callback              (gpointer                  user_data);
91
static void          desktop_location_changed_callback     (gpointer                  user_data);
92 93 94 95 96 97
static void          volume_mounted_callback               (NautilusVolumeMonitor    *monitor,
							    NautilusVolume           *volume,
							    NautilusApplication      *application);
static void          volume_unmounted_callback             (NautilusVolumeMonitor    *monitor,
							    NautilusVolume           *volume,
							    NautilusApplication      *application);
98
static void	     init_session 			    (void);
99

Ramiro Estrugo's avatar
Ramiro Estrugo committed
100
EEL_DEFINE_CLASS_BOILERPLATE (NautilusApplication, nautilus_application, BONOBO_OBJECT_TYPE)
101

102
static POA_GNOME_ObjectFactory__epv factory_epv = {
Gene Z. Ragan's avatar
CVS:  
Gene Z. Ragan committed
103
	NULL,
104 105 106 107
	&manufactures,
	&create_object
};
static PortableServer_ServantBase__epv base_epv;
108
static POA_GNOME_ObjectFactory__vepv vepv = {
109 110
	&base_epv,
	&factory_epv
111 112 113
};

static CORBA_boolean
114 115 116
manufactures (PortableServer_Servant servant,
	      const CORBA_char *iid,
	      CORBA_Environment *ev)
117
{
118
	return strcmp (iid, NAUTILUS_ICON_VIEW_IID) == 0
119
		|| strcmp (iid, NAUTILUS_DESKTOP_ICON_VIEW_IID) == 0
120
		|| strcmp (iid, NAUTILUS_LIST_VIEW_IID) == 0
121
		|| strcmp (iid, SEARCH_LIST_VIEW_IID) == 0
122 123
		|| strcmp (iid, SHELL_IID) == 0
		|| strcmp (iid, METAFILE_FACTORY_IID) == 0;
124 125 126
}

static CORBA_Object
127 128
create_object (PortableServer_Servant servant,
	       const CORBA_char *iid,
Maciej Stachowiak's avatar
Maciej Stachowiak committed
129
	       const GNOME_stringlist *params,
130
	       CORBA_Environment *ev)
131
{
132
	BonoboObject *object;
133
	FMDirectoryView *directory_view;
134
	NautilusApplication *application;
135

136
	if (strcmp (iid, NAUTILUS_ICON_VIEW_IID) == 0) {
137
		directory_view = FM_DIRECTORY_VIEW (gtk_object_new (fm_icon_view_get_type (), NULL));
138
		object = BONOBO_OBJECT (fm_directory_view_get_nautilus_view (directory_view));
139 140 141
	} else if (strcmp (iid, NAUTILUS_DESKTOP_ICON_VIEW_IID) == 0) {
		directory_view = FM_DIRECTORY_VIEW (gtk_object_new (fm_desktop_icon_view_get_type (), NULL));
		object = BONOBO_OBJECT (fm_directory_view_get_nautilus_view (directory_view));
142
	} else if (strcmp (iid, NAUTILUS_LIST_VIEW_IID) == 0) {
143
		directory_view = FM_DIRECTORY_VIEW (gtk_object_new (fm_list_view_get_type (), NULL));
144
		object = BONOBO_OBJECT (fm_directory_view_get_nautilus_view (directory_view));
145 146 147
	} else if (strcmp (iid, SEARCH_LIST_VIEW_IID) == 0) {
		directory_view = FM_DIRECTORY_VIEW (gtk_object_new (fm_search_list_view_get_type (), NULL));
		object = BONOBO_OBJECT (fm_directory_view_get_nautilus_view (directory_view));
148
	} else if (strcmp (iid, SHELL_IID) == 0) {
149 150
		application = NAUTILUS_APPLICATION (((BonoboObjectServant *) servant)->bonobo_object);
		object = BONOBO_OBJECT (nautilus_shell_new (application));
151 152
	} else if (strcmp (iid, METAFILE_FACTORY_IID) == 0) {
		object = BONOBO_OBJECT (nautilus_metafile_factory_get_instance ());
Gene Z. Ragan's avatar
CVS:  
Gene Z. Ragan committed
153
	} else {
154
		return CORBA_OBJECT_NIL;
Gene Z. Ragan's avatar
CVS:  
Gene Z. Ragan committed
155
	}
156 157

	return CORBA_Object_duplicate (bonobo_object_corba_objref (object), ev);
158 159
}

160
static CORBA_Object
161 162 163
create_factory (PortableServer_POA poa,
		NautilusApplication *bonobo_object,
		CORBA_Environment *ev)
164
{
165
	BonoboObjectServant *servant;
Gene Z. Ragan's avatar
CVS:  
Gene Z. Ragan committed
166

167
	servant = g_new0 (BonoboObjectServant, 1);
168 169
	((POA_GNOME_ObjectFactory *) servant)->vepv = &vepv;
	POA_GNOME_ObjectFactory__init ((PortableServer_Servant) servant, ev);
170
	return bonobo_object_activate_servant (BONOBO_OBJECT (bonobo_object), servant);
171 172
}

173 174
GList *
nautilus_application_get_window_list (void)
175 176 177 178
{
	return nautilus_application_window_list;
}

179
static void
180
nautilus_application_initialize_class (NautilusApplicationClass *klass)
181
{
182
	GTK_OBJECT_CLASS (klass)->destroy = nautilus_application_destroy;
183 184 185
}

static void
186
nautilus_application_initialize (NautilusApplication *application)
187
{
188
	CORBA_Environment ev;
189
	CORBA_Object corba_object;
190

191
	CORBA_exception_init (&ev);
192
	corba_object = create_factory (bonobo_poa (), application, &ev);
193 194 195 196
	if (ev._major != CORBA_NO_EXCEPTION) {
		g_error ("could not create factory");
	}
	CORBA_exception_free (&ev);
197
	
198
	bonobo_object_construct (BONOBO_OBJECT (application), corba_object);
199
	
200 201
	/* Create an undo manager */
	application->undo_manager = nautilus_undo_manager_new ();
202

203 204 205 206 207 208 209 210 211 212 213
	/* Watch for volume mounts so we can restore open windows */
	gtk_signal_connect (GTK_OBJECT (nautilus_volume_monitor_get ()),
			    "volume_mounted",
			    volume_mounted_callback,
			    application);

	/* Watch for volume unmounts so we can close open windows */
	gtk_signal_connect (GTK_OBJECT (nautilus_volume_monitor_get ()),
			    "volume_unmounted",
			    volume_unmounted_callback,
			    application);
214 215
}

216 217
NautilusApplication *
nautilus_application_new (void)
218
{
219
	return NAUTILUS_APPLICATION (gtk_object_new (nautilus_application_get_type (), NULL));
220 221 222
}

static void
223
nautilus_application_destroy (GtkObject *object)
224
{
225 226 227 228
	NautilusApplication *application;

	application = NAUTILUS_APPLICATION (object);

229
	nautilus_bookmarks_exiting ();
230
	
231 232
	bonobo_object_unref (BONOBO_OBJECT (application->undo_manager));

Ramiro Estrugo's avatar
Ramiro Estrugo committed
233
	EEL_CALL_PARENT (GTK_OBJECT_CLASS, destroy, (object));
234
}
235

236 237
static gboolean
check_required_directories (NautilusApplication *application)
238
{
239 240 241 242 243 244 245 246 247
	char *user_directory;
	char *user_main_directory;
	char *desktop_directory;
	EelStringList *directories;
	char *directories_as_string;
	char *error_string;
	char *dialog_title;
	GnomeDialog *dialog;
	int failed_count;
248
	
249
	g_assert (NAUTILUS_IS_APPLICATION (application));
250 251 252 253 254

	user_directory = nautilus_get_user_directory ();
	user_main_directory = nautilus_get_user_main_directory ();
	desktop_directory = nautilus_get_desktop_directory ();

255
	directories = eel_string_list_new (TRUE);
256 257
	
	if (!g_file_test (user_directory, G_FILE_TEST_ISDIR)) {
258
		eel_string_list_insert (directories, user_directory);
259
	}
260
	g_free (user_directory);
261 262
	    
	if (!g_file_test (user_main_directory, G_FILE_TEST_ISDIR)) {
263
		eel_string_list_insert (directories, user_main_directory);
264
	}
265
	g_free (user_main_directory);
266 267
	    
	if (!g_file_test (desktop_directory, G_FILE_TEST_ISDIR)) {
268
		eel_string_list_insert (directories, desktop_directory);
269
	}
270
	g_free (desktop_directory);
271

272
	failed_count = eel_string_list_get_length (directories);
273

274
	if (failed_count != 0) {
275
		directories_as_string = eel_string_list_as_string (directories, "\n", EEL_STRING_LIST_ALL_STRINGS);
276

277 278
		if (failed_count == 1) {
			dialog_title = g_strdup (_("Couldn't Create Required Folder"));
279 280 281
			error_string = g_strdup_printf (_("Nautilus could not create the required folder \"%s\". "
							  "Before running Nautilus, please create this folder, or "
							  "set permissions such that Nautilus can create it."),
282
							directories_as_string);
283 284
		} else {
			dialog_title = g_strdup (_("Couldn't Create Required Folders"));
285 286 287 288
			error_string = g_strdup_printf (_("Nautilus could not create the following required folders:\n\n"
							  "%s\n\n"
							  "Before running Nautilus, please create these folders, or "
							  "set permissions such that Nautilus can create them."),
289
							directories_as_string);
290 291
		}
		
Ramiro Estrugo's avatar
Ramiro Estrugo committed
292
		dialog = eel_show_error_dialog (error_string, dialog_title, NULL);
293 294
		/* We need the main event loop so the user has a chance to see the dialog. */
		nautilus_main_event_loop_register (GTK_OBJECT (dialog));
295

296
		g_free (directories_as_string);
297
		g_free (error_string);
298
		g_free (dialog_title);
299 300
	}

301
	eel_string_list_free (directories);
302 303

	return failed_count == 0;
304 305
}

306 307 308 309 310 311 312 313 314 315
static int
nautilus_strv_length (const char * const *strv)
{
	const char * const *p;

	for (p = strv; *p != NULL; p++) { }
	return p - strv;
}

static Nautilus_URIList *
316
nautilus_make_uri_list_from_shell_strv (const char * const *strv)
317 318 319
{
	int length, i;
	Nautilus_URIList *uri_list;
320
	char *translated_uri;
321 322 323 324 325 326 327 328

	length = nautilus_strv_length (strv);

	uri_list = Nautilus_URIList__alloc ();
	uri_list->_maximum = length;
	uri_list->_length = length;
	uri_list->_buffer = CORBA_sequence_Nautilus_URI_allocbuf (length);
	for (i = 0; i < length; i++) {
Ramiro Estrugo's avatar
Ramiro Estrugo committed
329
		translated_uri = eel_make_uri_from_shell_arg (strv[i]);
330 331 332
		uri_list->_buffer[i] = CORBA_string_dup (translated_uri);
		g_free (translated_uri);
		translated_uri = NULL;
333 334 335 336 337 338
	}
	CORBA_sequence_set_release (uri_list, CORBA_TRUE);

	return uri_list;
}

339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416
/* Find ~/.gnome-desktop/Trash and rename it to ~/.gnome-desktop/Trash-gmc
 * Only if it is a directory
 */
static void
migrate_gmc_trash (void)
{
	char *dp, *trash_dir, *dest;
	struct stat buf;

	dp = nautilus_get_desktop_directory ();
	trash_dir = g_strconcat (dp, "/", "Trash", NULL);
	dest = g_strconcat (dp, "/", "Trash.gmc", NULL);
	
	if (stat (trash_dir, &buf) == 0 && S_ISDIR (buf.st_mode)) {
		rename (trash_dir, dest);
		gnome_metadata_rename (trash_dir, dest);
	}
	
	g_free (dp);
	g_free (trash_dir);
	g_free (dest);
}

static void
migrate_old_nautilus_files (void)
{
	char *new_desktop_dir, *np;
	char *old_desktop_dir, *op;
	char *old_desktop_dir_new_name;
	struct stat buf;
	DIR *dir;
	struct dirent *de;
	
	old_desktop_dir = g_strconcat (g_get_home_dir (), "/.nautilus/desktop", NULL);
	if (stat (old_desktop_dir, &buf) == -1) {
		g_free (old_desktop_dir);
		return;
	}
	if (!S_ISLNK (buf.st_mode)){
		dir = opendir (old_desktop_dir);
		if (dir == NULL) {
			g_free (old_desktop_dir);
			return;
		}
	
		new_desktop_dir = nautilus_get_desktop_directory ();
		
		while ((de = readdir (dir)) != NULL){
			if (de->d_name [0] == '.'){
				if (de->d_name [0] == 0)
					continue;
				
				if (de->d_name [1] == '.' && de->d_name [2] == 0)
					continue;
			}
	
			op = g_strconcat (old_desktop_dir, "/", de->d_name, NULL);
			np = g_strconcat (new_desktop_dir, "/", de->d_name, NULL);
	
			rename (op, np);
	
			g_free (op);
			g_free (np);
		}

		closedir (dir);

		g_free (new_desktop_dir);
	}

	/* In case we miss something */
	old_desktop_dir_new_name = g_strconcat (old_desktop_dir, "-old", NULL);
	rename (old_desktop_dir, old_desktop_dir_new_name);
	g_free (old_desktop_dir_new_name);

	g_free (old_desktop_dir);
}

417
void
418
nautilus_application_startup (NautilusApplication *application,
419
			      gboolean kill_shell,
420
			      gboolean restart_shell,
421
			      gboolean no_default_window,
422
			      gboolean no_desktop,
423
			      gboolean do_first_time_druid_check,
424
			      const char *geometry,
425
			      const char *urls[])
426
{
427 428
	CORBA_Environment ev;
	Nautilus_Shell shell;
429 430 431
	OAF_RegistrationResult result;
	const char *message, *detailed_message;
	GnomeDialog *dialog;
432
	Nautilus_URIList *url_list;
433
	const CORBA_char *corba_geometry;
434 435 436
	int num_failures;

	num_failures = 0;
437

438
	/* Perform check for nautilus being run as super user */
439 440 441 442
	if (!(kill_shell || restart_shell)) {
		if (!confirm_ok_to_run_as_root ()) {
			return;
		}
443
	}
Ramiro Estrugo's avatar
Ramiro Estrugo committed
444

445
	/* Check the user's ~/.nautilus directories and post warnings
446
	 * if there are problems.
447
	 */
448
	if (!kill_shell && !check_required_directories (application)) {
449 450
		return;
	}
451

452 453
	/* Run the first time startup druid if needed. */
	if (do_first_time_druid_check && need_to_show_first_time_druid ()) {
454
		nautilus_first_time_druid_show (application, urls);
455 456
		return;
	}
457 458

	/* Make the desktop work with gmc and old Nautilus. */
459 460 461 462
	if (!kill_shell) {
		migrate_gmc_trash ();
		migrate_old_nautilus_files ();
	}
463
	
464 465 466
	/* initialize the sound machinery */
	nautilus_sound_initialize ();
	
467 468
	CORBA_exception_init (&ev);

469
	/* Start up the factory. */
470
	while (TRUE) {
471 472 473 474 475 476 477
		/* Try to register the file manager view factory with OAF. */
		result = oaf_active_server_register
			(FACTORY_IID,
			 bonobo_object_corba_objref (BONOBO_OBJECT (application)));
		switch (result) {
		case OAF_REG_SUCCESS:
			/* We are registered with OAF and all is right with the world. */
478
		case OAF_REG_ALREADY_ACTIVE:
479
			/* Another copy of nautilus already is running and registered. */
480
			message = NULL;
481
			detailed_message = NULL;
482 483 484
			break;
		case OAF_REG_NOT_LISTED:
			/* Can't register myself due to trouble locating the
485
			 * Nautilus_Shell.oaf file. This has happened when you
486 487 488
			 * launch Nautilus with an LD_LIBRARY_PATH that
			 * doesn't include the directory containg the oaf
			 * library. It could also happen if the
489
			 * Nautilus_Shell.oaf file was not present for some
490 491 492 493 494
			 * reason. Sometimes killing oafd and gconfd fixes
			 * this problem but we don't exactly understand why,
			 * since neither of the above causes explain it.
			 */
			message = _("Nautilus can't be used now. "
495 496 497 498
				    "Running the command \"nautilus-clean.sh -x\""
				    " from the console may fix the problem. If not,"
				    " you can try rebooting the computer or"
				    " installing Nautilus again.");
499
			/* FIXME bugzilla.eazel.com 2536: The guesses and stuff here are lame. */
500
			detailed_message = _("Nautilus can't be used now. "
501 502 503 504
					     "Running the command \"nautilus-clean.sh -x\""
					     " from the console may fix the problem. If not,"
					     " you can try rebooting the computer or"
					     " installing Nautilus again.\n\n"
505
					     "OAF couldn't locate the Nautilus_shell.oaf file. "
506 507 508
					     "One cause of this seems to be an LD_LIBRARY_PATH "
					     "that does not include the oaf library's directory. "
					     "Another possible cause would be bad install "
509
					     "with a missing Nautilus_Shell.oaf file.\n\n"
510 511 512
					     "Running \"nautilus-clean.sh -x\" will kill all "
					     "OAF and GConf processes, which may be needed by "
					     "other applications.\n\n"
513
					     "Sometimes killing oafd and gconfd fixes "
514
					     "the problem, but we don't know why.\n\n"
515 516
					     "We have also seen this error when a faulty "
					     "version of oaf was installed.");
517 518 519 520 521 522 523 524 525
			break;
		default:
			/* This should never happen. */
			g_warning ("bad error code from oaf_active_server_register");
		case OAF_REG_ERROR:
			/* Some misc. error (can never happen with current
			 * version of OAF). Show dialog and terminate the
			 * program.
			 */
526
			/* FIXME bugzilla.eazel.com 2537: Looks like this does happen with the
527 528 529 530
			 * current OAF. I guess I read the code
			 * wrong. Need to figure out when and make a
			 * good message.
			 */
531 532 533 534 535
			message = _("Nautilus can't be used now, due to an unexpected error.");
			detailed_message = _("Nautilus can't be used now, due to an unexpected error "
					     "from OAF when attempting to register the file manager view server.");
			break;
		}
536 537 538 539 540 541 542 543 544 545 546 547

		/* Get the shell object. */
		if (message == NULL) {
			shell = oaf_activate_from_id (SHELL_IID, 0, NULL, NULL);
			if (!CORBA_Object_is_nil (shell, &ev)) {
				break;
			}

			/* If we couldn't find ourselves it's a bad problem so
			 * we better stop looping.
			 */
			if (result == OAF_REG_SUCCESS) {
548
				/* FIXME bugzilla.eazel.com 2538: When can this happen? */
549 550
				message = _("Nautilus can't be used now, due to an unexpected error.");
				detailed_message = _("Nautilus can't be used now, due to an unexpected error "
551 552 553 554 555 556 557 558 559 560 561
						     "from OAF when attempting to locate the factory."
						     "Killing oafd and restarting Nautilus may help fix the problem.");
			} else {
				num_failures++;
				if (num_failures > 20) {
					message = _("Nautilus can't be used now, due to an unexpected error.");
					detailed_message = _("Nautilus can't be used now, due to an unexpected error "
							     "from OAF when attempting to locate the shell object. "
							     "Killing oafd and restarting Nautilus may help fix the problem.");
					
				}
562 563 564
			}
		}

565
		if (message != NULL) {
Ramiro Estrugo's avatar
Ramiro Estrugo committed
566
			dialog = eel_show_error_dialog_with_details (message, NULL, detailed_message, NULL);
567 568
			/* We need the main event loop so the user has a chance to see the dialog. */
			nautilus_main_event_loop_register (GTK_OBJECT (dialog));
569
			goto out;
570 571 572
		}
	}

573 574
	if (kill_shell) {
		Nautilus_Shell_quit (shell, &ev);
575 576
	} else if (restart_shell) {
		Nautilus_Shell_restart (shell, &ev);
577
	} else {
578
		if (!no_desktop && nautilus_preferences_get_boolean (NAUTILUS_PREFERENCES_SHOW_DESKTOP)) {
579 580
			Nautilus_Shell_start_desktop (shell, &ev);
		}
581 582 583 584 585 586
		
		/* Monitor the preference to show or hide the desktop */
		nautilus_preferences_add_callback_while_alive (NAUTILUS_PREFERENCES_SHOW_DESKTOP,
							       desktop_changed_callback,
							       application,
							       GTK_OBJECT (application));
587

588 589 590 591
		/* Monitor the preference to have the desktop */
		/* point to the Unix home folder */
		nautilus_preferences_add_callback_while_alive (NAUTILUS_PREFERENCES_DESKTOP_IS_HOME_DIR,
							       desktop_location_changed_callback,
592
							       NULL,
593 594
							       GTK_OBJECT (application));

595 596 597 598
		/* CORBA C mapping doesn't allow NULL to be passed
		   for string parameters */
		corba_geometry = (geometry != NULL) ? geometry : "";

599 600
	  	/* Create the other windows. */
		if (urls != NULL) {
601
			url_list = nautilus_make_uri_list_from_shell_strv (urls);
602
			Nautilus_Shell_open_windows (shell, url_list, corba_geometry, &ev);
603
			CORBA_free (url_list);
604
		} else if (!no_default_window) {
605
			Nautilus_Shell_open_default_window (shell, corba_geometry, &ev);
606
		}
607 608 609
		
		/* Add ourselves to the session */
		init_session ();
610
	}
611

612
	/* We're done with the shell now, so let it go. */
613 614 615 616 617
	/* HACK: Don't bother releasing the shell in the case where we
	 * just told it to quit -- that just leads to hangs and does
	 * no good. We could probably fix this in some fancier way if
	 * we could figure out a better lifetime rule.
	 */
618
	if (!(kill_shell || restart_shell)) {
619
		bonobo_object_release_unref (shell, NULL);
620
	}
621

622
 out:
623
	CORBA_exception_free (&ev);
624 625 626
}

static void
627 628
nautilus_application_create_desktop_window (NautilusApplication *application)
{
629 630
	g_return_if_fail (nautilus_application_desktop_window == NULL);
	g_return_if_fail (NAUTILUS_IS_APPLICATION (application));
631

632 633
	nautilus_application_desktop_window = nautilus_desktop_window_new (application);
	gtk_widget_show (GTK_WIDGET (nautilus_application_desktop_window));
634 635 636 637
}

void
nautilus_application_open_desktop (NautilusApplication *application)
638
{
639 640
	if (nautilus_application_desktop_window == NULL) {
		nautilus_application_create_desktop_window (application);
641 642 643 644 645 646
	}
}

void
nautilus_application_close_desktop (void)
{
647
	if (nautilus_application_desktop_window != NULL) {
648
		gtk_widget_destroy (GTK_WIDGET (nautilus_application_desktop_window));
649
		nautilus_application_desktop_window = NULL;
650
	}
651 652
}

653 654 655
void
nautilus_application_close_all_windows (void)
{
656
	while (nautilus_application_window_list != NULL) {
657 658 659 660
		nautilus_window_close (NAUTILUS_WINDOW (nautilus_application_window_list->data));
	}
}

661
static void
662
nautilus_application_destroyed_window (GtkObject *object, NautilusApplication *application)
663
{
664
	nautilus_application_window_list = g_list_remove (nautilus_application_window_list, object);
665 666
}

667 668 669 670 671 672 673 674 675 676 677 678 679
static gboolean
nautilus_window_delete_event_callback (GtkWidget *widget,
				       GdkEvent *event,
				       gpointer user_data)
{
	NautilusWindow *window;

	window = NAUTILUS_WINDOW (widget);
	nautilus_window_close (window);

	return TRUE;
}				       

680
NautilusWindow *
681
nautilus_application_create_window (NautilusApplication *application)
682
{
683
	NautilusWindow *window;
684 685

	g_return_val_if_fail (NAUTILUS_IS_APPLICATION (application), NULL);
686
	
687
	window = NAUTILUS_WINDOW (gtk_widget_new (nautilus_window_get_type (),
688
						  "app", GTK_OBJECT (application),
689
						  "app_id", "nautilus", NULL));
690
	
691 692 693 694
	gtk_signal_connect (GTK_OBJECT (window), 
			    "delete_event", GTK_SIGNAL_FUNC (nautilus_window_delete_event_callback),
                    	    NULL);

695
	gtk_signal_connect (GTK_OBJECT (window),
696
			    "destroy", nautilus_application_destroyed_window,
697 698
			    application);

699
	nautilus_application_window_list = g_list_prepend (nautilus_application_window_list, window);
700

Gene Z. Ragan's avatar
CVS:  
Gene Z. Ragan committed
701
	/* Do not yet show the window. It will be shown later on if it can
702 703 704
	 * successfully display its initial URI. Otherwise it will be destroyed
	 * without ever having seen the light of day.
	 */
705

706
	return window;
707
}
708 709

/*
710
 * confirm_ok_to_run_as_root:
711 712 713 714
 *
 * Puts out a warning if the user is running nautilus as root.
 */
static gboolean
715
confirm_ok_to_run_as_root (void)
716
{
717 718
	GtkWidget *dialog;
	int result;
719

720 721 722 723
	if (geteuid () != 0) {
		return TRUE;
	}

724 725 726 727 728 729
	if (g_getenv ("NAUTILUS_OK_TO_RUN_AS_ROOT") != NULL) {
		return TRUE;
	}

	dialog = gnome_message_box_new
		(_("You are about to run Nautilus as root.\n\n"
730 731 732
		   "As root, you can damage your system if you are not careful, and\n"
		   "Nautilus will not stop you from doing it."),
		 GNOME_MESSAGE_BOX_WARNING,
733 734
		 GNOME_STOCK_BUTTON_OK, _("Quit"), NULL);
	result = gnome_dialog_run_and_close (GNOME_DIALOG (dialog));
735
	
736
	return result == 0;
737
}
Ramiro Estrugo's avatar
Ramiro Estrugo committed
738

739 740 741 742
/* callback for changing the directory the desktop points to */
static void
desktop_location_changed_callback (gpointer user_data)
{
743 744 745
	if (nautilus_application_desktop_window != NULL) {
		nautilus_desktop_window_update_directory
			(nautilus_application_desktop_window);
746
	}
747 748
}

749 750 751 752 753 754 755
/* callback for showing or hiding the desktop based on the user's preference */
static void
desktop_changed_callback (gpointer user_data)
{
	NautilusApplication *application;
	
	application = NAUTILUS_APPLICATION (user_data);
756
	if ( nautilus_preferences_get_boolean (NAUTILUS_PREFERENCES_SHOW_DESKTOP)) {
757 758 759 760 761 762
		nautilus_application_open_desktop (application);
	} else {
		nautilus_application_close_desktop ();
	}
}

Ramiro Estrugo's avatar
Ramiro Estrugo committed
763 764 765 766 767 768
/*
 * need_to_show_first_time_druid
 *
 * Determine whether Nautilus needs to show the first time druid.
 * 
 * Note that the flag file indicating whether the druid has been
769
 * presented is: ~/.nautilus/first-time-flag.
Ramiro Estrugo's avatar
Ramiro Estrugo committed
770 771 772 773 774
 *
 * Another alternative could be to use preferences to store this flag
 * However, there because of bug 1229 this is not yet possible.
 *
 * Also, for debugging purposes, it is convenient to have just one file
775
 * to kill in order to test the startup druid:
Ramiro Estrugo's avatar
Ramiro Estrugo committed
776
 *
777
 * rm -f ~/.nautilus/first-time-flag
Ramiro Estrugo's avatar
Ramiro Estrugo committed
778 779 780 781 782 783 784 785 786 787 788
 *
 * In order to accomplish the same thing with preferences, you would have
 * to either kill ALL your preferences or spend time digging in ~/.gconf
 * xml files finding the right one.
 */
static gboolean
need_to_show_first_time_druid (void)
{
	gboolean result;
	char *user_directory;
	char *druid_flag_file_name;
789
	
Ramiro Estrugo's avatar
Ramiro Estrugo committed
790 791 792 793
	user_directory = nautilus_get_user_directory ();

	druid_flag_file_name = g_strdup_printf ("%s/%s",
						user_directory,
794
						"first-time-flag");
Ramiro Estrugo's avatar
Ramiro Estrugo committed
795

796
	result = !g_file_exists (druid_flag_file_name);	
Ramiro Estrugo's avatar
Ramiro Estrugo committed
797 798
	g_free (druid_flag_file_name);

799 800 801 802 803 804 805 806 807 808 809
	/* we changed the name of the flag for version 1.0, so we should
	 * check for and delete the old one, if the new one didn't exist 
	 */
	if (result) {
		druid_flag_file_name = g_strdup_printf ("file://%s/%s",
						user_directory,
						"first-time-wizard-flag");
		gnome_vfs_unlink (druid_flag_file_name);
		g_free (druid_flag_file_name);
	}
	g_free (user_directory); 
Ramiro Estrugo's avatar
Ramiro Estrugo committed
810 811
	return result;
}
812 813 814 815 816

static void
volume_mounted_callback (NautilusVolumeMonitor *monitor, NautilusVolume *volume,
			 NautilusApplication *application)
{
817 818 819 820 821 822 823 824 825 826 827 828 829 830 831
	NautilusWindow *window;
	char *uri;
	
	if (volume == NULL || application == NULL) {
		return;
	}
	
	/* Open a window to the CD if the user has set that preference. */
	if (volume->device_type == NAUTILUS_DEVICE_CD_ROM_DRIVE
		&& gnome_config_get_bool ("/magicdev/Options/do_fileman_window=true")) {		
		window = nautilus_application_create_window (application);
		uri = gnome_vfs_get_uri_from_local_path (volume->mount_path);
		nautilus_window_go_to (window, uri);
		g_free (uri);
	}
832 833
}

834 835 836 837 838 839 840 841 842 843 844 845 846 847 848 849 850 851 852 853 854 855 856 857 858 859 860
static gboolean
window_can_be_closed (NautilusWindow *window)
{
	if (!NAUTILUS_IS_DESKTOP_WINDOW (window)) {
		return TRUE;
	}
	
	return FALSE;
}

static gboolean
is_last_closable_window (NautilusWindow *window)
{
	GList *node, *window_list;
	
	window_list = nautilus_application_get_window_list ();
	
	for (node = window_list; node != NULL; node = node->next) {
		if (window != NAUTILUS_WINDOW (node->data) && window_can_be_closed (NAUTILUS_WINDOW (node->data))) {
			return FALSE;
		}
	}
	
	return TRUE;
}


861 862 863 864 865 866 867 868
/* Called whenever a volume is unmounted. Check and see if there are any windows open
 * displaying contents on the volume. If there are, close them.
 * It would also be cool to save open window and position info.
 */
static void
volume_unmounted_callback (NautilusVolumeMonitor *monitor, NautilusVolume *volume,
			   NautilusApplication *application)
{
869
	GList *window_list, *node, *close_list;
870
	NautilusWindow *window;
871 872
	char *uri;
	char *path;
873 874 875 876
		
	close_list = NULL;
	
	/* Check and see if any of the open windows are displaying contents from the unmounted volume */
877
	window_list = nautilus_application_get_window_list ();
878
	
879 880
	/* Construct a list of windows to be closed. Do not add the non-closable windows to the list. */
	for (node = window_list; node != NULL; node = node->next) {
881
		window = NAUTILUS_WINDOW (node->data);
882
		if (window != NULL && window_can_be_closed (window)) {
883 884
			uri = nautilus_window_get_location (window);
			path = gnome_vfs_get_local_path_from_uri (uri);
Ramiro Estrugo's avatar
Ramiro Estrugo committed
885
			if (eel_str_has_prefix (path, volume->mount_path)) {
886
				close_list = g_list_prepend (close_list, window);
887
			}
888