/* -*- Mode: C; indent-tabs-mode: t; c-basic-offset: 8; tab-width: 8 -*- */ /* CajaUndoTransaction - An object for an undoable transaction. * Used internally by undo machinery. * Not public. * * Copyright (C) 2000 Eazel, Inc. * * Author: Gene Z. Ragan <gzr@eazel.com> * * This 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. * * 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public * License along with this library; if not, write to the * Free Software Foundation, Inc., 51 Franklin St, Fifth Floor, * Boston, MA 02110-1301, USA. */ #include <config.h> #include <libcaja-private/caja-undo.h> #include <libcaja-private/caja-undo-manager.h> #include <libcaja-private/caja-undo-transaction.h> #include "caja-undo-private.h" #include <gtk/gtk.h> #define CAJA_UNDO_TRANSACTION_LIST_DATA "Caja undo transaction list" /* undo atoms */ static void undo_atom_list_free (GList *list); static void undo_atom_list_undo_and_free (GList *list); G_DEFINE_TYPE (CajaUndoTransaction, caja_undo_transaction, G_TYPE_OBJECT); #ifdef UIH static Caja_Undo_MenuItem * impl_Caja_Undo_Transaction__get_undo_menu_item (PortableServer_Servant servant, CORBA_Environment *ev) { CajaUndoTransaction *transaction; Caja_Undo_MenuItem *item; transaction = CAJA_UNDO_TRANSACTION (matecomponent_object_from_servant (servant)); item = Caja_Undo_MenuItem__alloc (); item->label = CORBA_string_dup (transaction->undo_menu_item_label); item->hint = CORBA_string_dup (transaction->undo_menu_item_hint); return item; } static Caja_Undo_MenuItem * impl_Caja_Undo_Transaction__get_redo_menu_item (PortableServer_Servant servant, CORBA_Environment *ev) { CajaUndoTransaction *transaction; Caja_Undo_MenuItem *item; transaction = CAJA_UNDO_TRANSACTION (matecomponent_object_from_servant (servant)); item = Caja_Undo_MenuItem__alloc (); item->label = CORBA_string_dup (transaction->redo_menu_item_label); item->hint = CORBA_string_dup (transaction->redo_menu_item_hint); return item; } static CORBA_char * impl_Caja_Undo_Transaction__get_operation_name (PortableServer_Servant servant, CORBA_Environment *ev) { CajaUndoTransaction *transaction; transaction = CAJA_UNDO_TRANSACTION (matecomponent_object_from_servant (servant)); return CORBA_string_dup (transaction->operation_name); } #endif CajaUndoTransaction * caja_undo_transaction_new (const char *operation_name, const char *undo_menu_item_label, const char *undo_menu_item_hint, const char *redo_menu_item_label, const char *redo_menu_item_hint) { CajaUndoTransaction *transaction; transaction = CAJA_UNDO_TRANSACTION (g_object_new (caja_undo_transaction_get_type (), NULL)); transaction->operation_name = g_strdup (operation_name); transaction->undo_menu_item_label = g_strdup (undo_menu_item_label); transaction->undo_menu_item_hint = g_strdup (undo_menu_item_hint); transaction->redo_menu_item_label = g_strdup (redo_menu_item_label); transaction->redo_menu_item_hint = g_strdup (redo_menu_item_hint); return transaction; } static void caja_undo_transaction_init (CajaUndoTransaction *transaction) { } static void remove_transaction_from_object (gpointer list_data, gpointer callback_data) { CajaUndoAtom *atom; CajaUndoTransaction *transaction; GList *list; g_assert (list_data != NULL); atom = list_data; transaction = CAJA_UNDO_TRANSACTION (callback_data); /* Remove the transaction from the list on the atom. */ list = g_object_get_data (atom->target, CAJA_UNDO_TRANSACTION_LIST_DATA); if (list != NULL) { list = g_list_remove (list, transaction); g_object_set_data (atom->target, CAJA_UNDO_TRANSACTION_LIST_DATA, list); } } static void remove_transaction_from_atom_targets (CajaUndoTransaction *transaction) { g_list_foreach (transaction->atom_list, remove_transaction_from_object, transaction); } static void caja_undo_transaction_finalize (GObject *object) { CajaUndoTransaction *transaction; transaction = CAJA_UNDO_TRANSACTION (object); remove_transaction_from_atom_targets (transaction); undo_atom_list_free (transaction->atom_list); g_free (transaction->operation_name); g_free (transaction->undo_menu_item_label); g_free (transaction->undo_menu_item_hint); g_free (transaction->redo_menu_item_label); g_free (transaction->redo_menu_item_hint); if (transaction->owner != NULL) { g_object_unref (transaction->owner); } G_OBJECT_CLASS (caja_undo_transaction_parent_class)->finalize (object); } void caja_undo_transaction_add_atom (CajaUndoTransaction *transaction, const CajaUndoAtom *atom) { GList *list; g_return_if_fail (CAJA_IS_UNDO_TRANSACTION (transaction)); g_return_if_fail (atom != NULL); g_return_if_fail (GTK_IS_OBJECT (atom->target)); /* Add the atom to the atom list in the transaction. */ transaction->atom_list = g_list_append (transaction->atom_list, g_memdup (atom, sizeof (*atom))); /* Add the transaction to the list on the atoms target object. */ list = g_object_get_data (atom->target, CAJA_UNDO_TRANSACTION_LIST_DATA); if (g_list_find (list, transaction) != NULL) { return; } /* If it's not already on that atom, this object is new. */ list = g_list_prepend (list, transaction); g_object_set_data (atom->target, CAJA_UNDO_TRANSACTION_LIST_DATA, list); /* Connect a signal handler to the atom so it will unregister * itself when it's destroyed. */ g_signal_connect (atom->target, "destroy", G_CALLBACK (caja_undo_transaction_unregister_object), NULL); } void caja_undo_transaction_undo (CajaUndoTransaction *transaction) { g_return_if_fail (CAJA_IS_UNDO_TRANSACTION (transaction)); remove_transaction_from_atom_targets (transaction); undo_atom_list_undo_and_free (transaction->atom_list); transaction->atom_list = NULL; } void caja_undo_transaction_add_to_undo_manager (CajaUndoTransaction *transaction, CajaUndoManager *manager) { g_return_if_fail (CAJA_IS_UNDO_TRANSACTION (transaction)); g_return_if_fail (transaction->owner == NULL); if (manager != NULL) { caja_undo_manager_append (manager, transaction); transaction->owner = g_object_ref (manager); } } static void remove_atoms (CajaUndoTransaction *transaction, GObject *object) { GList *p, *next; CajaUndoAtom *atom; g_assert (CAJA_IS_UNDO_TRANSACTION (transaction)); g_assert (G_IS_OBJECT (object)); /* Destroy any atoms for this object. */ for (p = transaction->atom_list; p != NULL; p = next) { atom = p->data; next = p->next; if (atom->target == object) { transaction->atom_list = g_list_remove_link (transaction->atom_list, p); undo_atom_list_free (p); } } /* If all the atoms are gone, forget this transaction. * This may end up freeing the transaction. */ if (transaction->atom_list == NULL) { caja_undo_manager_forget ( transaction->owner, transaction); } } static void remove_atoms_cover (gpointer list_data, gpointer callback_data) { if (CAJA_IS_UNDO_TRANSACTION (list_data)) { remove_atoms (CAJA_UNDO_TRANSACTION (list_data), G_OBJECT (callback_data)); } } void caja_undo_transaction_unregister_object (GObject *object) { GList *list; g_return_if_fail (G_IS_OBJECT (object)); /* Remove atoms from each transaction on the list. */ list = g_object_get_data (object, CAJA_UNDO_TRANSACTION_LIST_DATA); if (list != NULL) { g_list_foreach (list, remove_atoms_cover, object); g_list_free (list); g_object_set_data (object, CAJA_UNDO_TRANSACTION_LIST_DATA, NULL); } } static void undo_atom_free (CajaUndoAtom *atom) { /* Call the destroy-notify function if it's present. */ if (atom->callback_data_destroy_notify != NULL) { (* atom->callback_data_destroy_notify) (atom->callback_data); } /* Free the atom storage. */ g_free (atom); } static void undo_atom_undo_and_free (CajaUndoAtom *atom) { /* Call the function that does the actual undo. */ (* atom->callback) (atom->target, atom->callback_data); /* Get rid of the atom now that it's spent. */ undo_atom_free (atom); } static void undo_atom_free_cover (gpointer atom, gpointer callback_data) { g_assert (atom != NULL); g_assert (callback_data == NULL); undo_atom_free (atom); } static void undo_atom_undo_and_free_cover (gpointer atom, gpointer callback_data) { g_assert (atom != NULL); g_assert (callback_data == NULL); undo_atom_undo_and_free (atom); } static void undo_atom_list_free (GList *list) { g_list_foreach (list, undo_atom_free_cover, NULL); g_list_free (list); } static void undo_atom_list_undo_and_free (GList *list) { g_list_foreach (list, undo_atom_undo_and_free_cover, NULL); g_list_free (list); } static void caja_undo_transaction_class_init (CajaUndoTransactionClass *klass) { G_OBJECT_CLASS (klass)->finalize = caja_undo_transaction_finalize; }