medit/moo/moofileview/moofileview-tools.c

399 lines
10 KiB
C
Raw Normal View History

2006-05-21 11:17:14 -05:00
/*
* moofileview-tools.c
*
2010-11-07 01:20:45 -08:00
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@sourceforge.net>
2006-05-21 11:17:14 -05:00
*
* This file is part of medit. medit is free software; you can
* redistribute it and/or modify it under the terms of the
* GNU Lesser General Public License as published by the
* Free Software Foundation; either version 2.1 of the License,
* or (at your option) any later version.
2006-05-21 11:17:14 -05:00
*
* You should have received a copy of the GNU Lesser General Public
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
2006-05-21 11:17:14 -05:00
*/
2006-05-22 00:45:55 -05:00
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
2006-08-13 04:00:15 -05:00
#define MOO_FILE_VIEW_COMPILATION
2006-05-21 11:17:14 -05:00
#include "moofileview/moofileview-tools.h"
2006-08-13 04:00:15 -05:00
#include "moofileview/moofileview-private.h"
#include "moofileview/moofile-private.h"
2006-05-22 00:45:55 -05:00
#include "mooutils/mooprefs.h"
#include "mooutils/mooaction.h"
2006-12-19 21:57:52 -06:00
#include "mooutils/mooutils-fs.h"
#include "mooutils/mootype-macros.h"
#include "mooutils/mooi18n.h"
2008-08-31 01:40:29 -05:00
#include "mooutils/moo-mime.h"
2006-05-22 00:45:55 -05:00
#include <string.h>
typedef struct {
guint merge_id;
GSList *actions;
} ToolsInfo;
typedef struct {
MooAction parent;
2006-05-22 00:45:55 -05:00
MooFileView *fileview;
GSList *extensions;
GSList *mimetypes;
char *command;
} ToolAction;
typedef MooActionClass ToolActionClass;
2006-05-22 00:45:55 -05:00
MOO_DEFINE_TYPE_STATIC (ToolAction, _moo_file_view_tool_action, MOO_TYPE_ACTION)
2006-05-22 00:45:55 -05:00
static void
_moo_file_view_tool_action_init (G_GNUC_UNUSED ToolAction *action)
{
}
static void
moo_file_view_tool_action_finalize (GObject *object)
{
ToolAction *action = (ToolAction*) object;
g_slist_foreach (action->extensions, (GFunc) g_free, NULL);
g_slist_free (action->extensions);
g_slist_foreach (action->mimetypes, (GFunc) g_free, NULL);
g_slist_free (action->mimetypes);
g_free (action->command);
G_OBJECT_CLASS (_moo_file_view_tool_action_parent_class)->finalize (object);
}
static void
moo_file_view_tool_action_activate (GtkAction *_action)
{
ToolAction *action = (ToolAction*) _action;
GList *files;
const char *p;
GString *command;
GError *error = NULL;
g_return_if_fail (MOO_IS_FILE_VIEW (action->fileview));
g_return_if_fail (action->command && action->command[0]);
p = strstr (action->command, "%f");
g_return_if_fail (p != NULL);
files = _moo_file_view_get_filenames (action->fileview);
g_return_if_fail (files != NULL);
command = g_string_new_len (action->command, p - action->command);
while (files)
{
g_string_append_printf (command, " \"%s\"", (char*) files->data);
g_free (files->data);
files = g_list_delete_link (files, files);
}
g_string_append (command, p + 2);
if (!g_spawn_command_line_async (command->str, &error))
{
g_warning ("%s: %s", G_STRLOC, error->message);
g_error_free (error);
}
g_string_free (command, TRUE);
}
static void
_moo_file_view_tool_action_class_init (ToolActionClass *klass)
{
G_OBJECT_CLASS (klass)->finalize = moo_file_view_tool_action_finalize;
GTK_ACTION_CLASS (klass)->activate = moo_file_view_tool_action_activate;
2006-05-22 00:45:55 -05:00
}
2006-08-16 00:15:35 -05:00
static void
tools_info_free (ToolsInfo *info)
{
if (info)
{
2008-01-09 12:03:09 -06:00
g_slist_foreach (info->actions, (GFunc) g_object_unref, NULL);
2006-08-16 00:15:35 -05:00
g_slist_free (info->actions);
g_free (info);
}
}
2006-05-22 00:45:55 -05:00
static void
remove_old_tools (MooFileView *fileview,
2008-09-13 14:56:47 -05:00
MooUiXml *xml,
2006-05-22 00:45:55 -05:00
GtkActionGroup *group)
{
ToolsInfo *info;
info = g_object_get_data (G_OBJECT (fileview), "moo-file-view-tools-info");
if (info)
{
moo_ui_xml_remove_ui (xml, info->merge_id);
while (info->actions)
{
2008-01-09 12:03:09 -06:00
GtkAction *action = info->actions->data;
gtk_action_group_remove_action (group, action);
g_object_unref (action);
2006-08-16 00:15:35 -05:00
info->actions = g_slist_delete_link (info->actions, info->actions);
2006-05-22 00:45:55 -05:00
}
}
2006-08-16 00:15:35 -05:00
/* this will call tools_info_free */
g_object_set_data (G_OBJECT (fileview), "moo-file-view-tools-info", NULL);
2006-05-22 00:45:55 -05:00
}
static GtkAction *
tool_action_new (MooFileView *fileview,
const char *label,
const char *extensions,
const char *mimetypes,
const char *command)
{
ToolAction *action;
char *name;
char **set, **p;
g_return_val_if_fail (label != NULL, NULL);
g_return_val_if_fail (command && strstr (command, "%f"), NULL);
name = g_strdup_printf ("moofileview-action-%08x", g_random_int ());
2010-08-12 18:17:00 -07:00
action = (ToolAction*)
g_object_new (_moo_file_view_tool_action_get_type (),
"label", label, "name", name,
(const char*) NULL);
2006-05-22 00:45:55 -05:00
action->fileview = fileview;
action->command = g_strdup (command);
if (extensions && extensions[0])
{
set = g_strsplit_set (extensions, ";,", 0);
for (p = set; p && *p; ++p)
{
g_strstrip (*p);
if (**p)
action->extensions = g_slist_prepend (action->extensions,
g_strdup (*p));
}
action->extensions = g_slist_reverse (action->extensions);
g_strfreev (set);
}
if (mimetypes && mimetypes[0])
{
set = g_strsplit_set (mimetypes, ";,", 0);
for (p = set; p && *p; ++p)
{
g_strstrip (*p);
if (**p)
action->mimetypes = g_slist_prepend (action->mimetypes,
g_strdup (*p));
}
action->mimetypes = g_slist_reverse (action->mimetypes);
g_strfreev (set);
}
g_free (name);
return GTK_ACTION (action);
}
2006-05-21 11:17:14 -05:00
void
2006-05-22 00:45:55 -05:00
_moo_file_view_tools_load (MooFileView *fileview)
2006-05-21 11:17:14 -05:00
{
2006-05-22 00:45:55 -05:00
ToolsInfo *info;
MooMarkupDoc *doc;
MooMarkupNode *root, *child;
2008-09-13 14:56:47 -05:00
MooUiXml *xml;
MooActionCollection *actions;
2006-05-22 00:45:55 -05:00
GtkActionGroup *group;
MooUiNode *ph;
2006-05-22 00:45:55 -05:00
GSList *l;
g_return_if_fail (MOO_IS_FILE_VIEW (fileview));
xml = moo_file_view_get_ui_xml (fileview);
actions = moo_file_view_get_actions (fileview);
group = moo_action_collection_get_group (actions, NULL);
2006-05-22 00:45:55 -05:00
remove_old_tools (fileview, xml, group);
doc = moo_prefs_get_markup (MOO_PREFS_RC);
2006-05-22 00:45:55 -05:00
g_return_if_fail (doc != NULL);
info = g_new0 (ToolsInfo, 1);
info->merge_id = moo_ui_xml_new_merge_id (xml);
g_object_set_data_full (G_OBJECT (fileview), "moo-file-view-tools-info", info,
(GDestroyNotify) tools_info_free);
2006-05-22 00:45:55 -05:00
ph = moo_ui_xml_find_placeholder (xml, "OpenWith");
g_return_if_fail (ph != NULL);
root = moo_markup_get_element (MOO_MARKUP_NODE (doc), "MooFileView/Tools");
2006-05-22 00:45:55 -05:00
for (child = root ? root->children : NULL; child != NULL; child = child->next)
2006-05-22 00:45:55 -05:00
{
GtkAction *action;
const char *label, *extensions, *mimetypes;
const char *command;
if (!MOO_MARKUP_IS_ELEMENT (child))
continue;
if (strcmp (child->name, "tool"))
{
g_warning ("%s: unknown node '%s'", G_STRLOC, child->name);
continue;
}
label = moo_markup_get_prop (child, "label");
extensions = moo_markup_get_prop (child, "extensions");
mimetypes = moo_markup_get_prop (child, "mimetypes");
command = moo_markup_get_content (child);
if (!label)
{
g_warning ("%s: label missing", G_STRLOC);
continue;
}
if (!command)
{
g_warning ("%s: command missing", G_STRLOC);
continue;
}
action = tool_action_new (fileview, label, extensions, mimetypes, command);
if (!action)
continue;
info->actions = g_slist_prepend (info->actions, action);
}
{
2009-11-01 12:44:46 -08:00
GtkAction *action = tool_action_new (fileview, Q_("Open with|Default Application"), "*", NULL,
#ifndef __WIN32__
"xdg-open %f"
#else
"explorer %f"
#endif
);
info->actions = g_slist_prepend (info->actions, action);
2006-05-22 00:45:55 -05:00
}
info->actions = g_slist_reverse (info->actions);
for (l = info->actions; l != NULL; l = l->next)
{
GtkAction *action = l->data;
char *markup;
gtk_action_group_add_action (group, action);
markup = g_markup_printf_escaped ("<item action=\"%s\"/>",
gtk_action_get_name (action));
moo_ui_xml_insert (xml, info->merge_id, ph, -1, markup);
g_free (markup);
}
}
static gboolean
action_check_one (ToolAction *action,
MooFile *file)
{
GSList *l;
G_GNUC_UNUSED const char *mime;
2006-05-22 00:45:55 -05:00
g_return_val_if_fail (file != NULL, FALSE);
2006-12-19 21:57:52 -06:00
if (!action->extensions && !action->mimetypes)
return TRUE;
2006-05-22 00:45:55 -05:00
for (l = action->extensions; l != NULL; l = l->next)
2006-12-19 21:57:52 -06:00
if (_moo_glob_match_simple (l->data, _moo_file_display_name (file)))
2006-05-22 00:45:55 -05:00
return TRUE;
mime = _moo_file_get_mime_type (file);
g_return_val_if_fail (mime != NULL, FALSE);
2008-08-31 01:40:29 -05:00
if (mime == MOO_MIME_TYPE_UNKNOWN)
2006-05-22 00:45:55 -05:00
return FALSE;
for (l = action->mimetypes; l != NULL; l = l->next)
2008-08-31 01:40:29 -05:00
if (moo_mime_type_is_subclass (mime, l->data))
2006-05-22 00:45:55 -05:00
return TRUE;
return FALSE;
}
static void
action_check (ToolAction *action,
GList *files)
{
gboolean visible = TRUE;
while (files)
{
MooFile *f = files->data;
if (!MOO_FILE_EXISTS (f) || MOO_FILE_IS_DIR (f) || !action_check_one (action, f))
2006-05-22 00:45:55 -05:00
{
visible = FALSE;
break;
}
files = files->next;
}
g_object_set (action, "visible", visible, NULL);
}
void
_moo_file_view_tools_check (MooFileView *fileview)
{
GList *files;
ToolsInfo *info;
GSList *l;
info = g_object_get_data (G_OBJECT (fileview), "moo-file-view-tools-info");
if (!info)
return;
files = _moo_file_view_get_files (fileview);
if (!files)
{
for (l = info->actions; l != NULL; l = l->next)
g_object_set (l->data, "visible", FALSE, NULL);
return;
}
for (l = info->actions; l != NULL; l = l->next)
action_check (l->data, files);
g_list_foreach (files, (GFunc) _moo_file_unref, NULL);
g_list_free (files);
2006-05-21 11:17:14 -05:00
}