medit/moo/mooedit/mooeditwindow.c

4262 lines
138 KiB
C

/*
* mooeditwindow.c
*
* Copyright (C) 2004-2007 by Yevgen Muntyan <muntyan@math.tamu.edu>
*
* This library is free software; you can redistribute it and/or
* modify it under the terms of the GNU Lesser General Public
* License version 2.1 as published by the Free Software Foundation.
*
* See COPYING file that comes with this distribution.
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
#define MOOEDIT_COMPILATION
#include "mooedit/statusbar-glade.h"
#include "mooedit/mooedit-private.h"
#include "mooedit/mooeditor-private.h"
#include "mooedit/mooeditfiltersettings.h"
#include "mooedit/moolang.h"
#include "mooedit/mootextbuffer.h"
#include "mooedit/mooeditprefs.h"
#include "mooedit/mooplugin.h"
#include "mooedit/moocmdview.h"
#include "mooedit/mooeditaction.h"
#include "mooedit/mooedit-bookmarks.h"
#include "mooutils/moonotebook.h"
#include "mooutils/moostock.h"
#include "mooutils/moomarshals.h"
#include "mooutils/moomenuaction.h"
#include "mooutils/mooutils-misc.h"
#include "mooutils/moocompat.h"
#include "mooutils/mooglade.h"
#include "mooutils/mooi18n.h"
#include "mooutils/mooaction-private.h"
#include "mooutils/moofiledialog.h"
#include <string.h>
#include <gtk/gtk.h>
#include <math.h>
#if GTK_CHECK_VERSION(2,10,0)
#define ENABLE_PRINTING
#include "mooedit/mootextprint.h"
#endif
#define ENABLE_BOOKMARKS
#define ACTIVE_DOC moo_edit_window_get_active_doc
#define ACTIVE_PAGE(window) (moo_notebook_get_current_page (window->priv->notebook))
#define LANG_ACTION_ID "LanguageMenu"
#define STOP_ACTION_ID "StopJob"
#define DEFAULT_TITLE_FORMAT "%a - %f%s"
#define DEFAULT_TITLE_FORMAT_NO_DOC "%a"
typedef struct {
MooActionCheckFunc func;
gpointer data;
GDestroyNotify notify;
} OnePropCheck;
#define N_ACTION_CHECKS 3
typedef struct {
OnePropCheck checks[N_ACTION_CHECKS];
} ActionCheck;
static GHashTable *action_checks; /* char* -> ActionCheck* */
static GSList *windows;
typedef struct {
MooPaneParams *params;
int position;
} PaneParams;
struct _MooEditWindowPrivate {
MooEditor *editor;
GtkStatusbar *statusbar;
guint statusbar_idle;
guint last_msg_id;
GtkLabel *cursor_label;
GtkLabel *chars_label;
GtkLabel *insert_label;
GtkWidget *info;
MooNotebook *notebook;
GHashTable *panes;
GHashTable *panes_to_save; /* char* -> PaneParams* */
guint save_params_idle;
char *title_format;
char *title_format_no_doc;
MooUIXML *xml;
guint doc_list_merge_id;
guint doc_list_update_idle;
GSList *stop_clients;
GSList *jobs; /* Job* */
GList *history;
guint history_blocked : 1;
};
enum {
TARGET_MOO_EDIT_TAB = 1,
TARGET_URI_LIST = 2
};
static GdkAtom moo_edit_tab_atom;
static GdkAtom text_uri_atom;
static GtkTargetEntry dest_targets[] = {
{(char*) "MOO_EDIT_TAB", GTK_TARGET_SAME_APP, TARGET_MOO_EDIT_TAB},
{(char*) "text/uri-list", 0, TARGET_URI_LIST}
};
static void action_checks_init (void);
static void moo_edit_window_check_actions (MooEditWindow *window);
static GObject *moo_edit_window_constructor (GType type,
guint n_props,
GObjectConstructParam *props);
static void moo_edit_window_finalize (GObject *object);
static void moo_edit_window_destroy (GtkObject *object);
static void moo_edit_window_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec);
static void moo_edit_window_get_property (GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec);
static gboolean moo_edit_window_close (MooWindow *window);
static void setup_notebook (MooEditWindow *window);
static void update_window_title (MooEditWindow *window);
static void set_title_format_from_prefs (MooEditWindow *window);
static void proxy_boolean_property (MooEditWindow *window,
GParamSpec *prop,
MooEdit *doc);
static void edit_changed (MooEditWindow *window,
MooEdit *doc);
static void edit_filename_changed (MooEditWindow *window,
const char *filename,
MooEdit *doc);
static void edit_lang_changed (MooEditWindow *window,
guint var_id,
GParamSpec *pspec,
MooEdit *doc);
static void edit_overwrite_changed (MooEditWindow *window,
GParamSpec *pspec,
MooEdit *doc);
static void edit_wrap_mode_changed (MooEditWindow *window,
GParamSpec *pspec,
MooEdit *doc);
static void edit_show_line_numbers_changed (MooEditWindow *window,
GParamSpec *pspec,
MooEdit *doc);
static GtkWidget *create_tab_label (MooEditWindow *window,
MooEdit *edit);
static void update_tab_label (MooEditWindow *window,
MooEdit *doc);
static void edit_cursor_moved (MooEditWindow *window,
GtkTextIter *iter,
MooEdit *edit);
static void update_lang_menu (MooEditWindow *window);
static void update_doc_view_actions (MooEditWindow *window);
static void create_statusbar (MooEditWindow *window);
static void update_statusbar (MooEditWindow *window);
static MooEdit *get_nth_tab (MooEditWindow *window,
guint n);
static int get_page_num (MooEditWindow *window,
MooEdit *doc);
static GtkAction *create_lang_action (MooEditWindow *window);
static void create_paned (MooEditWindow *window);
static PaneParams *load_pane_params (const char *pane_id);
static gboolean save_pane_params (const char *pane_id,
PaneParams *params);
static void pane_params_changed (MooPane *pane,
GParamSpec *pspec,
MooEditWindow *window);
static void pane_size_changed (MooEditWindow *window,
MooPanePosition position);
static PaneParams *pane_params_new (void);
static void pane_params_free (PaneParams *params);
static void moo_edit_window_update_doc_list (MooEditWindow *window);
static void notebook_drag_data_recv (GtkWidget *widget,
GdkDragContext *context,
int x,
int y,
GtkSelectionData *data,
guint info,
guint time,
MooEditWindow *window);
static gboolean notebook_drag_drop (GtkWidget *widget,
GdkDragContext *context,
int x,
int y,
guint time,
MooEditWindow *window);
static gboolean notebook_drag_motion (GtkWidget *widget,
GdkDragContext *context,
int x,
int y,
guint time,
MooEditWindow *window);
/* actions */
static void action_new_doc (MooEditWindow *window);
static void action_open (MooEditWindow *window);
static void action_reload (MooEditWindow *window);
static void action_save (MooEditWindow *window);
static void action_save_as (MooEditWindow *window);
static void action_close_tab (MooEditWindow *window);
static void action_close_all (MooEditWindow *window);
static void action_previous_tab (MooEditWindow *window);
static void action_next_tab (MooEditWindow *window);
static void action_switch_to_tab (MooEditWindow *window,
guint n);
#ifdef ENABLE_BOOKMARKS
static void action_toggle_bookmark (MooEditWindow *window);
static void action_next_bookmark (MooEditWindow *window);
static void action_prev_bookmark (MooEditWindow *window);
static GtkAction *create_goto_bookmark_action (MooWindow *window,
gpointer data);
static GtkAction *create_bookmarks_menu_action (MooWindow *window,
gpointer data);
#endif
#if 0
static void action_next_ph (MooEditWindow *window);
static void action_prev_ph (MooEditWindow *window);
#endif
static void action_find_now_f (MooEditWindow *window);
static void action_find_now_b (MooEditWindow *window);
static void action_abort_jobs (MooEditWindow *window);
static void wrap_text_toggled (MooEditWindow *window,
gboolean active);
static void line_numbers_toggled (MooEditWindow *window,
gboolean active);
#ifdef ENABLE_PRINTING
static void action_page_setup (MooEditWindow *window);
static void action_print (MooEditWindow *window);
static void action_print_preview (MooEditWindow *window);
static void action_print_pdf (MooEditWindow *window);
#endif
/* MOO_TYPE_EDIT_WINDOW */
G_DEFINE_TYPE (MooEditWindow, moo_edit_window, MOO_TYPE_WINDOW)
enum {
PROP_0,
PROP_EDITOR,
PROP_ACTIVE_DOC,
/* aux properties */
PROP_CAN_RELOAD,
PROP_HAS_OPEN_DOCUMENT,
PROP_CAN_UNDO,
PROP_CAN_REDO,
PROP_HAS_SELECTION,
PROP_HAS_COMMENTS,
PROP_HAS_TEXT,
PROP_HAS_JOBS_RUNNING,
PROP_HAS_STOP_CLIENTS
};
enum {
NEW_DOC,
CLOSE_DOC,
CLOSE_DOC_AFTER,
NUM_SIGNALS
};
static guint signals[NUM_SIGNALS];
#define INSTALL_PROP(prop_id,name) \
g_object_class_install_property (gobject_class, prop_id, \
g_param_spec_boolean (name, name, name, FALSE, G_PARAM_READABLE))
static void
moo_edit_window_class_init (MooEditWindowClass *klass)
{
guint i;
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
GtkObjectClass *gtkobject_class = GTK_OBJECT_CLASS (klass);
MooWindowClass *window_class = MOO_WINDOW_CLASS (klass);
action_checks_init ();
gobject_class->constructor = moo_edit_window_constructor;
gobject_class->finalize = moo_edit_window_finalize;
gobject_class->set_property = moo_edit_window_set_property;
gobject_class->get_property = moo_edit_window_get_property;
gtkobject_class->destroy = moo_edit_window_destroy;
window_class->close = moo_edit_window_close;
moo_edit_tab_atom = gdk_atom_intern ("MOO_EDIT_TAB", FALSE);
text_uri_atom = gdk_atom_intern ("text/uri-list", FALSE);
g_type_class_add_private (klass, sizeof (MooEditWindowPrivate));
g_object_class_install_property (gobject_class,
PROP_EDITOR,
g_param_spec_object ("editor",
"editor",
"editor",
MOO_TYPE_EDITOR,
G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
g_object_class_install_property (gobject_class,
PROP_ACTIVE_DOC,
g_param_spec_object ("active-doc",
"active-doc",
"active-doc",
MOO_TYPE_EDIT,
G_PARAM_READWRITE));
signals[NEW_DOC] =
g_signal_new ("new-doc",
G_OBJECT_CLASS_TYPE (klass),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooEditWindowClass, new_doc),
NULL, NULL,
_moo_marshal_VOID__OBJECT,
G_TYPE_NONE, 1,
MOO_TYPE_EDIT);
signals[CLOSE_DOC] =
g_signal_new ("close-doc",
G_OBJECT_CLASS_TYPE (klass),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooEditWindowClass, close_doc),
NULL, NULL,
_moo_marshal_VOID__OBJECT,
G_TYPE_NONE, 1,
MOO_TYPE_EDIT);
signals[CLOSE_DOC_AFTER] =
g_signal_new ("close-doc-after",
G_OBJECT_CLASS_TYPE (klass),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (MooEditWindowClass, close_doc_after),
NULL, NULL,
_moo_marshal_VOID__VOID,
G_TYPE_NONE, 0);
INSTALL_PROP (PROP_CAN_RELOAD, "can-reload");
INSTALL_PROP (PROP_HAS_OPEN_DOCUMENT, "has-open-document");
INSTALL_PROP (PROP_CAN_UNDO, "can-undo");
INSTALL_PROP (PROP_CAN_REDO, "can-redo");
INSTALL_PROP (PROP_HAS_SELECTION, "has-selection");
INSTALL_PROP (PROP_HAS_COMMENTS, "has-comments");
INSTALL_PROP (PROP_HAS_TEXT, "has-text");
INSTALL_PROP (PROP_HAS_JOBS_RUNNING, "has-jobs-running");
INSTALL_PROP (PROP_HAS_STOP_CLIENTS, "has-stop-clients");
moo_window_class_set_id (window_class, "Editor", "Editor");
moo_window_class_new_action (window_class, "NewDoc", NULL,
"display-name", GTK_STOCK_NEW,
"label", GTK_STOCK_NEW,
"tooltip", _("Create new document"),
"stock-id", GTK_STOCK_NEW,
"accel", "<ctrl>N",
"closure-callback", action_new_doc,
NULL);
moo_window_class_new_action (window_class, "Open", NULL,
"display-name", GTK_STOCK_OPEN,
"label", _("_Open..."),
"tooltip", _("Open..."),
"stock-id", GTK_STOCK_OPEN,
"accel", "<ctrl>O",
"closure-callback", action_open,
NULL);
moo_window_class_new_action (window_class, "Reload", NULL,
"display-name", _("Reload"),
"label", _("_Reload"),
"tooltip", _("Reload document"),
"stock-id", GTK_STOCK_REFRESH,
"accel", "F5",
"closure-callback", action_reload,
"condition::sensitive", "can-reload",
NULL);
moo_window_class_new_action (window_class, "Save", NULL,
"display-name", GTK_STOCK_SAVE,
"label", GTK_STOCK_SAVE,
"tooltip", GTK_STOCK_SAVE,
"stock-id", GTK_STOCK_SAVE,
"accel", "<ctrl>S",
"closure-callback", action_save,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "SaveAs", NULL,
"display-name", GTK_STOCK_SAVE_AS,
"label", _("Save _As..."),
"tooltip", _("Save as..."),
"stock-id", GTK_STOCK_SAVE_AS,
"accel", "<ctrl><shift>S",
"closure-callback", action_save_as,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "Close", NULL,
"display-name", GTK_STOCK_CLOSE,
"label", GTK_STOCK_CLOSE,
"tooltip", _("Close document"),
"stock-id", GTK_STOCK_CLOSE,
"accel", "<ctrl>W",
"closure-callback", action_close_tab,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "CloseAll", NULL,
"display-name", _("Close All"),
"label", _("Close _All"),
"tooltip", _("Close all documents"),
"accel", "<shift><ctrl>W",
"closure-callback", action_close_all,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "Undo", NULL,
"display-name", GTK_STOCK_UNDO,
"label", GTK_STOCK_UNDO,
"tooltip", GTK_STOCK_UNDO,
"stock-id", GTK_STOCK_UNDO,
"accel", "<ctrl>Z",
"closure-signal", "undo",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "can-undo",
NULL);
moo_window_class_new_action (window_class, "Redo", NULL,
"display-name", GTK_STOCK_REDO,
"label", GTK_STOCK_REDO,
"tooltip", GTK_STOCK_REDO,
"stock-id", GTK_STOCK_REDO,
"accel", "<shift><ctrl>Z",
"closure-signal", "redo",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "can-redo",
NULL);
moo_window_class_new_action (window_class, "Cut", NULL,
"display-name", GTK_STOCK_CUT,
"label", GTK_STOCK_CUT,
"tooltip", GTK_STOCK_CUT,
"stock-id", GTK_STOCK_CUT,
"accel", "<ctrl>X",
"closure-signal", "cut-clipboard",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-selection",
NULL);
moo_window_class_new_action (window_class, "Copy", NULL,
"display-name", GTK_STOCK_COPY,
"label", GTK_STOCK_COPY,
"tooltip", GTK_STOCK_COPY,
"stock-id", GTK_STOCK_COPY,
"accel", "<ctrl>C",
"closure-signal", "copy-clipboard",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-selection",
NULL);
moo_window_class_new_action (window_class, "Paste", NULL,
"display-name", GTK_STOCK_PASTE,
"label", GTK_STOCK_PASTE,
"tooltip", GTK_STOCK_PASTE,
"stock-id", GTK_STOCK_PASTE,
"accel", "<ctrl>V",
"closure-signal", "paste-clipboard",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "Delete", NULL,
"display-name", GTK_STOCK_DELETE,
"label", GTK_STOCK_DELETE,
"tooltip", GTK_STOCK_DELETE,
"stock-id", GTK_STOCK_DELETE,
"closure-signal", "delete-selection",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-selection",
NULL);
moo_window_class_new_action (window_class, "SelectAll", NULL,
"display-name", GTK_STOCK_SELECT_ALL,
"label", GTK_STOCK_SELECT_ALL,
"tooltip", GTK_STOCK_SELECT_ALL,
"accel", "<ctrl>A",
"closure-callback", moo_text_view_select_all,
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-text",
NULL);
moo_window_class_new_action (window_class, "PreviousTab", NULL,
"display-name", _("Previous Tab"),
"label", _("_Previous Tab"),
"tooltip", _("Previous tab"),
"stock-id", GTK_STOCK_GO_BACK,
"accel", "<alt>Left",
"closure-callback", action_previous_tab,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "NextTab", NULL,
"display-name", _("Next Tab"),
"label", _("_Next Tab"),
"tooltip", _("Next tab"),
"stock-id", GTK_STOCK_GO_FORWARD,
"accel", "<alt>Right",
"closure-callback", action_next_tab,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "Find", NULL,
"display-name", GTK_STOCK_FIND,
"label", GTK_STOCK_FIND,
"tooltip", GTK_STOCK_FIND,
"stock-id", GTK_STOCK_FIND,
"accel", "<ctrl>F",
"closure-signal", "find-interactive",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "FindNext", NULL,
"display-name", _("Find Next"),
"label", _("Find _Next"),
"tooltip", _("Find next"),
"stock-id", GTK_STOCK_GO_FORWARD,
"accel", "F3",
"closure-signal", "find-next-interactive",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "FindPrevious", NULL,
"display-name", _("Find Previous"),
"label", _("Find _Previous"),
"tooltip", _("Find previous"),
"stock-id", GTK_STOCK_GO_BACK,
"accel", "<shift>F3",
"closure-signal", "find-prev-interactive",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "Replace", NULL,
"display-name", GTK_STOCK_FIND_AND_REPLACE,
"label", GTK_STOCK_FIND_AND_REPLACE,
"tooltip", GTK_STOCK_FIND_AND_REPLACE,
"stock-id", GTK_STOCK_FIND_AND_REPLACE,
"accel", "<ctrl>R",
"closure-signal", "replace-interactive",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "FindCurrent", NULL,
"display-name", _("Find Current Word"),
"label", _("Find Current _Word"),
"stock-id", GTK_STOCK_FIND,
"accel", "F4",
"closure-callback", action_find_now_f,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "FindCurrentBack", NULL,
"display-name", _("Find Current Word Backwards"),
"label", _("Find Current Word _Backwards"),
"stock-id", GTK_STOCK_FIND,
"accel", "<shift>F4",
"closure-callback", action_find_now_b,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "GoToLine", NULL,
"display-name", _("Go to Line"),
"label", _("_Go to Line..."),
"tooltip", _("Go to line..."),
"accel", "<ctrl>G",
"closure-signal", "goto-line-interactive",
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "DocumentSubmenu", NULL,
"label", _("_Document"),
"no-accel", TRUE,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "WrapText", NULL,
"action-type::", MOO_TYPE_TOGGLE_ACTION,
"display-name", _("Toggle Text Wrapping"),
"label", _("_Wrap Text"),
"toggled-callback", wrap_text_toggled,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "LineNumbers", NULL,
"action-type::", MOO_TYPE_TOGGLE_ACTION,
"display-name", _("Toggle Line Numbers Display"),
"label", _("_Show Line Numbers"),
"toggled-callback", line_numbers_toggled,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "FocusDoc", NULL,
"display-name", _("Focus Document"),
"label", _("_Focus Document"),
"accel", "<alt>C",
"closure-callback", gtk_widget_grab_focus,
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, STOP_ACTION_ID, NULL,
"display-name", GTK_STOCK_STOP,
"label", GTK_STOCK_STOP,
"tooltip", GTK_STOCK_STOP,
"stock-id", GTK_STOCK_STOP,
"accel", "Escape",
"closure-callback", action_abort_jobs,
"condition::sensitive", "has-jobs-running",
"condition::visible", "has-stop-clients",
NULL);
for (i = 1; i < 10; ++i)
{
char *action_id = g_strdup_printf ("SwitchToTab%u", i);
char *accel = g_strdup_printf ("<Alt>%u", i);
_moo_window_class_new_action_callback (window_class, action_id, NULL,
G_CALLBACK (action_switch_to_tab),
_moo_marshal_VOID__UINT,
G_TYPE_NONE, 1,
G_TYPE_UINT, i - 1,
"accel", accel,
"connect-accel", TRUE,
"accel-editable", FALSE,
NULL);
g_free (accel);
g_free (action_id);
}
#ifdef ENABLE_BOOKMARKS
for (i = 1; i < 10; ++i)
{
char *action_id = g_strdup_printf (MOO_EDIT_GOTO_BOOKMARK_ACTION "%u", i);
moo_window_class_new_action_custom (window_class, action_id, NULL,
create_goto_bookmark_action,
GUINT_TO_POINTER (i),
NULL);
g_free (action_id);
}
moo_window_class_new_action_custom (window_class, "BookmarksMenu", NULL,
create_bookmarks_menu_action,
NULL, NULL);
moo_window_class_new_action (window_class, "ToggleBookmark", NULL,
"display-name", _("Toggle Bookmark"),
"label", _("Toggle Bookmark"),
"stock-id", MOO_STOCK_EDIT_BOOKMARK,
"accel", "<ctrl>B",
"closure-callback", action_toggle_bookmark,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "NextBookmark", NULL,
"display-name", _("Next Bookmark"),
"label", _("Next Bookmark"),
"accel", "<alt>Down",
"connect-accel", TRUE,
"closure-callback", action_next_bookmark,
"condition::visible", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "PreviousBookmark", NULL,
"display-name", _("Previous Bookmark"),
"label", _("Previous Bookmark"),
"accel", "<alt>Up",
"connect-accel", TRUE,
"closure-callback", action_prev_bookmark,
"condition::visible", "has-open-document",
NULL);
#endif
#if 0
moo_window_class_new_action (window_class, "NextPlaceholder", NULL,
"display-name", "Next Placeholder",
"label", "Next Placeholder",
"tooltip", "Go to next placeholder",
"stock-id", GTK_STOCK_GO_FORWARD,
"closure-callback", action_next_ph,
"condition::visible", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "PrevPlaceholder", NULL,
"display-name", "Previous Placeholder",
"label", "Previous Placeholder",
"tooltip", "Go to previous placeholder",
"stock-id", GTK_STOCK_GO_BACK,
"closure-callback", action_prev_ph,
"condition::visible", "has-open-document",
NULL);
#endif
moo_window_class_new_action (window_class, "Comment", NULL,
/* action */
"display-name", _("Comment"),
"label", _("Comment"),
"tooltip", _("Comment"),
"closure-callback", moo_edit_comment,
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-comments",
NULL);
moo_window_class_new_action (window_class, "Uncomment", NULL,
/* action */
"display-name", _("Uncomment"),
"label", _("Uncomment"),
"tooltip", _("Uncomment"),
"closure-callback", moo_edit_uncomment,
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-comments",
NULL);
moo_window_class_new_action (window_class, "Indent", NULL,
"display-name", GTK_STOCK_INDENT,
"label", GTK_STOCK_INDENT,
"tooltip", GTK_STOCK_INDENT,
"stock-id", GTK_STOCK_INDENT,
"closure-callback", moo_text_view_indent,
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "Unindent", NULL,
"display-name", GTK_STOCK_UNINDENT,
"label", GTK_STOCK_UNINDENT,
"tooltip", GTK_STOCK_UNINDENT,
"stock-id", GTK_STOCK_UNINDENT,
"closure-callback", moo_text_view_unindent,
"closure-proxy-func", moo_edit_window_get_active_doc,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "NoDocuments", NULL,
/* Insensitive menu item which appears in Window menu with no documents open */
"label", _("No Documents"),
"no-accel", TRUE,
"sensitive", FALSE,
"condition::visible", "!has-open-document",
NULL);
#ifdef ENABLE_PRINTING
moo_window_class_new_action (window_class, "PageSetup", NULL,
"display-name", _("Page Setup"),
"label", _("Page Setup..."),
"tooltip", _("Page Setup..."),
"accel", "<ctrl><shift>P",
"closure-callback", action_page_setup,
NULL);
moo_window_class_new_action (window_class, "PrintPreview", NULL,
"display-name", GTK_STOCK_PRINT_PREVIEW,
"label", GTK_STOCK_PRINT_PREVIEW,
"tooltip", GTK_STOCK_PRINT_PREVIEW,
"stock-id", GTK_STOCK_PRINT_PREVIEW,
"closure-callback", action_print_preview,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "Print", NULL,
"display-name", GTK_STOCK_PRINT,
"label", _("Print..."),
"tooltip", _("Print..."),
"accel", "<ctrl>P",
"stock-id", GTK_STOCK_PRINT,
"closure-callback", action_print,
"condition::sensitive", "has-open-document",
NULL);
moo_window_class_new_action (window_class, "PrintPdf", NULL,
"display-name", _("Export as PDF"),
"label", _("Export as PDF..."),
"tooltip", _("Export as PDF..."),
"stock-id", GTK_STOCK_PRINT,
"closure-callback", action_print_pdf,
"condition::sensitive", "has-open-document",
NULL);
#endif
moo_window_class_new_action_custom (window_class, LANG_ACTION_ID, NULL,
(MooWindowActionFunc) create_lang_action,
NULL, NULL);
}
static void
moo_edit_window_init (MooEditWindow *window)
{
window->priv = G_TYPE_INSTANCE_GET_PRIVATE (window, MOO_TYPE_EDIT_WINDOW, MooEditWindowPrivate);
window->priv->history = NULL;
window->priv->history_blocked = FALSE;
window->priv->panes = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
window->priv->panes_to_save = g_hash_table_new_full (g_str_hash, g_str_equal, g_free,
(GDestroyNotify) pane_params_free);
g_object_set (G_OBJECT (window),
"menubar-ui-name", "Editor/Menubar",
"toolbar-ui-name", "Editor/Toolbar",
NULL);
set_title_format_from_prefs (window);
windows = g_slist_prepend (windows, window);
}
MooEditor *
moo_edit_window_get_editor (MooEditWindow *window)
{
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), NULL);
return window->priv->editor;
}
static void
moo_edit_window_destroy (GtkObject *object)
{
MooEditWindow *window = MOO_EDIT_WINDOW (object);
if (window->priv->doc_list_merge_id)
{
moo_ui_xml_remove_ui (window->priv->xml,
window->priv->doc_list_merge_id);
window->priv->doc_list_merge_id = 0;
}
if (window->priv->doc_list_update_idle)
{
g_source_remove (window->priv->doc_list_update_idle);
window->priv->doc_list_update_idle = 0;
}
if (window->priv->statusbar_idle)
{
g_source_remove (window->priv->statusbar_idle);
window->priv->statusbar_idle = 0;
}
if (window->priv->xml)
{
g_object_unref (window->priv->xml);
window->priv->xml = NULL;
}
if (window->priv->stop_clients || window->priv->jobs)
{
GSList *list, *l;
moo_edit_window_abort_jobs (window);
g_slist_foreach (window->priv->jobs, (GFunc) g_free, NULL);
g_slist_free (window->priv->jobs);
window->priv->jobs = NULL;
list = g_slist_copy (window->priv->stop_clients);
for (l = list; l != NULL; l = l->next)
moo_edit_window_remove_stop_client (window, l->data);
g_assert (window->priv->stop_clients == NULL);
g_slist_free (list);
}
windows = g_slist_remove (windows, window);
GTK_OBJECT_CLASS(moo_edit_window_parent_class)->destroy (object);
}
static void
moo_edit_window_finalize (GObject *object)
{
MooEditWindow *window = MOO_EDIT_WINDOW (object);
/* XXX */
g_hash_table_destroy (window->priv->panes);
if (window->priv->panes_to_save)
g_hash_table_destroy (window->priv->panes_to_save);
if (window->priv->save_params_idle)
{
g_warning ("%s: oops", G_STRLOC);
g_source_remove (window->priv->save_params_idle);
}
g_free (window->priv->title_format);
g_free (window->priv->title_format_no_doc);
G_OBJECT_CLASS (moo_edit_window_parent_class)->finalize (object);
}
static void
moo_edit_window_set_property (GObject *object,
guint prop_id,
const GValue *value,
GParamSpec *pspec)
{
MooEditWindow *window = MOO_EDIT_WINDOW (object);
switch (prop_id)
{
case PROP_EDITOR:
window->priv->editor = g_value_get_object (value);
break;
case PROP_ACTIVE_DOC:
moo_edit_window_set_active_doc (window, g_value_get_object (value));
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
static void moo_edit_window_get_property(GObject *object,
guint prop_id,
GValue *value,
GParamSpec *pspec)
{
MooEditWindow *window = MOO_EDIT_WINDOW (object);
MooEdit *doc;
switch (prop_id)
{
case PROP_EDITOR:
g_value_set_object (value, window->priv->editor);
break;
case PROP_ACTIVE_DOC:
g_value_set_object (value, moo_edit_window_get_active_doc (window));
break;
case PROP_CAN_RELOAD:
doc = ACTIVE_DOC (window);
g_value_set_boolean (value, doc && !moo_edit_is_untitled (doc));
break;
case PROP_HAS_OPEN_DOCUMENT:
g_value_set_boolean (value, ACTIVE_DOC (window) != NULL);
break;
case PROP_CAN_UNDO:
doc = ACTIVE_DOC (window);
g_value_set_boolean (value, doc && moo_text_view_can_undo (MOO_TEXT_VIEW (doc)));
break;
case PROP_CAN_REDO:
doc = ACTIVE_DOC (window);
g_value_set_boolean (value, doc && moo_text_view_can_redo (MOO_TEXT_VIEW (doc)));
break;
case PROP_HAS_SELECTION:
doc = ACTIVE_DOC (window);
g_value_set_boolean (value, doc && moo_text_view_has_selection (MOO_TEXT_VIEW (doc)));
break;
case PROP_HAS_COMMENTS:
doc = ACTIVE_DOC (window);
g_value_set_boolean (value, doc && _moo_edit_has_comments (doc, NULL, NULL));
break;
case PROP_HAS_TEXT:
doc = ACTIVE_DOC (window);
g_value_set_boolean (value, doc && moo_text_view_has_text (MOO_TEXT_VIEW (doc)));
break;
case PROP_HAS_JOBS_RUNNING:
g_value_set_boolean (value, window->priv->jobs != NULL);
break;
case PROP_HAS_STOP_CLIENTS:
g_value_set_boolean (value, window->priv->stop_clients != NULL);
break;
default:
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
break;
}
}
/****************************************************************************/
/* Constructing
*/
static GObject *
moo_edit_window_constructor (GType type,
guint n_props,
GObjectConstructParam *props)
{
GtkWidget *notebook;
MooEditWindow *window;
GtkWindowGroup *group;
GObject *object =
G_OBJECT_CLASS(moo_edit_window_parent_class)->constructor (type, n_props, props);
window = MOO_EDIT_WINDOW (object);
g_return_val_if_fail (window->priv->editor != NULL, object);
group = gtk_window_group_new ();
gtk_window_group_add_window (group, GTK_WINDOW (window));
g_object_unref (group);
create_paned (window);
notebook = g_object_new (MOO_TYPE_NOTEBOOK,
"show-tabs", TRUE,
"enable-popup", TRUE,
"enable-reordering", TRUE,
NULL);
gtk_widget_show (notebook);
moo_big_paned_add_child (window->paned, notebook);
window->priv->notebook = MOO_NOTEBOOK (notebook);
setup_notebook (window);
create_statusbar (window);
g_signal_connect (window, "realize", G_CALLBACK (update_window_title), NULL);
g_signal_connect (window, "notify::ui-xml",
G_CALLBACK (moo_edit_window_update_doc_list), NULL);
edit_changed (window, NULL);
moo_edit_window_check_actions (window);
return object;
}
static char *
get_doc_status_string (MooEdit *doc)
{
MooEditStatus status;
gboolean modified;
const char *doc_title_format;
status = moo_edit_get_status (doc);
modified = (status & MOO_EDIT_MODIFIED) && !(status & MOO_EDIT_CLEAN);
if (!modified)
{
if (status & MOO_EDIT_NEW)
/* Translators: this goes into window title, %s stands for filename, e.g. "/foo/bar.txt [new file]" */
doc_title_format = _("%s [new file]");
else if (status & MOO_EDIT_MODIFIED_ON_DISK)
/* Translators: this goes into window title, %s stands for filename, e.g. "/foo/bar.txt [modified on disk]" */
doc_title_format = _("%s [modified on disk]");
else if (status & MOO_EDIT_DELETED)
/* Translators: this goes into window title, %s stands for filename, e.g. "/foo/bar.txt [deleted]" */
doc_title_format = _("%s [deleted]");
else
doc_title_format = "%s";
}
else
{
if (status & MOO_EDIT_NEW)
/* Translators: this goes into window title, %s stands for filename, e.g. "/foo/bar.txt [new file] [modified]" */
doc_title_format = _("%s [new file] [modified]");
else if (status & MOO_EDIT_MODIFIED_ON_DISK)
/* Translators: this goes into window title, %s stands for filename, e.g. "/foo/bar.txt [modified on disk] [modified]" */
doc_title_format = _("%s [modified on disk] [modified]");
else if (status & MOO_EDIT_DELETED)
/* Translators: this goes into window title, %s stands for filename, e.g. "/foo/bar.txt [deleted] [modified]" */
doc_title_format = _("%s [deleted] [modified]");
else
/* Translators: this goes into window title, %s stands for filename, e.g. "/foo/bar.txt [modified]" */
doc_title_format = _("%s [modified]");
}
return g_strdup_printf (doc_title_format, "");
}
static char *
parse_title_format (const char *format,
MooEditWindow *window,
MooEdit *doc)
{
GString *str;
str = g_string_new (NULL);
while (*format)
{
if (*format == '%')
{
format++;
if (!*format)
{
g_critical ("%s: trailing percent sign", G_STRFUNC);
break;
}
switch (*format)
{
case 'a':
g_string_append (str, moo_editor_get_app_name (window->priv->editor));
break;
case 'b':
if (!doc)
g_critical ("%s: %%b used without document", G_STRFUNC);
else
g_string_append (str, moo_edit_get_display_basename (doc));
break;
case 'f':
if (!doc)
g_critical ("%s: %%f used without document", G_STRFUNC);
else
g_string_append (str, moo_edit_get_display_name (doc));
break;
case 'u':
if (!doc)
g_critical ("%s: %%u used without document", G_STRFUNC);
else
{
char *tmp = moo_edit_get_uri (doc);
if (tmp)
g_string_append (str, tmp);
else
g_string_append (str, moo_edit_get_display_name (doc));
g_free (tmp);
}
break;
case 's':
if (!doc)
g_critical ("%s: %%s used without document", G_STRFUNC);
else
{
char *tmp = get_doc_status_string (doc);
if (tmp)
g_string_append (str, tmp);
g_free (tmp);
}
break;
case '%':
g_string_append_c (str, '%');
break;
default:
g_critical ("%s: unknown format '%%%c'", G_STRFUNC, *format);
break;
}
}
else
{
g_string_append_c (str, *format);
}
format++;
}
return g_string_free (str, FALSE);
}
static void
update_window_title (MooEditWindow *window)
{
MooEdit *doc;
char *title;
doc = ACTIVE_DOC (window);
if (doc)
title = parse_title_format (window->priv->title_format, window, doc);
else
title = parse_title_format (window->priv->title_format_no_doc, window, NULL);
gtk_window_set_title (GTK_WINDOW (window), title);
g_free (title);
}
static const char *
check_format (const char *format)
{
if (!format || !format[0])
return DEFAULT_TITLE_FORMAT;
if (!g_utf8_validate (format, -1, NULL))
{
g_critical ("%s: window title format is not valid UTF8", G_STRLOC);
return DEFAULT_TITLE_FORMAT;
}
return format;
}
static void
set_title_format (MooEditWindow *window,
const char *format,
const char *format_no_doc)
{
format = check_format (format);
format_no_doc = check_format (format_no_doc);
g_free (window->priv->title_format);
g_free (window->priv->title_format_no_doc);
window->priv->title_format = g_strdup (format);
window->priv->title_format_no_doc = g_strdup (format_no_doc);
if (GTK_WIDGET_REALIZED (window))
update_window_title (window);
}
static void
set_title_format_from_prefs (MooEditWindow *window)
{
const char *format, *format_no_doc;
format = moo_prefs_get_string (moo_edit_setting (MOO_EDIT_PREFS_TITLE_FORMAT));
format_no_doc = moo_prefs_get_string (moo_edit_setting (MOO_EDIT_PREFS_TITLE_FORMAT_NO_DOC));
set_title_format (window, format, format_no_doc);
}
void
_moo_edit_window_update_title (void)
{
GSList *l;
for (l = windows; l != NULL; l = l->next)
set_title_format_from_prefs (l->data);
}
MooEditWindow *
moo_edit_get_window (MooEdit *edit)
{
GtkWidget *toplevel;
g_return_val_if_fail (MOO_IS_EDIT (edit), NULL);
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (edit));
if (MOO_IS_EDIT_WINDOW (toplevel))
return MOO_EDIT_WINDOW (toplevel);
else
return NULL;
}
gboolean
moo_edit_window_close_all (MooEditWindow *window)
{
GSList *docs;
gboolean result;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), FALSE);
docs = moo_edit_window_list_docs (window);
result = moo_editor_close_docs (window->priv->editor, docs, TRUE);
g_slist_free (docs);
return result;
}
static gboolean
moo_edit_window_close (MooWindow *window)
{
MooEditWindow *edit_window = MOO_EDIT_WINDOW (window);
moo_editor_close_window (edit_window->priv->editor, edit_window, TRUE);
return TRUE;
}
/****************************************************************************/
/* Actions
*/
static void
action_new_doc (MooEditWindow *window)
{
moo_editor_new_doc (window->priv->editor, window);
}
static void
action_open (MooEditWindow *window)
{
moo_editor_open (window->priv->editor, window, GTK_WIDGET (window), NULL);
}
static void
action_reload (MooEditWindow *window)
{
MooEdit *edit = moo_edit_window_get_active_doc (window);
g_return_if_fail (edit != NULL);
_moo_editor_reload (window->priv->editor, edit, NULL);
}
static void
action_save (MooEditWindow *window)
{
MooEdit *edit = moo_edit_window_get_active_doc (window);
g_return_if_fail (edit != NULL);
_moo_editor_save (window->priv->editor, edit, NULL);
}
static void
action_save_as (MooEditWindow *window)
{
MooEdit *edit = moo_edit_window_get_active_doc (window);
g_return_if_fail (edit != NULL);
_moo_editor_save_as (window->priv->editor, edit, NULL, NULL, NULL);
}
static void
action_close_tab (MooEditWindow *window)
{
MooEdit *edit = moo_edit_window_get_active_doc (window);
g_return_if_fail (edit != NULL);
moo_editor_close_doc (window->priv->editor, edit, TRUE);
}
static void
action_close_all (MooEditWindow *window)
{
moo_edit_window_close_all (window);
}
static void
switch_to_tab (MooEditWindow *window,
int n)
{
MooEdit *doc;
if (n < 0)
n = moo_edit_window_num_docs (window) - 1;
if (n < 0 || n >= moo_edit_window_num_docs (window))
return;
moo_notebook_set_current_page (window->priv->notebook, n);
if ((doc = moo_edit_window_get_active_doc (window)))
gtk_widget_grab_focus (GTK_WIDGET (doc));
}
static void
action_previous_tab (MooEditWindow *window)
{
int n;
n = moo_notebook_get_current_page (window->priv->notebook);
if (n > 0)
switch_to_tab (window, n - 1);
else
switch_to_tab (window, -1);
}
static void
action_next_tab (MooEditWindow *window)
{
int n;
n = moo_notebook_get_current_page (window->priv->notebook);
if (n < moo_notebook_get_n_pages (window->priv->notebook) - 1)
switch_to_tab (window, n + 1);
else
switch_to_tab (window, 0);
}
static void
action_switch_to_tab (MooEditWindow *window,
guint n)
{
switch_to_tab (window, n);
}
static void
moo_edit_window_find_now (MooEditWindow *window,
gboolean forward)
{
MooEdit *doc;
doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
g_signal_emit_by_name (doc, "find-word-at-cursor", forward);
}
static void
action_find_now_f (MooEditWindow *window)
{
moo_edit_window_find_now (window, TRUE);
}
static void
action_find_now_b (MooEditWindow *window)
{
moo_edit_window_find_now (window, FALSE);
}
static void
action_abort_jobs (MooEditWindow *window)
{
moo_edit_window_abort_jobs (window);
}
#ifdef ENABLE_BOOKMARKS
static void
action_toggle_bookmark (MooEditWindow *window)
{
MooEdit *doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
moo_edit_toggle_bookmark (doc, moo_text_view_get_cursor_line (MOO_TEXT_VIEW (doc)));
}
static void
action_next_bookmark (MooEditWindow *window)
{
int cursor;
GSList *bookmarks;
MooEdit *doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
cursor = moo_text_view_get_cursor_line (MOO_TEXT_VIEW (doc));
bookmarks = moo_edit_get_bookmarks_in_range (doc, cursor + 1, -1);
if (bookmarks)
{
moo_edit_goto_bookmark (doc, bookmarks->data);
g_slist_free (bookmarks);
}
}
static void
action_prev_bookmark (MooEditWindow *window)
{
int cursor;
GSList *bookmarks = NULL;
MooEdit *doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
cursor = moo_text_view_get_cursor_line (MOO_TEXT_VIEW (doc));
if (cursor > 0)
bookmarks = moo_edit_get_bookmarks_in_range (doc, 0, cursor - 1);
if (bookmarks)
{
GSList *last = g_slist_last (bookmarks);
moo_edit_goto_bookmark (doc, last->data);
g_slist_free (bookmarks);
}
}
static void
goto_bookmark_activated (GtkAction *action,
gpointer data)
{
MooEdit *doc;
MooEditWindow *window;
MooEditBookmark *bk;
guint n = GPOINTER_TO_UINT (data);
window = _moo_action_get_window (action);
g_return_if_fail (window != NULL);
doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
if ((bk = moo_edit_get_bookmark (doc, n)))
moo_edit_goto_bookmark (doc, bk);
}
static GtkAction *
create_goto_bookmark_action (MooWindow *window,
gpointer data)
{
GtkAction *action;
guint n = GPOINTER_TO_UINT (data);
char *accel;
char *name;
name = g_strdup_printf (MOO_EDIT_GOTO_BOOKMARK_ACTION "%u", n);
accel = g_strdup_printf ("<ctrl>%u", n);
action = g_object_new (MOO_TYPE_ACTION, "name", name, "accel", accel,
"connect-accel", TRUE, "accel-editable", FALSE, NULL);
g_signal_connect (action, "activate", G_CALLBACK (goto_bookmark_activated), data);
moo_bind_bool_property (action, "sensitive", window, "has-open-document", FALSE);
g_free (name);
g_free (accel);
return action;
}
static void
bookmark_item_activated (GtkWidget *item)
{
moo_edit_goto_bookmark (g_object_get_data (G_OBJECT (item), "moo-edit"),
g_object_get_data (G_OBJECT (item), "moo-bookmark"));
}
static GtkWidget *
create_bookmark_item (MooEditWindow *window,
MooEdit *doc,
MooEditBookmark *bk)
{
char *label, *bk_text;
GtkWidget *item = NULL;
bk_text = _moo_edit_bookmark_get_text (bk);
label = g_strdup_printf ("%d - \"%s\"", 1 + moo_line_mark_get_line (MOO_LINE_MARK (bk)),
bk_text ? bk_text : "");
g_free (bk_text);
if (bk->no)
{
GtkAction *action;
char *action_name;
action_name = g_strdup_printf (MOO_EDIT_GOTO_BOOKMARK_ACTION "%u", bk->no);
action = moo_window_get_action (MOO_WINDOW (window), action_name);
if (action)
{
g_object_set (action, "label", label, "use-underline", FALSE, NULL);
item = gtk_action_create_menu_item (action);
}
else
{
g_critical ("%s: oops", G_STRLOC);
}
g_free (action_name);
}
if (!item)
{
item = gtk_menu_item_new_with_label (label);
g_signal_connect (item, "activate", G_CALLBACK (bookmark_item_activated), NULL);
}
g_object_set_data_full (G_OBJECT (item), "moo-bookmark", g_object_ref (bk), g_object_unref);
g_object_set_data_full (G_OBJECT (item), "moo-edit", g_object_ref (doc), g_object_unref);
g_free (label);
return item;
}
static void
populate_bookmarks (MooEditWindow *window,
GtkWidget *menu)
{
GtkAction *pn;
MooEdit *doc;
GtkWidget *item;
const GSList *bookmarks;
doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
bookmarks = moo_edit_list_bookmarks (doc);
pn = moo_window_get_action (MOO_WINDOW (window), "PreviousBookmark");
item = gtk_action_create_menu_item (pn);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_object_set (pn, "sensitive", bookmarks != NULL, NULL);
pn = moo_window_get_action (MOO_WINDOW (window), "NextBookmark");
item = gtk_action_create_menu_item (pn);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_object_set (pn, "sensitive", bookmarks != NULL, NULL);
if (bookmarks)
{
item = gtk_separator_menu_item_new ();
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
}
while (bookmarks)
{
item = create_bookmark_item (window, doc, bookmarks->data);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
bookmarks = bookmarks->next;
}
gtk_widget_show_all (menu);
}
static gboolean
erase_bookmarks_menu (GtkMenuItem *bk_item)
{
if (g_object_get_data (G_OBJECT (bk_item), "moo-bookmarks-populated"))
{
GtkWidget *menu = gtk_menu_item_get_submenu (bk_item);
if (menu)
gtk_container_foreach (GTK_CONTAINER (menu),
(GtkCallback) gtk_widget_destroy,
NULL);
}
g_object_set_data (G_OBJECT (bk_item),
"moo-bookmarks-populated",
GINT_TO_POINTER (FALSE));
g_object_unref (bk_item);
return FALSE;
}
static void
bookmarks_toplevel_hide (GtkWidget *toplevel,
GtkMenuItem *bk_item)
{
moo_idle_add ((GSourceFunc) erase_bookmarks_menu, g_object_ref (bk_item));
g_signal_handlers_disconnect_by_func (toplevel,
(gpointer) bookmarks_toplevel_hide,
bk_item);
}
static void
bookmarks_item_selected (GtkMenuItem *bk_item,
MooEditWindow *window)
{
if (!g_object_get_data (G_OBJECT (bk_item), "moo-bookmarks-populated"))
{
GtkWidget *menu;
menu = gtk_menu_item_get_submenu (bk_item);
g_return_if_fail (menu != NULL);
populate_bookmarks (window, menu);
g_object_set_data (G_OBJECT (bk_item), "moo-bookmarks-populated",
GINT_TO_POINTER (TRUE));
menu = gtk_widget_get_parent (GTK_WIDGET (bk_item));
if (GTK_IS_MENU (menu))
g_signal_connect (GTK_MENU (menu)->toplevel, "hide",
G_CALLBACK (bookmarks_toplevel_hide), bk_item);
else
g_critical ("%s: oops", G_STRLOC);
}
}
static void
connect_bookmarks_item (GtkAction *action,
GtkWidget *widget)
{
MooEditWindow *window;
GtkWidget *menu;
g_return_if_fail (GTK_IS_MENU_ITEM (widget));
window = _moo_action_get_window (action);
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
menu = gtk_menu_new ();
gtk_menu_item_set_submenu (GTK_MENU_ITEM (widget), menu);
g_signal_connect (widget, "select", G_CALLBACK (bookmarks_item_selected), window);
}
static GtkAction *
create_bookmarks_menu_action (MooWindow *window,
G_GNUC_UNUSED gpointer data)
{
GtkAction *action;
action = g_object_new (MOO_TYPE_ACTION,
"name", "BookmarksMenu",
"label", _("Bookmarks"),
"no-accel", TRUE,
NULL);
g_signal_connect (action, "connect-proxy", G_CALLBACK (connect_bookmarks_item), NULL);
moo_bind_bool_property (action, "sensitive", window, "has-open-document", FALSE);
return action;
}
#endif
#if 0
static void
action_next_ph (MooEditWindow *window)
{
MooEdit *doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
moo_text_view_next_placeholder (MOO_TEXT_VIEW (doc));
}
static void
action_prev_ph (MooEditWindow *window)
{
MooEdit *doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
moo_text_view_prev_placeholder (MOO_TEXT_VIEW (doc));
}
#endif
#ifdef ENABLE_PRINTING
static void
action_page_setup (MooEditWindow *window)
{
_moo_edit_page_setup (GTK_WIDGET (window));
}
static void
action_print (MooEditWindow *window)
{
gpointer doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
_moo_edit_print (doc, GTK_WIDGET (window));
}
static void
action_print_preview (MooEditWindow *window)
{
gpointer doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
_moo_edit_print_preview (doc, GTK_WIDGET (window));
}
static void
action_print_pdf (MooEditWindow *window)
{
char *start_name;
const char *doc_name, *dot;
const char *filename;
gpointer doc = moo_edit_window_get_active_doc (window);
doc_name = doc ? moo_edit_get_display_basename (doc) : "output";
dot = strrchr (doc_name, '.');
if (dot && dot != doc_name)
{
start_name = g_new (char, (dot - doc_name) + 5);
memcpy (start_name, doc_name, dot - doc_name);
memcpy (start_name + (dot - doc_name), ".pdf", 5);
}
else
{
start_name = g_strdup_printf ("%s.pdf", doc_name);
}
doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
filename = moo_file_dialogp (GTK_WIDGET (window),
MOO_FILE_DIALOG_SAVE,
start_name,
_("Export as PDF"),
moo_edit_setting (MOO_EDIT_PREFS_PDF_LAST_DIR),
NULL);
if (filename)
_moo_edit_export_pdf (doc, filename);
g_free (start_name);
}
#endif
static void
wrap_text_toggled (MooEditWindow *window,
gboolean active)
{
MooEdit *doc;
GtkWrapMode mode;
doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
g_object_get (doc, "wrap-mode", &mode, NULL);
if ((active && mode != GTK_WRAP_NONE) || (!active && mode == GTK_WRAP_NONE))
return;
if (!active)
{
mode = GTK_WRAP_NONE;
}
else
{
if (moo_prefs_get_bool (moo_edit_setting (MOO_EDIT_PREFS_WRAP_WORDS)))
mode = GTK_WRAP_WORD;
else
mode = GTK_WRAP_CHAR;
}
moo_edit_config_set (doc->config, MOO_EDIT_CONFIG_SOURCE_USER, "wrap-mode", mode, NULL);
}
static void
line_numbers_toggled (MooEditWindow *window,
gboolean active)
{
MooEdit *doc;
gboolean show;
doc = moo_edit_window_get_active_doc (window);
g_return_if_fail (doc != NULL);
g_object_get (doc, "show-line-numbers", &show, NULL);
if ((active && show) || (!active && !show))
return;
moo_edit_config_set (doc->config, MOO_EDIT_CONFIG_SOURCE_USER,
"show-line-numbers", active,
NULL);
}
/****************************************************************************/
/* Notebook popup menu
*/
static void
close_activated (GtkWidget *item,
MooEditWindow *window)
{
MooEdit *edit = g_object_get_data (G_OBJECT (item), "moo-edit");
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (MOO_IS_EDIT (edit));
moo_editor_close_doc (window->priv->editor, edit, TRUE);
}
static void
close_others_activated (GtkWidget *item,
MooEditWindow *window)
{
GSList *list;
MooEdit *edit = g_object_get_data (G_OBJECT (item), "moo-edit");
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (MOO_IS_EDIT (edit));
list = moo_edit_window_list_docs (window);
list = g_slist_remove (list, edit);
if (list)
moo_editor_close_docs (window->priv->editor, list, TRUE);
g_slist_free (list);
}
static void
detach_activated (GtkWidget *item,
MooEditWindow *window)
{
MooEdit *doc = g_object_get_data (G_OBJECT (item), "moo-edit");
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (MOO_IS_EDIT (doc));
_moo_editor_move_doc (window->priv->editor, doc, NULL, TRUE);
}
/****************************************************************************/
/* Documents
*/
static void
notebook_switch_page (G_GNUC_UNUSED MooNotebook *notebook,
guint page_num,
MooEditWindow *window)
{
edit_changed (window, get_nth_tab (window, page_num));
moo_edit_window_check_actions (window);
moo_edit_window_update_doc_list (window);
g_object_notify (G_OBJECT (window), "active-doc");
}
static gboolean
notebook_populate_popup (MooNotebook *notebook,
GtkWidget *child,
GtkMenu *menu,
MooEditWindow *window)
{
MooEdit *edit;
GtkWidget *item;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), TRUE);
g_return_val_if_fail (window->priv->notebook == notebook, TRUE);
g_return_val_if_fail (GTK_IS_SCROLLED_WINDOW (child), TRUE);
edit = MOO_EDIT (gtk_bin_get_child (GTK_BIN (child)));
g_return_val_if_fail (MOO_IS_EDIT (edit), TRUE);
item = gtk_menu_item_new_with_label ("Close");
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_object_set_data (G_OBJECT (item), "moo-edit", edit);
g_signal_connect (item, "activate",
G_CALLBACK (close_activated),
window);
if (moo_edit_window_num_docs (window) > 1)
{
item = gtk_menu_item_new_with_label ("Close All Others");
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_object_set_data (G_OBJECT (item), "moo-edit", edit);
g_signal_connect (item, "activate",
G_CALLBACK (close_others_activated),
window);
}
if (moo_edit_window_num_docs (window) > 1)
{
gtk_menu_shell_append (GTK_MENU_SHELL (menu),
g_object_new (GTK_TYPE_SEPARATOR_MENU_ITEM,
"visible", TRUE, NULL));
item = gtk_menu_item_new_with_label ("Detach");
gtk_widget_show (item);
gtk_menu_shell_append (GTK_MENU_SHELL (menu), item);
g_object_set_data (G_OBJECT (item), "moo-edit", edit);
g_signal_connect (item, "activate",
G_CALLBACK (detach_activated),
window);
}
return FALSE;
}
static gboolean
notebook_button_press (MooNotebook *notebook,
GdkEventButton *event,
MooEditWindow *window)
{
int n;
if (event->button != 2 || event->type != GDK_BUTTON_PRESS)
return FALSE;
n = moo_notebook_get_event_tab (notebook, (GdkEvent*) event);
if (n < 0)
return FALSE;
moo_editor_close_doc (window->priv->editor,
get_nth_tab (window, n),
TRUE);
return TRUE;
}
static void
set_use_tabs (MooEditWindow *window)
{
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_object_set (window->priv->notebook, "show-tabs",
moo_prefs_get_bool (moo_edit_setting (MOO_EDIT_PREFS_USE_TABS)), NULL);
}
void
_moo_edit_window_set_use_tabs (void)
{
GSList *l;
for (l = windows; l != NULL; l = l->next)
set_use_tabs (l->data);
}
static void
setup_notebook (MooEditWindow *window)
{
GtkWidget *button, *icon, *frame;
set_use_tabs (window);
g_signal_connect_after (window->priv->notebook, "moo-switch-page",
G_CALLBACK (notebook_switch_page), window);
g_signal_connect (window->priv->notebook, "populate-popup",
G_CALLBACK (notebook_populate_popup), window);
g_signal_connect (window->priv->notebook, "button-press-event",
G_CALLBACK (notebook_button_press), window);
frame = gtk_aspect_frame_new (NULL, 0.5, 0.5, 1.0, FALSE);
gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_NONE);
button = gtk_button_new ();
gtk_button_set_focus_on_click (GTK_BUTTON (button), FALSE);
g_signal_connect_swapped (button, "clicked",
G_CALLBACK (action_close_tab), window);
moo_bind_bool_property (button, "sensitive", window, "has-open-document", FALSE);
icon = gtk_image_new_from_stock (MOO_STOCK_CLOSE, MOO_ICON_SIZE_REAL_SMALL);
gtk_container_add (GTK_CONTAINER (button), icon);
gtk_container_add (GTK_CONTAINER (frame), button);
gtk_widget_show_all (frame);
moo_notebook_set_action_widget (window->priv->notebook, frame, TRUE);
gtk_drag_dest_set (GTK_WIDGET (window->priv->notebook), 0,
dest_targets, G_N_ELEMENTS (dest_targets),
GDK_ACTION_COPY | GDK_ACTION_MOVE);
gtk_drag_dest_add_text_targets (GTK_WIDGET (window->priv->notebook));
g_signal_connect (window->priv->notebook, "drag-motion",
G_CALLBACK (notebook_drag_motion), window);
g_signal_connect (window->priv->notebook, "drag-drop",
G_CALLBACK (notebook_drag_drop), window);
g_signal_connect (window->priv->notebook, "drag-data-received",
G_CALLBACK (notebook_drag_data_recv), window);
}
static void
edit_changed (MooEditWindow *window,
MooEdit *doc)
{
if (doc == ACTIVE_DOC (window))
{
g_object_freeze_notify (G_OBJECT (window));
g_object_notify (G_OBJECT (window), "can-reload");
g_object_notify (G_OBJECT (window), "has-open-document");
g_object_notify (G_OBJECT (window), "can-undo");
g_object_notify (G_OBJECT (window), "can-redo");
g_object_notify (G_OBJECT (window), "has-selection");
g_object_notify (G_OBJECT (window), "has-comments");
g_object_notify (G_OBJECT (window), "has-text");
g_object_thaw_notify (G_OBJECT (window));
update_window_title (window);
update_statusbar (window);
update_lang_menu (window);
update_doc_view_actions (window);
}
if (doc)
update_tab_label (window, doc);
}
static void
edit_overwrite_changed (MooEditWindow *window,
G_GNUC_UNUSED GParamSpec *pspec,
MooEdit *doc)
{
if (doc == ACTIVE_DOC (window))
update_statusbar (window);
}
static void
edit_wrap_mode_changed (MooEditWindow *window,
G_GNUC_UNUSED GParamSpec *pspec,
MooEdit *doc)
{
gpointer action;
GtkWrapMode mode;
if (doc != ACTIVE_DOC (window))
return;
action = moo_window_get_action (MOO_WINDOW (window), "WrapText");
g_return_if_fail (action != NULL);
g_object_get (doc, "wrap-mode", &mode, NULL);
gtk_toggle_action_set_active (action, mode != GTK_WRAP_NONE);
}
static void
edit_show_line_numbers_changed (MooEditWindow *window,
G_GNUC_UNUSED GParamSpec *pspec,
MooEdit *doc)
{
gpointer action;
gboolean show;
if (doc != ACTIVE_DOC (window))
return;
action = moo_window_get_action (MOO_WINDOW (window), "LineNumbers");
g_return_if_fail (action != NULL);
g_object_get (doc, "show-line-numbers", &show, NULL);
gtk_toggle_action_set_active (action, show);
}
static void
update_doc_view_actions (MooEditWindow *window)
{
MooEdit *doc;
doc = moo_edit_window_get_active_doc (window);
if (!doc)
return;
edit_wrap_mode_changed (window, NULL, doc);
edit_show_line_numbers_changed (window, NULL, doc);
}
static void
edit_filename_changed (MooEditWindow *window,
G_GNUC_UNUSED const char *filename,
MooEdit *doc)
{
edit_changed (window, doc);
moo_edit_window_update_doc_list (window);
}
static void
proxy_boolean_property (MooEditWindow *window,
GParamSpec *prop,
MooEdit *doc)
{
if (doc == ACTIVE_DOC (window))
g_object_notify (G_OBJECT (window), prop->name);
}
MooEdit *
moo_edit_window_get_active_doc (MooEditWindow *window)
{
GtkWidget *swin;
int page;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), NULL);
if (!window->priv->notebook)
return NULL;
page = moo_notebook_get_current_page (window->priv->notebook);
if (page < 0)
return NULL;
swin = moo_notebook_get_nth_page (window->priv->notebook, page);
return MOO_EDIT (gtk_bin_get_child (GTK_BIN (swin)));
}
void
moo_edit_window_set_active_doc (MooEditWindow *window,
MooEdit *edit)
{
GtkWidget *swin;
int page;
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (MOO_IS_EDIT (edit));
swin = GTK_WIDGET(edit)->parent;
page = moo_notebook_page_num (window->priv->notebook, swin);
g_return_if_fail (page >= 0);
moo_notebook_set_current_page (window->priv->notebook, page);
}
GSList *
moo_edit_window_list_docs (MooEditWindow *window)
{
GSList *list = NULL;
int num, i;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), NULL);
num = moo_notebook_get_n_pages (window->priv->notebook);
for (i = 0; i < num; i++)
list = g_slist_prepend (list, get_nth_tab (window, i));
return g_slist_reverse (list);
}
int
moo_edit_window_num_docs (MooEditWindow *window)
{
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), 0);
if (!window->priv->notebook)
return 0;
else
return moo_notebook_get_n_pages (window->priv->notebook);
}
MooEdit *
moo_edit_window_get_nth_doc (MooEditWindow *window,
guint n)
{
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), NULL);
if (!window->priv->notebook || n >= (guint) moo_notebook_get_n_pages (window->priv->notebook))
return NULL;
return get_nth_tab (window, n);
}
static MooEdit *
get_nth_tab (MooEditWindow *window,
guint n)
{
GtkWidget *swin;
swin = moo_notebook_get_nth_page (window->priv->notebook, n);
if (swin)
return MOO_EDIT (gtk_bin_get_child (GTK_BIN (swin)));
else
return NULL;
}
static int
get_page_num (MooEditWindow *window,
MooEdit *doc)
{
GtkWidget *swin;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), -1);
g_return_val_if_fail (MOO_IS_EDIT (doc), -1);
swin = GTK_WIDGET(doc)->parent;
return moo_notebook_page_num (window->priv->notebook, swin);
}
int
_moo_edit_window_get_doc_no (MooEditWindow *window,
MooEdit *doc)
{
return get_page_num (window, doc);
}
void
_moo_edit_window_insert_doc (MooEditWindow *window,
MooEdit *edit,
int position)
{
GtkWidget *label;
GtkWidget *scrolledwindow;
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (MOO_IS_EDIT (edit));
label = create_tab_label (window, edit);
gtk_widget_show (label);
scrolledwindow = gtk_scrolled_window_new (NULL, NULL);
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolledwindow),
GTK_POLICY_AUTOMATIC,
GTK_POLICY_AUTOMATIC);
gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scrolledwindow),
GTK_SHADOW_ETCHED_IN);
gtk_container_add (GTK_CONTAINER (scrolledwindow), GTK_WIDGET (edit));
gtk_widget_show_all (scrolledwindow);
moo_notebook_insert_page (window->priv->notebook, scrolledwindow, label, position);
g_signal_connect_swapped (edit, "doc_status_changed",
G_CALLBACK (edit_changed), window);
g_signal_connect_swapped (edit, "notify::overwrite",
G_CALLBACK (edit_overwrite_changed), window);
g_signal_connect_swapped (edit, "notify::wrap-mode",
G_CALLBACK (edit_wrap_mode_changed), window);
g_signal_connect_swapped (edit, "notify::show-line-numbers",
G_CALLBACK (edit_show_line_numbers_changed), window);
g_signal_connect_swapped (edit, "filename_changed",
G_CALLBACK (edit_filename_changed), window);
g_signal_connect_swapped (edit, "notify::can-undo",
G_CALLBACK (proxy_boolean_property), window);
g_signal_connect_swapped (edit, "notify::can-redo",
G_CALLBACK (proxy_boolean_property), window);
g_signal_connect_swapped (edit, "notify::has-selection",
G_CALLBACK (proxy_boolean_property), window);
g_signal_connect_swapped (edit, "notify::has-comments",
G_CALLBACK (proxy_boolean_property), window);
g_signal_connect_swapped (edit, "notify::has-text",
G_CALLBACK (proxy_boolean_property), window);
g_signal_connect_swapped (edit, "config-notify::lang",
G_CALLBACK (edit_lang_changed), window);
g_signal_connect_swapped (edit, "cursor-moved",
G_CALLBACK (edit_cursor_moved), window);
moo_edit_window_update_doc_list (window);
g_signal_emit (window, signals[NEW_DOC], 0, edit);
_moo_doc_attach_plugins (window, edit);
moo_edit_window_set_active_doc (window, edit);
edit_changed (window, edit);
gtk_widget_grab_focus (GTK_WIDGET (edit));
}
void
_moo_edit_window_remove_doc (MooEditWindow *window,
MooEdit *doc,
gboolean destroy)
{
int page;
GtkAction *action;
MooEdit *new_doc;
gboolean had_focus;
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (MOO_IS_EDIT (doc));
page = get_page_num (window, doc);
g_return_if_fail (page >= 0);
had_focus = GTK_WIDGET_HAS_FOCUS (doc);
g_signal_emit (window, signals[CLOSE_DOC], 0, doc);
g_signal_handlers_disconnect_by_func (doc, (gpointer) edit_changed, window);
g_signal_handlers_disconnect_by_func (doc, (gpointer) edit_filename_changed, window);
g_signal_handlers_disconnect_by_func (doc, (gpointer) proxy_boolean_property, window);
g_signal_handlers_disconnect_by_func (doc, (gpointer) edit_cursor_moved, window);
g_signal_handlers_disconnect_by_func (doc, (gpointer) edit_lang_changed, window);
g_signal_handlers_disconnect_by_func (doc, (gpointer) edit_overwrite_changed, window);
g_signal_handlers_disconnect_by_func (doc, (gpointer) edit_wrap_mode_changed, window);
g_signal_handlers_disconnect_by_func (doc, (gpointer) edit_show_line_numbers_changed, window);
_moo_doc_detach_plugins (window, doc);
action = g_object_get_data (G_OBJECT (doc), "moo-doc-list-action");
if (action)
{
moo_action_collection_remove_action (moo_window_get_actions (MOO_WINDOW (window)), action);
g_object_set_data (G_OBJECT (doc), "moo-doc-list-action", NULL);
}
window->priv->history = g_list_remove (window->priv->history, doc);
window->priv->history_blocked = TRUE;
moo_edit_window_update_doc_list (window);
/* removing scrolled window from the notebook will destroy the scrolled window,
* and that in turn will destroy the doc if it's not removed before */
if (!destroy)
gtk_container_remove (GTK_CONTAINER (GTK_WIDGET(doc)->parent), GTK_WIDGET (doc));
moo_notebook_remove_page (window->priv->notebook, page);
window->priv->history_blocked = FALSE;
if (window->priv->history)
moo_edit_window_set_active_doc (window, window->priv->history->data);
edit_changed (window, NULL);
g_signal_emit (window, signals[CLOSE_DOC_AFTER], 0);
g_object_notify (G_OBJECT (window), "active-doc");
new_doc = ACTIVE_DOC (window);
if (!new_doc)
moo_edit_window_check_actions (window);
else if (had_focus)
gtk_widget_grab_focus (GTK_WIDGET (new_doc));
}
typedef struct {
int x;
int y;
gboolean drag_started;
} DragInfo;
static gboolean tab_icon_button_press (GtkWidget *evbox,
GdkEventButton *event,
MooEditWindow *window);
static gboolean tab_icon_button_release (GtkWidget *evbox,
GdkEventButton *event,
MooEditWindow *window);
static gboolean tab_icon_motion_notify (GtkWidget *evbox,
GdkEventMotion *event,
MooEditWindow *window);
static void tab_icon_drag_begin (GtkWidget *evbox,
GdkDragContext *context,
MooEditWindow *window);
static void tab_icon_drag_data_delete (GtkWidget *evbox,
GdkDragContext *context,
MooEditWindow *window);
static void tab_icon_drag_data_get (GtkWidget *evbox,
GdkDragContext *context,
GtkSelectionData *data,
guint info,
guint time,
MooEditWindow *window);
static void tab_icon_drag_end (GtkWidget *evbox,
GdkDragContext *context,
MooEditWindow *window);
static gboolean
tab_icon_button_release (GtkWidget *evbox,
G_GNUC_UNUSED GdkEventButton *event,
MooEditWindow *window)
{
g_object_set_data (G_OBJECT (evbox), "moo-drag-info", NULL);
g_signal_handlers_disconnect_by_func (evbox, (gpointer) tab_icon_button_release, window);
g_signal_handlers_disconnect_by_func (evbox, (gpointer) tab_icon_motion_notify, window);
return FALSE;
}
static void
tab_icon_start_drag (GtkWidget *evbox,
GdkEvent *event,
MooEditWindow *window)
{
GtkTargetList *targets;
MooEdit *edit;
edit = g_object_get_data (G_OBJECT (evbox), "moo-edit");
g_return_if_fail (MOO_IS_EDIT (edit));
g_signal_connect (evbox, "drag-begin", G_CALLBACK (tab_icon_drag_begin), window);
g_signal_connect (evbox, "drag-data-delete", G_CALLBACK (tab_icon_drag_data_delete), window);
g_signal_connect (evbox, "drag-data-get", G_CALLBACK (tab_icon_drag_data_get), window);
g_signal_connect (evbox, "drag-end", G_CALLBACK (tab_icon_drag_end), window);
targets = gtk_target_list_new (NULL, 0);
gtk_target_list_add (targets,
gdk_atom_intern ("text/uri-list", FALSE),
0, TARGET_URI_LIST);
gtk_target_list_add (targets,
gdk_atom_intern ("MOO_EDIT_TAB", FALSE),
GTK_TARGET_SAME_APP,
TARGET_MOO_EDIT_TAB);
gtk_drag_begin (evbox, targets,
GDK_ACTION_COPY | GDK_ACTION_MOVE | GDK_ACTION_LINK,
1, (GdkEvent*) event);
gtk_target_list_unref (targets);
}
static gboolean
tab_icon_motion_notify (GtkWidget *evbox,
GdkEventMotion *event,
MooEditWindow *window)
{
DragInfo *info;
info = g_object_get_data (G_OBJECT (evbox), "moo-drag-info");
g_return_val_if_fail (info != NULL, FALSE);
if (info->drag_started)
return TRUE;
if (gtk_drag_check_threshold (evbox, info->x, info->y, event->x, event->y))
{
info->drag_started = TRUE;
tab_icon_start_drag (evbox, (GdkEvent*) event, window);
}
return TRUE;
}
static gboolean
tab_icon_button_press (GtkWidget *evbox,
GdkEventButton *event,
MooEditWindow *window)
{
DragInfo *info;
if (event->window != evbox->window || event->button != 1 || event->type != GDK_BUTTON_PRESS)
return FALSE;
info = g_new0 (DragInfo, 1);
info->x = event->x;
info->y = event->y;
g_object_set_data_full (G_OBJECT (evbox), "moo-drag-info", info, g_free);
g_signal_connect (evbox, "motion-notify-event", G_CALLBACK (tab_icon_motion_notify), window);
g_signal_connect (evbox, "button-release-event", G_CALLBACK (tab_icon_button_release), window);
return FALSE;
}
static void
tab_icon_drag_begin (GtkWidget *evbox,
GdkDragContext *context,
G_GNUC_UNUSED MooEditWindow *window)
{
GdkPixbuf *pixbuf;
GtkImage *icon;
icon = g_object_get_data (G_OBJECT (evbox), "moo-edit-icon");
pixbuf = gtk_image_get_pixbuf (icon);
gtk_drag_set_icon_pixbuf (context, pixbuf, 0, 0);
}
static void
tab_icon_drag_data_delete (G_GNUC_UNUSED GtkWidget *evbox,
G_GNUC_UNUSED GdkDragContext *context,
G_GNUC_UNUSED MooEditWindow *window)
{
g_print ("delete!\n");
}
static void
tab_icon_drag_data_get (GtkWidget *evbox,
G_GNUC_UNUSED GdkDragContext *context,
GtkSelectionData *data,
guint info,
G_GNUC_UNUSED guint time,
G_GNUC_UNUSED MooEditWindow *window)
{
MooEdit *edit = g_object_get_data (G_OBJECT (evbox), "moo-edit");
g_return_if_fail (MOO_IS_EDIT (edit));
if (info == TARGET_MOO_EDIT_TAB)
{
_moo_selection_data_set_pointer (data,
gdk_atom_intern ("MOO_EDIT_TAB", FALSE),
edit);
}
else if (info == TARGET_URI_LIST)
{
char *uris[] = {NULL, NULL};
uris[0] = moo_edit_get_uri (edit);
gtk_selection_data_set_uris (data, uris);
g_free (uris[0]);
}
else
{
g_print ("drag-data-get WTF?\n");
gtk_selection_data_set_text (data, "", -1);
}
}
static void
tab_icon_drag_end (GtkWidget *evbox,
G_GNUC_UNUSED GdkDragContext *context,
MooEditWindow *window)
{
g_object_set_data (G_OBJECT (evbox), "moo-drag-info", NULL);
g_signal_handlers_disconnect_by_func (evbox, (gpointer) tab_icon_drag_begin, window);
g_signal_handlers_disconnect_by_func (evbox, (gpointer) tab_icon_drag_data_delete, window);
g_signal_handlers_disconnect_by_func (evbox, (gpointer) tab_icon_drag_data_get, window);
g_signal_handlers_disconnect_by_func (evbox, (gpointer) tab_icon_drag_end, window);
}
static void
update_evbox_shape (GtkWidget *image,
GtkWidget *evbox)
{
GtkMisc *misc;
GdkPixbuf *pixbuf;
GdkBitmap *mask;
int width, height;
int x, y;
g_return_if_fail (GTK_IS_EVENT_BOX (evbox));
g_return_if_fail (GTK_IS_IMAGE (image));
if (!GTK_WIDGET_REALIZED (image) || !(pixbuf = gtk_image_get_pixbuf (GTK_IMAGE (image))))
return;
width = gdk_pixbuf_get_width (pixbuf);
height = gdk_pixbuf_get_height (pixbuf);
g_return_if_fail (width < 2000 && height < 2000);
gdk_pixbuf_render_pixmap_and_mask (pixbuf, NULL, &mask, 1);
g_return_if_fail (mask != NULL);
misc = GTK_MISC (image);
x = floor (image->allocation.x + misc->xpad
+ ((image->allocation.width - image->requisition.width) * misc->xalign));
y = floor (image->allocation.y + misc->ypad
+ ((image->allocation.height - image->requisition.height) * misc->yalign));
gtk_widget_shape_combine_mask (evbox, NULL, 0, 0);
gtk_widget_shape_combine_mask (evbox, mask, x, y);
g_object_unref (mask);
}
static void
icon_size_allocate (GtkWidget *image,
GtkAllocation *allocation,
GtkWidget *evbox)
{
GtkAllocation *old_allocation;
old_allocation = g_object_get_data (G_OBJECT (image), "moo-icon-allocation");
if (!old_allocation ||
old_allocation->x != allocation->x ||
old_allocation->y != allocation->y ||
old_allocation->width != allocation->width ||
old_allocation->height != allocation->height)
{
GtkAllocation *copy = g_memdup (allocation, sizeof *allocation);
g_object_set_data_full (G_OBJECT (image), "moo-icon-allocation", copy, g_free);
update_evbox_shape (image, evbox);
}
}
static GtkWidget *
create_tab_label (MooEditWindow *window,
MooEdit *edit)
{
GtkWidget *hbox, *icon, *label, *evbox;
GtkSizeGroup *group;
group = gtk_size_group_new (GTK_SIZE_GROUP_VERTICAL);
hbox = gtk_hbox_new (FALSE, 3);
gtk_widget_show (hbox);
evbox = gtk_event_box_new ();
gtk_box_pack_start (GTK_BOX (hbox), evbox, FALSE, FALSE, 0);
icon = gtk_image_new ();
gtk_container_add (GTK_CONTAINER (evbox), icon);
gtk_widget_show_all (evbox);
g_signal_connect (icon, "realize", G_CALLBACK (update_evbox_shape), evbox);
g_signal_connect (icon, "size-allocate", G_CALLBACK (icon_size_allocate), evbox);
label = gtk_label_new (moo_edit_get_display_basename (edit));
gtk_label_set_single_line_mode (GTK_LABEL (label), TRUE);
gtk_widget_show (label);
gtk_box_pack_start (GTK_BOX (hbox), label, TRUE, TRUE, 0);
gtk_size_group_add_widget (group, evbox);
gtk_size_group_add_widget (group, label);
g_object_set_data (G_OBJECT (hbox), "moo-edit-icon", icon);
g_object_set_data (G_OBJECT (hbox), "moo-edit-icon-evbox", evbox);
g_object_set_data (G_OBJECT (hbox), "moo-edit-label", label);
g_object_set_data (G_OBJECT (evbox), "moo-edit-icon", icon);
g_object_set_data (G_OBJECT (evbox), "moo-edit", edit);
g_object_set_data (G_OBJECT (icon), "moo-edit", edit);
g_signal_connect (evbox, "button-press-event",
G_CALLBACK (tab_icon_button_press),
window);
g_object_unref (group);
return hbox;
}
static void
set_tab_icon (GtkWidget *image,
GtkWidget *evbox,
GdkPixbuf *pixbuf)
{
GdkPixbuf *old_pixbuf;
/* file icons are cached, so it's likely the same pixbuf
* object as before (and it happens every time you switch tabs) */
old_pixbuf = gtk_image_get_pixbuf (GTK_IMAGE (image));
if (old_pixbuf != pixbuf)
{
gtk_image_set_from_pixbuf (GTK_IMAGE (image), pixbuf);
if (GTK_WIDGET_REALIZED (evbox))
update_evbox_shape (image, evbox);
}
}
static void
update_tab_label (MooEditWindow *window,
MooEdit *doc)
{
GtkWidget *hbox, *icon, *label, *evbox;
MooEditStatus status;
char *label_text;
gboolean modified, deleted;
GdkPixbuf *pixbuf;
int page;
page = get_page_num (window, doc);
g_return_if_fail (page >= 0);
hbox = moo_notebook_get_tab_label (window->priv->notebook,
GTK_WIDGET(doc)->parent);
g_return_if_fail (GTK_IS_WIDGET (hbox));
icon = g_object_get_data (G_OBJECT (hbox), "moo-edit-icon");
label = g_object_get_data (G_OBJECT (hbox), "moo-edit-label");
evbox = g_object_get_data (G_OBJECT (hbox), "moo-edit-icon-evbox");
g_return_if_fail (GTK_IS_WIDGET (icon) && GTK_IS_WIDGET (label));
g_return_if_fail (GTK_IS_WIDGET (evbox));
status = moo_edit_get_status (doc);
deleted = status & (MOO_EDIT_DELETED | MOO_EDIT_MODIFIED_ON_DISK);
modified = (status & MOO_EDIT_MODIFIED) && !(status & MOO_EDIT_CLEAN);
label_text = g_strdup_printf ("%s%s%s",
deleted ? "!" : "",
modified ? "*" : "",
moo_edit_get_display_basename (doc));
gtk_label_set_text (GTK_LABEL (label), label_text);
pixbuf = _moo_edit_get_icon (doc, icon, GTK_ICON_SIZE_MENU);
set_tab_icon (icon, evbox, pixbuf);
g_free (label_text);
}
/****************************************************************************/
/* Panes
*/
static const char *PANE_POSITIONS[4] = {
"left",
"right",
"top",
"bottom"
};
static void
create_paned (MooEditWindow *window)
{
MooBigPaned *paned;
guint i;
paned = g_object_new (MOO_TYPE_BIG_PANED,
"handle-cursor-type", GDK_FLEUR,
"enable-detaching", TRUE,
NULL);
gtk_widget_show (GTK_WIDGET (paned));
gtk_box_pack_start (GTK_BOX (MOO_WINDOW(window)->vbox),
GTK_WIDGET (paned), TRUE, TRUE, 0);
window->paned = paned;
for (i = 0; i < 4; ++i)
{
int size;
char *key = g_strdup_printf (MOO_EDIT_PREFS_PREFIX "/window/panes/%s",
PANE_POSITIONS[i]);
moo_prefs_create_key (key, MOO_PREFS_STATE, G_TYPE_INT, -1);
size = moo_prefs_get_int (key);
if (size > 0)
moo_paned_set_pane_size (MOO_PANED (paned->paned[i]), size);
g_free (key);
}
g_signal_connect_swapped (paned, "set-pane-size",
G_CALLBACK (pane_size_changed),
window);
}
gboolean
moo_edit_window_add_pane (MooEditWindow *window,
const char *user_id,
GtkWidget *widget,
MooPaneLabel *label,
MooPanePosition position)
{
MooPane *pane;
PaneParams *params;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), FALSE);
g_return_val_if_fail (user_id != NULL, FALSE);
g_return_val_if_fail (GTK_IS_WIDGET (widget), FALSE);
g_return_val_if_fail (label != NULL, FALSE);
g_return_val_if_fail (moo_edit_window_get_pane (window, user_id) == NULL, FALSE);
MOO_OBJECT_REF_SINK (widget);
params = load_pane_params (user_id);
position = (params && params->position >= 0) ?
(MooPanePosition) params->position : position;
pane = moo_big_paned_insert_pane (window->paned, widget, label, position, -1);
if (pane != NULL)
{
moo_pane_set_removable (pane, FALSE);
g_hash_table_insert (window->priv->panes,
g_strdup (user_id), widget);
g_object_set_data_full (G_OBJECT (widget), "moo-edit-window-pane-user-id",
g_strdup (user_id), g_free);
if (params)
moo_pane_set_params (pane, params->params);
g_signal_connect (pane, "notify::params", G_CALLBACK (pane_params_changed), window);
}
pane_params_free (params);
g_object_unref (widget);
return pane != NULL;
}
gboolean
moo_edit_window_remove_pane (MooEditWindow *window,
const char *user_id)
{
MooPane *pane;
GtkWidget *widget;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), FALSE);
g_return_val_if_fail (user_id != NULL, FALSE);
widget = g_hash_table_lookup (window->priv->panes, user_id);
if (!widget)
return FALSE;
pane = moo_big_paned_find_pane (window->paned, widget, NULL);
g_signal_handlers_disconnect_by_func (pane, (gpointer) pane_params_changed, window);
g_hash_table_remove (window->priv->panes, user_id);
moo_big_paned_remove_pane (window->paned, widget);
return TRUE;
}
GtkWidget*
moo_edit_window_get_pane (MooEditWindow *window,
const char *user_id)
{
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), NULL);
g_return_val_if_fail (user_id != NULL, NULL);
return g_hash_table_lookup (window->priv->panes, user_id);
}
#define PANE_PREFS_ROOT MOO_EDIT_PREFS_PREFIX "/panes"
#define ELEMENT_PANE "pane"
#define PROP_PANE_ID "id"
#define PROP_PANE_POSITION "position"
#define PROP_PANE_WINDOW_X "x"
#define PROP_PANE_WINDOW_Y "y"
#define PROP_PANE_WINDOW_WIDTH "width"
#define PROP_PANE_WINDOW_HEIGHT "height"
#define PROP_PANE_WINDOW_DETACHED "detached"
#define PROP_PANE_WINDOW_MAXIMIZED "maximized"
#define PROP_PANE_WINDOW_KEEP_ON_TOP "keep-on-top"
static MooMarkupNode*
get_pane_element (const char *pane_id,
gboolean create)
{
MooMarkupDoc *xml;
MooMarkupNode *root, *node;
xml = moo_prefs_get_markup (MOO_PREFS_STATE);
root = moo_markup_get_element (MOO_MARKUP_NODE (xml), PANE_PREFS_ROOT);
if (!root)
{
if (create)
{
node = moo_markup_create_element (MOO_MARKUP_NODE (xml),
PANE_PREFS_ROOT "/" ELEMENT_PANE);
moo_markup_set_prop (node, PROP_PANE_ID, pane_id);
return node;
}
else
{
return NULL;
}
}
for (node = root->children; node != NULL; node = node->next)
{
const char *id;
if (!MOO_MARKUP_IS_ELEMENT (node))
continue;
if (strcmp (node->name, ELEMENT_PANE))
{
g_warning ("%s: invalid element '%s'", G_STRLOC, node->name);
continue;
}
id = moo_markup_get_prop (node, PROP_PANE_ID);
if (!id || !id[0])
{
g_warning ("%s: id missing", G_STRLOC);
continue;
}
if (strcmp (id, pane_id))
continue;
return node;
}
if (create)
{
node = moo_markup_create_element (MOO_MARKUP_NODE (xml),
PANE_PREFS_ROOT "/" ELEMENT_PANE);
moo_markup_set_prop (node, PROP_PANE_ID, pane_id);
return node;
}
return NULL;
}
static PaneParams *
load_pane_params (const char *pane_id)
{
MooMarkupNode *node;
PaneParams *params;
GdkRectangle rect;
node = get_pane_element (pane_id, FALSE);
if (!node)
return NULL;
rect.x = moo_markup_get_int_prop (node, PROP_PANE_WINDOW_X, 0);
rect.y = moo_markup_get_int_prop (node, PROP_PANE_WINDOW_Y, 0);
rect.width = moo_markup_get_int_prop (node, PROP_PANE_WINDOW_WIDTH, 0);
rect.height = moo_markup_get_int_prop (node, PROP_PANE_WINDOW_HEIGHT, 0);
params = pane_params_new ();
params->params = moo_pane_params_new (&rect, moo_markup_get_bool_prop (node, PROP_PANE_WINDOW_DETACHED, FALSE),
moo_markup_get_bool_prop (node, PROP_PANE_WINDOW_MAXIMIZED, FALSE),
moo_markup_get_bool_prop (node, PROP_PANE_WINDOW_KEEP_ON_TOP, FALSE));
params->position = moo_markup_get_int_prop (node, PROP_PANE_POSITION, -1);
return params;
}
static void
set_if_not_zero (MooMarkupNode *node,
const char *attr,
int val)
{
if (val)
moo_markup_set_int_prop (node, attr, val);
else
moo_markup_set_prop (node, attr, NULL);
}
static void
set_if_not_false (MooMarkupNode *node,
const char *attr,
gboolean val)
{
if (val)
moo_markup_set_bool_prop (node, attr, val);
else
moo_markup_set_prop (node, attr, NULL);
}
static gboolean
save_pane_params (const char *pane_id,
PaneParams *params)
{
MooMarkupNode *node;
if (!params)
{
node = get_pane_element (pane_id, FALSE);
if (node)
moo_markup_delete_node (node);
return TRUE;
}
node = get_pane_element (pane_id, TRUE);
g_return_val_if_fail (node != NULL, TRUE);
set_if_not_zero (node, PROP_PANE_WINDOW_X, params->params->window_position.x);
set_if_not_zero (node, PROP_PANE_WINDOW_Y, params->params->window_position.y);
set_if_not_zero (node, PROP_PANE_WINDOW_WIDTH, params->params->window_position.width);
set_if_not_zero (node, PROP_PANE_WINDOW_HEIGHT, params->params->window_position.height);
set_if_not_false (node, PROP_PANE_WINDOW_DETACHED, params->params->detached);
set_if_not_false (node, PROP_PANE_WINDOW_MAXIMIZED, params->params->maximized);
set_if_not_false (node, PROP_PANE_WINDOW_KEEP_ON_TOP, params->params->keep_on_top);
if (params->position >= 0)
moo_markup_set_prop (node, PROP_PANE_POSITION,
_moo_convert_int_to_string (params->position));
else
moo_markup_set_prop (node, PROP_PANE_POSITION, NULL);
return TRUE;
}
static gboolean
do_save_panes (MooEditWindow *window)
{
g_hash_table_foreach_remove (window->priv->panes_to_save,
(GHRFunc) save_pane_params, NULL);
window->priv->save_params_idle = 0;
return FALSE;
}
static PaneParams *
pane_params_new (void)
{
PaneParams *params = g_new0 (PaneParams, 1);
params->position = -1;
return params;
}
static void
pane_params_free (PaneParams *params)
{
if (params)
{
moo_pane_params_free (params->params);
g_free (params);
}
}
static void
pane_params_changed (MooPane *pane,
G_GNUC_UNUSED GParamSpec *pspec,
MooEditWindow *window)
{
GtkWidget *widget;
const char *id;
PaneParams *params;
widget = moo_pane_get_child (pane);
g_return_if_fail (widget != NULL);
id = g_object_get_data (G_OBJECT (widget), "moo-edit-window-pane-user-id");
if (id)
{
params = pane_params_new ();
params->params = moo_pane_get_params (pane);
params->position = _moo_paned_get_position (_moo_pane_get_parent (pane));
g_hash_table_insert (window->priv->panes_to_save,
g_strdup (id), params);
if (!window->priv->save_params_idle)
window->priv->save_params_idle =
moo_idle_add ((GSourceFunc) do_save_panes, window);
}
}
static void
pane_size_changed (MooEditWindow *window,
MooPanePosition pos)
{
GtkWidget *paned = window->paned->paned[pos];
char *key;
g_return_if_fail (MOO_IS_PANED (paned));
g_return_if_fail (pos < 4);
key = g_strdup_printf (MOO_EDIT_PREFS_PREFIX "/window/panes/%s",
PANE_POSITIONS[pos]);
moo_prefs_set_int (key, moo_paned_get_pane_size (MOO_PANED (paned)));
g_free (key);
}
/****************************************************************************/
/* Statusbar
*/
static void
clear_statusbar (MooEditWindow *window)
{
gtk_statusbar_pop (window->priv->statusbar, 0);
}
static void
set_statusbar_numbers (MooEditWindow *window,
int line,
int column,
int chars)
{
char *text, *text2;
if (line > 0 && column > 0)
/* Label in the statusbar - line and column numbers */
text = g_strdup_printf (_("Line: %d Col: %d"), line, column);
else
/* Disabled label in the statusbar when no document is open */
text = g_strdup (_("Line: Col: "));
if (chars >= 0)
/* Label in the statusbar - number of characters in the document */
text2 = g_strdup_printf (_("Chars: %d"), chars);
else
/* Disabled label in the statusbar when no document is open */
text2 = g_strdup (_("Chars: "));
gtk_label_set_text (window->priv->cursor_label, text);
gtk_label_set_text (window->priv->chars_label, text2);
clear_statusbar (window);
g_free (text2);
g_free (text);
}
static void
do_update_statusbar (MooEditWindow *window)
{
MooEdit *edit;
int line, column, chars;
GtkTextIter iter;
GtkTextBuffer *buffer;
gboolean ovr;
edit = ACTIVE_DOC (window);
if (!edit)
{
gtk_widget_set_sensitive (window->priv->info, FALSE);
set_statusbar_numbers (window, 0, 0, -1);
return;
}
gtk_widget_set_sensitive (window->priv->info, TRUE);
buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (edit));
gtk_text_buffer_get_iter_at_mark (buffer, &iter,
gtk_text_buffer_get_insert (buffer));
line = gtk_text_iter_get_line (&iter) + 1;
column = moo_text_iter_get_visual_line_offset (&iter, 8) + 1;
chars = gtk_text_buffer_get_char_count (buffer);
set_statusbar_numbers (window, line, column, chars);
ovr = gtk_text_view_get_overwrite (GTK_TEXT_VIEW (edit));
/* Label in the editor window statusbar - Overwrite or Insert mode */
gtk_label_set_text (window->priv->insert_label, ovr ? _("OVR") : _("INS"));
}
static gboolean
update_statusbar_idle (MooEditWindow *window)
{
window->priv->statusbar_idle = 0;
do_update_statusbar (window);
return FALSE;
}
static void
update_statusbar (MooEditWindow *window)
{
if (!window->priv->statusbar_idle)
window->priv->statusbar_idle =
g_idle_add_full (G_PRIORITY_HIGH,
(GSourceFunc) update_statusbar_idle,
window, NULL);
}
static void
edit_cursor_moved (MooEditWindow *window,
G_GNUC_UNUSED GtkTextIter *iter,
MooEdit *edit)
{
if (edit == ACTIVE_DOC (window))
update_statusbar (window);
}
static void
create_statusbar (MooEditWindow *window)
{
MooGladeXML *xml;
GtkWidget *hbox;
xml = moo_glade_xml_new_from_buf (statusbar_glade_xml, -1,
"hbox", GETTEXT_PACKAGE, NULL);
hbox = moo_glade_xml_get_widget (xml, "hbox");
g_return_if_fail (hbox != NULL);
gtk_box_pack_start (GTK_BOX (MOO_WINDOW (window)->vbox),
hbox, FALSE, FALSE, 0);
window->priv->statusbar = moo_glade_xml_get_widget (xml, "statusbar");
window->priv->cursor_label = moo_glade_xml_get_widget (xml, "cursor");
window->priv->chars_label = moo_glade_xml_get_widget (xml, "chars");
window->priv->insert_label = moo_glade_xml_get_widget (xml, "insert");
window->priv->info = moo_glade_xml_get_widget (xml, "info");
g_object_unref (xml);
}
void
moo_edit_window_message (MooEditWindow *window,
const char *message)
{
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
gtk_statusbar_pop (window->priv->statusbar, 0);
if (message)
gtk_statusbar_push (window->priv->statusbar, 0, message);
}
/*****************************************************************************/
/* Language menu
*/
static int
cmp_langs (MooLang *lang1,
MooLang *lang2)
{
return g_utf8_collate (_moo_lang_display_name (lang1),
_moo_lang_display_name (lang2));
}
static void
lang_item_activated (MooEditWindow *window,
const char *lang_name)
{
MooEdit *doc = ACTIVE_DOC (window);
const char *old_val;
gboolean do_set = FALSE;
g_return_if_fail (doc != NULL);
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
old_val = moo_edit_config_get_string (doc->config, "lang");
if (old_val)
do_set = !lang_name || strcmp (old_val, lang_name);
else
do_set = !!lang_name;
if (do_set)
moo_edit_config_set (doc->config, MOO_EDIT_CONFIG_SOURCE_USER,
"lang", lang_name, NULL);
}
static GtkAction*
create_lang_action (MooEditWindow *window)
{
GtkAction *action;
MooMenuMgr *menu_mgr;
MooLangMgr *lang_mgr;
GSList *langs, *sections, *l;
lang_mgr = moo_editor_get_lang_mgr (window->priv->editor);
/* TODO display names, etc. */
sections = moo_lang_mgr_get_sections (lang_mgr);
sections = g_slist_sort (sections, (GCompareFunc) strcmp);
langs = moo_lang_mgr_get_available_langs (lang_mgr);
langs = g_slist_sort (langs, (GCompareFunc) cmp_langs);
action = moo_menu_action_new (LANG_ACTION_ID, _("_Language"));
menu_mgr = moo_menu_action_get_mgr (MOO_MENU_ACTION (action));
moo_menu_mgr_append (menu_mgr, NULL,
MOO_LANG_NONE, "None", NULL,
MOO_MENU_ITEM_RADIO, NULL, NULL);
for (l = sections; l != NULL; l = l->next)
moo_menu_mgr_append (menu_mgr, NULL,
l->data, l->data, NULL,
0, NULL, NULL);
for (l = langs; l != NULL; l = l->next)
{
MooLang *lang = l->data;
if (!_moo_lang_get_hidden (lang))
moo_menu_mgr_append (menu_mgr, _moo_lang_get_section (lang),
_moo_lang_id (lang),
_moo_lang_display_name (lang),
NULL, MOO_MENU_ITEM_RADIO,
g_strdup (_moo_lang_id (lang)),
g_free);
}
g_signal_connect_swapped (menu_mgr, "radio-set-active",
G_CALLBACK (lang_item_activated), window);
g_slist_foreach (langs, (GFunc) g_object_unref, NULL);
g_slist_free (langs);
g_slist_foreach (sections, (GFunc) g_free, NULL);
g_slist_free (sections);
moo_bind_bool_property (action, "sensitive", window, "has-open-document", FALSE);
return action;
}
static void
update_lang_menu (MooEditWindow *window)
{
MooEdit *doc;
GtkAction *action;
MooLang *lang;
doc = ACTIVE_DOC (window);
if (!doc)
return;
lang = moo_text_view_get_lang (MOO_TEXT_VIEW (doc));
action = moo_window_get_action (MOO_WINDOW (window), LANG_ACTION_ID);
g_return_if_fail (action != NULL);
moo_menu_mgr_set_active (moo_menu_action_get_mgr (MOO_MENU_ACTION (action)),
_moo_lang_id (lang), TRUE);
}
static void
edit_lang_changed (MooEditWindow *window,
G_GNUC_UNUSED guint var_id,
G_GNUC_UNUSED GParamSpec *pspec,
MooEdit *doc)
{
if (doc == ACTIVE_DOC (window))
{
update_lang_menu (window);
moo_edit_window_check_actions (window);
}
}
/*****************************************************************************/
/* Action properties checks
*/
static void
window_check_one_action (const char *action_id,
ActionCheck *set,
MooEditWindow *window,
MooEdit *doc)
{
MooActionCheckFunc func;
GtkAction *action;
gboolean visible = TRUE, sensitive = TRUE;
action = moo_window_get_action (MOO_WINDOW (window), action_id);
if (!action)
return;
if ((func = set->checks[MOO_ACTION_CHECK_ACTIVE].func))
{
if (!func (action, window, doc,
set->checks[MOO_ACTION_CHECK_ACTIVE].data))
{
visible = FALSE;
sensitive = FALSE;
}
}
if (visible && (func = set->checks[MOO_ACTION_CHECK_VISIBLE].func))
{
gpointer data = set->checks[MOO_ACTION_CHECK_VISIBLE].data;
visible = func (action, window, doc, data);
}
if (sensitive && (func = set->checks[MOO_ACTION_CHECK_SENSITIVE].func))
{
gpointer data = set->checks[MOO_ACTION_CHECK_SENSITIVE].data;
sensitive = func (action, window, doc, data);
}
g_object_set (action, "visible", visible, "sensitive", sensitive, NULL);
}
static void
check_action_hash_cb (const char *action_id,
ActionCheck *check,
gpointer user_data)
{
struct {
MooEdit *doc;
MooEditWindow *window;
} *data = user_data;
window_check_one_action (action_id, check, data->window, data->doc);
}
static void
moo_edit_window_check_actions (MooEditWindow *window)
{
struct {
MooEdit *doc;
MooEditWindow *window;
} data;
data.window = window;
data.doc = ACTIVE_DOC (window);
g_hash_table_foreach (action_checks,
(GHFunc) check_action_hash_cb,
&data);
}
void
moo_edit_window_set_action_check (const char *action_id,
MooActionCheckType type,
MooActionCheckFunc func,
gpointer data,
GDestroyNotify notify)
{
ActionCheck *check;
GSList *l;
g_return_if_fail (action_id != NULL);
g_return_if_fail (type < N_ACTION_CHECKS);
g_return_if_fail (func != NULL);
action_checks_init ();
check = g_hash_table_lookup (action_checks, action_id);
if (!check)
{
check = g_new0 (ActionCheck, 1);
g_hash_table_insert (action_checks, g_strdup (action_id), check);
}
if (check->checks[type].func)
{
check->checks[type].func = NULL;
if (check->checks[type].notify)
check->checks[type].notify (check->checks[type].data);
}
check->checks[type].func = func;
check->checks[type].data = data;
check->checks[type].notify = notify;
for (l = windows; l != NULL; l = l->next)
{
MooEditWindow *window = l->data;
MooEdit *doc = ACTIVE_DOC (window);
window_check_one_action (action_id, check, window, doc);
}
}
static void
moo_edit_window_remove_action_check (const char *action_id,
MooActionCheckType type)
{
ActionCheck *check;
gboolean remove = TRUE;
guint i;
g_return_if_fail (action_id != NULL);
g_return_if_fail (type <= N_ACTION_CHECKS);
if (!action_checks)
return;
check = g_hash_table_lookup (action_checks, action_id);
if (!check)
return;
if (type < N_ACTION_CHECKS)
{
for (i = 0; i < N_ACTION_CHECKS; ++i)
{
if (check->checks[i].func && i != type)
{
remove = FALSE;
break;
}
}
if (!remove && check->checks[type].func && check->checks[type].notify)
{
check->checks[type].func = NULL;
check->checks[type].notify (check->checks[type].data);
}
}
if (remove)
{
g_hash_table_remove (action_checks, action_id);
for (i = 0; i < N_ACTION_CHECKS; ++i)
if (check->checks[i].func && check->checks[i].notify)
check->checks[i].notify (check->checks[i].data);
g_free (check);
}
}
GSList *
_moo_edit_parse_langs (const char *string)
{
char **pieces, **p;
GSList *list = NULL;
if (!string)
return NULL;
pieces = g_strsplit_set (string, " \t\r\n;,", 0);
if (!pieces)
return NULL;
for (p = pieces; *p != NULL; ++p)
{
g_strstrip (*p);
if (**p)
list = g_slist_prepend (list, _moo_lang_id_from_name (*p));
}
g_strfreev (pieces);
return g_slist_reverse (list);
}
static gboolean
check_action_filter (G_GNUC_UNUSED GtkAction *action,
G_GNUC_UNUSED MooEditWindow *window,
MooEdit *doc,
gpointer filter)
{
gboolean value = FALSE;
if (doc)
value = _moo_edit_filter_match (filter, doc);
return value;
}
void
moo_edit_window_set_action_filter (const char *action_id,
MooActionCheckType type,
const char *filter_string)
{
MooEditFilter *filter = NULL;
g_return_if_fail (action_id != NULL);
g_return_if_fail (type < N_ACTION_CHECKS);
if (filter_string && filter_string[0])
filter = _moo_edit_filter_new (filter_string);
if (filter)
moo_edit_window_set_action_check (action_id, type,
check_action_filter,
filter,
(GDestroyNotify) _moo_edit_filter_free);
else
moo_edit_window_remove_action_check (action_id, type);
}
static void
action_checks_init (void)
{
if (!action_checks)
action_checks =
g_hash_table_new_full (g_str_hash, g_str_equal, g_free, NULL);
}
/*****************************************************************************/
/* Stop button
*/
typedef struct {
gpointer job;
MooAbortJobFunc abort;
} Job;
static void moo_edit_window_job_finished (MooEditWindow *window,
gpointer job);
static void moo_edit_window_job_started (MooEditWindow *window,
const char *name,
MooAbortJobFunc func,
gpointer job);
static void
client_died (MooEditWindow *window,
gpointer client)
{
window->priv->stop_clients = g_slist_remove (window->priv->stop_clients, client);
moo_edit_window_job_finished (window, client);
}
static void
abort_client_job (gpointer client)
{
gboolean ret;
g_signal_emit_by_name (client, "abort", &ret);
}
static void
client_job_started (gpointer client,
const char *job_name,
MooEditWindow *window)
{
moo_edit_window_job_started (window, job_name, abort_client_job, client);
}
static void
client_job_finished (gpointer client,
MooEditWindow *window)
{
moo_edit_window_job_finished (window, client);
}
void
moo_edit_window_add_stop_client (MooEditWindow *window,
gpointer client)
{
GType type, return_type;
guint signal_abort, signal_started, signal_finished;
GSignalQuery query;
gboolean had_clients;
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (G_IS_OBJECT (client));
g_return_if_fail (!g_slist_find (window->priv->stop_clients, client));
type = G_OBJECT_TYPE (client);
signal_abort = g_signal_lookup ("abort", type);
signal_started = g_signal_lookup ("job-started", type);
signal_finished = g_signal_lookup ("job-finished", type);
g_return_if_fail (signal_abort && signal_started && signal_finished);
#define REAL_TYPE(t__) ((t__) & ~(G_SIGNAL_TYPE_STATIC_SCOPE))
g_signal_query (signal_abort, &query);
return_type = REAL_TYPE (query.return_type);
g_return_if_fail (return_type == G_TYPE_NONE || return_type == G_TYPE_BOOLEAN);
g_return_if_fail (query.n_params == 0);
g_signal_query (signal_started, &query);
g_return_if_fail (REAL_TYPE (query.return_type) == G_TYPE_NONE);
g_return_if_fail (query.n_params == 1);
g_return_if_fail (REAL_TYPE (query.param_types[0]) == G_TYPE_STRING);
g_signal_query (signal_finished, &query);
g_return_if_fail (REAL_TYPE (query.return_type) == G_TYPE_NONE);
g_return_if_fail (query.n_params == 0);
#undef REAL_TYPE
had_clients = window->priv->stop_clients != NULL;
window->priv->stop_clients = g_slist_prepend (window->priv->stop_clients, client);
g_object_weak_ref (client, (GWeakNotify) client_died, window);
g_signal_connect (client, "job-started", G_CALLBACK (client_job_started), window);
g_signal_connect (client, "job-finished", G_CALLBACK (client_job_finished), window);
if (!had_clients)
g_object_notify (G_OBJECT (window), "has-stop-clients");
}
void
moo_edit_window_remove_stop_client (MooEditWindow *window,
gpointer client)
{
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (g_slist_find (window->priv->stop_clients, client));
window->priv->stop_clients = g_slist_remove (window->priv->stop_clients, client);
if (G_IS_OBJECT (client))
{
g_object_weak_unref (client, (GWeakNotify) client_died, window);
g_signal_handlers_disconnect_by_func (client, (gpointer)client_job_started, window);
g_signal_handlers_disconnect_by_func (client, (gpointer)client_job_finished, window);
}
if (!window->priv->stop_clients)
g_object_notify (G_OBJECT (window), "has-stop-clients");
}
void
moo_edit_window_abort_jobs (MooEditWindow *window)
{
GSList *l, *jobs;
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
jobs = g_slist_copy (window->priv->jobs);
for (l = jobs; l != NULL; l = l->next)
{
Job *j = l->data;
j->abort (j->job);
}
g_slist_free (jobs);
}
static void
moo_edit_window_job_started (MooEditWindow *window,
G_GNUC_UNUSED const char *name,
MooAbortJobFunc func,
gpointer job)
{
Job *j;
gboolean had_jobs;
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (func != NULL);
g_return_if_fail (job != NULL);
j = g_new0 (Job, 1);
j->abort = func;
j->job = job;
had_jobs = window->priv->jobs != NULL;
window->priv->jobs = g_slist_prepend (window->priv->jobs, j);
if (!had_jobs)
g_object_notify (G_OBJECT (window), "has-jobs-running");
}
static void
moo_edit_window_job_finished (MooEditWindow *window,
gpointer job)
{
GSList *l;
Job *j = NULL;
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
g_return_if_fail (job != NULL);
for (l = window->priv->jobs; l != NULL; l = l->next)
{
j = l->data;
if (j->job == job)
break;
else
j = NULL;
}
if (j)
{
window->priv->jobs = g_slist_remove (window->priv->jobs, j);
if (!window->priv->jobs)
g_object_notify (G_OBJECT (window), "has-jobs-running");
g_free (j);
}
}
GtkWidget *
moo_edit_window_get_output (MooEditWindow *window)
{
MooPaneLabel *label;
GtkWidget *cmd_view;
GtkWidget *scrolled_window;
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), NULL);
scrolled_window = moo_edit_window_get_pane (window, "moo-edit-window-output");
if (!scrolled_window)
{
scrolled_window = gtk_scrolled_window_new (NULL, NULL);
gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window),
GTK_POLICY_AUTOMATIC, GTK_POLICY_ALWAYS);
cmd_view = moo_cmd_view_new ();
moo_text_view_set_font_from_string (MOO_TEXT_VIEW (cmd_view), "Monospace");
gtk_container_add (GTK_CONTAINER (scrolled_window), cmd_view);
gtk_widget_show_all (scrolled_window);
g_object_set_data (G_OBJECT (scrolled_window), "moo-output", cmd_view);
label = moo_pane_label_new (MOO_STOCK_TERMINAL, NULL, "Output", "Output");
if (!moo_edit_window_add_pane (window, "moo-edit-window-output",
scrolled_window, label, MOO_PANE_POS_BOTTOM))
{
g_critical ("%s: oops", G_STRLOC);
moo_pane_label_free (label);
return NULL;
}
moo_edit_window_add_stop_client (window, cmd_view);
moo_pane_label_free (label);
return cmd_view;
}
return g_object_get_data (G_OBJECT (scrolled_window), "moo-output");
}
GtkWidget *
moo_edit_window_get_output_pane (MooEditWindow *window)
{
g_return_val_if_fail (MOO_IS_EDIT_WINDOW (window), NULL);
return moo_edit_window_get_pane (window, "moo-edit-window-output");
}
void
moo_edit_window_present_output (MooEditWindow *window)
{
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
moo_edit_window_get_output (window);
moo_big_paned_present_pane (window->paned,
moo_edit_window_get_output_pane (window));
}
/************************************************************************/
/* Doc list
*/
static void
doc_list_action_toggled (gpointer action,
MooEditWindow *window)
{
MooEdit *doc;
if (window->priv->doc_list_update_idle ||
!gtk_toggle_action_get_active (action))
return;
doc = g_object_get_data (action, "moo-edit");
g_return_if_fail (MOO_IS_EDIT (doc));
g_return_if_fail (MOO_IS_EDIT_WINDOW (window));
if (doc != ACTIVE_DOC (window))
moo_edit_window_set_active_doc (window, doc);
}
static int
compare_doc_list_actions (gpointer a1,
gpointer a2)
{
MooEdit *d1, *d2;
int result;
d1 = g_object_get_data (a1, "moo-edit");
d2 = g_object_get_data (a2, "moo-edit");
g_return_val_if_fail (d1 && d2, -1);
result = strcmp (moo_edit_get_display_basename (d1),
moo_edit_get_display_basename (d2));
return result;
}
static gboolean
do_update_doc_list (MooEditWindow *window)
{
MooUIXML *xml;
GSList *actions = NULL, *docs;
GSList *group = NULL;
MooUINode *ph;
gpointer active_doc;
active_doc = ACTIVE_DOC (window);
xml = moo_window_get_ui_xml (MOO_WINDOW (window));
g_return_val_if_fail (xml != NULL, FALSE);
if (xml != window->priv->xml)
{
if (window->priv->xml)
{
if (window->priv->doc_list_merge_id)
moo_ui_xml_remove_ui (window->priv->xml,
window->priv->doc_list_merge_id);
g_object_unref (window->priv->xml);
}
window->priv->xml = g_object_ref (xml);
}
else if (window->priv->doc_list_merge_id)
{
moo_ui_xml_remove_ui (xml, window->priv->doc_list_merge_id);
}
window->priv->doc_list_merge_id = 0;
ph = moo_ui_xml_find_placeholder (xml, "DocList");
g_return_val_if_fail (ph != NULL, FALSE);
docs = moo_edit_window_list_docs (window);
if (!docs)
goto out;
while (docs)
{
GtkRadioAction *action;
gpointer doc = docs->data;
action = g_object_get_data (doc, "moo-doc-list-action");
if (action)
{
g_object_set (action,
"label", moo_edit_get_display_basename (doc),
"tooltip", moo_edit_get_display_name (doc),
NULL);
}
else
{
GtkActionGroup *action_group;
char *name = g_strdup_printf ("MooEdit-%p", doc);
action = g_object_new (MOO_TYPE_RADIO_ACTION ,
"name", name,
"label", moo_edit_get_display_basename (doc),
"tooltip", moo_edit_get_display_name (doc),
"use-underline", FALSE,
NULL);
g_object_set_data_full (doc, "moo-doc-list-action", action, g_object_unref);
g_object_set_data (G_OBJECT (action), "moo-edit", doc);
_moo_action_set_no_accel (action, TRUE);
g_signal_connect (action, "toggled", G_CALLBACK (doc_list_action_toggled), window);
action_group = moo_action_collection_get_group (moo_window_get_actions (MOO_WINDOW (window)), NULL);
gtk_action_group_add_action (action_group, GTK_ACTION (action));
g_free (name);
}
gtk_radio_action_set_group (action, group);
group = gtk_radio_action_get_group (action);
actions = g_slist_prepend (actions, action);
docs = g_slist_delete_link (docs, docs);
}
window->priv->doc_list_merge_id = moo_ui_xml_new_merge_id (xml);
actions = g_slist_sort (actions, (GCompareFunc) compare_doc_list_actions);
while (actions)
{
gpointer action = actions->data;
gpointer doc = g_object_get_data (G_OBJECT (action), "moo-edit");
char *markup = g_markup_printf_escaped ("<item action=\"%s\"/>",
gtk_action_get_name (action));
moo_ui_xml_insert (xml, window->priv->doc_list_merge_id, ph, -1, markup);
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action),
doc == active_doc);
g_free (markup);
actions = g_slist_delete_link (actions, actions);
}
out:
window->priv->doc_list_update_idle = 0;
return FALSE;
}
static void
moo_edit_window_update_doc_list (MooEditWindow *window)
{
MooEdit *doc;
if (!window->priv->doc_list_update_idle)
window->priv->doc_list_update_idle =
_moo_idle_add_full (G_PRIORITY_HIGH,
(GSourceFunc) do_update_doc_list,
window, NULL);
if (!window->priv->history_blocked &&
(doc = moo_edit_window_get_active_doc (window)))
{
GList *link = g_list_find (window->priv->history, doc);
if (link && link != window->priv->history)
{
window->priv->history = g_list_delete_link (window->priv->history, link);
window->priv->history = g_list_prepend (window->priv->history, doc);
}
else if (!link)
{
window->priv->history = g_list_prepend (window->priv->history, doc);
}
}
}
/************************************************************************/
/* Drag into the window
*/
static gboolean
notebook_drag_motion (GtkWidget *widget,
GdkDragContext *context,
G_GNUC_UNUSED int x,
G_GNUC_UNUSED int y,
guint time,
G_GNUC_UNUSED MooEditWindow *window)
{
GdkAtom target;
target = gtk_drag_dest_find_target (widget, context, NULL);
if (target == GDK_NONE)
return FALSE;
if (target == moo_edit_tab_atom)
gtk_drag_get_data (widget, context, moo_edit_tab_atom, time);
else
gdk_drag_status (context, context->suggested_action, time);
return TRUE;
}
static gboolean
notebook_drag_drop (GtkWidget *widget,
GdkDragContext *context,
G_GNUC_UNUSED int x,
G_GNUC_UNUSED int y,
guint time,
G_GNUC_UNUSED MooEditWindow *window)
{
GdkAtom target;
target = gtk_drag_dest_find_target (widget, context, NULL);
if (target == GDK_NONE)
{
gtk_drag_finish (context, FALSE, FALSE, time);
}
else
{
g_object_set_data (G_OBJECT (widget), "moo-edit-window-drop",
GINT_TO_POINTER (TRUE));
gtk_drag_get_data (widget, context, target, time);
}
return TRUE;
}
static void
notebook_drag_data_recv (GtkWidget *widget,
GdkDragContext *context,
G_GNUC_UNUSED int x,
G_GNUC_UNUSED int y,
GtkSelectionData *data,
guint info,
guint time,
MooEditWindow *window)
{
gboolean finished = FALSE;
if (g_object_get_data (G_OBJECT (widget), "moo-edit-window-drop"))
{
g_object_set_data (G_OBJECT (widget), "moo-edit-window-drop", NULL);
if (data->target == moo_edit_tab_atom)
{
GtkWidget *toplevel;
MooEdit *doc = _moo_selection_data_get_pointer (data, moo_edit_tab_atom);
if (!doc)
goto out;
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (doc));
if (toplevel != GTK_WIDGET (window))
_moo_editor_move_doc (window->priv->editor, doc, window, TRUE);
goto out;
}
else if (data->target == text_uri_atom)
{
char **uris;
char **u;
/* XXX this is wrong but works. gtk_selection_data_get_uris()
* does not work on windows */
uris = g_uri_list_extract_uris ((char*) data->data);
if (!uris)
goto out;
for (u = uris; *u; ++u)
moo_editor_open_uri (window->priv->editor, window,
NULL, *u, NULL);
g_strfreev (uris);
gtk_drag_finish (context, TRUE, FALSE, time);
finished = TRUE;
}
else
{
MooEdit *doc;
GtkTextBuffer *buf;
char *text = (char *) gtk_selection_data_get_text (data);
if (!text)
goto out;
doc = moo_editor_new_doc (window->priv->editor, window);
if (!doc)
{
g_free (text);
goto out;
}
/* XXX */
buf = gtk_text_view_get_buffer (GTK_TEXT_VIEW (doc));
gtk_text_buffer_set_text (buf, text, -1);
g_free (text);
gtk_drag_finish (context, TRUE,
context->suggested_action == GDK_ACTION_MOVE,
time);
finished = TRUE;
}
}
else
{
if (info == TARGET_MOO_EDIT_TAB)
{
GtkWidget *toplevel;
MooEdit *doc = _moo_selection_data_get_pointer (data, moo_edit_tab_atom);
if (!doc)
{
g_critical ("%s: oops", G_STRLOC);
gdk_drag_status (context, 0, time);
return;
}
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (doc));
if (toplevel == GTK_WIDGET (window))
{
gdk_drag_status (context, 0, time);
return;
}
gdk_drag_status (context, GDK_ACTION_MOVE, time);
}
else
{
gdk_drag_status (context, 0, time);
}
return;
}
out:
if (!finished)
gtk_drag_finish (context, FALSE, FALSE, time);
}