2013-09-26 02:02:59 +08:00
|
|
|
/* GIMP - The GNU Image Manipulation Program
|
2014-02-19 06:24:48 +08:00
|
|
|
* Copyright (C) 1995 Spencer Kimball and Peter Mattis
|
2013-09-26 02:02:59 +08:00
|
|
|
*
|
|
|
|
* gimpaction-history.c
|
2014-02-19 06:24:48 +08:00
|
|
|
* Copyright (C) 2013 Jehan <jehan at girinstud.io>
|
2013-09-26 02:02:59 +08:00
|
|
|
*
|
|
|
|
* 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "config.h"
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
#include <gtk/gtk.h>
|
|
|
|
|
|
|
|
#include "libgimpbase/gimpbase.h"
|
|
|
|
|
|
|
|
#include "widgets-types.h"
|
|
|
|
|
|
|
|
#include "config/gimpguiconfig.h"
|
|
|
|
|
|
|
|
#include "gimpuimanager.h"
|
|
|
|
#include "gimpaction.h"
|
|
|
|
#include "gimpaction-history.h"
|
|
|
|
|
|
|
|
|
2014-02-19 06:24:48 +08:00
|
|
|
#define GIMP_ACTION_HISTORY_FILENAME "action-history"
|
|
|
|
|
|
|
|
|
|
|
|
typedef struct
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GtkAction *action;
|
|
|
|
const gchar *name;
|
|
|
|
gint count;
|
2013-09-26 02:02:59 +08:00
|
|
|
} GimpActionHistoryItem;
|
|
|
|
|
2014-02-19 06:24:48 +08:00
|
|
|
static struct
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GList *items;
|
2013-09-26 02:02:59 +08:00
|
|
|
} history;
|
|
|
|
|
2014-02-19 06:24:48 +08:00
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
static GimpActionHistoryItem *
|
|
|
|
gimp_action_history_item_new (GtkAction *action,
|
|
|
|
gint count);
|
2013-09-26 02:02:59 +08:00
|
|
|
static void gimp_action_history_item_free (GimpActionHistoryItem *item);
|
|
|
|
|
|
|
|
static gint gimp_action_history_init_compare_func (GimpActionHistoryItem *a,
|
|
|
|
GimpActionHistoryItem *b);
|
|
|
|
static gint gimp_action_history_compare_func (GimpActionHistoryItem *a,
|
|
|
|
GimpActionHistoryItem *b);
|
|
|
|
|
2014-02-19 06:24:48 +08:00
|
|
|
|
2013-09-26 02:02:59 +08:00
|
|
|
/* public functions */
|
|
|
|
|
|
|
|
void
|
|
|
|
gimp_action_history_init (GimpGuiConfig *config)
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GimpUIManager *manager;
|
|
|
|
gchar *filename;
|
|
|
|
gint count;
|
|
|
|
FILE *fp;
|
|
|
|
gint i;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
|
|
|
if (history.items != NULL)
|
|
|
|
{
|
|
|
|
g_warning ("%s: must be run only once.", G_STRFUNC);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
filename = gimp_personal_rc_file (GIMP_ACTION_HISTORY_FILENAME);
|
|
|
|
fp = fopen (filename, "r");
|
|
|
|
g_free (filename);
|
2014-02-19 06:24:48 +08:00
|
|
|
|
2013-09-26 02:02:59 +08:00
|
|
|
if (fp == NULL)
|
|
|
|
/* Probably a first use case. Not necessarily an error. */
|
|
|
|
return;
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
manager = gimp_ui_managers_from_name ("<Image>")->data;
|
|
|
|
|
2013-09-26 02:02:59 +08:00
|
|
|
for (i = 0; i < config->action_history_size; i++)
|
|
|
|
{
|
|
|
|
/* Let's assume an action name will never be more than 256 character. */
|
2014-07-28 17:22:20 +08:00
|
|
|
gchar action_name[256];
|
|
|
|
GtkAction *action;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
|
|
|
if (fscanf (fp, "%s %d", action_name, &count) == EOF)
|
|
|
|
break;
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
action = gimp_ui_manager_find_action (manager, NULL, action_name);
|
|
|
|
|
|
|
|
if (action)
|
|
|
|
{
|
|
|
|
history.items =
|
|
|
|
g_list_insert_sorted (history.items,
|
|
|
|
gimp_action_history_item_new (action, count),
|
|
|
|
(GCompareFunc) gimp_action_history_init_compare_func);
|
|
|
|
}
|
2013-09-26 02:02:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (count > 1)
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GList *actions;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
for (actions = history.items, i = 0;
|
|
|
|
actions && i < config->action_history_size;
|
|
|
|
actions = g_list_next (actions), i++)
|
2013-09-26 02:02:59 +08:00
|
|
|
{
|
|
|
|
GimpActionHistoryItem *action = actions->data;
|
|
|
|
|
|
|
|
action->count -= count - 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fclose (fp);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gimp_action_history_exit (GimpGuiConfig *config)
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GimpActionHistoryItem *item;
|
|
|
|
GList *actions;
|
|
|
|
gchar *filename;
|
|
|
|
gint min_count = 0;
|
|
|
|
FILE *fp;
|
|
|
|
gint i;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
|
|
|
/* If we have more items than current history size, trim the history
|
2014-07-28 17:22:20 +08:00
|
|
|
* and move down all count so that 1 is lower.
|
|
|
|
*/
|
|
|
|
item = g_list_nth_data (history.items, config->action_history_size);
|
|
|
|
if (item)
|
|
|
|
min_count = item->count - 1;
|
|
|
|
|
|
|
|
filename = gimp_personal_rc_file (GIMP_ACTION_HISTORY_FILENAME);
|
|
|
|
fp = fopen (filename, "w");
|
|
|
|
g_free (filename);
|
|
|
|
|
|
|
|
for (actions = history.items, i = 0;
|
|
|
|
actions && i < config->action_history_size;
|
|
|
|
actions = g_list_next (actions), i++)
|
2013-09-26 02:02:59 +08:00
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
item = actions->data;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
fprintf (fp, "%s %d \n", item->name, item->count - min_count);
|
2013-09-26 02:02:59 +08:00
|
|
|
}
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
fclose (fp);
|
2013-09-26 02:02:59 +08:00
|
|
|
|
|
|
|
gimp_action_history_empty ();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* gimp_action_history_excluded_action:
|
|
|
|
*
|
|
|
|
* Returns whether an action should be excluded from history.
|
|
|
|
*/
|
|
|
|
gboolean
|
|
|
|
gimp_action_history_excluded_action (const gchar *action_name)
|
|
|
|
{
|
2014-04-20 21:57:57 +08:00
|
|
|
if (gimp_action_is_gui_blacklisted (action_name))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
return (g_str_has_suffix (action_name, "-set") ||
|
2013-09-26 02:02:59 +08:00
|
|
|
g_str_has_suffix (action_name, "-accel") ||
|
|
|
|
g_str_has_prefix (action_name, "context-") ||
|
|
|
|
g_str_has_prefix (action_name, "plug-in-recent-") ||
|
|
|
|
g_strcmp0 (action_name, "plug-in-repeat") == 0 ||
|
|
|
|
g_strcmp0 (action_name, "plug-in-reshow") == 0 ||
|
|
|
|
g_strcmp0 (action_name, "help-action-search") == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Callback run on the `activate` signal of an action.
|
2014-02-19 06:24:48 +08:00
|
|
|
* It allows us to log all used action.
|
|
|
|
*/
|
2013-09-26 02:02:59 +08:00
|
|
|
void
|
|
|
|
gimp_action_history_activate_callback (GtkAction *action,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GList *actions;
|
|
|
|
const gchar *action_name;
|
|
|
|
gint previous_count = 0;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
|
|
|
action_name = gtk_action_get_name (action);
|
|
|
|
|
|
|
|
/* Some specific actions are of no log interest. */
|
|
|
|
if (gimp_action_history_excluded_action (action_name))
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (actions = history.items; actions; actions = g_list_next (actions))
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GimpActionHistoryItem *item = actions->data;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
if (g_strcmp0 (action_name, item->name) == 0)
|
2013-09-26 02:02:59 +08:00
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GimpActionHistoryItem *next_item;
|
|
|
|
|
|
|
|
next_item = g_list_next (actions) ? g_list_next (actions)->data : NULL;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
2014-02-19 06:24:48 +08:00
|
|
|
/* Is there any other item with the same count? We don't
|
|
|
|
* want to leave any count gap to always accept new items.
|
|
|
|
* This means that if we increment the only item with a
|
|
|
|
* given count, we must decrement the next item. Other
|
|
|
|
* consequence is that an item with higher count won't be
|
2014-07-28 17:22:20 +08:00
|
|
|
* incremented at all if no other items have the same count.
|
2014-02-19 06:24:48 +08:00
|
|
|
*/
|
2014-07-28 17:22:20 +08:00
|
|
|
if (previous_count == item->count ||
|
|
|
|
(next_item && next_item->count == item->count))
|
2013-09-26 02:02:59 +08:00
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
item->count++;
|
|
|
|
|
|
|
|
history.items = g_list_remove (history.items, item);
|
|
|
|
history.items = g_list_insert_sorted (history.items, item,
|
2013-09-26 02:02:59 +08:00
|
|
|
(GCompareFunc) gimp_action_history_compare_func);
|
|
|
|
}
|
2014-07-28 17:22:20 +08:00
|
|
|
else if (previous_count != 0 &&
|
|
|
|
previous_count != item->count)
|
2013-09-26 02:02:59 +08:00
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GimpActionHistoryItem *previous_item = g_list_previous (actions)->data;
|
|
|
|
|
|
|
|
item->count++;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
history.items = g_list_remove (history.items, item);
|
|
|
|
history.items = g_list_insert_sorted (history.items, item,
|
2013-09-26 02:02:59 +08:00
|
|
|
(GCompareFunc) gimp_action_history_compare_func);
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
previous_item->count--;
|
|
|
|
|
|
|
|
history.items = g_list_remove (history.items, previous_item);
|
|
|
|
history.items = g_list_insert_sorted (history.items, previous_item,
|
2013-09-26 02:02:59 +08:00
|
|
|
(GCompareFunc) gimp_action_history_compare_func);
|
|
|
|
}
|
2014-07-28 17:22:20 +08:00
|
|
|
|
2013-09-26 02:02:59 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
previous_count = item->count;
|
2013-09-26 02:02:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
/* If we are here, this action is not logged yet. */
|
2014-07-28 17:22:20 +08:00
|
|
|
history.items =
|
|
|
|
g_list_insert_sorted (history.items,
|
|
|
|
gimp_action_history_item_new (action, 1),
|
|
|
|
(GCompareFunc) gimp_action_history_compare_func);
|
2013-09-26 02:02:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
gimp_action_history_empty (void)
|
|
|
|
{
|
2014-02-19 06:24:48 +08:00
|
|
|
g_list_free_full (history.items,
|
|
|
|
(GDestroyNotify) gimp_action_history_item_free);
|
2013-09-26 02:02:59 +08:00
|
|
|
history.items = NULL;
|
|
|
|
}
|
|
|
|
|
2014-02-19 06:24:48 +08:00
|
|
|
/* Search all history actions which match "keyword" with function
|
|
|
|
* match_func(action, keyword).
|
|
|
|
*
|
|
|
|
* @return a list of GtkAction*, to free with:
|
|
|
|
* g_list_free_full (result, (GDestroyNotify) g_object_unref);
|
|
|
|
*/
|
2014-07-28 17:22:20 +08:00
|
|
|
GList *
|
2013-09-26 02:02:59 +08:00
|
|
|
gimp_action_history_search (const gchar *keyword,
|
|
|
|
GimpActionMatchFunc match_func,
|
|
|
|
GimpGuiConfig *config)
|
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GList *actions;
|
|
|
|
GList *result = NULL;
|
|
|
|
gint i;
|
|
|
|
|
|
|
|
for (actions = history.items, i = 0;
|
|
|
|
actions && i < config->action_history_size;
|
|
|
|
actions = g_list_next (actions), i++)
|
2013-09-26 02:02:59 +08:00
|
|
|
{
|
2014-07-28 17:22:20 +08:00
|
|
|
GimpActionHistoryItem *item = actions->data;
|
|
|
|
GtkAction *action = item->action;
|
2013-09-26 02:02:59 +08:00
|
|
|
|
2014-02-19 06:24:48 +08:00
|
|
|
if (! gtk_action_is_sensitive (action) &&
|
|
|
|
! config->search_show_unavailable)
|
2013-09-26 02:02:59 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
if (match_func (action, keyword, NULL, FALSE))
|
2014-07-28 17:22:20 +08:00
|
|
|
result = g_list_prepend (result, g_object_ref (action));
|
2013-09-26 02:02:59 +08:00
|
|
|
}
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
return g_list_reverse (result);
|
2013-09-26 02:02:59 +08:00
|
|
|
}
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
|
2013-09-26 02:02:59 +08:00
|
|
|
/* private functions */
|
|
|
|
|
2014-07-28 17:22:20 +08:00
|
|
|
static GimpActionHistoryItem *
|
|
|
|
gimp_action_history_item_new (GtkAction *action,
|
|
|
|
gint count)
|
|
|
|
{
|
|
|
|
GimpActionHistoryItem *item = g_new0 (GimpActionHistoryItem, 1);
|
|
|
|
|
|
|
|
item->action = g_object_ref (action);
|
|
|
|
item->name = gtk_action_get_name (action);
|
|
|
|
item->count = count;
|
|
|
|
|
|
|
|
return item;
|
|
|
|
}
|
|
|
|
|
2013-09-26 02:02:59 +08:00
|
|
|
static void
|
|
|
|
gimp_action_history_item_free (GimpActionHistoryItem *item)
|
|
|
|
{
|
|
|
|
g_object_unref (item->action);
|
|
|
|
g_free (item);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compare function used at list initialization.
|
2014-02-19 06:24:48 +08:00
|
|
|
* We use a slightly different compare function as for runtime insert,
|
|
|
|
* because we want to keep history file order for equal values.
|
|
|
|
*/
|
2013-09-26 02:02:59 +08:00
|
|
|
static gint
|
|
|
|
gimp_action_history_init_compare_func (GimpActionHistoryItem *a,
|
|
|
|
GimpActionHistoryItem *b)
|
|
|
|
{
|
|
|
|
return (a->count <= b->count);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Compare function used when updating the list.
|
2014-02-19 06:24:48 +08:00
|
|
|
* There is no equality case. If they have the same count,
|
|
|
|
* I ensure that the first action (last inserted) will be before.
|
|
|
|
*/
|
2013-09-26 02:02:59 +08:00
|
|
|
static gint
|
|
|
|
gimp_action_history_compare_func (GimpActionHistoryItem *a,
|
|
|
|
GimpActionHistoryItem *b)
|
|
|
|
{
|
|
|
|
return (a->count < b->count);
|
|
|
|
}
|