423 lines
11 KiB
C
423 lines
11 KiB
C
/*
|
|
* moooutputfilter.c
|
|
*
|
|
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@sourceforge.net>
|
|
*
|
|
* 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.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "mooedit/moooutputfilter.h"
|
|
#include "mooedit/mooeditor.h"
|
|
#include "marshals.h"
|
|
#include "mooutils/mooutils-debug.h"
|
|
#include "mooutils/mooutils-misc.h"
|
|
#include "mooutils/mootype-macros.h"
|
|
#include <string.h>
|
|
|
|
|
|
struct _MooOutputFilterPrivate {
|
|
MooLineView *view;
|
|
char **active_dirs;
|
|
char *active_file;
|
|
MooEditWindow *window;
|
|
};
|
|
|
|
|
|
G_DEFINE_TYPE (MooOutputFilter, moo_output_filter, G_TYPE_OBJECT)
|
|
|
|
|
|
static void moo_output_filter_open_file_line (MooOutputFilter *filter,
|
|
MooFileLineData *data);
|
|
|
|
|
|
enum {
|
|
STDOUT_LINE,
|
|
STDERR_LINE,
|
|
CMD_START,
|
|
CMD_EXIT,
|
|
N_SIGNALS
|
|
};
|
|
|
|
static guint signals[N_SIGNALS];
|
|
|
|
|
|
static void
|
|
moo_output_filter_finalize (GObject *object)
|
|
{
|
|
MooOutputFilter *filter = MOO_OUTPUT_FILTER (object);
|
|
|
|
g_strfreev (filter->priv->active_dirs);
|
|
g_free (filter->priv->active_file);
|
|
|
|
G_OBJECT_CLASS (moo_output_filter_parent_class)->finalize (object);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_output_filter_activate (MooOutputFilter *filter,
|
|
int line)
|
|
{
|
|
MooFileLineData *data;
|
|
|
|
data = moo_line_view_get_boxed (filter->priv->view, line, MOO_TYPE_FILE_LINE_DATA);
|
|
|
|
if (data)
|
|
{
|
|
moo_output_filter_open_file_line (filter, data);
|
|
moo_file_line_data_free (data);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
moo_output_filter_class_init (MooOutputFilterClass *klass)
|
|
{
|
|
G_OBJECT_CLASS (klass)->finalize = moo_output_filter_finalize;
|
|
klass->activate = moo_output_filter_activate;
|
|
|
|
g_type_class_add_private (klass, sizeof (MooOutputFilterPrivate));
|
|
|
|
signals[STDOUT_LINE] =
|
|
g_signal_new ("stdout-line",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooOutputFilterClass, stdout_line),
|
|
g_signal_accumulator_true_handled, NULL,
|
|
_moo_marshal_BOOL__STRING,
|
|
G_TYPE_BOOLEAN, 1,
|
|
G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE);
|
|
|
|
signals[STDERR_LINE] =
|
|
g_signal_new ("stderr-line",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooOutputFilterClass, stderr_line),
|
|
g_signal_accumulator_true_handled, NULL,
|
|
_moo_marshal_BOOL__STRING,
|
|
G_TYPE_BOOLEAN, 1,
|
|
G_TYPE_STRING | G_SIGNAL_TYPE_STATIC_SCOPE);
|
|
|
|
signals[CMD_START] =
|
|
g_signal_new ("cmd-start",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooOutputFilterClass, cmd_start),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
|
|
signals[CMD_EXIT] =
|
|
g_signal_new ("cmd-exit",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooOutputFilterClass, cmd_exit),
|
|
g_signal_accumulator_true_handled, NULL,
|
|
_moo_marshal_BOOL__INT,
|
|
G_TYPE_BOOLEAN, 1,
|
|
G_TYPE_INT);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_output_filter_init (MooOutputFilter *filter)
|
|
{
|
|
filter->priv = G_TYPE_INSTANCE_GET_PRIVATE (filter, MOO_TYPE_OUTPUT_FILTER, MooOutputFilterPrivate);
|
|
}
|
|
|
|
|
|
static void
|
|
view_activate (MooLineView *view,
|
|
int line,
|
|
MooOutputFilter *filter)
|
|
{
|
|
g_return_if_fail (MOO_IS_LINE_VIEW (view));
|
|
g_return_if_fail (MOO_IS_OUTPUT_FILTER (filter));
|
|
|
|
if (MOO_OUTPUT_FILTER_GET_CLASS (filter)->activate)
|
|
MOO_OUTPUT_FILTER_GET_CLASS (filter)->activate (filter, line);
|
|
}
|
|
|
|
|
|
void
|
|
moo_output_filter_set_view (MooOutputFilter *filter,
|
|
MooLineView *view)
|
|
{
|
|
g_return_if_fail (MOO_IS_OUTPUT_FILTER (filter));
|
|
g_return_if_fail (!view || MOO_IS_LINE_VIEW (view));
|
|
|
|
if (filter->priv->view == view)
|
|
return;
|
|
|
|
if (filter->priv->view)
|
|
{
|
|
g_signal_handlers_disconnect_by_func (filter->priv->view,
|
|
(gpointer) view_activate,
|
|
filter);
|
|
|
|
if (MOO_OUTPUT_FILTER_GET_CLASS (filter)->detach)
|
|
MOO_OUTPUT_FILTER_GET_CLASS (filter)->detach (filter);
|
|
}
|
|
|
|
filter->priv->view = view;
|
|
|
|
if (view)
|
|
{
|
|
g_signal_connect (view, "activate", G_CALLBACK (view_activate), filter);
|
|
|
|
if (MOO_OUTPUT_FILTER_GET_CLASS (filter)->attach)
|
|
MOO_OUTPUT_FILTER_GET_CLASS (filter)->attach (filter);
|
|
}
|
|
}
|
|
|
|
|
|
MooLineView *
|
|
moo_output_filter_get_view (MooOutputFilter *filter)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_OUTPUT_FILTER (filter), NULL);
|
|
return filter->priv->view;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
moo_output_filter_output_line (MooOutputFilter *filter,
|
|
const char *line,
|
|
guint sig)
|
|
{
|
|
gboolean result = FALSE;
|
|
|
|
if (line[0] && line[1] && line[strlen(line) - 1] == '\n')
|
|
g_warning ("%s: oops", G_STRLOC);
|
|
|
|
g_signal_emit (filter, signals[sig], 0, line, &result);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
gboolean
|
|
moo_output_filter_stdout_line (MooOutputFilter *filter,
|
|
const char *line)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_OUTPUT_FILTER (filter), FALSE);
|
|
g_return_val_if_fail (line != NULL, FALSE);
|
|
return moo_output_filter_output_line (filter, line, STDOUT_LINE);
|
|
}
|
|
|
|
|
|
gboolean
|
|
moo_output_filter_stderr_line (MooOutputFilter *filter,
|
|
const char *line)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_OUTPUT_FILTER (filter), FALSE);
|
|
g_return_val_if_fail (line != NULL, FALSE);
|
|
return moo_output_filter_output_line (filter, line, STDERR_LINE);
|
|
}
|
|
|
|
|
|
void
|
|
moo_output_filter_cmd_start (MooOutputFilter *filter,
|
|
const char *working_dir)
|
|
{
|
|
char *dirs[2];
|
|
|
|
g_return_if_fail (MOO_IS_OUTPUT_FILTER (filter));
|
|
|
|
dirs[0] = (char*) working_dir;
|
|
dirs[1] = NULL;
|
|
MOO_ASSIGN_STRV (filter->priv->active_dirs, dirs);
|
|
|
|
g_signal_emit (filter, signals[CMD_START], 0);
|
|
}
|
|
|
|
void
|
|
moo_output_filter_add_active_dirs (MooOutputFilter *filter,
|
|
char **dirs)
|
|
{
|
|
char **new_dirs, **p, **t;
|
|
int n;
|
|
|
|
g_return_if_fail (MOO_IS_OUTPUT_FILTER (filter));
|
|
|
|
if (!dirs || !dirs[0])
|
|
return;
|
|
|
|
n = g_strv_length (filter->priv->active_dirs) + g_strv_length (dirs);
|
|
new_dirs = g_new (char*, n + 1);
|
|
|
|
for (p = new_dirs, t = filter->priv->active_dirs; t && *t; ++p, ++t)
|
|
*p = *t;
|
|
for (t = dirs; t && *t; ++p, ++t)
|
|
*p = g_strdup (*t);
|
|
*p = NULL;
|
|
|
|
g_free (filter->priv->active_dirs);
|
|
filter->priv->active_dirs = new_dirs;
|
|
}
|
|
|
|
|
|
gboolean
|
|
moo_output_filter_cmd_exit (MooOutputFilter *filter,
|
|
int status)
|
|
{
|
|
gboolean result = FALSE;
|
|
|
|
g_return_val_if_fail (MOO_IS_OUTPUT_FILTER (filter), FALSE);
|
|
|
|
g_signal_emit (filter, signals[CMD_EXIT], 0, status, &result);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
MooFileLineData *
|
|
moo_file_line_data_new (const char *file,
|
|
int line,
|
|
int character)
|
|
{
|
|
MooFileLineData *data;
|
|
|
|
data = g_new0 (MooFileLineData, 1);
|
|
data->file = file && file[0] ? g_strdup (file) : NULL;
|
|
data->line = line;
|
|
data->character = character;
|
|
|
|
return data;
|
|
}
|
|
|
|
static MooFileLineData *
|
|
moo_file_line_data_copy (MooFileLineData *data)
|
|
{
|
|
MooFileLineData *copy = NULL;
|
|
|
|
if (data)
|
|
{
|
|
copy = g_memdup (data, sizeof (MooFileLineData));
|
|
copy->file = g_strdup (data->file);
|
|
}
|
|
|
|
return copy;
|
|
}
|
|
|
|
void
|
|
moo_file_line_data_free (MooFileLineData *data)
|
|
{
|
|
if (data)
|
|
{
|
|
g_free (data->file);
|
|
g_free (data);
|
|
}
|
|
}
|
|
|
|
MOO_DEFINE_BOXED_TYPE_C (MooFileLineData, moo_file_line_data)
|
|
|
|
|
|
const char *
|
|
moo_output_filter_get_active_file (MooOutputFilter *filter)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_OUTPUT_FILTER (filter), NULL);
|
|
return filter->priv->active_file;
|
|
}
|
|
|
|
|
|
void
|
|
moo_output_filter_set_active_file (MooOutputFilter *filter,
|
|
const char *filename)
|
|
{
|
|
g_return_if_fail (MOO_IS_OUTPUT_FILTER (filter));
|
|
MOO_ASSIGN_STRING (filter->priv->active_file, filename);
|
|
}
|
|
|
|
|
|
const char * const *
|
|
moo_output_filter_get_active_dirs (MooOutputFilter *filter)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_OUTPUT_FILTER (filter), NULL);
|
|
return (const char*const*) filter->priv->active_dirs;
|
|
}
|
|
|
|
|
|
void
|
|
moo_output_filter_set_window (MooOutputFilter *filter,
|
|
gpointer window)
|
|
{
|
|
g_return_if_fail (MOO_IS_OUTPUT_FILTER (filter));
|
|
g_return_if_fail (!window || MOO_IS_EDIT_WINDOW (window));
|
|
filter->priv->window = window;
|
|
}
|
|
|
|
#if 0
|
|
gpointer
|
|
moo_output_filter_get_window (MooOutputFilter *filter)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_OUTPUT_FILTER (filter), NULL);
|
|
return filter->priv->window;
|
|
}
|
|
#endif
|
|
|
|
|
|
static char *
|
|
find_file_in_dirs (const char *filename,
|
|
char **dirs)
|
|
{
|
|
for ( ; dirs && *dirs; ++dirs)
|
|
{
|
|
char *path = g_build_filename (*dirs, filename, NULL);
|
|
if (g_file_test (path, G_FILE_TEST_EXISTS))
|
|
return path;
|
|
g_free (path);
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
static void
|
|
moo_output_filter_open_file_line (MooOutputFilter *filter,
|
|
MooFileLineData *data)
|
|
{
|
|
const char *filename;
|
|
const char *path = NULL;
|
|
char *freeme = NULL;
|
|
|
|
g_return_if_fail (MOO_IS_OUTPUT_FILTER (filter));
|
|
g_return_if_fail (data != NULL);
|
|
|
|
filename = data->file ? data->file : filter->priv->active_file;
|
|
g_return_if_fail (filename != NULL);
|
|
|
|
if (g_path_is_absolute (filename))
|
|
{
|
|
path = filename;
|
|
}
|
|
else
|
|
{
|
|
freeme = find_file_in_dirs (filename, filter->priv->active_dirs);
|
|
path = freeme;
|
|
}
|
|
|
|
if (path)
|
|
{
|
|
if (g_file_test (path, G_FILE_TEST_EXISTS))
|
|
{
|
|
MooEditor *editor = moo_editor_instance ();
|
|
moo_editor_open_file_line (editor, path, data->line, filter->priv->window);
|
|
}
|
|
else
|
|
{
|
|
_moo_message ("file '%s' does not exist", path);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
_moo_message ("could not find file '%s'", filename);
|
|
}
|
|
|
|
g_free (freeme);
|
|
}
|