[nautilus/wip/undo: 1/2] undo: introduce undo support
- From: Cosimo Cecchi <cosimoc src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [nautilus/wip/undo: 1/2] undo: introduce undo support
- Date: Thu, 19 Jan 2012 18:52:59 +0000 (UTC)
commit 2d06be28efc67c421a09b9f21b2781c8f0455fa3
Author: Amos Brocco <amos brocco gmail com>
Date: Mon Jan 16 22:20:36 2012 -0500
undo: introduce undo support
libnautilus-private/Makefile.am | 5 +
libnautilus-private/nautilus-debug.c | 1 +
libnautilus-private/nautilus-debug.h | 1 +
libnautilus-private/nautilus-file-operations.c | 164 +++-
libnautilus-private/nautilus-file-private.h | 2 +
libnautilus-private/nautilus-file-undo-manager.c | 600 +++++++++
libnautilus-private/nautilus-file-undo-manager.h | 87 ++
.../nautilus-file-undo-operations.c | 1297 ++++++++++++++++++++
.../nautilus-file-undo-operations.h | 78 ++
libnautilus-private/nautilus-file-undo-types.h | 168 +++
libnautilus-private/nautilus-file.c | 62 +-
src/nautilus-directory-view-ui.xml | 4 +
src/nautilus-shell-ui.xml | 2 +
src/nautilus-view.c | 125 ++-
src/nautilus-window-menus.c | 4 +
15 files changed, 2591 insertions(+), 9 deletions(-)
---
diff --git a/libnautilus-private/Makefile.am b/libnautilus-private/Makefile.am
index 6c8e015..1dda15e 100644
--- a/libnautilus-private/Makefile.am
+++ b/libnautilus-private/Makefile.am
@@ -188,6 +188,11 @@ libnautilus_private_la_SOURCES = \
nautilus-vfs-directory.h \
nautilus-vfs-file.c \
nautilus-vfs-file.h \
+ nautilus-file-undo-operations.c \
+ nautilus-file-undo-operations.h \
+ nautilus-file-undo-manager.c \
+ nautilus-file-undo-manager.h \
+ nautilus-file-undo-types.h \
$(nautilus_tracker_engine_sources) \
$(NULL)
diff --git a/libnautilus-private/nautilus-debug.c b/libnautilus-private/nautilus-debug.c
index f4ee2c7..4f874d8 100644
--- a/libnautilus-private/nautilus-debug.c
+++ b/libnautilus-private/nautilus-debug.c
@@ -49,6 +49,7 @@ static GDebugKey keys[] = {
{ "Previewer", NAUTILUS_DEBUG_PREVIEWER },
{ "Smclient", NAUTILUS_DEBUG_SMCLIENT },
{ "Window", NAUTILUS_DEBUG_WINDOW },
+ { "Undo", NAUTILUS_DEBUG_UNDO },
{ 0, }
};
diff --git a/libnautilus-private/nautilus-debug.h b/libnautilus-private/nautilus-debug.h
index 56afff4..ea2e1d1 100644
--- a/libnautilus-private/nautilus-debug.h
+++ b/libnautilus-private/nautilus-debug.h
@@ -46,6 +46,7 @@ typedef enum {
NAUTILUS_DEBUG_PREVIEWER = 1 << 11,
NAUTILUS_DEBUG_SMCLIENT = 1 << 12,
NAUTILUS_DEBUG_WINDOW = 1 << 13,
+ NAUTILUS_DEBUG_UNDO = 1 << 14,
} DebugFlags;
void nautilus_debug_set_flags (DebugFlags flags);
diff --git a/libnautilus-private/nautilus-file-operations.c b/libnautilus-private/nautilus-file-operations.c
index 55fca11..87b3554 100644
--- a/libnautilus-private/nautilus-file-operations.c
+++ b/libnautilus-private/nautilus-file-operations.c
@@ -63,6 +63,8 @@
#include "nautilus-trash-monitor.h"
#include "nautilus-file-utilities.h"
#include "nautilus-file-conflict-dialog.h"
+#include "nautilus-file-undo-operations.h"
+#include "nautilus-file-undo-manager.h"
/* TODO: TESTING!!! */
@@ -76,6 +78,7 @@ typedef struct {
GCancellable *cancellable;
GHashTable *skip_files;
GHashTable *skip_readdir_error;
+ NautilusFileUndoData *undo_redo_data;
gboolean skip_all_error;
gboolean skip_all_conflict;
gboolean merge_all;
@@ -953,6 +956,12 @@ finalize_common (CommonJob *common)
if (common->skip_readdir_error) {
g_hash_table_destroy (common->skip_readdir_error);
}
+
+ if (common->undo_redo_data != NULL) {
+ nautilus_file_undo_manager_add_action (nautilus_file_undo_manager_get (),
+ common->undo_redo_data);
+ }
+
g_object_unref (common->progress);
g_object_unref (common->cancellable);
g_free (common);
@@ -1257,7 +1266,10 @@ static void
abort_job (CommonJob *job)
{
g_cancellable_cancel (job->cancellable);
-
+
+ /* destroy the undo action data too */
+ nautilus_file_undo_data_free (job->undo_redo_data);
+ job->undo_redo_data = NULL;
}
static gboolean
@@ -1752,6 +1764,8 @@ trash_files (CommonJob *job, GList *files, int *files_skipped)
char *primary, *secondary, *details;
int response;
+ guint64 mtime;
+
if (job_aborted (job)) {
return;
}
@@ -1768,6 +1782,9 @@ trash_files (CommonJob *job, GList *files, int *files_skipped)
file = l->data;
error = NULL;
+
+ mtime = nautilus_file_undo_manager_get_file_modification_time (file);
+
if (!g_file_trash (file, job->cancellable, &error)) {
if (job->skip_all_error) {
(*files_skipped)++;
@@ -1814,7 +1831,11 @@ trash_files (CommonJob *job, GList *files, int *files_skipped)
total_files--;
} else {
nautilus_file_changes_queue_file_removed (file);
-
+
+ if (job->undo_redo_data != NULL) {
+ nautilus_file_undo_data_add_trashed_file (job->undo_redo_data, file, mtime);
+ }
+
files_trashed++;
report_trash_progress (job, files_trashed, total_files);
}
@@ -1958,6 +1979,10 @@ trash_or_delete_internal (GList *files,
inhibit_power_manager ((CommonJob *)job, _("Deleting Files"));
}
+ if (try_trash && !nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_MOVE_TO_TRASH, g_list_length(files));
+ }
+
g_io_scheduler_push_job (delete_job,
job,
NULL,
@@ -3367,6 +3392,11 @@ create_dest_dir (CommonJob *job,
return CREATE_DEST_DIR_FAILED;
}
nautilus_file_changes_queue_file_added (*dest);
+
+ if (job->undo_redo_data != NULL) {
+ nautilus_file_undo_data_add_origin_target_pair (job->undo_redo_data, src, *dest);
+ }
+
return CREATE_DEST_DIR_SUCCESS;
}
@@ -4121,7 +4151,11 @@ copy_move_file (CopyMoveJob *copy_job,
dest,
FALSE);
}
-
+
+ if (job->undo_redo_data != NULL) {
+ nautilus_file_undo_data_add_origin_target_pair (job->undo_redo_data, src, dest);
+ }
+
g_object_unref (dest);
return;
}
@@ -4581,6 +4615,18 @@ nautilus_file_operations_copy (GList *files,
inhibit_power_manager ((CommonJob *)job, _("Copying Files"));
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ GFile* src_dir;
+
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_COPY, g_list_length(files));
+ src_dir = g_file_get_parent (files->data);
+
+ nautilus_file_undo_data_set_src_dir (job->common.undo_redo_data, src_dir);
+ nautilus_file_undo_data_set_dest_dir (job->common.undo_redo_data, target_dir);
+
+ g_object_unref (src_dir);
+ }
+
g_io_scheduler_push_job (copy_job,
job,
NULL, /* destroy notify */
@@ -4737,7 +4783,11 @@ move_file_prepare (CopyMoveJob *move_job,
} else {
nautilus_file_changes_queue_schedule_position_remove (dest);
}
-
+
+ if (job->undo_redo_data != NULL) {
+ nautilus_file_undo_data_add_origin_target_pair (job->undo_redo_data, src, dest);
+ }
+
return;
}
@@ -5098,6 +5148,23 @@ nautilus_file_operations_move (GList *files,
inhibit_power_manager ((CommonJob *)job, _("Moving Files"));
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ GFile* src_dir;
+
+ if (g_file_has_uri_scheme (g_list_first(files)->data, "trash")) {
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_RESTORE_FROM_TRASH, g_list_length(files));
+ } else {
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_MOVE, g_list_length(files));
+ }
+
+ src_dir = g_file_get_parent (files->data);
+
+ nautilus_file_undo_data_set_src_dir (job->common.undo_redo_data, src_dir);
+ nautilus_file_undo_data_set_dest_dir (job->common.undo_redo_data, target_dir);
+
+ g_object_unref (src_dir);
+ }
+
g_io_scheduler_push_job (move_job,
job,
NULL, /* destroy notify */
@@ -5191,6 +5258,11 @@ link_file (CopyMoveJob *job,
path,
common->cancellable,
&error)) {
+
+ if (common->undo_redo_data != NULL) {
+ nautilus_file_undo_data_add_origin_target_pair (common->undo_redo_data, src, dest);
+ }
+
g_free (path);
if (debuting_files) {
g_hash_table_replace (debuting_files, g_object_ref (dest), GINT_TO_POINTER (TRUE));
@@ -5396,6 +5468,18 @@ nautilus_file_operations_link (GList *files,
}
job->debuting_files = g_hash_table_new_full (g_file_hash, (GEqualFunc)g_file_equal, g_object_unref, NULL);
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ GFile* src_dir;
+
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_CREATE_LINK, g_list_length(files));
+ src_dir = g_file_get_parent (files->data);
+
+ nautilus_file_undo_data_set_src_dir (job->common.undo_redo_data, src_dir);
+ nautilus_file_undo_data_set_dest_dir (job->common.undo_redo_data, target_dir);
+
+ g_object_unref (src_dir);
+ }
+
g_io_scheduler_push_job (link_job,
job,
NULL, /* destroy notify */
@@ -5427,6 +5511,19 @@ nautilus_file_operations_duplicate (GList *files,
}
job->debuting_files = g_hash_table_new_full (g_file_hash, (GEqualFunc)g_file_equal, g_object_unref, NULL);
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ GFile* src_dir;
+
+ job->common.undo_redo_data =
+ nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_DUPLICATE, g_list_length(files));
+ src_dir = g_file_get_parent (files->data);
+
+ nautilus_file_undo_data_set_src_dir (job->common.undo_redo_data, src_dir);
+ nautilus_file_undo_data_set_dest_dir (job->common.undo_redo_data, src_dir);
+
+ g_object_unref (src_dir);
+ }
+
g_io_scheduler_push_job (copy_job,
job,
NULL, /* destroy notify */
@@ -5496,6 +5593,11 @@ set_permissions_file (SetPermissionsJob *job,
if (!job_aborted (common) &&
g_file_info_has_attribute (info, G_FILE_ATTRIBUTE_UNIX_MODE)) {
current = g_file_info_get_attribute_uint32 (info, G_FILE_ATTRIBUTE_UNIX_MODE);
+
+ if (common->undo_redo_data != NULL) {
+ nautilus_file_undo_data_add_file_permissions (common->undo_redo_data, file, current);
+ }
+
current = (current & ~mask) | value;
g_file_set_attribute_uint32 (file, G_FILE_ATTRIBUTE_UNIX_MODE,
@@ -5578,7 +5680,16 @@ nautilus_file_set_permissions_recursive (const char *directory,
job->dir_mask = dir_mask;
job->done_callback = callback;
job->done_callback_data = callback_data;
-
+
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_RECURSIVE_SET_PERMISSIONS, 1);
+
+ nautilus_file_undo_data_set_recursive_permissions_dest_dir (job->common.undo_redo_data, job->file);
+ nautilus_file_undo_data_set_recursive_permissions
+ (job->common.undo_redo_data,
+ file_permissions, file_mask, dir_permissions, dir_mask);
+ }
+
g_io_scheduler_push_job (set_permissions_job,
job,
NULL,
@@ -5669,6 +5780,7 @@ nautilus_file_operations_copy_move (const GList *item_uris,
if (target_dir == NULL ||
(src_dir != NULL &&
g_file_equal (src_dir, dest))) {
+
nautilus_file_operations_duplicate (locations,
relative_item_points,
parent_window,
@@ -5691,11 +5803,13 @@ nautilus_file_operations_copy_move (const GList *item_uris,
cb_data = g_slice_new0 (MoveTrashCBData);
cb_data->real_callback = done_callback;
cb_data->real_data = done_callback_data;
+
nautilus_file_operations_trash_or_delete (locations,
parent_window,
(NautilusDeleteCallback) callback_for_move_to_trash,
cb_data);
} else {
+
nautilus_file_operations_move (locations,
relative_item_points,
dest,
@@ -5703,6 +5817,7 @@ nautilus_file_operations_copy_move (const GList *item_uris,
done_callback, done_callback_data);
}
} else {
+
nautilus_file_operations_link (locations,
relative_item_points,
dest,
@@ -5829,6 +5944,13 @@ create_job (GIOSchedulerJob *io_job,
res = g_file_make_directory (dest,
common->cancellable,
&error);
+
+ if (res && common->undo_redo_data != NULL) {
+ nautilus_file_undo_data_set_create_data (common->undo_redo_data,
+ dest,
+ NULL);
+ }
+
} else {
if (job->src) {
res = g_file_copy (job->src,
@@ -5837,6 +5959,17 @@ create_job (GIOSchedulerJob *io_job,
common->cancellable,
NULL, NULL,
&error);
+
+ if (res && common->undo_redo_data != NULL) {
+ gchar *uri;
+
+ uri = g_file_get_uri (job->src);
+ nautilus_file_undo_data_set_create_data (common->undo_redo_data,
+ dest, uri);
+
+ g_free (uri);
+ }
+
} else {
data = "";
length = 0;
@@ -5859,6 +5992,11 @@ create_job (GIOSchedulerJob *io_job,
res = g_output_stream_close (G_OUTPUT_STREAM (out),
common->cancellable,
&error);
+
+ if (res && common->undo_redo_data != NULL) {
+ nautilus_file_undo_data_set_create_data (common->undo_redo_data,
+ dest, data);
+ }
}
/* This will close if the write failed and we didn't close */
@@ -6029,6 +6167,10 @@ nautilus_file_operations_new_folder (GtkWidget *parent_view,
job->has_position = TRUE;
}
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_CREATE_FOLDER, 1);
+ }
+
g_io_scheduler_push_job (create_job,
job,
NULL, /* destroy notify */
@@ -6067,6 +6209,10 @@ nautilus_file_operations_new_file_from_template (GtkWidget *parent_view,
job->src = g_file_new_for_uri (template_uri);
}
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_CREATE_FILE_FROM_TEMPLATE, 1);
+ }
+
g_io_scheduler_push_job (create_job,
job,
NULL, /* destroy notify */
@@ -6104,6 +6250,10 @@ nautilus_file_operations_new_file (GtkWidget *parent_view,
job->length = length;
job->filename = g_strdup (target_filename);
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ job->common.undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_CREATE_EMPTY_FILE, length);
+ }
+
g_io_scheduler_push_job (create_job,
job,
NULL, /* destroy notify */
@@ -6148,7 +6298,7 @@ delete_trash_file (CommonJob *job,
g_object_unref (enumerator);
}
}
-
+
if (!job_aborted (job) && del_file) {
g_file_delete (file, job->cancellable, NULL);
}
@@ -6166,6 +6316,8 @@ empty_trash_job_done (gpointer user_data)
if (job->done_callback) {
job->done_callback (job->done_callback_data);
}
+
+ nautilus_file_undo_manager_trash_has_emptied (nautilus_file_undo_manager_get ());
finalize_common ((CommonJob *)job);
return FALSE;
diff --git a/libnautilus-private/nautilus-file-private.h b/libnautilus-private/nautilus-file-private.h
index cc14b3e..f06df02 100644
--- a/libnautilus-private/nautilus-file-private.h
+++ b/libnautilus-private/nautilus-file-private.h
@@ -28,6 +28,7 @@
#include <libnautilus-private/nautilus-directory.h>
#include <libnautilus-private/nautilus-file.h>
#include <libnautilus-private/nautilus-monitor.h>
+#include <libnautilus-private/nautilus-file-undo-types.h>
#include <eel/eel-glib-extensions.h>
#include <eel/eel-string.h>
@@ -230,6 +231,7 @@ typedef struct {
gpointer data;
GDestroyNotify free_data;
+ NautilusFileUndoData *undo_redo_data;
} NautilusFileOperation;
NautilusFile *nautilus_file_new_from_info (NautilusDirectory *directory,
diff --git a/libnautilus-private/nautilus-file-undo-manager.c b/libnautilus-private/nautilus-file-undo-manager.c
new file mode 100644
index 0000000..81a451d
--- /dev/null
+++ b/libnautilus-private/nautilus-file-undo-manager.c
@@ -0,0 +1,600 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+
+/* nautilus-file-undo-manager.c - Manages the undo/redo stack
+ *
+ * Copyright (C) 2007-2011 Amos Brocco
+ * Copyright (C) 2010 Red Hat, Inc.
+ *
+ * This library is free software; you can redistribute it and/or
+ * 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.
+ *
+ * This 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
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ * Authors: Amos Brocco <amos brocco gmail com>
+ * Cosimo Cecchi <cosimoc redhat com>
+ *
+ */
+
+#include <config.h>
+
+#include "nautilus-file-undo-manager.h"
+
+#include "nautilus-file-operations.h"
+#include "nautilus-file.h"
+#include "nautilus-file-undo-operations.h"
+#include "nautilus-file-undo-types.h"
+
+#include <gio/gio.h>
+#include <glib.h>
+#include <glib/gi18n.h>
+#include <locale.h>
+#include <gdk/gdk.h>
+#include <eel/eel-glib-extensions.h>
+
+#define DEBUG_FLAG NAUTILUS_DEBUG_UNDO
+#include "nautilus-debug.h"
+
+/* Default depth of the undo/redo stack. */
+#define DEFAULT_UNDO_DEPTH 32
+
+#define NAUTILUS_FILE_UNDO_MANAGER_GET_PRIVATE(o) \
+ (G_TYPE_INSTANCE_GET_PRIVATE ((o), NAUTILUS_TYPE_FILE_UNDO_MANAGER, NautilusFileUndoManagerPrivate))
+
+enum {
+ SIGNAL_UNDO_CHANGED,
+ NUM_SIGNALS,
+};
+
+static guint signals[NUM_SIGNALS] = { 0, };
+
+G_DEFINE_TYPE (NautilusFileUndoManager, nautilus_file_undo_manager, G_TYPE_OBJECT)
+
+static NautilusFileUndoManager *singleton = NULL;
+
+/* must be called with the mutex locked */
+static NautilusFileUndoData *
+get_next_redo_action (NautilusFileUndoManagerPrivate *priv)
+{
+ NautilusFileUndoData *retval = NULL;
+
+ if (g_queue_is_empty (priv->stack)) {
+ DEBUG ("Queue is empty, no redo actions to return");
+ goto out;
+ }
+
+ if (priv->index == 0) {
+ DEBUG ("Queue has only undoable actions, no redo actions to return");
+ goto out;
+ }
+
+ DEBUG ("Getting redo action for index %d", priv->index);
+ retval = g_queue_peek_nth (priv->stack, priv->index - 1);
+
+ if (retval && retval->locked) {
+ DEBUG ("Action is locked, no redo actions to return");
+ retval = NULL;
+ }
+
+ out:
+ DEBUG ("Returning %p as next redo action", retval);
+ return retval;
+}
+
+/* must be called with the mutex locked */
+static NautilusFileUndoData *
+get_next_undo_action (NautilusFileUndoManagerPrivate *priv)
+{
+ NautilusFileUndoData *retval = NULL;
+ guint stack_size;
+
+ if (g_queue_is_empty (priv->stack)) {
+ DEBUG ("Queue is empty, no undo actions to return");
+ goto out;
+ }
+
+ stack_size = g_queue_get_length (priv->stack);
+
+ if (priv->index == stack_size) {
+ DEBUG ("Queue has only of undone actions, no undo actions to return");
+ goto out;
+ }
+
+ DEBUG ("Getting undo action for index %d", priv->index);
+ retval = g_queue_peek_nth (priv->stack, priv->index);
+
+ if (retval->locked) {
+ DEBUG ("Action is locked, no undo actions to return");
+ retval = NULL;
+ }
+
+ out:
+ DEBUG ("Returning %p as next undo action", retval);
+ return retval;
+}
+
+/* must be called with the mutex locked */
+static void
+clear_redo_actions (NautilusFileUndoManagerPrivate *priv)
+{
+ DEBUG ("Clearing redo actions, index is %d", priv->index);
+
+ while (priv->index > 0) {
+ NautilusFileUndoData *head;
+
+ head = g_queue_pop_head (priv->stack);
+ nautilus_file_undo_data_free (head);
+
+ DEBUG ("Removed action, index was %d", priv->index);
+
+ priv->index--;
+ }
+}
+
+/* must be called with the mutex locked */
+static gboolean
+can_undo (NautilusFileUndoManagerPrivate *priv)
+{
+ return (get_next_undo_action (priv) != NULL);
+}
+
+/* must be called with the mutex locked */
+static gboolean
+can_redo (NautilusFileUndoManagerPrivate *priv)
+{
+ return (get_next_redo_action (priv) != NULL);
+}
+
+/* must be called with the mutex locked */
+static NautilusFileUndoData *
+stack_scroll_right (NautilusFileUndoManagerPrivate *priv)
+{
+ NautilusFileUndoData *data;
+
+ if (!can_undo (priv)) {
+ DEBUG ("Scrolling stack right, but no undo actions");
+ return NULL;
+ }
+
+ data = g_queue_peek_nth (priv->stack, priv->index);
+
+ if (priv->index < g_queue_get_length (priv->stack)) {
+ priv->index++;
+ DEBUG ("Scrolling stack right, increasing index to %d", priv->index);
+ }
+
+ return data;
+}
+
+/* must be called with the mutex locked */
+static NautilusFileUndoData *
+stack_scroll_left (NautilusFileUndoManagerPrivate *priv)
+{
+ NautilusFileUndoData *data;
+
+ if (!can_redo (priv)) {
+ DEBUG ("Scrolling stack left, but no redo actions");
+ return NULL;
+ }
+
+ priv->index--;
+ data = g_queue_peek_nth (priv->stack, priv->index);
+
+ DEBUG ("Scrolling stack left, decreasing index to %d", priv->index);
+
+ return data;
+}
+
+/* must be called with the mutex locked */
+static void
+stack_clear_n_oldest (GQueue *stack,
+ guint n)
+{
+ NautilusFileUndoData *action;
+ guint i;
+
+ DEBUG ("Clearing %u oldest actions from the undo stack", n);
+
+ for (i = 0; i < n; i++) {
+ action = (NautilusFileUndoData *) g_queue_pop_tail (stack);
+ if (action->locked) {
+ action->freed = TRUE;
+ } else {
+ nautilus_file_undo_data_free (action);
+ }
+ }
+}
+
+/* must be called with the mutex locked */
+static void
+stack_ensure_size (NautilusFileUndoManager *self)
+{
+ NautilusFileUndoManagerPrivate *priv = self->priv;
+ guint length;
+
+ length = g_queue_get_length (priv->stack);
+
+ if (length > priv->undo_levels) {
+ if (priv->index > (priv->undo_levels + 1)) {
+ /* If the index will fall off the stack
+ * move it back to the maximum position */
+ priv->index = priv->undo_levels + 1;
+ }
+ stack_clear_n_oldest (priv->stack, length - (priv->undo_levels));
+ }
+}
+
+/* must be called with the mutex locked */
+static void
+stack_push_action (NautilusFileUndoManager *self,
+ NautilusFileUndoData *action)
+{
+ NautilusFileUndoManagerPrivate *priv = self->priv;
+
+ /* clear all the redo items, as we're pushing a new undoable action */
+ clear_redo_actions (priv);
+
+ g_queue_push_head (priv->stack, action);
+
+ /* cleanup items in excess, if any */
+ stack_ensure_size (self);
+}
+
+static void
+nautilus_file_undo_manager_init (NautilusFileUndoManager * self)
+{
+ NautilusFileUndoManagerPrivate *priv;
+
+ priv = NAUTILUS_FILE_UNDO_MANAGER_GET_PRIVATE (self);
+
+ self->priv = priv;
+
+ /* Initialize private fields */
+ priv->stack = g_queue_new ();
+ g_mutex_init(&priv->mutex);
+ priv->index = 0;
+ priv->undo_redo_flag = FALSE;
+
+ /* initialize the undo stack depth */
+ priv->undo_levels = DEFAULT_UNDO_DEPTH;
+
+ g_mutex_lock (&priv->mutex);
+ stack_ensure_size (self);
+ g_mutex_unlock (&priv->mutex);
+}
+
+static void
+nautilus_file_undo_manager_finalize (GObject * object)
+{
+ NautilusFileUndoManager *self = NAUTILUS_FILE_UNDO_MANAGER (object);
+ NautilusFileUndoManagerPrivate *priv = self->priv;
+
+ g_mutex_lock (&priv->mutex);
+
+ g_queue_foreach (priv->stack, (GFunc) nautilus_file_undo_data_free, NULL);
+ g_queue_free (priv->stack);
+
+ g_mutex_unlock (&priv->mutex);
+ g_mutex_clear (&priv->mutex);
+
+ G_OBJECT_CLASS (nautilus_file_undo_manager_parent_class)->finalize (object);
+}
+
+static GObject *
+nautilus_file_undo_manager_constructor (GType type,
+ guint n_construct_params,
+ GObjectConstructParam *construct_params)
+{
+ GObject *retval;
+
+ if (singleton != NULL) {
+ return G_OBJECT (singleton);
+ }
+
+ retval = G_OBJECT_CLASS (nautilus_file_undo_manager_parent_class)->constructor
+ (type, n_construct_params, construct_params);
+
+ singleton = NAUTILUS_FILE_UNDO_MANAGER (retval);
+ g_object_add_weak_pointer (retval, (gpointer) &singleton);
+
+ return retval;
+}
+
+static void
+nautilus_file_undo_manager_class_init (NautilusFileUndoManagerClass *klass)
+{
+ GObjectClass *oclass;
+
+ oclass = G_OBJECT_CLASS (klass);
+
+ oclass->constructor = nautilus_file_undo_manager_constructor;
+ oclass->finalize = nautilus_file_undo_manager_finalize;
+
+ signals[SIGNAL_UNDO_CHANGED] =
+ g_signal_new ("undo-changed",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0, NULL, NULL,
+ g_cclosure_marshal_VOID__VOID,
+ G_TYPE_NONE, 0);
+
+ g_type_class_add_private (klass, sizeof (NautilusFileUndoManagerPrivate));
+}
+
+void
+nautilus_file_undo_manager_trash_has_emptied (NautilusFileUndoManager *manager)
+{
+ NautilusFileUndoManagerPrivate *priv;
+ NautilusFileUndoData *action;
+ guint newest_move_to_trash_position;
+ guint i, length;
+
+ priv = manager->priv;
+
+ g_mutex_lock (&priv->mutex);
+
+ /* Clear actions from the oldest to the newest move to trash */
+ clear_redo_actions (priv);
+
+ /* Search newest move to trash */
+ length = g_queue_get_length (priv->stack);
+ newest_move_to_trash_position = -1;
+ action = NULL;
+
+ for (i = 0; i < length; i++) {
+ action = (NautilusFileUndoData *)g_queue_peek_nth (priv->stack, i);
+ if (action->type == NAUTILUS_FILE_UNDO_MOVE_TO_TRASH) {
+ newest_move_to_trash_position = i;
+ break;
+ }
+ }
+
+ if (newest_move_to_trash_position >= 0) {
+ guint to_clear;
+ to_clear = length - newest_move_to_trash_position;
+ stack_clear_n_oldest (priv->stack, to_clear);
+ }
+
+ g_mutex_unlock (&priv->mutex);
+}
+
+guint64
+nautilus_file_undo_manager_get_file_modification_time (GFile * file)
+{
+ GFileInfo *info;
+ guint64 mtime;
+
+ /* TODO: Synch-I/O, Error checking. */
+ info = g_file_query_info (file, G_FILE_ATTRIBUTE_TIME_MODIFIED, G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS, FALSE, NULL);
+ if (info == NULL) {
+ return -1;
+ }
+
+ mtime = g_file_info_get_attribute_uint64 (info, G_FILE_ATTRIBUTE_TIME_MODIFIED);
+
+ g_object_unref (info);
+
+ return mtime;
+}
+
+static void
+refresh_strings (NautilusFileUndoData *action)
+{
+ gchar **descriptions;
+ gchar **labels;
+
+ descriptions = (gchar **) g_malloc0 (3 * sizeof (gchar *));
+ descriptions[2] = NULL;
+
+ labels = (gchar **) g_malloc0 (3 * sizeof (gchar *));
+ labels[2] = NULL;
+
+ action->strings_func (action, action->count,
+ labels, descriptions);
+
+ action->undo_label = labels[0];
+ action->redo_label = labels[1];
+
+ action->undo_description = descriptions[0];
+ action->redo_description = descriptions[1];
+
+ g_free (descriptions);
+ g_free (labels);
+}
+
+static const gchar *
+get_redo_label (NautilusFileUndoData *action)
+{
+ if (action->redo_label == NULL) {
+ refresh_strings (action);
+ }
+
+ return action->redo_label;
+}
+
+static const gchar *
+get_undo_label (NautilusFileUndoData *action)
+{
+ if (action->undo_label == NULL) {
+ refresh_strings (action);
+ }
+
+ return action->undo_label;
+}
+
+static const char *
+get_undo_description (NautilusFileUndoData *action)
+{
+ if (action->undo_description == NULL) {
+ refresh_strings (action);
+ }
+
+ return action->undo_description;
+}
+
+static const char *
+get_redo_description (NautilusFileUndoData *action)
+{
+ if (action->redo_description == NULL) {
+ refresh_strings (action);
+ }
+
+ return action->redo_description;
+}
+
+static void
+do_undo_redo (NautilusFileUndoManager *self,
+ gboolean undo,
+ NautilusFileUndoFinishCallback callback,
+ gpointer user_data)
+{
+ NautilusFileUndoManagerPrivate *priv = self->priv;
+ NautilusFileUndoData *action;
+
+ /* Update the menus invalidating undo/redo while an operation is already underway */
+ g_mutex_lock (&priv->mutex);
+
+ if (undo) {
+ action = stack_scroll_right (priv);
+ } else {
+ action = stack_scroll_left (priv);
+ }
+
+ /* Action will be NULL if redo is not possible */
+ if (action != NULL) {
+ action->locked = TRUE; /* Remember to unlock when finished */
+ }
+
+ g_mutex_unlock (&priv->mutex);
+
+ g_signal_emit (self, signals[SIGNAL_UNDO_CHANGED], 0);
+
+ if (action != NULL) {
+ priv->undo_redo_flag = TRUE;
+
+ if (undo) {
+ action->undo_func (action);
+ } else {
+ action->redo_func (action);
+ }
+ }
+
+ if (callback != NULL) {
+ callback (user_data);
+ }
+}
+
+void
+nautilus_file_undo_manager_redo (NautilusFileUndoManager *manager,
+ NautilusFileUndoFinishCallback callback,
+ gpointer user_data)
+{
+ do_undo_redo (manager, FALSE, callback, user_data);
+}
+
+void
+nautilus_file_undo_manager_undo (NautilusFileUndoManager *manager,
+ NautilusFileUndoFinishCallback callback,
+ gpointer user_data)
+{
+ do_undo_redo (manager, TRUE, callback, user_data);
+}
+
+void
+nautilus_file_undo_manager_add_action (NautilusFileUndoManager *self,
+ NautilusFileUndoData *action)
+{
+ NautilusFileUndoManagerPrivate *priv;
+
+ priv = self->priv;
+
+ DEBUG ("Adding action %p, type %d", action, action->type);
+
+ if (!(action && action->is_valid)) {
+ DEBUG ("Action %p is not valid, ignoring", action);
+ nautilus_file_undo_data_free (action);
+ return;
+ }
+
+ action->manager = self;
+
+ g_mutex_lock (&priv->mutex);
+ stack_push_action (self, action);
+ g_mutex_unlock (&priv->mutex);
+
+ g_signal_emit (self, signals[SIGNAL_UNDO_CHANGED], 0);
+}
+
+gboolean
+nautilus_file_undo_manager_is_undo_redo (NautilusFileUndoManager *manager)
+{
+ NautilusFileUndoManagerPrivate *priv;
+
+ priv = manager->priv;
+
+ if (priv->undo_redo_flag) {
+ priv->undo_redo_flag = FALSE;
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+NautilusFileUndoManager *
+nautilus_file_undo_manager_get (void)
+{
+ NautilusFileUndoManager *manager;
+
+ manager = g_object_new (NAUTILUS_TYPE_FILE_UNDO_MANAGER,
+ NULL);
+
+ return manager;
+}
+
+NautilusFileUndoMenuData *
+nautilus_file_undo_manager_get_menu_data (NautilusFileUndoManager *self)
+{
+ NautilusFileUndoData *action;
+ NautilusFileUndoManagerPrivate *priv;
+ NautilusFileUndoMenuData *data;
+
+ priv = self->priv;
+ data = g_slice_new0 (NautilusFileUndoMenuData);
+
+ DEBUG ("Getting menu data");
+
+ g_mutex_lock (&priv->mutex);
+
+ action = get_next_undo_action (priv);
+
+ if (action != NULL) {
+ data->undo_label = get_undo_label (action);
+ data->undo_description = get_undo_description (action);
+ }
+
+ action = get_next_redo_action (priv);
+
+ if (action != NULL) {
+ data->redo_label = get_redo_label (action);
+ data->redo_description = get_redo_description (action);
+ }
+
+ g_mutex_unlock (&priv->mutex);
+
+ return data;
+}
+
+void
+nautilus_file_undo_menu_data_free (NautilusFileUndoMenuData *data)
+{
+ g_slice_free (NautilusFileUndoMenuData, data);
+}
diff --git a/libnautilus-private/nautilus-file-undo-manager.h b/libnautilus-private/nautilus-file-undo-manager.h
new file mode 100644
index 0000000..f6a63a9
--- /dev/null
+++ b/libnautilus-private/nautilus-file-undo-manager.h
@@ -0,0 +1,87 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+
+/* nautilus-file-undo-manager.h - Manages the undo/redo stack
+ *
+ * Copyright (C) 2007-2011 Amos Brocco
+ *
+ * This library is free software; you can redistribute it and/or
+ * 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.
+ *
+ * This 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
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ * Author: Amos Brocco <amos brocco gmail com>
+ */
+
+#ifndef __NAUTILUS_FILE_UNDO_MANAGER_H__
+#define __NAUTILUS_FILE_UNDO_MANAGER_H__
+
+#include <glib.h>
+#include <glib-object.h>
+#include <gtk/gtk.h>
+#include <gio/gio.h>
+
+#include <libnautilus-private/nautilus-file-undo-types.h>
+
+typedef struct _NautilusFileUndoManagerClass NautilusFileUndoManagerClass;
+typedef struct _NautilusFileUndoManagerPrivate NautilusFileUndoManagerPrivate;
+
+#define NAUTILUS_TYPE_FILE_UNDO_MANAGER\
+ (nautilus_file_undo_manager_get_type())
+#define NAUTILUS_FILE_UNDO_MANAGER(object)\
+ (G_TYPE_CHECK_INSTANCE_CAST((object), NAUTILUS_TYPE_FILE_UNDO_MANAGER,\
+ NautilusFileUndoManager))
+#define NAUTILUS_FILE_UNDO_MANAGER_CLASS(klass)\
+ (G_TYPE_CHECK_CLASS_CAST((klass), NAUTILUS_TYPE_FILE_UNDO_MANAGER,\
+ NautilusFileUndoManagerClass))
+#define NAUTILUS_IS_FILE_UNDO_MANAGER(object)\
+ (G_TYPE_CHECK_INSTANCE_TYPE((object), NAUTILUS_TYPE_FILE_UNDO_MANAGER))
+#define NAUTILUS_IS_FILE_UNDO_MANAGER_CLASS(klass)\
+ (G_TYPE_CHECK_CLASS_TYPE((klass), NAUTILUS_TYPE_FILE_UNDO_MANAGER))
+#define NAUTILUS_FILE_UNDO_MANAGER_GET_CLASS(object)\
+ (G_TYPE_INSTANCE_GET_CLASS((object), NAUTILUS_TYPE_FILE_UNDO_MANAGER,\
+ NautilusFileUndoManagerClass))
+
+struct _NautilusFileUndoManager {
+ GObject parent_instance;
+
+ /* < private > */
+ NautilusFileUndoManagerPrivate* priv;
+};
+
+struct _NautilusFileUndoManagerClass {
+ GObjectClass parent_class;
+};
+
+
+GType nautilus_file_undo_manager_get_type (void) G_GNUC_CONST;
+
+NautilusFileUndoManager * nautilus_file_undo_manager_get (void);
+
+void nautilus_file_undo_manager_add_action (NautilusFileUndoManager *manager,
+ NautilusFileUndoData *action);
+void nautilus_file_undo_manager_undo (NautilusFileUndoManager *manager,
+ NautilusFileUndoFinishCallback callback,
+ gpointer user_data);
+void nautilus_file_undo_manager_redo (NautilusFileUndoManager *manager,
+ NautilusFileUndoFinishCallback callback,
+ gpointer user_data);
+
+gboolean nautilus_file_undo_manager_is_undo_redo (NautilusFileUndoManager *manager);
+void nautilus_file_undo_manager_trash_has_emptied (NautilusFileUndoManager *manager);
+void nautilus_file_undo_manager_request_menu_update (NautilusFileUndoManager *manager);
+guint64 nautilus_file_undo_manager_get_file_modification_time (GFile *file);
+
+NautilusFileUndoMenuData * nautilus_file_undo_manager_get_menu_data (NautilusFileUndoManager *self);
+void nautilus_file_undo_menu_data_free (NautilusFileUndoMenuData *data);
+
+#endif /* __NAUTILUS_FILE_UNDO_MANAGER_H__ */
diff --git a/libnautilus-private/nautilus-file-undo-operations.c b/libnautilus-private/nautilus-file-undo-operations.c
new file mode 100644
index 0000000..f2d61c9
--- /dev/null
+++ b/libnautilus-private/nautilus-file-undo-operations.c
@@ -0,0 +1,1297 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+
+/* nautilus-file-undo-operations.c - Manages undo/redo of file operations
+ *
+ * Copyright (C) 2007-2011 Amos Brocco
+ * Copyright (C) 2010 Red Hat, Inc.
+ *
+ * This library is free software; you can redistribute it and/or
+ * 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.
+ *
+ * This 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
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ * Authors: Amos Brocco <amos brocco gmail com>
+ * Cosimo Cecchi <cosimoc redhat com>
+ *
+ */
+
+#include <config.h>
+
+#include "nautilus-file-undo-operations.h"
+
+#include <glib/gi18n.h>
+
+#include "nautilus-file-operations.h"
+#include "nautilus-file.h"
+#include "nautilus-file-undo-manager.h"
+#include "nautilus-file-undo-types.h"
+
+void
+nautilus_file_undo_data_free (NautilusFileUndoData *action)
+{
+ g_return_if_fail (action != NULL);
+
+ g_free (action->undo_label);
+ g_free (action->undo_description);
+ g_free (action->redo_label);
+ g_free (action->redo_description);
+
+ action->finalize_func (action);
+}
+
+static char *
+get_first_target_short_name (NautilusFileUndoDataExt *eaction)
+{
+ GList *targets_first;
+ char *file_name = NULL;
+
+ targets_first = g_list_first (eaction->destinations);
+
+ if (targets_first != NULL &&
+ targets_first->data != NULL) {
+ file_name = g_file_get_basename (targets_first->data);
+ }
+
+ return file_name;
+}
+
+static GList *
+uri_list_to_gfile_list (GList * urilist)
+{
+ const GList *l;
+ GList *file_list = NULL;
+ GFile *file;
+
+ for (l = urilist; l != NULL; l = l->next) {
+ file = g_file_new_for_uri (l->data);
+ file_list = g_list_append (file_list, file);
+ }
+
+ return file_list;
+}
+
+/* TODO: Synch-I/O, error handling */
+static GHashTable *
+retrieve_files_to_restore (GHashTable * trashed)
+{
+ GFileEnumerator *enumerator;
+ GHashTable *to_restore;
+ GFile *trash;
+
+ to_restore = g_hash_table_new_full (g_direct_hash, g_direct_equal, g_object_unref, g_free);
+
+ trash = g_file_new_for_uri ("trash:///");
+
+ enumerator = g_file_enumerate_children (trash,
+ G_FILE_ATTRIBUTE_STANDARD_NAME","
+ G_FILE_ATTRIBUTE_TIME_MODIFIED","
+ G_FILE_ATTRIBUTE_TRASH_ORIG_PATH,
+ G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS,
+ NULL, NULL);
+
+ if (enumerator) {
+ GFileInfo *info;
+ guint64 *mtime;
+ gpointer lookupvalue;
+ GFile *item;
+ guint64 mtime_item;
+ char *origpath;
+ GFile *origfile;
+ char *origuri;
+
+ mtime = 0;
+ while ((info = g_file_enumerator_next_file (enumerator, NULL, NULL)) != NULL) {
+ /* Retrieve the original file uri */
+ origpath = g_file_info_get_attribute_as_string (info, G_FILE_ATTRIBUTE_TRASH_ORIG_PATH);
+ origfile = g_file_new_for_path (origpath);
+ origuri = g_file_get_uri (origfile);
+ g_object_unref (origfile);
+ g_free (origpath);
+
+ lookupvalue = g_hash_table_lookup (trashed, origuri);
+
+ if (lookupvalue) {
+ mtime = (guint64 *)lookupvalue;
+ mtime_item = g_file_info_get_attribute_uint64 (info, G_FILE_ATTRIBUTE_TIME_MODIFIED);
+ if (*mtime == mtime_item) {
+ /* File in the trash */
+ item = g_file_get_child (trash, g_file_info_get_name (info));
+ g_hash_table_insert (to_restore, item, origuri);
+ }
+ } else {
+ g_free (origuri);
+ }
+
+ }
+ g_file_enumerator_close (enumerator, FALSE, NULL);
+ g_object_unref (enumerator);
+ }
+ g_object_unref (trash);
+
+ return to_restore;
+}
+
+static void
+undo_redo_done_transfer_callback (GHashTable * debuting_uris,
+ gpointer data)
+{
+ NautilusFileUndoData *action = data;
+ NautilusFileUndoManager *manager = action->manager;
+
+ /* If the action needed to be freed but was locked, free now */
+ if (action->freed) {
+ nautilus_file_undo_data_free (action);
+ } else {
+ action->locked = FALSE;
+ }
+
+ /* Update menus */
+ g_signal_emit_by_name (manager, "undo-changed", 0);
+}
+
+static void
+undo_redo_done_rename_callback (NautilusFile * file,
+ GFile * result_location,
+ GError * error,
+ gpointer callback_data)
+{
+ undo_redo_done_transfer_callback (NULL, callback_data);
+}
+
+static void
+undo_redo_done_create_callback (GFile * new_file,
+ gpointer callback_data)
+{
+ undo_redo_done_transfer_callback (NULL, callback_data);
+}
+
+static void
+undo_redo_done_delete_callback (GHashTable *debuting_uris,
+ gboolean user_cancel,
+ gpointer callback_data)
+{
+ undo_redo_done_transfer_callback (debuting_uris, callback_data);
+}
+
+static void
+undo_redo_op_callback (gpointer callback_data)
+{
+ undo_redo_done_transfer_callback (NULL, callback_data);
+}
+
+/* undo helpers */
+
+static void
+delete_files (NautilusFileUndoManager *self,
+ GList *files,
+ NautilusFileUndoData *action)
+{
+ nautilus_file_operations_delete (files, NULL,
+ undo_redo_done_delete_callback, action);
+}
+
+static void
+create_undo_func (NautilusFileUndoData *action)
+{
+ GList *files = NULL;
+ NautilusFileUndoDataCreate* eaction = (NautilusFileUndoDataCreate*) action;
+
+ files = g_list_append (files, g_object_ref (eaction->target_file));
+ delete_files (action->manager, files, action);
+
+ g_list_free_full (files, g_object_unref);
+}
+
+static void
+copy_or_link_undo_func (NautilusFileUndoData *action)
+{
+ GList *files;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ files = g_list_copy (eaction->destinations);
+ files = g_list_reverse (files); /* Deleting must be done in reverse */
+
+ delete_files (action->manager, files, action);
+
+ g_list_free (files);
+}
+
+static void
+restore_undo_func (NautilusFileUndoData *action)
+{
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+ nautilus_file_operations_trash_or_delete (eaction->destinations, NULL,
+ undo_redo_done_delete_callback, action);
+}
+
+static void
+trash_undo_func (NautilusFileUndoData *action)
+{
+ NautilusFileUndoManagerPrivate *priv = action->manager->priv;
+ GHashTable *files_to_restore;
+ NautilusFileUndoDataTrash* eaction = (NautilusFileUndoDataTrash*) action;
+
+ /* Internally managed op, clear the undo_redo_flag.
+ * Same as calling nautilus_file_undo_manager_is_undo_redo()
+ * minus the function call and unused return val.
+ */
+
+ priv->undo_redo_flag = FALSE;
+ files_to_restore = retrieve_files_to_restore (eaction->trashed);
+
+ if (g_hash_table_size (files_to_restore) > 0) {
+ GList *gfiles_in_trash, *l;
+ GFile *item;
+ GFile *dest;
+ char *value;
+
+ gfiles_in_trash = g_hash_table_get_keys (files_to_restore);
+
+ for (l = gfiles_in_trash; l != NULL; l = l->next) {
+ item = l->data;
+ value = g_hash_table_lookup (files_to_restore, item);
+ dest = g_file_new_for_uri (value);
+ g_file_move (item, dest, G_FILE_COPY_NOFOLLOW_SYMLINKS, NULL, NULL, NULL, NULL);
+ g_object_unref (dest);
+ }
+
+ g_list_free (gfiles_in_trash);
+ }
+
+ g_hash_table_destroy (files_to_restore);
+
+ /* Here we must do what's necessary for the callback */
+ undo_redo_done_transfer_callback (NULL, action);
+}
+
+static void
+move_undo_func (NautilusFileUndoData *action)
+{
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+ nautilus_file_operations_move (eaction->destinations, NULL,
+ eaction->src_dir, NULL,
+ undo_redo_done_transfer_callback, action);
+}
+
+static void
+rename_undo_func (NautilusFileUndoData *action)
+{
+ gchar *new_name;
+ NautilusFile *file;
+ NautilusFileUndoDataRename* eaction = (NautilusFileUndoDataRename*) action;
+
+ new_name = g_file_get_basename (eaction->old_file);
+ file = nautilus_file_get (eaction->new_file);
+
+ nautilus_file_rename (file, new_name,
+ undo_redo_done_rename_callback, action);
+
+ nautilus_file_unref (file);
+ g_free (new_name);
+}
+
+static void
+set_permissions_undo_func (NautilusFileUndoData *action)
+{
+ NautilusFile *file;
+ NautilusFileUndoDataPermissions* eaction = (NautilusFileUndoDataPermissions*) action;
+
+ file = nautilus_file_get (eaction->target_file);
+
+ nautilus_file_set_permissions (file,
+ eaction->current_permissions,
+ undo_redo_done_rename_callback, action);
+
+ nautilus_file_unref (file);
+}
+
+static void
+recursive_permissions_undo_func (NautilusFileUndoData *action)
+{
+ NautilusFileUndoManagerPrivate *priv = action->manager->priv;
+ NautilusFileUndoDataRecursivePermissions* eaction = (NautilusFileUndoDataRecursivePermissions*) action;
+
+ /* Internally managed op, clear the undo_redo_flag. */
+ priv->undo_redo_flag = FALSE;
+
+ if (g_hash_table_size (eaction->original_permissions) > 0) {
+ GList *gfiles_list;
+ guint32 *perm;
+ GList *l;
+ GFile *dest;
+ char *item;
+
+ gfiles_list = g_hash_table_get_keys (eaction->original_permissions);
+ for (l = gfiles_list; l != NULL; l = l->next) {
+ item = l->data;
+ perm = g_hash_table_lookup (eaction->original_permissions, item);
+ dest = g_file_new_for_uri (item);
+ g_file_set_attribute_uint32 (dest,
+ G_FILE_ATTRIBUTE_UNIX_MODE,
+ *perm, G_FILE_QUERY_INFO_NOFOLLOW_SYMLINKS, NULL, NULL);
+ g_object_unref (dest);
+ }
+
+ g_list_free (gfiles_list);
+ /* Here we must do what's necessary for the callback */
+ undo_redo_done_transfer_callback (NULL, action);
+ }
+}
+
+static void
+change_group_undo_func (NautilusFileUndoData *action)
+{
+ NautilusFile *file;
+ NautilusFileUndoDataOwnership* eaction = (NautilusFileUndoDataOwnership*) action;
+
+ file = nautilus_file_get (eaction->target_file);
+
+ nautilus_file_set_group (file,
+ eaction->original_ownership,
+ undo_redo_done_rename_callback, action);
+
+ nautilus_file_unref (file);
+}
+
+static void
+change_owner_undo_func (NautilusFileUndoData *action)
+{
+ NautilusFile *file;
+ NautilusFileUndoDataOwnership* eaction = (NautilusFileUndoDataOwnership*) action;
+
+ file = nautilus_file_get (eaction->target_file);
+
+ nautilus_file_set_owner (file,
+ eaction->original_ownership,
+ undo_redo_done_rename_callback, action);
+
+ nautilus_file_unref (file);
+}
+
+/* redo helpers */
+
+static void
+copy_redo_func (NautilusFileUndoData *action)
+{
+ GList *locations;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ locations = eaction->sources;
+ nautilus_file_operations_copy (locations, NULL,
+ eaction->dest_dir, NULL,
+ undo_redo_done_transfer_callback, action);
+}
+
+static void
+create_from_template_redo_func (NautilusFileUndoData *action)
+{
+ GFile *parent;
+ gchar *parent_uri, *new_name;
+ NautilusFileUndoDataCreate* eaction = (NautilusFileUndoDataCreate*) action;
+
+ parent = g_file_get_parent (eaction->target_file);
+ parent_uri = g_file_get_uri (parent);
+ new_name = g_file_get_parse_name (eaction->target_file);
+ nautilus_file_operations_new_file_from_template (NULL, NULL,
+ parent_uri, new_name,
+ eaction->template,
+ undo_redo_done_create_callback, action);
+
+ g_free (parent_uri);
+ g_free (new_name);
+ g_object_unref (parent);
+}
+
+static void
+duplicate_redo_func (NautilusFileUndoData *action)
+{
+ GList *locations;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ locations = eaction->sources;
+ nautilus_file_operations_duplicate (locations, NULL, NULL,
+ undo_redo_done_transfer_callback, action);
+}
+
+static void
+move_restore_redo_func (NautilusFileUndoData *action)
+{
+ GList *locations;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ locations = eaction->sources;
+ nautilus_file_operations_move (locations, NULL,
+ eaction->dest_dir, NULL,
+ undo_redo_done_transfer_callback, action);
+}
+
+static void
+rename_redo_func (NautilusFileUndoData *action)
+{
+ gchar *new_name;
+ NautilusFile *file;
+ NautilusFileUndoDataRename* eaction = (NautilusFileUndoDataRename*) action;
+
+ new_name = g_file_get_basename (eaction->new_file);
+ file = nautilus_file_get (eaction->old_file);
+ nautilus_file_rename (file, new_name,
+ undo_redo_done_rename_callback, action);
+
+ nautilus_file_unref (file);
+ g_free (new_name);
+}
+
+static void
+create_empty_redo_func (NautilusFileUndoData *action)
+{
+ GFile *parent;
+ gchar *parent_uri;
+ gchar *new_name;
+ NautilusFileUndoDataCreate* eaction = (NautilusFileUndoDataCreate*) action;
+
+ parent = g_file_get_parent (eaction->target_file);
+ parent_uri = g_file_get_uri (parent);
+ new_name = g_file_get_parse_name (eaction->target_file);
+ nautilus_file_operations_new_file (NULL, NULL, parent_uri,
+ new_name,
+ eaction->template,
+ action->count, undo_redo_done_create_callback, action);
+
+ g_free (parent_uri);
+ g_free (new_name);
+ g_object_unref (parent);
+}
+
+static void
+create_folder_redo_func (NautilusFileUndoData *action)
+{
+ GFile *parent;
+ gchar *parent_uri;
+ NautilusFileUndoDataCreate* eaction = (NautilusFileUndoDataCreate*) action;
+
+ parent = g_file_get_parent (eaction->target_file);
+ parent_uri = g_file_get_uri (parent);
+ nautilus_file_operations_new_folder (NULL, NULL, parent_uri,
+ undo_redo_done_create_callback, action);
+
+ g_free (parent_uri);
+ g_object_unref (parent);
+}
+
+static void
+trash_redo_func (NautilusFileUndoData *action)
+{
+ NautilusFileUndoDataTrash* eaction = (NautilusFileUndoDataTrash*) action;
+
+ if (g_hash_table_size (eaction->trashed) > 0) {
+ GList *uri_to_trash, *locations;
+
+ uri_to_trash = g_hash_table_get_keys (eaction->trashed);
+ locations = uri_list_to_gfile_list (uri_to_trash);
+ nautilus_file_operations_trash_or_delete (locations, NULL,
+ undo_redo_done_delete_callback, action);
+ g_list_free (uri_to_trash);
+ g_list_free_full (locations, g_object_unref);
+ }
+}
+
+static void
+create_link_redo_func (NautilusFileUndoData *action)
+{
+ GList *locations;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ locations = eaction->sources;
+ nautilus_file_operations_link (locations, NULL,
+ eaction->dest_dir, NULL,
+ undo_redo_done_transfer_callback, action);
+}
+
+static void
+set_permissions_redo_func (NautilusFileUndoData *action)
+{
+ NautilusFile *file;
+ NautilusFileUndoDataPermissions* eaction = (NautilusFileUndoDataPermissions*) action;
+
+ file = nautilus_file_get (eaction->target_file);
+ nautilus_file_set_permissions (file, eaction->new_permissions,
+ undo_redo_done_rename_callback, action);
+
+ nautilus_file_unref (file);
+}
+
+static void
+recursive_permissions_redo_func (NautilusFileUndoData *action)
+{
+ gchar *parent_uri;
+ NautilusFileUndoDataRecursivePermissions* eaction = (NautilusFileUndoDataRecursivePermissions*) action;
+
+ parent_uri = g_file_get_uri (eaction->dest_dir);
+ nautilus_file_set_permissions_recursive (parent_uri,
+ eaction->file_permissions,
+ eaction->file_mask,
+ eaction->dir_permissions,
+ eaction->dir_mask,
+ undo_redo_op_callback,
+ action);
+ g_free (parent_uri);
+}
+
+static void
+change_group_redo_func (NautilusFileUndoData *action)
+{
+ NautilusFile *file;
+ NautilusFileUndoDataOwnership* eaction = (NautilusFileUndoDataOwnership*) action;
+
+ file = nautilus_file_get (eaction->target_file);
+
+ nautilus_file_set_group (file,
+ eaction->new_ownership,
+ undo_redo_done_rename_callback,
+ action);
+
+ nautilus_file_unref (file);
+}
+
+static void
+change_owner_redo_func (NautilusFileUndoData *action)
+{
+ NautilusFile *file;
+ NautilusFileUndoDataOwnership* eaction = (NautilusFileUndoDataOwnership*) action;
+
+ file = nautilus_file_get (eaction->target_file);
+ nautilus_file_set_owner (file,
+ eaction->new_ownership,
+ undo_redo_done_rename_callback,
+ action);
+
+ nautilus_file_unref (file);
+}
+
+/* description helpers */
+
+static void
+copy_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ gchar *destination;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ destination = g_file_get_path (eaction->dest_dir);
+
+ if (count != 1) {
+ descriptions[0] = g_strdup_printf (_("Delete %d copied items"), count);
+ descriptions[1] = g_strdup_printf (_("Copy %d items to '%s'"), count, destination);
+
+ labels[0] = g_strdup_printf (_("_Undo Copy %d items"), count);
+ labels[1] = g_strdup_printf (_("_Redo Copy %d items"), count);
+ } else {
+ gchar *name;
+
+ name = get_first_target_short_name (eaction);
+ descriptions[0] = g_strdup_printf (_("Delete '%s'"), name);
+ descriptions[1] = g_strdup_printf (_("Copy '%s' to '%s'"), name, destination);
+
+ labels[0] = g_strdup (_("_Undo Copy"));
+ labels[1] = g_strdup (_("_Redo Copy"));
+
+ g_free (name);
+ }
+}
+
+static void
+duplicate_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ gchar *destination;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ destination = g_file_get_path (eaction->dest_dir);
+
+ if (count != 1) {
+ descriptions[0] = g_strdup_printf (_("Delete %d duplicated items"), count);
+ descriptions[1] = g_strdup_printf (_("Duplicate of %d items in '%s'"),
+ count, destination);
+
+ labels[0] = g_strdup_printf (_("_Undo Duplicate %d items"), count);
+ labels[1] = g_strdup_printf (_("_Redo Duplicate %d items"), count);
+ } else {
+ char *name;
+
+ name = get_first_target_short_name (eaction);
+ descriptions[0] = g_strdup_printf (_("Delete '%s'"), name);
+ descriptions[1] = g_strdup_printf (_("Duplicate '%s' in '%s'"),
+ name, destination);
+
+ labels[0] = g_strdup (_("_Undo Duplicate"));
+ labels[1] = g_strdup (_("_Redo Duplicate"));
+
+ g_free (name);
+ }
+
+}
+
+static void
+move_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ gchar *source, *destination;
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ source = g_file_get_path (eaction->src_dir);
+ destination = g_file_get_path (eaction->dest_dir);
+
+ if (count != 1) {
+ descriptions[0] = g_strdup_printf (_("Move %d items back to '%s'"), count, source);
+ descriptions[1] = g_strdup_printf (_("Move %d items to '%s'"), count, destination);
+
+ labels[0] = g_strdup_printf (_("_Undo Move %d items"), count);
+ labels[1] = g_strdup_printf (_("_Redo Move %d items"), count);
+ } else {
+ char *name;
+
+ name = get_first_target_short_name (eaction);
+ descriptions[0] = g_strdup_printf (_("Move '%s' back to '%s'"), name, source);
+ descriptions[1] = g_strdup_printf (_("Move '%s' to '%s'"), name, destination);
+
+ labels[0] = g_strdup (_("_Undo Move"));
+ labels[1] = g_strdup (_("_Redo Move"));
+
+ g_free (name);
+ }
+}
+
+static void
+rename_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ gchar *new_name, *old_name;
+ NautilusFileUndoDataRename* eaction = (NautilusFileUndoDataRename*) action;
+
+ new_name = g_file_get_parse_name (eaction->new_file);
+ old_name = g_file_get_parse_name (eaction->old_file);
+
+ descriptions[0] = g_strdup_printf (_("Rename '%s' as '%s'"), new_name, old_name);
+ descriptions[1] = g_strdup_printf (_("Rename '%s' as '%s'"), old_name, new_name);
+
+ labels[0] = g_strdup (_("_Undo Rename"));
+ labels[1] = g_strdup (_("_Redo Rename"));
+
+ g_free (old_name);
+ g_free (new_name);
+}
+
+static void
+create_undo_common (NautilusFileUndoDataCreate *eaction,
+ guint count,
+ gchar **descriptions)
+{
+ char *name;
+
+ name = g_file_get_parse_name (eaction->target_file);
+ descriptions[0] = g_strdup_printf (_("Delete '%s'"), name);
+
+ g_free (name);
+}
+
+static void
+create_empty_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ gchar *name;
+ NautilusFileUndoDataCreate* eaction = (NautilusFileUndoDataCreate*) action;
+
+ create_undo_common (eaction, count, descriptions);
+
+ name = g_file_get_parse_name (eaction->target_file);
+ descriptions[1] = g_strdup_printf (_("Create an empty file '%s'"), name);
+
+ labels[0] = g_strdup (_("_Undo Create Empty File"));
+ labels[1] = g_strdup (_("_Redo Create Empty File"));
+
+ g_free (name);
+}
+
+static void
+create_from_template_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ char *name;
+ NautilusFileUndoDataCreate* eaction = (NautilusFileUndoDataCreate*) action;
+
+ create_undo_common (eaction, count, descriptions);
+
+ name = g_file_get_parse_name (eaction->target_file);
+ descriptions[1] = g_strdup_printf (_("Create new file '%s' from template "), name);
+ g_free (name);
+
+ labels[0] = g_strdup (_("_Undo Create from Template"));
+ labels[1] = g_strdup (_("_Redo Create from Template"));
+}
+
+static void
+create_folder_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ char *name;
+ NautilusFileUndoDataCreate* eaction = (NautilusFileUndoDataCreate*) action;
+
+ create_undo_common (eaction, count, descriptions);
+
+ name = g_file_get_parse_name (eaction->target_file);
+ descriptions[1] = g_strdup_printf (_("Create a new folder '%s'"), name);
+
+ labels[0] = g_strdup (_("_Undo Create Folder"));
+ labels[1] = g_strdup (_("_Redo Create Folder"));
+
+ g_free (name);
+}
+
+static void
+trash_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ NautilusFileUndoDataTrash* eaction = (NautilusFileUndoDataTrash*) action;
+ count = g_hash_table_size (eaction->trashed);
+
+ if (count != 1) {
+ descriptions[0] = g_strdup_printf (_("Restore %d items from trash"), count);
+ descriptions[1] = g_strdup_printf (_("Move %d items to trash"), count);
+ } else {
+ GList *keys;
+ char *name, *orig_path;
+ GFile *file;
+
+ keys = g_hash_table_get_keys (eaction->trashed);
+ file = g_file_new_for_commandline_arg (keys->data);
+ name = g_file_get_basename (file);
+ orig_path = g_file_get_path (file);
+ descriptions[0] = g_strdup_printf (_("Restore '%s' to '%s'"), name, orig_path);
+
+ g_free (name);
+ g_free (orig_path);
+ g_list_free (keys);
+
+ name = g_file_get_parse_name (file);
+ descriptions[1] = g_strdup_printf (_("Move '%s' to trash"), name);
+
+ g_free (name);
+ g_object_unref (file);
+
+ labels[0] = g_strdup (_("_Undo Trash"));
+ labels[1] = g_strdup (_("_Redo Trash"));
+ }
+}
+
+static void
+restore_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ if (count != 1) {
+ descriptions[0] = g_strdup_printf (_("Move %d items back to trash"), count);
+ descriptions[1] = g_strdup_printf (_("Restore %d items from trash"), count);
+ } else {
+ char *name;
+
+ name = get_first_target_short_name (eaction);
+ descriptions[0] = g_strdup_printf (_("Move '%s' back to trash"), name);
+ descriptions[1] = g_strdup_printf (_("Restore '%s' from trash"), name);
+
+ g_free (name);
+
+ labels[0] = g_strdup (_("_Undo Restore from Trash"));
+ labels[1] = g_strdup (_("_Redo Restore from Trash"));
+ }
+}
+
+static void
+create_link_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ NautilusFileUndoDataExt* eaction = (NautilusFileUndoDataExt*) action;
+
+ if (count != 1) {
+ descriptions[0] = g_strdup_printf (_("Delete links to %d items"), count);
+ descriptions[1] = g_strdup_printf (_("Create links to %d items"), count);
+ } else {
+ char *name;
+
+ name = get_first_target_short_name (eaction);
+ descriptions[0] = g_strdup_printf (_("Delete link to '%s'"), name);
+ descriptions[1] = g_strdup_printf (_("Create link to '%s'"), name);
+
+ labels[0] = g_strdup (_("_Undo Create Link"));
+ labels[1] = g_strdup (_("_Redo Create Link"));
+
+ g_free (name);
+ }
+}
+
+static void
+recursive_permissions_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ char *name;
+ NautilusFileUndoDataRecursivePermissions* eaction = (NautilusFileUndoDataRecursivePermissions*) action;
+
+ name = g_file_get_path (eaction->dest_dir);
+
+ descriptions[0] = g_strdup_printf (_("Restore original permissions of items enclosed in '%s'"), name);
+ descriptions[1] = g_strdup_printf (_("Set permissions of items enclosed in '%s'"), name);
+
+ labels[0] = g_strdup (_("_Undo Change Permissions"));
+ labels[1] = g_strdup (_("_Redo Change Permissions"));
+
+ g_free (name);
+}
+
+static void
+set_permissions_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ char *name;
+ NautilusFileUndoDataPermissions* eaction = (NautilusFileUndoDataPermissions*) action;
+
+ name = g_file_get_parse_name (eaction->target_file);
+ descriptions[0] = g_strdup_printf (_("Restore original permissions of '%s'"), name);
+ descriptions[1] = g_strdup_printf (_("Set permissions of '%s'"), name);
+
+ labels[0] = g_strdup (_("_Undo Change Permissions"));
+ labels[1] = g_strdup (_("_Redo Change Permissions"));
+
+ g_free (name);
+}
+
+static void
+change_group_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ gchar *name;
+ NautilusFileUndoDataOwnership* eaction = (NautilusFileUndoDataOwnership*) action;
+
+ name = g_file_get_parse_name (eaction->target_file);
+ descriptions[0] = g_strdup_printf (_("Restore group of '%s' to '%s'"),
+ name, eaction->original_ownership);
+ descriptions[1] = g_strdup_printf (_("Set group of '%s' to '%s'"),
+ name, eaction->new_ownership);
+
+ labels[0] = g_strdup (_("_Undo Change Group"));
+ labels[1] = g_strdup (_("_Redo Change Group"));
+
+ g_free (name);
+}
+
+static void
+change_owner_description_func (NautilusFileUndoData *action,
+ guint count,
+ gchar **labels,
+ gchar **descriptions)
+{
+ gchar *name;
+ NautilusFileUndoDataOwnership* eaction = (NautilusFileUndoDataOwnership*) action;
+
+ name = g_file_get_parse_name (eaction->target_file);
+ descriptions[0] = g_strdup_printf (_("Restore owner of '%s' to '%s'"),
+ name, eaction->original_ownership);
+ descriptions[1] = g_strdup_printf (_("Set owner of '%s' to '%s'"),
+ name, eaction->new_ownership);
+
+ labels[0] = g_strdup (_("_Undo Change Owner"));
+ labels[1] = g_strdup (_("_Redo Change Owner"));
+
+ g_free (name);
+}
+
+static void
+finalize_undo_data_ext (NautilusFileUndoData* data)
+{
+ NautilusFileUndoDataExt* extdata = (NautilusFileUndoDataExt*) data;
+ if (extdata->sources) {
+ g_list_foreach (extdata->sources, (GFunc) g_free, NULL);
+ g_list_free (extdata->sources);
+ }
+ if (extdata->destinations) {
+ g_list_foreach (extdata->destinations, (GFunc) g_free, NULL);
+ g_list_free (extdata->destinations);
+ }
+ if (extdata->src_dir) {
+ g_object_unref (extdata->src_dir);
+ }
+ if (extdata->dest_dir) {
+ g_object_unref (extdata->dest_dir);
+ }
+ g_slice_free (NautilusFileUndoDataExt, extdata);
+}
+
+static void
+finalize_undo_data_rename (NautilusFileUndoData* data)
+{
+ NautilusFileUndoDataRename* extdata = (NautilusFileUndoDataRename*) data;
+ if (extdata->old_file) {
+ g_object_unref (extdata->old_file);
+ }
+ if (extdata->new_file) {
+ g_free (extdata->new_file);
+ }
+ g_slice_free (NautilusFileUndoDataRename, extdata);
+}
+
+static void
+finalize_undo_data_create (NautilusFileUndoData* data)
+{
+ NautilusFileUndoDataCreate* extdata = (NautilusFileUndoDataCreate*) data;
+ if (extdata->target_file) {
+ g_object_unref (extdata->target_file);
+ }
+ g_free (extdata->template);
+ g_slice_free (NautilusFileUndoDataCreate, extdata);
+}
+
+static void
+finalize_undo_data_trash (NautilusFileUndoData* data)
+{
+ NautilusFileUndoDataTrash* extdata = (NautilusFileUndoDataTrash*) data;
+ if (extdata->trashed) {
+ g_hash_table_destroy (extdata->trashed);
+ }
+ g_slice_free (NautilusFileUndoDataTrash, extdata);
+}
+
+static void
+finalize_undo_data_permissions (NautilusFileUndoData* data)
+{
+ NautilusFileUndoDataPermissions* extdata = (NautilusFileUndoDataPermissions*) data;
+ if (extdata->target_file) {
+ g_object_unref (extdata->target_file);
+ }
+ g_slice_free (NautilusFileUndoDataPermissions, extdata);
+}
+
+static void
+finalize_undo_data_recursivepermissions (NautilusFileUndoData* data)
+{
+ NautilusFileUndoDataRecursivePermissions* extdata = (NautilusFileUndoDataRecursivePermissions*) data;
+ if (extdata->original_permissions) {
+ g_hash_table_destroy (extdata->original_permissions);
+ }
+ if (extdata->dest_dir) {
+ g_object_unref (extdata->dest_dir);
+ }
+ g_slice_free (NautilusFileUndoDataRecursivePermissions, extdata);
+}
+
+static void
+finalize_undo_data_ownership (NautilusFileUndoData* data)
+{
+ NautilusFileUndoDataOwnership* extdata = (NautilusFileUndoDataOwnership*) data;
+ if (extdata->target_file) {
+ g_object_unref (extdata->target_file);
+ }
+ g_free (extdata->original_ownership);
+ g_free (extdata->new_ownership);
+ g_slice_free (NautilusFileUndoDataOwnership, extdata);
+}
+
+
+static NautilusFileUndoData *
+create_from_type (NautilusFileUndoDataType type)
+{
+ struct {
+ void (* undo_func) (NautilusFileUndoData *data);
+ void (* redo_func) (NautilusFileUndoData *data);
+ void (* strings_func) (NautilusFileUndoData *data,
+ guint count,
+ gchar **labels,
+ gchar **descriptions);
+ void (* finalize_func) (NautilusFileUndoData *data);
+ gsize alloc_size;
+ } const mappings[NAUTILUS_FILE_UNDO_NUM_TYPES] = {
+ /* copy action */
+ { copy_or_link_undo_func, copy_redo_func,
+ copy_description_func, finalize_undo_data_ext,
+ sizeof(NautilusFileUndoDataExt) },
+ /* duplicate action */
+ { copy_or_link_undo_func, duplicate_redo_func,
+ duplicate_description_func, finalize_undo_data_ext,
+ sizeof(NautilusFileUndoDataExt) },
+ /* move action */
+ { move_undo_func, move_restore_redo_func,
+ move_description_func, finalize_undo_data_ext,
+ sizeof(NautilusFileUndoDataExt) },
+ /* rename action */
+ { rename_undo_func, rename_redo_func,
+ rename_description_func, finalize_undo_data_rename,
+ sizeof(NautilusFileUndoDataRename) },
+ /* create empty action */
+ { create_undo_func, create_empty_redo_func,
+ create_empty_description_func, finalize_undo_data_create,
+ sizeof(NautilusFileUndoDataCreate) },
+ /* create from template action */
+ { create_undo_func, create_from_template_redo_func,
+ create_from_template_description_func, finalize_undo_data_create,
+ sizeof(NautilusFileUndoDataCreate) },
+ /* create folder action */
+ { create_undo_func, create_folder_redo_func,
+ create_folder_description_func, finalize_undo_data_create,
+ sizeof(NautilusFileUndoDataCreate) },
+ /* move to trash action */
+ { trash_undo_func, trash_redo_func,
+ trash_description_func, finalize_undo_data_trash,
+ sizeof(NautilusFileUndoDataTrash) },
+ /* restore from trash action */
+ { restore_undo_func, move_restore_redo_func,
+ restore_description_func, finalize_undo_data_ext,
+ sizeof(NautilusFileUndoDataExt) },
+ /* create link action */
+ { create_undo_func, create_link_redo_func,
+ create_link_description_func, finalize_undo_data_ext,
+ sizeof(NautilusFileUndoDataExt) },
+ /* recursive permissions action */
+ { recursive_permissions_undo_func, recursive_permissions_redo_func,
+ recursive_permissions_description_func, finalize_undo_data_recursivepermissions,
+ sizeof(NautilusFileUndoDataRecursivePermissions) },
+ /* set permissions action */
+ { set_permissions_undo_func, set_permissions_redo_func,
+ set_permissions_description_func , finalize_undo_data_permissions,
+ sizeof(NautilusFileUndoDataPermissions) },
+ /* change group action */
+ { change_group_undo_func, change_group_redo_func,
+ change_group_description_func, finalize_undo_data_ownership,
+ sizeof(NautilusFileUndoDataOwnership) },
+ /* change owner action */
+ { change_owner_undo_func, change_owner_redo_func,
+ change_owner_description_func, finalize_undo_data_ownership,
+ sizeof(NautilusFileUndoDataOwnership) },
+ };
+
+ NautilusFileUndoData *retval;
+
+ retval = g_slice_alloc0 (mappings[type].alloc_size);
+ retval->undo_func = mappings[type].undo_func;
+ retval->redo_func = mappings[type].redo_func;
+ retval->strings_func = mappings[type].strings_func;
+ retval->finalize_func = mappings[type].finalize_func;
+
+ return retval;
+}
+
+/* functions to manipulate the action data */
+NautilusFileUndoData *
+nautilus_file_undo_data_new (NautilusFileUndoDataType type,
+ gint items_count)
+{
+ NautilusFileUndoData *data;
+
+ data = create_from_type (type);
+ data->type = type;
+ data->count = items_count;
+
+ return data;
+}
+
+void
+nautilus_file_undo_data_set_src_dir (NautilusFileUndoData *action_data,
+ GFile *src)
+{
+ NautilusFileUndoDataExt* eaction_data = (NautilusFileUndoDataExt*) action_data;
+ eaction_data->src_dir = g_object_ref (src);
+}
+
+void
+nautilus_file_undo_data_set_dest_dir (NautilusFileUndoData *action_data,
+ GFile *dest)
+{
+ NautilusFileUndoDataExt* eaction_data = (NautilusFileUndoDataExt*) action_data;
+ eaction_data->dest_dir = g_object_ref (dest);
+}
+
+void
+nautilus_file_undo_data_add_origin_target_pair (NautilusFileUndoData *action_data,
+ GFile *origin,
+ GFile *target)
+{
+ NautilusFileUndoDataExt* eaction_data = (NautilusFileUndoDataExt*) action_data;
+ eaction_data->sources =
+ g_list_append (eaction_data->sources, g_object_ref (origin));
+ eaction_data->destinations =
+ g_list_append (eaction_data->destinations, g_object_ref (target));
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_add_trashed_file (NautilusFileUndoData *action_data,
+ GFile *file,
+ guint64 mtime)
+{
+ guint64 *modification_time;
+ char *original_uri;
+ NautilusFileUndoDataTrash* eaction_data = (NautilusFileUndoDataTrash*) action_data;
+
+ if (eaction_data->trashed == NULL) {
+ eaction_data->trashed =
+ g_hash_table_new_full (g_str_hash, g_str_equal,
+ g_free, g_free);
+ }
+
+ modification_time = g_new (guint64, 1);
+ *modification_time = mtime;
+
+ original_uri = g_file_get_uri (file);
+
+ g_hash_table_insert (eaction_data->trashed, original_uri, modification_time);
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_add_file_permissions (NautilusFileUndoData *action_data,
+ GFile *file,
+ guint32 permission)
+{
+ guint32 *current_permissions;
+ char *original_uri;
+ NautilusFileUndoDataRecursivePermissions* eaction_data = (NautilusFileUndoDataRecursivePermissions*) action_data;
+
+ if (eaction_data->original_permissions == NULL) {
+ eaction_data->original_permissions =
+ g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
+ }
+
+ current_permissions = g_new (guint32, 1);
+ *current_permissions = permission;
+
+ original_uri = g_file_get_uri (file);
+
+ g_hash_table_insert (eaction_data->original_permissions, original_uri, current_permissions);
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_set_file_permissions (NautilusFileUndoData *action_data,
+ GFile *file,
+ guint32 current_permissions,
+ guint32 new_permissions)
+{
+ NautilusFileUndoDataPermissions* eaction_data = (NautilusFileUndoDataPermissions*) action_data;
+ eaction_data->target_file = g_object_ref (file);
+ eaction_data->current_permissions = current_permissions;
+ eaction_data->new_permissions = new_permissions;
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_set_owner_change_information (NautilusFileUndoData *action_data,
+ GFile *file,
+ const char *current_user,
+ const char *new_user)
+{
+ NautilusFileUndoDataOwnership* eaction_data = (NautilusFileUndoDataOwnership*) action_data;
+ eaction_data->target_file = g_object_ref (file);
+ eaction_data->original_ownership = g_strdup (current_user);
+ eaction_data->new_ownership = g_strdup (new_user);
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_set_group_change_information (NautilusFileUndoData *action_data,
+ GFile *file,
+ const char *current_group,
+ const char *new_group)
+{
+ NautilusFileUndoDataOwnership* eaction_data = (NautilusFileUndoDataOwnership*) action_data;
+ eaction_data->target_file = g_object_ref (file);
+ eaction_data->original_ownership = g_strdup (current_group);
+ eaction_data->new_ownership = g_strdup (new_group);
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_set_recursive_permissions (NautilusFileUndoData *action_data,
+ guint32 file_permissions,
+ guint32 file_mask,
+ guint32 dir_permissions,
+ guint32 dir_mask)
+{
+ NautilusFileUndoDataRecursivePermissions* eaction_data = (NautilusFileUndoDataRecursivePermissions*) action_data;
+ eaction_data->file_permissions = file_permissions;
+ eaction_data->file_mask = file_mask;
+ eaction_data->dir_permissions = dir_permissions;
+ eaction_data->dir_mask = dir_mask;
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_set_recursive_permissions_dest_dir (NautilusFileUndoData *action_data,
+ GFile *dest)
+{
+ NautilusFileUndoDataRecursivePermissions* eaction_data = (NautilusFileUndoDataRecursivePermissions*) action_data;
+ eaction_data->dest_dir = g_object_ref (dest);
+}
+
+void
+nautilus_file_undo_data_set_create_data (NautilusFileUndoData *action_data,
+ GFile *file,
+ const char *template)
+{
+ NautilusFileUndoDataCreate* eaction_data = (NautilusFileUndoDataCreate*) action_data;
+ eaction_data->target_file = g_object_ref (file);
+ eaction_data->template = g_strdup (template);
+
+ action_data->is_valid = TRUE;
+}
+
+void
+nautilus_file_undo_data_set_rename_information (NautilusFileUndoData *action_data,
+ GFile *old_file,
+ GFile *new_file)
+{
+ NautilusFileUndoDataRename* eaction_data = (NautilusFileUndoDataRename*) action_data;
+ eaction_data->old_file = g_object_ref (old_file);
+ eaction_data->new_file = g_object_ref (new_file);
+
+ action_data->is_valid = TRUE;
+}
diff --git a/libnautilus-private/nautilus-file-undo-operations.h b/libnautilus-private/nautilus-file-undo-operations.h
new file mode 100644
index 0000000..69a79c2
--- /dev/null
+++ b/libnautilus-private/nautilus-file-undo-operations.h
@@ -0,0 +1,78 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+
+/* nautilus-file-undo-operations.h - Manages undo/redo of file operations
+ *
+ * Copyright (C) 2007-2011 Amos Brocco
+ * Copyright (C) 2010 Red Hat, Inc.
+ *
+ * This library is free software; you can redistribute it and/or
+ * 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.
+ *
+ * This 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
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public
+ * License along with this library; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ * Authors: Amos Brocco <amos brocco gmail com>
+ * Cosimo Cecchi <cosimoc redhat com>
+ *
+ */
+
+#ifndef __NAUTILUS_FILE_UNDO_OPERATIONS_H__
+#define __NAUTILUS_FILE_UNDO_OPERATIONS_H__
+
+#include <glib.h>
+#include <libnautilus-private/nautilus-file-undo-types.h>
+
+NautilusFileUndoData *nautilus_file_undo_data_new (NautilusFileUndoDataType type,
+ gint items_count);
+void nautilus_file_undo_data_free (NautilusFileUndoData *action);
+
+void nautilus_file_undo_data_set_src_dir (NautilusFileUndoData *data,
+ GFile *src);
+void nautilus_file_undo_data_set_dest_dir (NautilusFileUndoData *data,
+ GFile *dest);
+
+void nautilus_file_undo_data_add_origin_target_pair (NautilusFileUndoData *data,
+ GFile *origin,
+ GFile *target);
+void nautilus_file_undo_data_set_create_data (NautilusFileUndoData *data,
+ GFile *target_file,
+ const char *template_uri);
+void nautilus_file_undo_data_set_rename_information (NautilusFileUndoData *data,
+ GFile *old_file,
+ GFile *new_file);
+void nautilus_file_undo_data_add_trashed_file (NautilusFileUndoData *data,
+ GFile *file,
+ guint64 mtime);
+void nautilus_file_undo_data_add_file_permissions (NautilusFileUndoData *data,
+ GFile *file,
+ guint32 permission);
+void nautilus_file_undo_data_set_recursive_permissions (NautilusFileUndoData *data,
+ guint32 file_permissions,
+ guint32 file_mask,
+ guint32 dir_permissions,
+ guint32 dir_mask);
+void nautilus_file_undo_data_set_recursive_permissions_dest_dir (NautilusFileUndoData *data,
+ GFile *dest);
+void nautilus_file_undo_data_set_file_permissions (NautilusFileUndoData *data,
+ GFile *file,
+ guint32 current_permissions,
+ guint32 new_permissions);
+void nautilus_file_undo_data_set_owner_change_information (NautilusFileUndoData *data,
+ GFile *file,
+ const char *current_user,
+ const char *new_user);
+void nautilus_file_undo_data_set_group_change_information (NautilusFileUndoData *data,
+ GFile *file,
+ const char *current_group,
+ const char *new_group);
+
+#endif /* __NAUTILUS_FILE_UNDO_OPERATIONS_H__ */
diff --git a/libnautilus-private/nautilus-file-undo-types.h b/libnautilus-private/nautilus-file-undo-types.h
new file mode 100644
index 0000000..b582d6d
--- /dev/null
+++ b/libnautilus-private/nautilus-file-undo-types.h
@@ -0,0 +1,168 @@
+/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */
+
+/* nautilus-file-undo-types.h - Data structures used by undo/redo
+ *
+ * Copyright (C) 2007-2011 Amos Brocco
+ * Copyright (C) 2010 Red Hat, Inc.
+ *
+ * This program is free software; you can redistribute it and/or
+ * 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.
+ *
+ * This program 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
+ * General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public
+ * License along with this program; if not, write to the
+ * Free Software Foundation, Inc., 59 Temple Place - Suite 330,
+ * Boston, MA 02111-1307, USA.
+ *
+ * Authors: Amos Brocco <amos brocco gmail com>
+ * Cosimo Cecchi <cosimoc redhat com>
+ *
+ */
+
+#ifndef __NAUTILUS_FILE_UNDO_TYPES_H__
+#define __NAUTILUS_FILE_UNDO_TYPES_H__
+
+#include <glib.h>
+#include <gio/gio.h>
+
+typedef struct _NautilusFileUndoManager NautilusFileUndoManager;
+typedef void (* NautilusFileUndoFinishCallback) (gpointer data);
+
+typedef enum {
+ NAUTILUS_FILE_UNDO_COPY,
+ NAUTILUS_FILE_UNDO_DUPLICATE,
+ NAUTILUS_FILE_UNDO_MOVE,
+ NAUTILUS_FILE_UNDO_RENAME,
+ NAUTILUS_FILE_UNDO_CREATE_EMPTY_FILE,
+ NAUTILUS_FILE_UNDO_CREATE_FILE_FROM_TEMPLATE,
+ NAUTILUS_FILE_UNDO_CREATE_FOLDER,
+ NAUTILUS_FILE_UNDO_MOVE_TO_TRASH,
+ NAUTILUS_FILE_UNDO_RESTORE_FROM_TRASH,
+ NAUTILUS_FILE_UNDO_CREATE_LINK,
+ NAUTILUS_FILE_UNDO_RECURSIVE_SET_PERMISSIONS,
+ NAUTILUS_FILE_UNDO_SET_PERMISSIONS,
+ NAUTILUS_FILE_UNDO_CHANGE_GROUP,
+ NAUTILUS_FILE_UNDO_CHANGE_OWNER,
+ NAUTILUS_FILE_UNDO_NUM_TYPES,
+} NautilusFileUndoDataType;
+
+typedef struct {
+ const char *undo_label;
+ const char *undo_description;
+ const char *redo_label;
+ const char *redo_description;
+} NautilusFileUndoMenuData;
+
+typedef struct _NautilusFileUndoData NautilusFileUndoData;
+
+struct _NautilusFileUndoData
+{
+ NautilusFileUndoDataType type;
+
+ void (* undo_func) (NautilusFileUndoData *data);
+ void (* redo_func) (NautilusFileUndoData *data);
+
+ NautilusFileUndoFinishCallback callback;
+ gpointer callback_user_data;
+
+ NautilusFileUndoManager *manager;
+ guint is_valid : 1;
+ guint locked : 1; /* True if the action is being undone/redone */
+ guint freed : 1; /* True if the action must be freed after undo/redo */
+ guint count; /* Number of items */
+
+ void (* strings_func) (NautilusFileUndoData *data,
+ guint count,
+ gchar **labels,
+ gchar **descriptions);
+
+ gchar *undo_label;
+ gchar *redo_label;
+ gchar *undo_description;
+ gchar *redo_description;
+
+ void (* finalize_func) (NautilusFileUndoData *data);
+};
+
+typedef struct
+{
+ NautilusFileUndoData base_data;
+ /* Copy / Move / Duplicate / Link / Restore from trash stuff */
+ GFile *src_dir;
+ GFile *dest_dir;
+ GList *sources; /* Relative to src_dir */
+ GList *destinations; /* Relative to dest_dir */
+} NautilusFileUndoDataExt;
+
+typedef struct
+{
+ NautilusFileUndoData base_data;
+ /* Create new file/folder stuff/set permissions */
+ char *template;
+ GFile *target_file;
+} NautilusFileUndoDataCreate;
+
+typedef struct
+{
+ NautilusFileUndoData base_data;
+ /* Rename stuff */
+ GFile *old_file;
+ GFile *new_file;
+} NautilusFileUndoDataRename;
+
+typedef struct
+{
+ NautilusFileUndoData base_data;
+ /* Trash stuff */
+ GHashTable *trashed;
+} NautilusFileUndoDataTrash;
+
+typedef struct
+{
+ NautilusFileUndoData base_data;
+ /* Recursive change permissions stuff */
+ GFile *dest_dir;
+ GHashTable *original_permissions;
+ guint32 dir_mask;
+ guint32 dir_permissions;
+ guint32 file_mask;
+ guint32 file_permissions;
+} NautilusFileUndoDataRecursivePermissions;
+
+typedef struct
+{
+ NautilusFileUndoData base_data;
+ /* Single file change permissions stuff */
+ GFile *target_file;
+ guint32 current_permissions;
+ guint32 new_permissions;
+} NautilusFileUndoDataPermissions;
+
+typedef struct
+{
+ NautilusFileUndoData base_data;
+ /* Group and Owner change stuff */
+ GFile *target_file;
+ char *original_ownership;
+ char *new_ownership;
+} NautilusFileUndoDataOwnership;
+
+struct _NautilusFileUndoManagerPrivate
+{
+ GQueue *stack;
+
+ /* Used to protect access to stack (because of async file ops) */
+ GMutex mutex;
+
+ guint undo_levels;
+ guint index;
+ guint undo_redo_flag : 1;
+};
+
+#endif /* __NAUTILUS_FILE_UNDO_TYPES_H__ */
diff --git a/libnautilus-private/nautilus-file.c b/libnautilus-private/nautilus-file.c
index e69a794..58b49ef 100644
--- a/libnautilus-private/nautilus-file.c
+++ b/libnautilus-private/nautilus-file.c
@@ -44,6 +44,8 @@
#include "nautilus-search-directory-file.h"
#include "nautilus-thumbnails.h"
#include "nautilus-vfs-file.h"
+#include "nautilus-file-undo-operations.h"
+#include "nautilus-file-undo-manager.h"
#include "nautilus-saved-search-file.h"
#include <eel/eel-debug.h>
#include <eel/eel-glib-extensions.h>
@@ -1651,6 +1653,11 @@ nautilus_file_operation_free (NautilusFileOperation *op)
if (op->free_data) {
op->free_data (op->data);
}
+
+ if (op->undo_redo_data != NULL) {
+ nautilus_file_undo_manager_add_action (nautilus_file_undo_manager_get (), op->undo_redo_data);
+ }
+
g_free (op);
}
@@ -1755,6 +1762,10 @@ rename_callback (GObject *source_object,
res, &error);
if (new_file != NULL) {
+ if (op->undo_redo_data != NULL) {
+ nautilus_file_undo_data_set_rename_information (op->undo_redo_data, G_FILE (source_object), new_file);
+ }
+
g_file_query_info_async (new_file,
NAUTILUS_FILE_DEFAULT_ATTRIBUTES,
0,
@@ -1925,10 +1936,14 @@ nautilus_file_rename (NautilusFile *file,
/* Set up a renaming operation. */
op = nautilus_file_operation_new (file, callback, callback_data);
op->is_rename = TRUE;
+ location = nautilus_file_get_location (file);
- /* Do the renaming. */
+ /* Tell the undo manager a rename is taking place */
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ op->undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_RENAME, 1);
+ }
- location = nautilus_file_get_location (file);
+ /* Do the renaming. */
g_file_set_display_name_async (location,
new_file_name,
G_PRIORITY_DEFAULT,
@@ -5055,9 +5070,21 @@ nautilus_file_set_permissions (NautilusFile *file,
return;
}
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ NautilusFileUndoData *undo_redo_data;
+
+ undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_SET_PERMISSIONS, 1);
+ nautilus_file_undo_data_set_file_permissions (undo_redo_data,
+ nautilus_file_get_location (file),
+ file->details->permissions,
+ new_permissions);
+ nautilus_file_undo_manager_add_action (nautilus_file_undo_manager_get (), undo_redo_data);
+ }
+
info = g_file_info_new ();
g_file_info_set_attribute_uint32 (info, G_FILE_ATTRIBUTE_UNIX_MODE, new_permissions);
nautilus_file_set_attributes (file, info, callback, callback_data);
+
g_object_unref (info);
}
@@ -5357,6 +5384,22 @@ nautilus_file_set_owner (NautilusFile *file,
(* callback) (file, NULL, NULL, callback_data);
return;
}
+
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ NautilusFileUndoData *undo_redo_data;
+ char* current_owner;
+
+ current_owner = nautilus_file_get_owner_as_string (file, FALSE);
+ undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_CHANGE_OWNER, 1);
+
+ nautilus_file_undo_data_set_owner_change_information (undo_redo_data,
+ nautilus_file_get_location (file),
+ current_owner,
+ user_name_or_id);
+ nautilus_file_undo_manager_add_action (nautilus_file_undo_manager_get (), undo_redo_data);
+
+ g_free (current_owner);
+ }
info = g_file_info_new ();
g_file_info_set_attribute_uint32 (info, G_FILE_ATTRIBUTE_UNIX_UID, new_id);
@@ -5621,6 +5664,21 @@ nautilus_file_set_group (NautilusFile *file,
return;
}
+ if (!nautilus_file_undo_manager_is_undo_redo (nautilus_file_undo_manager_get ())) {
+ NautilusFileUndoData *undo_redo_data;
+ char *current_group;
+
+ current_group = nautilus_file_get_group_name (file);
+ undo_redo_data = nautilus_file_undo_data_new (NAUTILUS_FILE_UNDO_CHANGE_GROUP, 1);
+
+ nautilus_file_undo_data_set_group_change_information (undo_redo_data,
+ nautilus_file_get_location (file),
+ current_group,
+ group_name_or_id);
+ nautilus_file_undo_manager_add_action (nautilus_file_undo_manager_get (), undo_redo_data);
+
+ g_free (current_group);
+ }
info = g_file_info_new ();
g_file_info_set_attribute_uint32 (info, G_FILE_ATTRIBUTE_UNIX_GID, new_id);
diff --git a/src/nautilus-directory-view-ui.xml b/src/nautilus-directory-view-ui.xml
index 62d1a1b..30dc7e2 100644
--- a/src/nautilus-directory-view-ui.xml
+++ b/src/nautilus-directory-view-ui.xml
@@ -56,6 +56,10 @@
<menuitem name="Copy" action="Copy"/>
<menuitem name="Paste" action="Paste"/>
</placeholder>
+ <placeholder name="Undostack Actions">
+ <menuitem name="Undo" action="Undo"/>
+ <menuitem name="Redo" action="Redo"/>
+ </placeholder>
<placeholder name="Select Items">
<menuitem name="Select All" action="Select All"/>
<menuitem name="Select Pattern" action="Select Pattern"/>
diff --git a/src/nautilus-shell-ui.xml b/src/nautilus-shell-ui.xml
index b7ec379..0861fa6 100644
--- a/src/nautilus-shell-ui.xml
+++ b/src/nautilus-shell-ui.xml
@@ -32,6 +32,8 @@
<menuitem name="Close" action="Close"/>
</menu>
<menu action="Edit">
+ <placeholder name="Undostack Actions"/>
+ <separator/>
<placeholder name="Clipboard Actions">
</placeholder>
<separator/>
diff --git a/src/nautilus-view.c b/src/nautilus-view.c
index fd8bb0c..1470bb3 100644
--- a/src/nautilus-view.c
+++ b/src/nautilus-view.c
@@ -78,6 +78,7 @@
#include <libnautilus-private/nautilus-ui-utilities.h>
#include <libnautilus-private/nautilus-signaller.h>
#include <libnautilus-private/nautilus-icon-names.h>
+#include <libnautilus-private/nautilus-file-undo-manager.h>
#define DEBUG_FLAG NAUTILUS_DEBUG_DIRECTORY_VIEW
#include <libnautilus-private/nautilus-debug.h>
@@ -2351,6 +2352,13 @@ nautilus_view_get_selection_count (NautilusView *view)
return len;
}
+static void
+undo_manager_changed_cb (NautilusFileUndoManager* manager,
+ NautilusView *view)
+{
+ schedule_update_menus (view);
+}
+
void
nautilus_view_set_selection (NautilusView *nautilus_view,
GList *selection)
@@ -2404,6 +2412,7 @@ nautilus_view_init (NautilusView *view)
NautilusDirectory *scripts_directory;
NautilusDirectory *templates_directory;
char *templates_uri;
+ NautilusFileUndoManager* manager;
view->details = G_TYPE_INSTANCE_GET_PRIVATE (view, NAUTILUS_TYPE_VIEW,
NautilusViewDetails);
@@ -2481,6 +2490,10 @@ nautilus_view_init (NautilusView *view)
g_signal_connect_swapped (nautilus_window_state,
"changed::" NAUTILUS_WINDOW_STATE_START_WITH_STATUS_BAR,
G_CALLBACK (nautilus_view_display_selection_info), view);
+
+ manager = nautilus_file_undo_manager_get ();
+ g_signal_connect_object (manager, "undo-changed",
+ G_CALLBACK (undo_manager_changed_cb), view, 0);
/* Accessibility */
atk_object = gtk_widget_get_accessible (GTK_WIDGET (view));
@@ -3146,6 +3159,8 @@ copy_move_done_callback (GHashTable *debuting_files, gpointer data)
(GClosureNotify) debuting_files_data_free,
G_CONNECT_AFTER);
}
+ /* Schedule menu update for undo items */
+ schedule_update_menus (directory_view);
}
copy_move_done_data_free (copy_move_done_data);
@@ -5989,6 +6004,104 @@ invoke_external_bulk_rename_utility (NautilusView *view,
}
static void
+update_undo_actions (NautilusView *view,
+ gboolean invalidate)
+{
+ NautilusFileUndoManager *manager;
+ NautilusFileUndoMenuData *menu_data;
+ GtkAction *action;
+ const gchar *label, *tooltip;
+ gboolean available;
+ gboolean undo_active, redo_active;
+
+ manager = nautilus_file_undo_manager_get ();
+ menu_data = nautilus_file_undo_manager_get_menu_data (manager);
+
+ undo_active = menu_data->undo_label != NULL && !invalidate;
+ redo_active = menu_data->redo_label != NULL && !invalidate;
+
+ /* Update undo entry */
+ action = gtk_action_group_get_action (view->details->dir_action_group,
+ "Undo");
+ available = undo_active;
+ if (available) {
+ label = menu_data->undo_label;
+ tooltip = menu_data->undo_description;
+ } else {
+ /* Reset to default info */
+ label = _("Undo");
+ tooltip = _("Undo the last action");
+ }
+
+ g_object_set (action,
+ "label", label,
+ "tooltip", tooltip,
+ NULL);
+ gtk_action_set_sensitive (action, available);
+
+ /* Update redo entry */
+ action = gtk_action_group_get_action (view->details->dir_action_group,
+ "Redo");
+ available = redo_active;
+ if (available) {
+ label = menu_data->redo_label;
+ tooltip = menu_data->redo_description;
+ } else {
+ /* Reset to default info */
+ label = _("Redo");
+ tooltip = _("Redo the last undone action");
+ }
+
+ g_object_set (action,
+ "label", label,
+ "tooltip", tooltip,
+ NULL);
+ gtk_action_set_sensitive (action, available);
+
+ nautilus_file_undo_menu_data_free (menu_data);
+}
+
+static void
+real_action_undo (NautilusView *view)
+{
+ NautilusFileUndoManager *manager;
+
+ manager = nautilus_file_undo_manager_get ();
+
+ /* Disable menus because they are in an untrustworthy status */
+ update_undo_actions (view, TRUE);
+
+ nautilus_file_undo_manager_undo (manager, NULL, NULL);
+}
+
+static void
+real_action_redo (NautilusView *view)
+{
+ NautilusFileUndoManager *manager;
+
+ manager = nautilus_file_undo_manager_get ();
+
+ /* Disable menus because they are in an untrustworthy status */
+ update_undo_actions (view, TRUE);
+
+ nautilus_file_undo_manager_redo (manager, NULL, NULL);
+}
+
+static void
+action_undo_callback (GtkAction *action,
+ gpointer callback_data)
+{
+ real_action_undo (NAUTILUS_VIEW (callback_data));
+}
+
+static void
+action_redo_callback (GtkAction *action,
+ gpointer callback_data)
+{
+ real_action_redo (NAUTILUS_VIEW (callback_data));
+}
+
+static void
real_action_rename (NautilusView *view,
gboolean select_all)
{
@@ -6934,7 +7047,15 @@ static const GtkActionEntry directory_view_entries[] = {
/* name, stock id */ { "Restore From Trash", NULL,
/* label, accelerator */ N_("_Restore"), NULL,
NULL,
- G_CALLBACK (action_restore_from_trash_callback) },
+ G_CALLBACK (action_restore_from_trash_callback) },
+ /* name, stock id */ { "Undo", GTK_STOCK_UNDO,
+ /* label, accelerator */ N_("_Undo"), "<control>Z",
+ /* tooltip */ N_("Undo the last action"),
+ G_CALLBACK (action_undo_callback) },
+ /* name, stock id */ { "Redo", GTK_STOCK_REDO,
+ /* label, accelerator */ N_("_Redo"), "<control>Y",
+ /* tooltip */ N_("Redo the last undone action"),
+ G_CALLBACK (action_redo_callback) },
/*
* multiview-TODO: decide whether "Reset to Defaults" should
* be window-wide, and not just view-wide.
@@ -8454,6 +8575,8 @@ real_update_menus (NautilusView *view)
real_update_menus_volumes (view, selection, selection_count);
+ update_undo_actions (view, FALSE);
+
nautilus_file_list_free (selection);
if (view->details->scripts_invalid) {
diff --git a/src/nautilus-window-menus.c b/src/nautilus-window-menus.c
index 4dc4a22..149ba2a 100644
--- a/src/nautilus-window-menus.c
+++ b/src/nautilus-window-menus.c
@@ -106,6 +106,7 @@ action_stop_callback (GtkAction *action,
nautilus_window_slot_stop_loading (slot);
}
+#ifdef TEXT_CHANGE_UNDO
static void
action_undo_callback (GtkAction *action,
gpointer user_data)
@@ -115,6 +116,7 @@ action_undo_callback (GtkAction *action,
app = nautilus_application_get_singleton ();
nautilus_undo_manager_undo (app->undo_manager);
}
+#endif
static void
action_home_callback (GtkAction *action,
@@ -918,9 +920,11 @@ static const GtkActionEntry main_entries[] = {
N_("Prefere_nces"),
NULL, N_("Edit Nautilus preferences"),
G_CALLBACK (action_preferences_callback) },
+#ifdef TEXT_CHANGE_UNDO
/* name, stock id, label */ { "Undo", NULL, N_("_Undo"),
"<control>Z", N_("Undo the last text change"),
G_CALLBACK (action_undo_callback) },
+#endif
/* name, stock id, label */ { "Up", GTK_STOCK_GO_UP, N_("Open _Parent"),
"<alt>Up", N_("Open the parent folder"),
G_CALLBACK (action_up_callback) },
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]