/* nautilus-error-reporting.h - implementation of file manager functions that report
* errors to the user.
*
* Copyright (C) 2000 Eazel, Inc.
*
* The Gnome Library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Library General Public License as
* published by the Free Software Foundation; either version 2 of the
* License, or (at your option) any later version.
*
* The Gnome Library 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
* Library General Public License for more details.
*
* You should have received a copy of the GNU Library General Public
* License along with the Gnome Library; see the file COPYING.LIB. If not,
* see .
*
* Authors: John Sullivan
*/
#include
#include "nautilus-error-reporting.h"
#include
#include
#include "nautilus-file.h"
#include
#include
#include "nautilus-ui-utilities.h"
#define DEBUG_FLAG NAUTILUS_DEBUG_DIRECTORY_VIEW
#include "nautilus-debug.h"
#define NEW_NAME_TAG "Nautilus: new name"
static void finish_rename (NautilusFile *file,
gboolean stop_timer,
GError *error);
static char *
get_truncated_name_for_file (NautilusFile *file)
{
g_autofree char *file_name = NULL;
g_assert (NAUTILUS_IS_FILE (file));
file_name = nautilus_file_get_display_name (file);
return eel_str_middle_truncate (file_name, MAXIMUM_DISPLAYED_FILE_NAME_LENGTH);
}
void
nautilus_report_error_loading_directory (NautilusFile *file,
GError *error,
GtkWindow *parent_window)
{
g_autofree char *truncated_name = NULL;
g_autofree char *message = NULL;
if (error == NULL ||
error->message == NULL)
{
return;
}
if (error->domain == G_IO_ERROR &&
error->code == G_IO_ERROR_NOT_MOUNTED)
{
/* This case is retried automatically */
return;
}
truncated_name = get_truncated_name_for_file (file);
if (error->domain == G_IO_ERROR)
{
switch (error->code)
{
case G_IO_ERROR_PERMISSION_DENIED:
{
message = g_strdup_printf (_("You do not have the permissions necessary to view the contents of “%s”."),
truncated_name);
}
break;
case G_IO_ERROR_NOT_FOUND:
{
message = g_strdup_printf (_("“%s” could not be found. Perhaps it has recently been deleted."),
truncated_name);
}
break;
default:
{
g_autofree char *truncated_error_message = NULL;
truncated_error_message = eel_str_middle_truncate (error->message,
MAXIMUM_DISPLAYED_ERROR_MESSAGE_LENGTH);
message = g_strdup_printf (_("Sorry, could not display all the contents of “%s”: %s"), truncated_name,
truncated_error_message);
}
break;
}
}
else
{
message = g_strdup (error->message);
}
show_dialog (_("This location could not be displayed."),
message,
parent_window,
GTK_MESSAGE_ERROR);
}
void
nautilus_report_error_setting_group (NautilusFile *file,
GError *error,
GtkWindow *parent_window)
{
g_autofree char *truncated_name = NULL;
g_autofree char *message = NULL;
if (error == NULL)
{
return;
}
truncated_name = get_truncated_name_for_file (file);
if (error->domain == G_IO_ERROR)
{
switch (error->code)
{
case G_IO_ERROR_PERMISSION_DENIED:
{
message = g_strdup_printf (_("You do not have the permissions necessary to change the group of “%s”."),
truncated_name);
}
break;
default:
{
}
break;
}
}
if (message == NULL)
{
g_autofree char *truncated_error_message = NULL;
truncated_error_message = eel_str_middle_truncate (error->message,
MAXIMUM_DISPLAYED_ERROR_MESSAGE_LENGTH);
/* We should invent decent error messages for every case we actually experience. */
g_warning ("Hit unhandled case %s:%d in nautilus_report_error_setting_group",
g_quark_to_string (error->domain), error->code);
/* fall through */
message = g_strdup_printf (_("Sorry, could not change the group of “%s”: %s"), truncated_name,
truncated_error_message);
}
show_dialog (_("The group could not be changed."), message, parent_window, GTK_MESSAGE_ERROR);
}
void
nautilus_report_error_setting_owner (NautilusFile *file,
GError *error,
GtkWindow *parent_window)
{
g_autofree char *truncated_name = NULL;
g_autofree char *truncated_error_message = NULL;
g_autofree char *message = NULL;
if (error == NULL)
{
return;
}
truncated_name = get_truncated_name_for_file (file);
truncated_error_message = eel_str_middle_truncate (error->message,
MAXIMUM_DISPLAYED_ERROR_MESSAGE_LENGTH);
message = g_strdup_printf (_("Sorry, could not change the owner of “%s”: %s"),
truncated_name, truncated_error_message);
show_dialog (_("The owner could not be changed."), message, parent_window, GTK_MESSAGE_ERROR);
}
void
nautilus_report_error_setting_permissions (NautilusFile *file,
GError *error,
GtkWindow *parent_window)
{
g_autofree char *truncated_name = NULL;
g_autofree char *truncated_error_message = NULL;
g_autofree char *message = NULL;
if (error == NULL)
{
return;
}
truncated_name = get_truncated_name_for_file (file);
truncated_error_message = eel_str_middle_truncate (error->message,
MAXIMUM_DISPLAYED_ERROR_MESSAGE_LENGTH);
message = g_strdup_printf (_("Sorry, could not change the permissions of “%s”: %s"),
truncated_name, truncated_error_message);
show_dialog (_("The permissions could not be changed."),
message,
parent_window,
GTK_MESSAGE_ERROR);
}
typedef struct _NautilusRenameData
{
char *name;
NautilusFileOperationCallback callback;
gpointer callback_data;
} NautilusRenameData;
void
nautilus_report_error_renaming_file (NautilusFile *file,
const char *new_name,
GError *error,
GtkWindow *parent_window)
{
g_autofree char *truncated_old_name = NULL;
g_autofree char *truncated_new_name = NULL;
g_autofree char *message = NULL;
/* Truncate names for display since very long file names with no spaces
* in them won't get wrapped, and can create insanely wide dialog boxes.
*/
truncated_old_name = get_truncated_name_for_file (file);
truncated_new_name = eel_str_middle_truncate (new_name, MAXIMUM_DISPLAYED_FILE_NAME_LENGTH);
if (error->domain == G_IO_ERROR)
{
switch (error->code)
{
case G_IO_ERROR_EXISTS:
{
message = g_strdup_printf (_("The name “%s” is already used in this location. "
"Please use a different name."),
truncated_new_name);
}
break;
case G_IO_ERROR_NOT_FOUND:
{
message = g_strdup_printf (_("There is no “%s” in this location. "
"Perhaps it was just moved or deleted?"),
truncated_old_name);
}
break;
case G_IO_ERROR_PERMISSION_DENIED:
{
message = g_strdup_printf (_("You do not have the permissions necessary to rename “%s”."),
truncated_old_name);
}
break;
case G_IO_ERROR_INVALID_FILENAME:
{
if (strchr (new_name, '/') != NULL)
{
message = g_strdup_printf (_("The name “%s” is not valid because it contains the character “/”. "
"Please use a different name."),
truncated_new_name);
}
else
{
message = g_strdup_printf (_("The name “%s” is not valid. "
"Please use a different name."),
truncated_new_name);
}
}
break;
case G_IO_ERROR_FILENAME_TOO_LONG:
{
message = g_strdup_printf (_("The name “%s” is too long. "
"Please use a different name."),
truncated_new_name);
}
break;
default:
{
}
break;
}
}
if (message == NULL)
{
g_autofree char *truncated_error_message = NULL;
truncated_error_message = eel_str_middle_truncate (error->message,
MAXIMUM_DISPLAYED_ERROR_MESSAGE_LENGTH);
/* We should invent decent error messages for every case we actually experience. */
g_warning ("Hit unhandled case %s:%d in nautilus_report_error_renaming_file",
g_quark_to_string (error->domain), error->code);
/* fall through */
message = g_strdup_printf (_("Sorry, could not rename “%s” to “%s”: %s"),
truncated_old_name, truncated_new_name,
truncated_error_message);
}
show_dialog (_("The item could not be renamed."), message, parent_window, GTK_MESSAGE_ERROR);
}
static void
nautilus_rename_data_free (NautilusRenameData *data)
{
g_free (data->name);
g_free (data);
}
static void
rename_callback (NautilusFile *file,
GFile *result_location,
GError *error,
gpointer callback_data)
{
NautilusRenameData *data;
gboolean cancelled = FALSE;
g_assert (NAUTILUS_IS_FILE (file));
g_assert (callback_data == NULL);
data = g_object_get_data (G_OBJECT (file), NEW_NAME_TAG);
g_assert (data != NULL);
if (error)
{
if (!(error->domain == G_IO_ERROR && error->code == G_IO_ERROR_CANCELLED))
{
GtkApplication *app = GTK_APPLICATION (g_application_get_default ());
GtkWindow *window = gtk_application_get_active_window (app);
/* If rename failed, notify the user. */
nautilus_report_error_renaming_file (file, data->name, error, window);
}
else
{
cancelled = TRUE;
}
}
finish_rename (file, !cancelled, error);
}
static void
cancel_rename_callback (gpointer callback_data)
{
nautilus_file_cancel (NAUTILUS_FILE (callback_data), rename_callback, NULL);
}
static void
finish_rename (NautilusFile *file,
gboolean stop_timer,
GError *error)
{
NautilusRenameData *data;
data = g_object_get_data (G_OBJECT (file), NEW_NAME_TAG);
if (data == NULL)
{
return;
}
/* Cancel both the rename and the timed wait. */
nautilus_file_cancel (file, rename_callback, NULL);
if (stop_timer)
{
eel_timed_wait_stop (cancel_rename_callback, file);
}
if (data->callback != NULL)
{
data->callback (file, NULL, error, data->callback_data);
}
/* Let go of file name. */
g_object_set_data (G_OBJECT (file), NEW_NAME_TAG, NULL);
}
void
nautilus_rename_file (NautilusFile *file,
const char *new_name,
NautilusFileOperationCallback callback,
gpointer callback_data)
{
g_autoptr (GError) error = NULL;
NautilusRenameData *data;
g_autofree char *truncated_old_name = NULL;
g_autofree char *truncated_new_name = NULL;
g_autofree char *wait_message = NULL;
g_autofree char *uri = NULL;
g_return_if_fail (NAUTILUS_IS_FILE (file));
g_return_if_fail (new_name != NULL);
/* Stop any earlier rename that's already in progress. */
error = g_error_new (G_IO_ERROR, G_IO_ERROR_CANCELLED, "Cancelled");
finish_rename (file, TRUE, error);
data = g_new0 (NautilusRenameData, 1);
data->name = g_strdup (new_name);
data->callback = callback;
data->callback_data = callback_data;
/* Attach the new name to the file. */
g_object_set_data_full (G_OBJECT (file),
NEW_NAME_TAG,
data, (GDestroyNotify) nautilus_rename_data_free);
/* Start the timed wait to cancel the rename. */
truncated_old_name = get_truncated_name_for_file (file);
truncated_new_name = eel_str_middle_truncate (new_name, MAXIMUM_DISPLAYED_FILE_NAME_LENGTH);
wait_message = g_strdup_printf (_("Renaming “%s” to “%s”."),
truncated_old_name,
truncated_new_name);
eel_timed_wait_start (cancel_rename_callback, file, wait_message,
NULL); /* FIXME bugzilla.gnome.org 42395: Parent this? */
uri = nautilus_file_get_uri (file);
DEBUG ("Renaming file %s to %s", uri, new_name);
/* Start the rename. */
nautilus_file_rename (file, new_name,
rename_callback, NULL);
}