4196 lines
134 KiB
C
4196 lines
134 KiB
C
#include <config.h>
|
|
#include <time.h>
|
|
#include <sys/types.h>
|
|
#include <string.h>
|
|
#include <glib/gi18n.h>
|
|
#include <gtk/gtk.h>
|
|
#include <libpeas/peas-extension-set.h>
|
|
|
|
#include "xed-ui.h"
|
|
#include "xed-window.h"
|
|
#include "xed-window-private.h"
|
|
#include "xed-app.h"
|
|
#include "xed-notebook.h"
|
|
#include "xed-statusbar.h"
|
|
#include "xed-searchbar.h"
|
|
#include "xed-view-frame.h"
|
|
#include "xed-utils.h"
|
|
#include "xed-commands.h"
|
|
#include "xed-debug.h"
|
|
#include "xed-document.h"
|
|
#include "xed-document-private.h"
|
|
#include "xed-panel.h"
|
|
#include "xed-documents-panel.h"
|
|
#include "xed-plugins-engine.h"
|
|
#include "xed-window-activatable.h"
|
|
#include "xed-enum-types.h"
|
|
#include "xed-dirs.h"
|
|
#include "xed-status-menu-button.h"
|
|
#include "xed-highlight-mode-selector.h"
|
|
#include "xed-settings.h"
|
|
|
|
#define LANGUAGE_NONE (const gchar *)"LangNone"
|
|
#define TAB_WIDTH_DATA "XedWindowTabWidthData"
|
|
#define LANGUAGE_DATA "XedWindowLanguageData"
|
|
|
|
#define XED_WINDOW_DEFAULT_WIDTH 650
|
|
#define XED_WINDOW_DEFAULT_HEIGHT 500
|
|
|
|
/* Signals */
|
|
enum
|
|
{
|
|
TAB_ADDED,
|
|
TAB_REMOVED,
|
|
TABS_REORDERED,
|
|
ACTIVE_TAB_CHANGED,
|
|
ACTIVE_TAB_STATE_CHANGED,
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL] = { 0 };
|
|
|
|
enum
|
|
{
|
|
PROP_0,
|
|
PROP_STATE
|
|
};
|
|
|
|
enum
|
|
{
|
|
TARGET_URI_LIST = 100
|
|
};
|
|
|
|
G_DEFINE_TYPE_WITH_PRIVATE (XedWindow, xed_window, GTK_TYPE_APPLICATION_WINDOW)
|
|
|
|
static void recent_manager_changed (GtkRecentManager *manager, XedWindow *window);
|
|
|
|
static void
|
|
xed_window_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
XedWindow *window = XED_WINDOW(object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_STATE:
|
|
g_value_set_enum (value, xed_window_get_state (window));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID(object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
save_panes_state (XedWindow *window)
|
|
{
|
|
gint panel_page;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
if (window->priv->side_panel_size > 0)
|
|
{
|
|
g_settings_set_int (window->priv->window_settings, XED_SETTINGS_SIDE_PANEL_SIZE, window->priv->side_panel_size);
|
|
}
|
|
|
|
panel_page = _xed_panel_get_active_item_id (XED_PANEL (window->priv->side_panel));
|
|
if (panel_page != 0)
|
|
{
|
|
g_settings_set_int (window->priv->window_settings, XED_SETTINGS_SIDE_PANEL_ACTIVE_PAGE, panel_page);
|
|
}
|
|
|
|
if (window->priv->bottom_panel_size > 0)
|
|
{
|
|
g_settings_set_int (window->priv->window_settings, XED_SETTINGS_BOTTOM_PANEL_SIZE, window->priv->bottom_panel_size);
|
|
}
|
|
|
|
panel_page = _xed_panel_get_active_item_id (XED_PANEL(window->priv->bottom_panel));
|
|
if (panel_page != 0)
|
|
{
|
|
g_settings_set_int (window->priv->window_settings, XED_SETTINGS_BOTTOM_PANEL_ACTIVE_PAGE, panel_page);
|
|
}
|
|
|
|
g_settings_apply (window->priv->window_settings);
|
|
}
|
|
|
|
static gint
|
|
on_key_pressed (GtkWidget *widget,
|
|
GdkEventKey *event,
|
|
XedWindow *window)
|
|
{
|
|
if (event->keyval == GDK_KEY_Escape)
|
|
{
|
|
XedTab *tab;
|
|
XedViewFrame *frame;
|
|
|
|
tab = xed_window_get_active_tab (window);
|
|
frame = XED_VIEW_FRAME (_xed_tab_get_view_frame (tab));
|
|
|
|
if (xed_view_frame_get_search_popup_visible (frame))
|
|
{
|
|
return GDK_EVENT_PROPAGATE;
|
|
}
|
|
else
|
|
{
|
|
xed_searchbar_hide (XED_SEARCHBAR (window->priv->searchbar));
|
|
return GDK_EVENT_STOP;
|
|
}
|
|
}
|
|
|
|
return GDK_EVENT_PROPAGATE;
|
|
}
|
|
|
|
static void
|
|
save_window_state (GtkWidget *widget)
|
|
{
|
|
XedWindow *window = XED_WINDOW (widget);
|
|
|
|
if ((window->priv->window_state &
|
|
(GDK_WINDOW_STATE_MAXIMIZED | GDK_WINDOW_STATE_FULLSCREEN)) == 0)
|
|
{
|
|
GtkAllocation allocation;
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
window->priv->width = allocation.width;
|
|
window->priv->height = allocation.height;
|
|
|
|
g_settings_set (window->priv->window_settings, XED_SETTINGS_WINDOW_SIZE,
|
|
"(ii)", window->priv->width, window->priv->height);
|
|
}
|
|
}
|
|
|
|
static void
|
|
xed_window_dispose (GObject *object)
|
|
{
|
|
XedWindow *window;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
window = XED_WINDOW (object);
|
|
|
|
/* Stop tracking removal of panes otherwise we always
|
|
* end up with thinking we had no pane active, since they
|
|
* should all be removed below */
|
|
if (window->priv->bottom_panel_item_removed_handler_id != 0)
|
|
{
|
|
g_signal_handler_disconnect (window->priv->bottom_panel, window->priv->bottom_panel_item_removed_handler_id);
|
|
window->priv->bottom_panel_item_removed_handler_id = 0;
|
|
}
|
|
|
|
/* First of all, force collection so that plugins
|
|
* really drop some of the references.
|
|
*/
|
|
peas_engine_garbage_collect (PEAS_ENGINE (xed_plugins_engine_get_default()));
|
|
|
|
/* save the panes position and make sure to deactivate plugins
|
|
* for this window, but only once */
|
|
if (!window->priv->dispose_has_run)
|
|
{
|
|
save_window_state (GTK_WIDGET (window));
|
|
save_panes_state (window);
|
|
|
|
/* Note that unreffing the extension will automatically remove
|
|
all extensions which in turn will deactivate the extension */
|
|
g_object_unref (window->priv->extensions);
|
|
|
|
peas_engine_garbage_collect (PEAS_ENGINE (xed_plugins_engine_get_default ()));
|
|
|
|
window->priv->dispose_has_run = TRUE;
|
|
}
|
|
|
|
if (window->priv->recents_handler_id != 0)
|
|
{
|
|
GtkRecentManager *recent_manager;
|
|
recent_manager = gtk_recent_manager_get_default ();
|
|
g_signal_handler_disconnect (recent_manager, window->priv->recents_handler_id);
|
|
window->priv->recents_handler_id = 0;
|
|
}
|
|
|
|
if (window->priv->favorites_handler_id != 0)
|
|
{
|
|
XAppFavorites *favorites;
|
|
favorites = xapp_favorites_get_default ();
|
|
g_signal_handler_disconnect (favorites, window->priv->favorites_handler_id);
|
|
window->priv->favorites_handler_id = 0;
|
|
}
|
|
|
|
g_clear_object (&window->priv->manager);
|
|
g_clear_object (&window->priv->message_bus);
|
|
g_clear_object (&window->priv->window_group);
|
|
g_clear_object (&window->priv->default_location);
|
|
|
|
/* We must free the settings after saving the panels */
|
|
g_clear_object (&window->priv->editor_settings);
|
|
g_clear_object (&window->priv->ui_settings);
|
|
g_clear_object (&window->priv->window_settings);
|
|
|
|
/* Now that there have broken some reference loops, force collection again. */
|
|
peas_engine_garbage_collect (PEAS_ENGINE (xed_plugins_engine_get_default ()));
|
|
|
|
G_OBJECT_CLASS (xed_window_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
xed_window_finalize (GObject *object)
|
|
{
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
G_OBJECT_CLASS (xed_window_parent_class)->finalize (object);
|
|
}
|
|
|
|
static gboolean
|
|
xed_window_window_state_event (GtkWidget *widget,
|
|
GdkEventWindowState *event)
|
|
{
|
|
XedWindow *window = XED_WINDOW (widget);
|
|
|
|
window->priv->window_state = event->new_window_state;
|
|
g_settings_set_int (window->priv->window_settings, XED_SETTINGS_WINDOW_STATE, window->priv->window_state);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
xed_window_configure_event (GtkWidget *widget,
|
|
GdkEventConfigure *event)
|
|
{
|
|
XedWindow *window = XED_WINDOW (widget);
|
|
|
|
if (gtk_widget_get_realized (widget) &&
|
|
(window->priv->window_state & (GDK_WINDOW_STATE_MAXIMIZED | GDK_WINDOW_STATE_FULLSCREEN)) == 0)
|
|
{
|
|
save_window_state (widget);
|
|
}
|
|
|
|
return GTK_WIDGET_CLASS (xed_window_parent_class)->configure_event (widget, event);
|
|
}
|
|
|
|
/*
|
|
* GtkWindow catches keybindings for the menu items _before_ passing them to
|
|
* the focused widget. This is unfortunate and means that pressing ctrl+V
|
|
* in an entry on a panel ends up pasting text in the TextView.
|
|
* Here we override GtkWindow's handler to do the same things that it
|
|
* does, but in the opposite order and then we chain up to the grand
|
|
* parent handler, skipping gtk_window_key_press_event.
|
|
*/
|
|
static gboolean
|
|
xed_window_key_press_event (GtkWidget *widget,
|
|
GdkEventKey *event)
|
|
{
|
|
static gpointer grand_parent_class = NULL;
|
|
GtkWindow *window = GTK_WINDOW(widget);
|
|
gboolean handled = FALSE;
|
|
|
|
if (grand_parent_class == NULL)
|
|
{
|
|
grand_parent_class = g_type_class_peek_parent (xed_window_parent_class);
|
|
}
|
|
|
|
/* handle focus widget key events */
|
|
if (!handled)
|
|
{
|
|
handled = gtk_window_propagate_key_event (window, event);
|
|
}
|
|
|
|
/* handle mnemonics and accelerators */
|
|
if (!handled)
|
|
{
|
|
handled = gtk_window_activate_key (window, event);
|
|
}
|
|
|
|
/* Chain up, invokes binding set */
|
|
if (!handled)
|
|
{
|
|
handled = GTK_WIDGET_CLASS (grand_parent_class)->key_press_event (widget, event);
|
|
}
|
|
|
|
return handled;
|
|
}
|
|
|
|
static void
|
|
xed_window_tab_removed (XedWindow *window,
|
|
XedTab *tab)
|
|
{
|
|
peas_engine_garbage_collect (PEAS_ENGINE (xed_plugins_engine_get_default ()));
|
|
}
|
|
|
|
static void
|
|
xed_window_class_init (XedWindowClass *klass)
|
|
{
|
|
GObjectClass *object_class = G_OBJECT_CLASS(klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS(klass);
|
|
|
|
klass->tab_removed = xed_window_tab_removed;
|
|
|
|
object_class->dispose = xed_window_dispose;
|
|
object_class->finalize = xed_window_finalize;
|
|
object_class->get_property = xed_window_get_property;
|
|
|
|
widget_class->window_state_event = xed_window_window_state_event;
|
|
widget_class->configure_event = xed_window_configure_event;
|
|
widget_class->key_press_event = xed_window_key_press_event;
|
|
|
|
signals[TAB_ADDED] = g_signal_new ("tab_added", G_OBJECT_CLASS_TYPE(object_class),
|
|
G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET(XedWindowClass, tab_added),
|
|
NULL, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, XED_TYPE_TAB);
|
|
|
|
signals[TAB_REMOVED] = g_signal_new ("tab_removed", G_OBJECT_CLASS_TYPE(object_class),
|
|
G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET(XedWindowClass, tab_removed),
|
|
NULL, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, XED_TYPE_TAB);
|
|
|
|
signals[TABS_REORDERED] = g_signal_new ("tabs_reordered", G_OBJECT_CLASS_TYPE(object_class),
|
|
G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET(XedWindowClass, tabs_reordered),
|
|
NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
|
|
|
|
signals[ACTIVE_TAB_CHANGED] = g_signal_new ("active_tab_changed", G_OBJECT_CLASS_TYPE(object_class),
|
|
G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET(XedWindowClass, active_tab_changed),
|
|
NULL, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, XED_TYPE_TAB);
|
|
|
|
signals[ACTIVE_TAB_STATE_CHANGED] = g_signal_new ("active_tab_state_changed", G_OBJECT_CLASS_TYPE(object_class),
|
|
G_SIGNAL_RUN_FIRST, G_STRUCT_OFFSET(XedWindowClass, active_tab_state_changed),
|
|
NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0);
|
|
|
|
g_object_class_install_property (object_class, PROP_STATE, g_param_spec_flags ("state",
|
|
"State",
|
|
"The window's state",
|
|
XED_TYPE_WINDOW_STATE,
|
|
XED_WINDOW_STATE_NORMAL,
|
|
G_PARAM_READABLE | G_PARAM_STATIC_STRINGS));
|
|
}
|
|
|
|
static void
|
|
menu_item_select_cb (GtkMenuItem *proxy,
|
|
XedWindow *window)
|
|
{
|
|
GtkAction *action;
|
|
char *message;
|
|
|
|
action = gtk_activatable_get_related_action (GTK_ACTIVATABLE(proxy));
|
|
g_return_if_fail(action != NULL);
|
|
|
|
g_object_get (G_OBJECT(action), "tooltip", &message, NULL);
|
|
if (message)
|
|
{
|
|
gtk_statusbar_push (GTK_STATUSBAR(window->priv->statusbar), window->priv->tip_message_cid, message);
|
|
g_free (message);
|
|
}
|
|
}
|
|
|
|
static void
|
|
menu_item_deselect_cb (GtkMenuItem *proxy,
|
|
XedWindow *window)
|
|
{
|
|
gtk_statusbar_pop (GTK_STATUSBAR(window->priv->statusbar), window->priv->tip_message_cid);
|
|
}
|
|
|
|
static void
|
|
connect_proxy_cb (GtkUIManager *manager,
|
|
GtkAction *action,
|
|
GtkWidget *proxy,
|
|
XedWindow *window)
|
|
{
|
|
if (GTK_IS_MENU_ITEM(proxy))
|
|
{
|
|
g_signal_connect(proxy, "select", G_CALLBACK (menu_item_select_cb), window);
|
|
g_signal_connect(proxy, "deselect", G_CALLBACK (menu_item_deselect_cb), window);
|
|
}
|
|
}
|
|
|
|
static void
|
|
disconnect_proxy_cb (GtkUIManager *manager,
|
|
GtkAction *action,
|
|
GtkWidget *proxy,
|
|
XedWindow *window)
|
|
{
|
|
if (GTK_IS_MENU_ITEM(proxy))
|
|
{
|
|
g_signal_handlers_disconnect_by_func(proxy, G_CALLBACK (menu_item_select_cb), window);
|
|
g_signal_handlers_disconnect_by_func(proxy, G_CALLBACK (menu_item_deselect_cb), window);
|
|
}
|
|
}
|
|
|
|
/* Returns TRUE if toolbar is visible */
|
|
static gboolean
|
|
set_toolbar_style (XedWindow *window,
|
|
XedWindow *origin)
|
|
{
|
|
gboolean visible;
|
|
GtkAction *action;
|
|
|
|
if (origin == NULL)
|
|
{
|
|
visible = g_settings_get_boolean (window->priv->ui_settings, XED_SETTINGS_TOOLBAR_VISIBLE);
|
|
}
|
|
else
|
|
{
|
|
visible = gtk_widget_get_visible (origin->priv->toolbar);
|
|
}
|
|
|
|
/* Set visibility */
|
|
if (visible)
|
|
{
|
|
gtk_widget_show (window->priv->toolbar);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_hide (window->priv->toolbar);
|
|
}
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewToolbar");
|
|
|
|
if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)) != visible)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action), visible);
|
|
}
|
|
|
|
return visible;
|
|
}
|
|
|
|
static void
|
|
update_next_prev_doc_sensitivity (XedWindow *window,
|
|
XedTab *tab)
|
|
{
|
|
gint tab_number;
|
|
GtkNotebook *notebook;
|
|
GtkAction *action;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
notebook = GTK_NOTEBOOK(_xed_window_get_notebook (window));
|
|
|
|
tab_number = gtk_notebook_page_num (notebook, GTK_WIDGET(tab));
|
|
g_return_if_fail(tab_number >= 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "DocumentsPreviousDocument");
|
|
gtk_action_set_sensitive (action, tab_number != 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "DocumentsNextDocument");
|
|
gtk_action_set_sensitive (action, tab_number < gtk_notebook_get_n_pages (notebook) - 1);
|
|
}
|
|
|
|
static void
|
|
update_next_prev_doc_sensitivity_per_window (XedWindow *window)
|
|
{
|
|
XedTab *tab;
|
|
GtkAction *action;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
tab = xed_window_get_active_tab (window);
|
|
if (tab != NULL)
|
|
{
|
|
update_next_prev_doc_sensitivity (window, tab);
|
|
return;
|
|
}
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "DocumentsPreviousDocument");
|
|
gtk_action_set_sensitive (action, FALSE);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "DocumentsNextDocument");
|
|
gtk_action_set_sensitive (action, FALSE);
|
|
|
|
}
|
|
|
|
static void
|
|
received_clipboard_contents (GtkClipboard *clipboard,
|
|
GtkSelectionData *selection_data,
|
|
XedWindow *window)
|
|
{
|
|
gboolean sensitive;
|
|
GtkAction *action;
|
|
|
|
/* getting clipboard contents is async, so we need to
|
|
* get the current tab and its state */
|
|
|
|
if (window->priv->active_tab != NULL)
|
|
{
|
|
XedTabState state;
|
|
gboolean state_normal;
|
|
|
|
state = xed_tab_get_state (window->priv->active_tab);
|
|
state_normal = (state == XED_TAB_STATE_NORMAL);
|
|
sensitive = state_normal && gtk_selection_data_targets_include_text (selection_data);
|
|
}
|
|
else
|
|
{
|
|
sensitive = FALSE;
|
|
}
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditPaste");
|
|
|
|
if (action != NULL)
|
|
{
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
}
|
|
|
|
g_object_unref (window);
|
|
}
|
|
|
|
static void
|
|
set_paste_sensitivity_according_to_clipboard (XedWindow *window,
|
|
GtkClipboard *clipboard)
|
|
{
|
|
GdkDisplay *display;
|
|
|
|
display = gtk_clipboard_get_display (clipboard);
|
|
if (gdk_display_supports_selection_notification (display))
|
|
{
|
|
gtk_clipboard_request_contents (clipboard, gdk_atom_intern_static_string ("TARGETS"),
|
|
(GtkClipboardReceivedFunc) received_clipboard_contents, g_object_ref (window));
|
|
}
|
|
else
|
|
{
|
|
GtkAction *action;
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditPaste");
|
|
|
|
/* XFIXES extension not available, make
|
|
* Paste always sensitive */
|
|
gtk_action_set_sensitive (action, TRUE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
set_sensitivity_according_to_tab (XedWindow *window,
|
|
XedTab *tab)
|
|
{
|
|
XedDocument *doc;
|
|
XedView *view;
|
|
GtkAction *action;
|
|
gboolean b;
|
|
gboolean state_normal;
|
|
gboolean editable;
|
|
XedTabState state;
|
|
GtkClipboard *clipboard;
|
|
gboolean enable_syntax_hl;
|
|
|
|
g_return_if_fail (XED_TAB (tab));
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
enable_syntax_hl = g_settings_get_boolean (window->priv->editor_settings, XED_SETTINGS_SYNTAX_HIGHLIGHTING);
|
|
|
|
state = xed_tab_get_state (tab);
|
|
state_normal = (state == XED_TAB_STATE_NORMAL);
|
|
|
|
view = xed_tab_get_view (tab);
|
|
editable = gtk_text_view_get_editable (GTK_TEXT_VIEW(view));
|
|
|
|
doc = XED_DOCUMENT (gtk_text_view_get_buffer (GTK_TEXT_VIEW (view)));
|
|
|
|
clipboard = gtk_widget_get_clipboard (GTK_WIDGET(window),
|
|
GDK_SELECTION_CLIPBOARD);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileSave");
|
|
gtk_action_set_sensitive (
|
|
action,
|
|
(state_normal || (state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION)
|
|
|| (state == XED_TAB_STATE_SHOWING_PRINT_PREVIEW))
|
|
&& !xed_document_get_readonly (doc));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileSaveAs");
|
|
gtk_action_set_sensitive (
|
|
action,
|
|
(state_normal || (state == XED_TAB_STATE_SAVING_ERROR)
|
|
|| (state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION)
|
|
|| (state == XED_TAB_STATE_SHOWING_PRINT_PREVIEW)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileRevert");
|
|
gtk_action_set_sensitive (
|
|
action,
|
|
(state_normal || (state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION))
|
|
&& !xed_document_is_untitled (doc));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FilePrintPreview");
|
|
gtk_action_set_sensitive (action, state_normal);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FilePrint");
|
|
gtk_action_set_sensitive (action, (state_normal || (state == XED_TAB_STATE_SHOWING_PRINT_PREVIEW)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->close_action_group, "FileClose");
|
|
|
|
gtk_action_set_sensitive (
|
|
action,
|
|
(state != XED_TAB_STATE_CLOSING) && (state != XED_TAB_STATE_SAVING)
|
|
&& (state != XED_TAB_STATE_SHOWING_PRINT_PREVIEW)
|
|
&& (state != XED_TAB_STATE_PRINTING) && (state != XED_TAB_STATE_PRINT_PREVIEWING)
|
|
&& (state != XED_TAB_STATE_SAVING_ERROR));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditUndo");
|
|
gtk_action_set_sensitive (action, state_normal && gtk_source_buffer_can_undo (GTK_SOURCE_BUFFER(doc)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditRedo");
|
|
gtk_action_set_sensitive (action, state_normal && gtk_source_buffer_can_redo (GTK_SOURCE_BUFFER(doc)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCut");
|
|
gtk_action_set_sensitive (action,
|
|
state_normal && editable && gtk_text_buffer_get_has_selection (GTK_TEXT_BUFFER(doc)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCopy");
|
|
gtk_action_set_sensitive (action,
|
|
(state_normal || state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION)
|
|
&& gtk_text_buffer_get_has_selection (GTK_TEXT_BUFFER(doc)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditPaste");
|
|
if (state_normal && editable)
|
|
{
|
|
set_paste_sensitivity_according_to_clipboard (window, clipboard);
|
|
}
|
|
else
|
|
{
|
|
gtk_action_set_sensitive (action, FALSE);
|
|
}
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditDelete");
|
|
gtk_action_set_sensitive (action,
|
|
state_normal && editable && gtk_text_buffer_get_has_selection (GTK_TEXT_BUFFER(doc)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchFind");
|
|
gtk_action_set_sensitive (action, (state_normal || state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchReplace");
|
|
gtk_action_set_sensitive (action, state_normal && editable);
|
|
|
|
b = TRUE;
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchFindNext");
|
|
gtk_action_set_sensitive (action, (state_normal || state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION) && b);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchFindPrevious");
|
|
gtk_action_set_sensitive (action, (state_normal || state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION) && b);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchGoToLine");
|
|
gtk_action_set_sensitive (action, (state_normal || state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "ViewHighlightMode");
|
|
gtk_action_set_sensitive (action, (state != XED_TAB_STATE_CLOSING) && enable_syntax_hl);
|
|
|
|
update_next_prev_doc_sensitivity (window, tab);
|
|
|
|
peas_extension_set_call (window->priv->extensions, "update_state");
|
|
}
|
|
|
|
void
|
|
_xed_recent_add (XedWindow *window,
|
|
GFile *location,
|
|
const gchar *mime)
|
|
{
|
|
GtkRecentManager *recent_manager;
|
|
GtkRecentData *recent_data;
|
|
gchar *uri;
|
|
|
|
static gchar *groups[2] = { "xed", NULL };
|
|
|
|
recent_manager = gtk_recent_manager_get_default ();
|
|
|
|
recent_data = g_slice_new(GtkRecentData);
|
|
|
|
recent_data->display_name = NULL;
|
|
recent_data->description = NULL;
|
|
recent_data->mime_type = (gchar *) mime;
|
|
recent_data->app_name = (gchar *) g_get_application_name ();
|
|
recent_data->app_exec = g_strjoin (" ", g_get_prgname (), "%u", NULL);
|
|
recent_data->groups = groups;
|
|
recent_data->is_private = FALSE;
|
|
|
|
uri = g_file_get_uri (location);
|
|
|
|
gtk_recent_manager_add_full (recent_manager, uri, recent_data);
|
|
|
|
g_free (uri);
|
|
g_free (recent_data->app_exec);
|
|
|
|
g_slice_free(GtkRecentData, recent_data);
|
|
}
|
|
|
|
void
|
|
_xed_recent_remove (XedWindow *window,
|
|
GFile *location)
|
|
{
|
|
GtkRecentManager *recent_manager;
|
|
gchar *uri;
|
|
|
|
recent_manager = gtk_recent_manager_get_default ();
|
|
|
|
uri = g_file_get_uri (location);
|
|
gtk_recent_manager_remove_item (recent_manager, uri, NULL);
|
|
g_free (uri);
|
|
}
|
|
|
|
static void
|
|
open_recent_file (GFile *location,
|
|
XedWindow *window)
|
|
{
|
|
GSList *locations = NULL;
|
|
GSList *loaded = NULL;
|
|
|
|
locations = g_slist_prepend (locations, (gpointer) location);
|
|
|
|
loaded = xed_commands_load_locations (window, locations, NULL, 0);
|
|
|
|
if (!loaded || loaded->next) /* if it doesn't contain just 1 element */
|
|
{
|
|
_xed_recent_remove (window, location);
|
|
}
|
|
|
|
g_slist_free (locations);
|
|
g_slist_free (loaded);
|
|
}
|
|
|
|
static void
|
|
recents_menu_activate (GtkAction *action,
|
|
XedWindow *window)
|
|
{
|
|
GtkRecentInfo *info;
|
|
GFile *location;
|
|
const gchar *uri;
|
|
|
|
info = g_object_get_data (G_OBJECT (action), "gtk-recent-info");
|
|
g_return_if_fail (info != NULL);
|
|
|
|
uri = gtk_recent_info_get_uri (info);
|
|
location = g_file_new_for_uri (uri);
|
|
|
|
if (location)
|
|
{
|
|
open_recent_file (location, window);
|
|
g_object_unref (location);
|
|
}
|
|
}
|
|
|
|
static gint
|
|
sort_recents_mru (GtkRecentInfo *a,
|
|
GtkRecentInfo *b)
|
|
{
|
|
return (gtk_recent_info_get_modified (b) - gtk_recent_info_get_modified (a));
|
|
}
|
|
|
|
static void
|
|
update_recent_files_menu (XedWindow *window);
|
|
|
|
static void
|
|
recent_manager_changed (GtkRecentManager *manager,
|
|
XedWindow *window)
|
|
{
|
|
/* regenerate the menu when the model changes */
|
|
update_recent_files_menu (window);
|
|
}
|
|
|
|
/*
|
|
* Manually construct the inline recents list in the File menu.
|
|
* Hopefully gtk 2.12 will add support for it.
|
|
*/
|
|
static void
|
|
update_recent_files_menu (XedWindow *window)
|
|
{
|
|
XedWindowPrivate *p = window->priv;
|
|
GtkRecentManager *recent_manager;
|
|
gint max_recents;
|
|
GList *actions, *l, *items;
|
|
GList *filtered_items = NULL;
|
|
guint i;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
max_recents = g_settings_get_uint (window->priv->ui_settings, XED_SETTINGS_MAX_RECENTS);
|
|
|
|
g_return_if_fail(p->recents_action_group != NULL);
|
|
|
|
if (p->recents_menu_ui_id != 0)
|
|
{
|
|
gtk_ui_manager_remove_ui (p->manager, p->recents_menu_ui_id);
|
|
}
|
|
|
|
actions = gtk_action_group_list_actions (p->recents_action_group);
|
|
for (l = actions; l != NULL; l = l->next)
|
|
{
|
|
g_signal_handlers_disconnect_by_func(GTK_ACTION (l->data), G_CALLBACK (recents_menu_activate), window);
|
|
gtk_action_group_remove_action (p->recents_action_group, GTK_ACTION(l->data));
|
|
}
|
|
g_list_free (actions);
|
|
|
|
p->recents_menu_ui_id = gtk_ui_manager_new_merge_id (p->manager);
|
|
|
|
recent_manager = gtk_recent_manager_get_default ();
|
|
items = gtk_recent_manager_get_items (recent_manager);
|
|
|
|
/* filter */
|
|
for (l = items; l != NULL; l = l->next)
|
|
{
|
|
GtkRecentInfo *info = l->data;
|
|
if (!gtk_recent_info_has_group (info, "xed"))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
filtered_items = g_list_prepend (filtered_items, info);
|
|
}
|
|
|
|
/* sort */
|
|
filtered_items = g_list_sort (filtered_items, (GCompareFunc) sort_recents_mru);
|
|
|
|
i = 0;
|
|
for (l = filtered_items; l != NULL; l = l->next)
|
|
{
|
|
gchar *action_name;
|
|
const gchar *display_name;
|
|
gchar *escaped;
|
|
gchar *label;
|
|
gchar *uri;
|
|
gchar *ruri;
|
|
gchar *tip;
|
|
GtkAction *action;
|
|
GtkRecentInfo *info = l->data;
|
|
GFile *location;
|
|
|
|
/* clamp */
|
|
if (i >= max_recents)
|
|
{
|
|
break;
|
|
}
|
|
|
|
i++;
|
|
|
|
action_name = g_strdup_printf ("recent-info-%d", i);
|
|
|
|
display_name = gtk_recent_info_get_display_name (info);
|
|
escaped = xed_utils_escape_underscores (display_name, -1);
|
|
if (i >= 10)
|
|
{
|
|
label = g_strdup_printf ("%d. %s", i, escaped);
|
|
}
|
|
else
|
|
label = g_strdup_printf ("_%d. %s", i, escaped);
|
|
g_free (escaped);
|
|
|
|
/* gtk_recent_info_get_uri_display (info) is buggy and
|
|
* works only for local files */
|
|
location = g_file_new_for_uri (gtk_recent_info_get_uri (info));
|
|
uri = g_file_get_parse_name (location);
|
|
g_object_unref (location);
|
|
ruri = xed_utils_replace_home_dir_with_tilde (uri);
|
|
g_free (uri);
|
|
|
|
/* Translators: %s is a URI */
|
|
tip = g_strdup_printf (_("Open '%s'"), ruri);
|
|
g_free (ruri);
|
|
|
|
action = gtk_action_new (action_name, label, tip, NULL);
|
|
|
|
g_object_set_data_full (G_OBJECT(action), "gtk-recent-info", gtk_recent_info_ref (info),
|
|
(GDestroyNotify) gtk_recent_info_unref);
|
|
|
|
g_signal_connect(action, "activate", G_CALLBACK (recents_menu_activate), window);
|
|
|
|
gtk_action_group_add_action (p->recents_action_group, action);
|
|
g_object_unref (action);
|
|
|
|
gtk_ui_manager_add_ui (p->manager, p->recents_menu_ui_id, "/MenuBar/FileMenu/FileRecentsMenu/FileRecentsPlaceholder",
|
|
action_name, action_name, GTK_UI_MANAGER_MENUITEM, FALSE);
|
|
|
|
g_free (action_name);
|
|
g_free (label);
|
|
g_free (tip);
|
|
}
|
|
|
|
g_list_free (filtered_items);
|
|
|
|
g_list_foreach (items, (GFunc) gtk_recent_info_unref, NULL);
|
|
g_list_free (items);
|
|
}
|
|
|
|
static void
|
|
toolbar_visibility_changed (GtkWidget *toolbar,
|
|
XedWindow *window)
|
|
{
|
|
gboolean visible;
|
|
GtkAction *action;
|
|
|
|
visible = gtk_widget_get_visible (toolbar);
|
|
|
|
g_settings_set_boolean (window->priv->ui_settings, XED_SETTINGS_TOOLBAR_VISIBLE, visible);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewToolbar");
|
|
|
|
if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION(action)) != visible)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION(action), visible);
|
|
}
|
|
}
|
|
|
|
static void
|
|
favorite_activated (GtkAction *action,
|
|
gpointer user_data)
|
|
{
|
|
GFile *location;
|
|
|
|
location = g_file_new_for_uri (gtk_action_get_name (action));
|
|
xed_commands_load_location (XED_WINDOW (user_data), location, NULL, 0);
|
|
|
|
g_object_unref (location);
|
|
}
|
|
|
|
static void
|
|
update_favorites_menu (XedWindow *window)
|
|
{
|
|
XedWindowPrivate *p = window->priv;
|
|
XAppFavorites *favorites;
|
|
GList *actions, *l;
|
|
GList *items = NULL;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
g_return_if_fail (p->favorites_action_group != NULL);
|
|
|
|
if (p->favorites_menu_ui_id != 0)
|
|
{
|
|
gtk_ui_manager_remove_ui (p->manager, p->favorites_menu_ui_id);
|
|
}
|
|
|
|
actions = gtk_action_group_list_actions (p->favorites_action_group);
|
|
for (l = actions; l != NULL; l = l->next)
|
|
{
|
|
g_signal_handlers_disconnect_by_func(GTK_ACTION (l->data), G_CALLBACK (favorite_activated), window);
|
|
gtk_action_group_remove_action (p->favorites_action_group, GTK_ACTION(l->data));
|
|
}
|
|
g_list_free (actions);
|
|
|
|
p->favorites_menu_ui_id = gtk_ui_manager_new_merge_id (p->manager);
|
|
|
|
favorites = xapp_favorites_get_default ();
|
|
|
|
const gchar *supported_mimetypes[2] = {
|
|
"text/plain",
|
|
NULL
|
|
};
|
|
|
|
items = xapp_favorites_create_actions (favorites, supported_mimetypes);
|
|
|
|
for (l = items; l != NULL; l = l->next)
|
|
{
|
|
GtkAction *action = GTK_ACTION (l->data);
|
|
const gchar *name = gtk_action_get_name (action);
|
|
|
|
g_signal_connect (action, "activate", G_CALLBACK (favorite_activated), window);
|
|
|
|
gtk_action_group_add_action (p->favorites_action_group, action);
|
|
|
|
gtk_ui_manager_add_ui (p->manager, p->favorites_menu_ui_id, "/MenuBar/FileMenu/XAppFavoritesMenu/XAppFavorites",
|
|
name, name, GTK_UI_MANAGER_MENUITEM, FALSE);
|
|
}
|
|
|
|
g_list_free_full (items, g_object_unref);
|
|
}
|
|
|
|
static GtkWidget *
|
|
create_toolbar_button (GtkAction *action)
|
|
{
|
|
GtkWidget *button;
|
|
GtkWidget *image;
|
|
|
|
button = gtk_button_new ();
|
|
image = gtk_image_new ();
|
|
|
|
gtk_button_set_image (GTK_BUTTON (button), image);
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (button), "flat");
|
|
gtk_activatable_set_related_action (GTK_ACTIVATABLE (button), action);
|
|
gtk_button_set_label (GTK_BUTTON (button), NULL);
|
|
gtk_widget_set_tooltip_text (button, gtk_action_get_tooltip (action));
|
|
|
|
return button;
|
|
}
|
|
|
|
static void
|
|
create_menu_bar_and_toolbar (XedWindow *window,
|
|
GtkWidget *main_box)
|
|
{
|
|
GtkActionGroup *action_group;
|
|
GtkAction *action;
|
|
GtkUIManager *manager;
|
|
GtkRecentManager *recent_manager;
|
|
GError *error = NULL;
|
|
GtkToolItem *tool_item;
|
|
GtkWidget *tool_box;
|
|
GtkWidget *box;
|
|
GtkWidget *separator;
|
|
GtkWidget *button;
|
|
XAppFavorites *favorites;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
manager = gtk_ui_manager_new ();
|
|
window->priv->manager = manager;
|
|
|
|
gtk_window_add_accel_group (GTK_WINDOW(window), gtk_ui_manager_get_accel_group (manager));
|
|
|
|
action_group = gtk_action_group_new ("XedWindowAlwaysSensitiveActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
gtk_action_group_add_actions (action_group, xed_always_sensitive_menu_entries,
|
|
G_N_ELEMENTS(xed_always_sensitive_menu_entries), window);
|
|
gtk_action_group_add_toggle_actions (action_group, xed_always_sensitive_toggle_menu_entries,
|
|
G_N_ELEMENTS(xed_always_sensitive_toggle_menu_entries), window);
|
|
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
window->priv->always_sensitive_action_group = action_group;
|
|
|
|
action_group = gtk_action_group_new ("XedWindowActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
gtk_action_group_add_actions (action_group, xed_menu_entries, G_N_ELEMENTS(xed_menu_entries), window);
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
window->priv->action_group = action_group;
|
|
|
|
action_group = gtk_action_group_new ("XedQuitWindowActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
gtk_action_group_add_actions (action_group, xed_quit_menu_entries, G_N_ELEMENTS(xed_quit_menu_entries), window);
|
|
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
window->priv->quit_action_group = action_group;
|
|
|
|
action_group = gtk_action_group_new ("XedCloseWindowActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
gtk_action_group_add_actions (action_group, xed_close_menu_entries, G_N_ELEMENTS(xed_close_menu_entries), window);
|
|
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
window->priv->close_action_group = action_group;
|
|
|
|
action_group = gtk_action_group_new ("XedWindowPanesActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
gtk_action_group_add_toggle_actions (action_group, xed_panes_toggle_menu_entries,
|
|
G_N_ELEMENTS(xed_panes_toggle_menu_entries), window);
|
|
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
window->priv->panes_action_group = action_group;
|
|
|
|
gtk_ui_manager_add_ui_from_resource (manager, "/org/x/editor/ui/xed-ui.xml", &error);
|
|
if (error != NULL)
|
|
{
|
|
g_warning ("Could not add ui definition: %s", error->message);
|
|
g_error_free (error);
|
|
}
|
|
|
|
/* show tooltips in the statusbar */
|
|
g_signal_connect(manager, "connect_proxy", G_CALLBACK (connect_proxy_cb), window);
|
|
g_signal_connect(manager, "disconnect_proxy", G_CALLBACK (disconnect_proxy_cb), window);
|
|
|
|
/* recent files menu */
|
|
action_group = gtk_action_group_new ("RecentFilesActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
window->priv->recents_action_group = action_group;
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
|
|
recent_manager = gtk_recent_manager_get_default ();
|
|
window->priv->recents_handler_id = g_signal_connect(recent_manager, "changed", G_CALLBACK (recent_manager_changed),
|
|
window);
|
|
update_recent_files_menu (window);
|
|
|
|
action_group = gtk_action_group_new ("FavoriteFilesActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
window->priv->favorites_action_group = action_group;
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
|
|
favorites = xapp_favorites_get_default ();
|
|
window->priv->favorites_handler_id = g_signal_connect_swapped (favorites,
|
|
"changed", G_CALLBACK (update_favorites_menu),
|
|
window);
|
|
|
|
update_favorites_menu (window);
|
|
|
|
/* list of open documents menu */
|
|
action_group = gtk_action_group_new ("DocumentsListActions");
|
|
gtk_action_group_set_translation_domain (action_group, NULL);
|
|
window->priv->documents_list_action_group = action_group;
|
|
gtk_ui_manager_insert_action_group (manager, action_group, 0);
|
|
g_object_unref (action_group);
|
|
|
|
window->priv->menubar = gtk_ui_manager_get_widget (manager, "/MenuBar");
|
|
gtk_box_pack_start (GTK_BOX(main_box), window->priv->menubar, FALSE, FALSE, 0);
|
|
|
|
window->priv->toolbar = gtk_toolbar_new ();
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (window->priv->toolbar), GTK_STYLE_CLASS_PRIMARY_TOOLBAR);
|
|
gtk_box_pack_start (GTK_BOX(main_box), window->priv->toolbar, FALSE, FALSE, 0);
|
|
|
|
tool_item = gtk_tool_item_new ();
|
|
gtk_tool_item_set_expand ((tool_item), TRUE);
|
|
gtk_toolbar_insert (GTK_TOOLBAR (window->priv->toolbar), (tool_item), 0);
|
|
|
|
tool_box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 6);
|
|
gtk_container_add (GTK_CONTAINER (tool_item), tool_box);
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_box_pack_start (GTK_BOX (tool_box), box, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "FileNew");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "FileOpen");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileSave");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
separator = gtk_separator_new (GTK_ORIENTATION_VERTICAL);
|
|
gtk_box_pack_start (GTK_BOX (tool_box), separator, FALSE, FALSE, 0);
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_box_pack_start (GTK_BOX (tool_box), box, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditUndo");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditRedo");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
separator = gtk_separator_new (GTK_ORIENTATION_VERTICAL);
|
|
gtk_box_pack_start (GTK_BOX (tool_box), separator, FALSE, FALSE, 0);
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_box_pack_start (GTK_BOX (tool_box), box, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCut");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCopy");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditPaste");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
separator = gtk_separator_new (GTK_ORIENTATION_VERTICAL);
|
|
gtk_box_pack_start (GTK_BOX (tool_box), separator, FALSE, FALSE, 0);
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_box_pack_start (GTK_BOX (tool_box), box, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchFind");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchReplace");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show_all (GTK_WIDGET (window->priv->toolbar));
|
|
|
|
set_toolbar_style (window, NULL);
|
|
|
|
g_signal_connect_after(G_OBJECT (window->priv->toolbar), "show", G_CALLBACK (toolbar_visibility_changed), window);
|
|
g_signal_connect_after(G_OBJECT (window->priv->toolbar), "hide", G_CALLBACK (toolbar_visibility_changed), window);
|
|
}
|
|
|
|
static void
|
|
documents_list_menu_activate (GtkToggleAction *action,
|
|
XedWindow *window)
|
|
{
|
|
gint n;
|
|
if (gtk_toggle_action_get_active (action) == FALSE)
|
|
{
|
|
return;
|
|
}
|
|
n = gtk_radio_action_get_current_value (GTK_RADIO_ACTION(action));
|
|
gtk_notebook_set_current_page (GTK_NOTEBOOK(window->priv->notebook), n);
|
|
}
|
|
|
|
static gchar *
|
|
get_menu_tip_for_tab (XedTab *tab)
|
|
{
|
|
XedDocument *doc;
|
|
gchar *uri;
|
|
gchar *ruri;
|
|
gchar *tip;
|
|
|
|
doc = xed_tab_get_document (tab);
|
|
|
|
uri = xed_document_get_uri_for_display (doc);
|
|
ruri = xed_utils_replace_home_dir_with_tilde (uri);
|
|
g_free (uri);
|
|
|
|
/* Translators: %s is a URI */
|
|
tip = g_strdup_printf (_("Activate '%s'"), ruri);
|
|
g_free (ruri);
|
|
|
|
return tip;
|
|
}
|
|
|
|
static void
|
|
update_documents_list_menu (XedWindow *window)
|
|
{
|
|
XedWindowPrivate *p = window->priv;
|
|
GList *actions, *l;
|
|
gint n, i;
|
|
guint id;
|
|
GSList *group = NULL;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
g_return_if_fail(p->documents_list_action_group != NULL);
|
|
|
|
if (p->documents_list_menu_ui_id != 0)
|
|
{
|
|
gtk_ui_manager_remove_ui (p->manager, p->documents_list_menu_ui_id);
|
|
}
|
|
|
|
actions = gtk_action_group_list_actions (p->documents_list_action_group);
|
|
for (l = actions; l != NULL; l = l->next)
|
|
{
|
|
g_signal_handlers_disconnect_by_func(GTK_ACTION (l->data), G_CALLBACK (documents_list_menu_activate), window);
|
|
gtk_action_group_remove_action (p->documents_list_action_group, GTK_ACTION(l->data));
|
|
}
|
|
g_list_free (actions);
|
|
|
|
n = gtk_notebook_get_n_pages (GTK_NOTEBOOK(p->notebook));
|
|
|
|
id = (n > 0) ? gtk_ui_manager_new_merge_id (p->manager) : 0;
|
|
|
|
for (i = 0; i < n; i++)
|
|
{
|
|
GtkWidget *tab;
|
|
GtkRadioAction *action;
|
|
gchar *action_name;
|
|
gchar *tab_name;
|
|
gchar *name;
|
|
gchar *tip;
|
|
gchar *accel;
|
|
|
|
tab = gtk_notebook_get_nth_page (GTK_NOTEBOOK(p->notebook), i);
|
|
|
|
/* NOTE: the action is associated to the position of the tab in
|
|
* the notebook not to the tab itself! This is needed to work
|
|
* around the gtk+ bug #170727: gtk leaves around the accels
|
|
* of the action. Since the accel depends on the tab position
|
|
* the problem is worked around, action with the same name always
|
|
* get the same accel.
|
|
*/
|
|
action_name = g_strdup_printf ("Tab_%d", i);
|
|
tab_name = _xed_tab_get_name (XED_TAB(tab));
|
|
name = xed_utils_escape_underscores (tab_name, -1);
|
|
tip = get_menu_tip_for_tab (XED_TAB(tab));
|
|
|
|
/* alt + 1, 2, 3... 0 to switch to the first ten tabs */
|
|
accel = (i < 10) ? g_strdup_printf ("<alt>%d", (i + 1) % 10) : NULL;
|
|
|
|
action = gtk_radio_action_new (action_name, name, tip, NULL, i);
|
|
|
|
if (group != NULL)
|
|
{
|
|
gtk_radio_action_set_group (action, group);
|
|
}
|
|
|
|
/* note that group changes each time we add an action, so it must be updated */
|
|
group = gtk_radio_action_get_group (action);
|
|
|
|
gtk_action_group_add_action_with_accel (p->documents_list_action_group, GTK_ACTION(action), accel);
|
|
|
|
g_signal_connect(action, "activate", G_CALLBACK (documents_list_menu_activate), window);
|
|
|
|
gtk_ui_manager_add_ui (p->manager, id, "/MenuBar/DocumentsMenu/DocumentsListPlaceholder", action_name,
|
|
action_name, GTK_UI_MANAGER_MENUITEM, FALSE);
|
|
|
|
if (XED_TAB (tab) == p->active_tab)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION(action), TRUE);
|
|
}
|
|
|
|
g_object_unref (action);
|
|
|
|
g_free (action_name);
|
|
g_free (tab_name);
|
|
g_free (name);
|
|
g_free (tip);
|
|
g_free (accel);
|
|
}
|
|
|
|
p->documents_list_menu_ui_id = id;
|
|
}
|
|
|
|
/* Returns TRUE if status bar is visible */
|
|
static gboolean
|
|
set_statusbar_style (XedWindow *window,
|
|
XedWindow *origin)
|
|
{
|
|
GtkAction *action;
|
|
gboolean visible;
|
|
|
|
if (origin == NULL)
|
|
{
|
|
visible = g_settings_get_boolean (window->priv->ui_settings, XED_SETTINGS_STATUSBAR_VISIBLE);
|
|
}
|
|
else
|
|
{
|
|
visible = gtk_widget_get_visible (origin->priv->statusbar);
|
|
}
|
|
|
|
if (visible)
|
|
{
|
|
gtk_widget_show (window->priv->statusbar);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_hide (window->priv->statusbar);
|
|
}
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewStatusbar");
|
|
|
|
if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION(action)) != visible)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION(action), visible);
|
|
}
|
|
|
|
return visible;
|
|
}
|
|
|
|
static void
|
|
statusbar_visibility_changed (GtkWidget *statusbar,
|
|
XedWindow *window)
|
|
{
|
|
gboolean visible;
|
|
GtkAction *action;
|
|
|
|
visible = gtk_widget_get_visible (statusbar);
|
|
|
|
g_settings_set_boolean (window->priv->ui_settings, XED_SETTINGS_STATUSBAR_VISIBLE, visible);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewStatusbar");
|
|
|
|
if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION(action)) != visible)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION(action), visible);
|
|
}
|
|
}
|
|
|
|
static void
|
|
tab_width_button_clicked (GtkMenuItem *item,
|
|
XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
guint width_data = 0;
|
|
|
|
view = xed_window_get_active_view (window);
|
|
|
|
if (!view)
|
|
{
|
|
return;
|
|
}
|
|
|
|
width_data = GPOINTER_TO_INT(g_object_get_data (G_OBJECT (item), TAB_WIDTH_DATA));
|
|
|
|
if (width_data == 0)
|
|
{
|
|
return;
|
|
}
|
|
|
|
gtk_source_view_set_tab_width (GTK_SOURCE_VIEW(view), width_data);
|
|
}
|
|
|
|
static void
|
|
set_tab_spaces_label (XedView *view,
|
|
XedWindow *window,
|
|
gboolean use_spaces)
|
|
{
|
|
gchar *label;
|
|
guint tab_width;
|
|
|
|
tab_width = gtk_source_view_get_tab_width (GTK_SOURCE_VIEW (view));
|
|
|
|
if (use_spaces)
|
|
{
|
|
label = g_strdup_printf (_("Spaces: %u"), tab_width);
|
|
}
|
|
else
|
|
{
|
|
label = g_strdup_printf (_("Tabs: %u"), tab_width);
|
|
}
|
|
|
|
xed_status_menu_button_set_label (XED_STATUS_MENU_BUTTON (window->priv->tab_width_button), label);
|
|
|
|
g_free (label);
|
|
}
|
|
|
|
static void
|
|
use_spaces_toggled (GtkCheckMenuItem *item,
|
|
XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
gboolean use_spaces;
|
|
|
|
view = xed_window_get_active_view (window);
|
|
use_spaces = gtk_check_menu_item_get_active (item);
|
|
|
|
set_tab_spaces_label (view, window, use_spaces);
|
|
|
|
g_signal_handler_block (view, window->priv->spaces_instead_of_tabs_id);
|
|
gtk_source_view_set_insert_spaces_instead_of_tabs (GTK_SOURCE_VIEW (view), use_spaces);
|
|
g_signal_handler_unblock (view, window->priv->spaces_instead_of_tabs_id);
|
|
}
|
|
|
|
typedef struct
|
|
{
|
|
const gchar *label;
|
|
guint width;
|
|
} TabWidthDefinition;
|
|
|
|
static void
|
|
tab_width_menu_popped_up (GtkMenu *menu,
|
|
gpointer flipped_rect,
|
|
gpointer final_rect,
|
|
gboolean flipped_x,
|
|
gboolean flipped_y,
|
|
gpointer user_data)
|
|
{
|
|
XedWindow *window = XED_WINDOW (user_data);
|
|
|
|
gtk_menu_shell_select_item (GTK_MENU_SHELL (window->priv->tab_width_menu), GTK_WIDGET (window->priv->tab_width_item));
|
|
}
|
|
|
|
static void
|
|
setup_tab_width_menu (XedWindow *window)
|
|
{
|
|
static TabWidthDefinition defs[] = { { "2", 2 }, { "4", 4 }, { "8", 8 }, { "", 0 }, /* custom size */
|
|
{ NULL, 0 } };
|
|
|
|
guint i = 0;
|
|
GtkWidget *item;
|
|
|
|
window->priv->tab_width_menu = gtk_menu_new ();
|
|
|
|
while (defs[i].label != NULL)
|
|
{
|
|
item = gtk_menu_item_new_with_label (defs[i].label);
|
|
g_object_set_data (G_OBJECT(item), TAB_WIDTH_DATA, GINT_TO_POINTER (defs[i].width));
|
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (window->priv->tab_width_menu), item);
|
|
|
|
g_signal_connect(G_OBJECT (item), "activate",
|
|
G_CALLBACK (tab_width_button_clicked), window);
|
|
|
|
if (defs[i].width != 0)
|
|
{
|
|
gtk_widget_show (item);
|
|
}
|
|
|
|
++i;
|
|
}
|
|
|
|
item = gtk_separator_menu_item_new ();
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (window->priv->tab_width_menu), item);
|
|
gtk_widget_show (item);
|
|
|
|
item = gtk_check_menu_item_new_with_label (_("Use Spaces"));
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (window->priv->tab_width_menu), item);
|
|
gtk_widget_show (item);
|
|
|
|
g_signal_connect (item, "toggled", G_CALLBACK (use_spaces_toggled), window);
|
|
|
|
g_signal_connect (window->priv->tab_width_menu, "popped-up", G_CALLBACK (tab_width_menu_popped_up), window);
|
|
}
|
|
|
|
static void
|
|
on_language_selector_shown (XedHighlightModeSelector *sel,
|
|
XedWindow *window)
|
|
{
|
|
XedDocument *doc;
|
|
|
|
doc = xed_window_get_active_document (window);
|
|
if (doc)
|
|
{
|
|
xed_highlight_mode_selector_select_language (sel,
|
|
xed_document_get_language (doc));
|
|
}
|
|
}
|
|
|
|
static void
|
|
on_language_selected (XedHighlightModeSelector *sel,
|
|
GtkSourceLanguage *language,
|
|
XedWindow *window)
|
|
{
|
|
XedDocument *doc;
|
|
|
|
doc = xed_window_get_active_document (window);
|
|
if (doc)
|
|
{
|
|
xed_document_set_language (doc, language);
|
|
}
|
|
|
|
gtk_widget_hide (GTK_WIDGET (window->priv->language_popover));
|
|
}
|
|
|
|
static void
|
|
create_statusbar (XedWindow *window,
|
|
GtkWidget *main_box)
|
|
{
|
|
GtkWidget *image;
|
|
GtkWidget *button_box;
|
|
GtkAction *action;
|
|
XedHighlightModeSelector *sel;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
window->priv->statusbar = xed_statusbar_new ();
|
|
window->priv->searchbar = xed_searchbar_new (GTK_WINDOW (window));
|
|
|
|
window->priv->generic_message_cid = gtk_statusbar_get_context_id (GTK_STATUSBAR (window->priv->statusbar),
|
|
"generic_message");
|
|
window->priv->tip_message_cid = gtk_statusbar_get_context_id (GTK_STATUSBAR (window->priv->statusbar),
|
|
"tip_message");
|
|
|
|
gtk_box_pack_end (GTK_BOX( main_box), window->priv->statusbar, FALSE, TRUE, 0);
|
|
|
|
gtk_widget_set_margin_start (GTK_WIDGET (window->priv->statusbar), 0);
|
|
gtk_widget_set_margin_end (GTK_WIDGET (window->priv->statusbar), 0);
|
|
|
|
window->priv->tab_width_button = xed_status_menu_button_new ();
|
|
gtk_widget_show (window->priv->tab_width_button);
|
|
gtk_box_pack_end (GTK_BOX (window->priv->statusbar), window->priv->tab_width_button, FALSE, FALSE, 0);
|
|
gtk_widget_set_margin_bottom (GTK_WIDGET (window->priv->tab_width_button), 2);
|
|
gtk_widget_set_margin_top (GTK_WIDGET (window->priv->tab_width_button), 2);
|
|
|
|
setup_tab_width_menu (window);
|
|
|
|
gtk_menu_button_set_popup (GTK_MENU_BUTTON (window->priv->tab_width_button),
|
|
window->priv->tab_width_menu);
|
|
|
|
window->priv->language_button = xed_status_menu_button_new ();
|
|
gtk_widget_show (window->priv->language_button);
|
|
gtk_widget_set_margin_bottom (GTK_WIDGET (window->priv->language_button), 2);
|
|
gtk_widget_set_margin_top (GTK_WIDGET (window->priv->language_button), 2);
|
|
gtk_box_pack_end (GTK_BOX(window->priv->statusbar), window->priv->language_button, FALSE, FALSE, 0);
|
|
|
|
window->priv->language_popover = gtk_popover_new (window->priv->language_button);
|
|
gtk_menu_button_set_popover (GTK_MENU_BUTTON (window->priv->language_button),
|
|
window->priv->language_popover);
|
|
|
|
sel = xed_highlight_mode_selector_new ();
|
|
g_signal_connect (sel,
|
|
"show",
|
|
G_CALLBACK (on_language_selector_shown),
|
|
window);
|
|
g_signal_connect (sel,
|
|
"language-selected",
|
|
G_CALLBACK (on_language_selected),
|
|
window);
|
|
|
|
gtk_container_add (GTK_CONTAINER (window->priv->language_popover), GTK_WIDGET (sel));
|
|
gtk_widget_show (GTK_WIDGET (sel));
|
|
|
|
button_box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 4);
|
|
gtk_widget_set_margin_top (button_box, 4);
|
|
gtk_widget_set_margin_bottom (button_box, 4);
|
|
gtk_widget_set_margin_start (button_box, 6);
|
|
gtk_box_pack_start (GTK_BOX (window->priv->statusbar), button_box, FALSE, FALSE, 0);
|
|
|
|
window->priv->show_side_pane_button = gtk_toggle_button_new ();
|
|
image = gtk_image_new_from_icon_name ("view-left-pane-symbolic", GTK_ICON_SIZE_INVALID);
|
|
gtk_image_set_pixel_size (GTK_IMAGE (image), 12);
|
|
gtk_container_add (GTK_CONTAINER (window->priv->show_side_pane_button), image);
|
|
gtk_box_pack_start (GTK_BOX (button_box), window->priv->show_side_pane_button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->panes_action_group, "ViewSidePane");
|
|
gtk_activatable_set_related_action (GTK_ACTIVATABLE (window->priv->show_side_pane_button), action);
|
|
gtk_widget_set_tooltip_text (window->priv->show_side_pane_button, gtk_action_get_tooltip (action));
|
|
|
|
window->priv->bottom_pane_button_revealer = gtk_revealer_new ();
|
|
gtk_revealer_set_transition_type (GTK_REVEALER (window->priv->bottom_pane_button_revealer),
|
|
GTK_REVEALER_TRANSITION_TYPE_SLIDE_RIGHT);
|
|
gtk_box_pack_start (GTK_BOX (button_box), window->priv->bottom_pane_button_revealer, FALSE, FALSE, 0);
|
|
|
|
window->priv->show_bottom_pane_button = gtk_toggle_button_new ();
|
|
image = gtk_image_new_from_icon_name ("view-bottom-pane-symbolic", GTK_ICON_SIZE_INVALID);
|
|
gtk_image_set_pixel_size (GTK_IMAGE (image), 12);
|
|
gtk_container_add (GTK_CONTAINER (window->priv->show_bottom_pane_button), image);
|
|
gtk_container_add (GTK_CONTAINER (window->priv->bottom_pane_button_revealer), window->priv->show_bottom_pane_button);
|
|
|
|
action = gtk_action_group_get_action (window->priv->panes_action_group, "ViewBottomPane");
|
|
gtk_activatable_set_related_action (GTK_ACTIVATABLE (window->priv->show_bottom_pane_button), action);
|
|
gtk_widget_set_tooltip_text (window->priv->show_bottom_pane_button, gtk_action_get_tooltip (action));
|
|
|
|
gtk_widget_show_all (button_box);
|
|
|
|
g_signal_connect_after(G_OBJECT (window->priv->statusbar), "show",
|
|
G_CALLBACK (statusbar_visibility_changed), window);
|
|
|
|
g_signal_connect_after(G_OBJECT (window->priv->statusbar), "hide",
|
|
G_CALLBACK (statusbar_visibility_changed), window);
|
|
|
|
set_statusbar_style (window, NULL);
|
|
|
|
gtk_box_pack_end (GTK_BOX (main_box), window->priv->searchbar, FALSE, FALSE, 0);
|
|
|
|
gtk_box_reorder_child (GTK_BOX (window->priv->statusbar), button_box, 0);
|
|
}
|
|
|
|
static XedWindow *
|
|
clone_window (XedWindow *origin)
|
|
{
|
|
XedWindow *window;
|
|
GdkScreen *screen;
|
|
XedApp *app;
|
|
gint panel_page;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
app = XED_APP (g_application_get_default ());
|
|
|
|
screen = gtk_window_get_screen (GTK_WINDOW (origin));
|
|
window = xed_app_create_window (app, screen);
|
|
|
|
gtk_window_set_default_size (GTK_WINDOW (window), origin->priv->width, origin->priv->height);
|
|
|
|
if ((origin->priv->window_state & GDK_WINDOW_STATE_MAXIMIZED) != 0)
|
|
{
|
|
gtk_window_maximize (GTK_WINDOW (window));
|
|
}
|
|
else
|
|
{
|
|
gtk_window_unmaximize (GTK_WINDOW (window));
|
|
}
|
|
|
|
if ((origin->priv->window_state & GDK_WINDOW_STATE_STICKY) != 0)
|
|
{
|
|
gtk_window_stick (GTK_WINDOW(window));
|
|
}
|
|
else
|
|
{
|
|
gtk_window_unstick (GTK_WINDOW(window));
|
|
}
|
|
|
|
/* set the panes size, the paned position will be set when
|
|
* they are mapped */
|
|
window->priv->side_panel_size = origin->priv->side_panel_size;
|
|
window->priv->bottom_panel_size = origin->priv->bottom_panel_size;
|
|
|
|
panel_page = _xed_panel_get_active_item_id (XED_PANEL(origin->priv->side_panel));
|
|
_xed_panel_set_active_item_by_id (XED_PANEL(window->priv->side_panel), panel_page);
|
|
|
|
panel_page = _xed_panel_get_active_item_id (XED_PANEL(origin->priv->bottom_panel));
|
|
_xed_panel_set_active_item_by_id (XED_PANEL(window->priv->bottom_panel), panel_page);
|
|
|
|
if (gtk_widget_get_visible (origin->priv->side_panel))
|
|
{
|
|
gtk_widget_show (window->priv->side_panel);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_hide (window->priv->side_panel);
|
|
}
|
|
|
|
if (gtk_widget_get_visible (origin->priv->bottom_panel))
|
|
{
|
|
gtk_widget_show (window->priv->bottom_panel);
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_hide (window->priv->bottom_panel);
|
|
}
|
|
|
|
set_statusbar_style (window, origin);
|
|
set_toolbar_style (window, origin);
|
|
|
|
return window;
|
|
}
|
|
|
|
static void
|
|
update_cursor_position_statusbar (GtkTextBuffer *buffer,
|
|
XedWindow *window)
|
|
{
|
|
gint row, col;
|
|
GtkTextIter iter;
|
|
GtkTextIter start;
|
|
guint tab_size;
|
|
XedView *view;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
if (buffer != GTK_TEXT_BUFFER(xed_window_get_active_document (window)))
|
|
{
|
|
return;
|
|
}
|
|
|
|
view = xed_window_get_active_view (window);
|
|
|
|
gtk_text_buffer_get_iter_at_mark (buffer, &iter, gtk_text_buffer_get_insert (buffer));
|
|
|
|
row = gtk_text_iter_get_line (&iter);
|
|
|
|
start = iter;
|
|
gtk_text_iter_set_line_offset (&start, 0);
|
|
col = 0;
|
|
|
|
tab_size = gtk_source_view_get_tab_width (GTK_SOURCE_VIEW(view));
|
|
|
|
while (!gtk_text_iter_equal (&start, &iter))
|
|
{
|
|
/* FIXME: Are we Unicode compliant here? */
|
|
if (gtk_text_iter_get_char (&start) == '\t')
|
|
{
|
|
col += (tab_size - (col % tab_size));
|
|
}
|
|
else
|
|
{
|
|
++col;
|
|
}
|
|
gtk_text_iter_forward_char (&start);
|
|
}
|
|
|
|
xed_statusbar_set_cursor_position (XED_STATUSBAR(window->priv->statusbar), row + 1, col + 1);
|
|
}
|
|
|
|
static void
|
|
update_overwrite_mode_statusbar (GtkTextView *view,
|
|
XedWindow *window)
|
|
{
|
|
if (view != GTK_TEXT_VIEW (xed_window_get_active_view (window)))
|
|
{
|
|
return;
|
|
}
|
|
|
|
/* Note that we have to use !gtk_text_view_get_overwrite since we
|
|
are in the in the signal handler of "toggle overwrite" that is
|
|
G_SIGNAL_RUN_LAST
|
|
*/
|
|
xed_statusbar_set_overwrite (XED_STATUSBAR(window->priv->statusbar), !gtk_text_view_get_overwrite (view));
|
|
}
|
|
|
|
#define MAX_TITLE_LENGTH 100
|
|
|
|
static void
|
|
set_title (XedWindow *window)
|
|
{
|
|
XedDocument *doc = NULL;
|
|
gchar *name;
|
|
gchar *dirname = NULL;
|
|
gchar *title = NULL;
|
|
gint len;
|
|
|
|
if (window->priv->active_tab == NULL)
|
|
{
|
|
xed_app_set_window_title (XED_APP (g_application_get_default ()), window, "Xed");
|
|
return;
|
|
}
|
|
|
|
doc = xed_tab_get_document (window->priv->active_tab);
|
|
g_return_if_fail(doc != NULL);
|
|
|
|
name = xed_document_get_short_name_for_display (doc);
|
|
|
|
len = g_utf8_strlen (name, -1);
|
|
|
|
/* if the name is awfully long, truncate it and be done with it,
|
|
* otherwise also show the directory (ellipsized if needed)
|
|
*/
|
|
if (len > MAX_TITLE_LENGTH)
|
|
{
|
|
gchar *tmp;
|
|
tmp = xed_utils_str_middle_truncate (name, MAX_TITLE_LENGTH);
|
|
g_free (name);
|
|
name = tmp;
|
|
}
|
|
else
|
|
{
|
|
GtkSourceFile *file = xed_document_get_file (doc);
|
|
GFile *location = gtk_source_file_get_location (file);
|
|
|
|
if (location != NULL)
|
|
{
|
|
gchar *str = xed_utils_location_get_dirname_for_display (location);
|
|
|
|
/* use the remaining space for the dir, but use a min of 20 chars
|
|
* so that we do not end up with a dirname like "(a...b)".
|
|
* This means that in the worst case when the filename is long 99
|
|
* we have a title long 99 + 20, but I think it's a rare enough
|
|
* case to be acceptable. It's justa darn title afterall :)
|
|
*/
|
|
dirname = xed_utils_str_middle_truncate (str, MAX(20, MAX_TITLE_LENGTH - len));
|
|
g_free (str);
|
|
}
|
|
}
|
|
|
|
if (gtk_text_buffer_get_modified (GTK_TEXT_BUFFER(doc)))
|
|
{
|
|
gchar *tmp_name;
|
|
tmp_name = g_strdup_printf ("*%s", name);
|
|
g_free (name);
|
|
name = tmp_name;
|
|
}
|
|
|
|
if (xed_document_get_readonly (doc))
|
|
{
|
|
if (dirname != NULL)
|
|
{
|
|
title = g_strdup_printf ("%s [%s] (%s)", name, _("Read-Only"), dirname);
|
|
}
|
|
else
|
|
{
|
|
title = g_strdup_printf ("%s [%s]", name, _("Read-Only"));
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (dirname != NULL)
|
|
{
|
|
title = g_strdup_printf ("%s (%s)", name, dirname);
|
|
}
|
|
else
|
|
{
|
|
title = g_strdup_printf ("%s", name);
|
|
}
|
|
}
|
|
|
|
xed_app_set_window_title (XED_APP (g_application_get_default ()), window, title);
|
|
|
|
g_free (dirname);
|
|
g_free (name);
|
|
g_free (title);
|
|
}
|
|
|
|
#undef MAX_TITLE_LENGTH
|
|
|
|
static void
|
|
spaces_instead_of_tabs_changed (GObject *object,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
XedView *view = XED_VIEW(object);
|
|
gboolean active = gtk_source_view_get_insert_spaces_instead_of_tabs (GTK_SOURCE_VIEW(view));
|
|
GList *children = gtk_container_get_children (GTK_CONTAINER (window->priv->tab_width_menu));
|
|
GtkCheckMenuItem *item;
|
|
item = GTK_CHECK_MENU_ITEM(g_list_last (children)->data);
|
|
gtk_check_menu_item_set_active (item, active);
|
|
g_list_free (children);
|
|
}
|
|
|
|
static void
|
|
tab_width_changed (GObject *object,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
GList *items;
|
|
GList *item;
|
|
guint new_tab_width;
|
|
gboolean use_spaces;
|
|
gboolean found = FALSE;
|
|
|
|
items = gtk_container_get_children (GTK_CONTAINER (window->priv->tab_width_menu));
|
|
|
|
new_tab_width = gtk_source_view_get_tab_width (GTK_SOURCE_VIEW (object));
|
|
use_spaces = gtk_source_view_get_insert_spaces_instead_of_tabs (GTK_SOURCE_VIEW (object));
|
|
|
|
for (item = items; item; item = item->next)
|
|
{
|
|
guint tab_width = GPOINTER_TO_INT(g_object_get_data (G_OBJECT (item->data), TAB_WIDTH_DATA));
|
|
|
|
if (tab_width == new_tab_width)
|
|
{
|
|
window->priv->tab_width_item = item->data;
|
|
if (gtk_widget_get_realized (window->priv->tab_width_menu))
|
|
{
|
|
gtk_menu_shell_select_item (GTK_MENU_SHELL (window->priv->tab_width_menu), GTK_WIDGET (item->data));
|
|
}
|
|
found = TRUE;
|
|
}
|
|
|
|
if (GTK_IS_SEPARATOR_MENU_ITEM(item->next->data))
|
|
{
|
|
if (!found)
|
|
{
|
|
/* Set for the last item the custom thing */
|
|
gchar *text;
|
|
|
|
text = g_strdup_printf ("%u", new_tab_width);
|
|
gtk_menu_item_set_label (GTK_MENU_ITEM(item->data), text);
|
|
|
|
window->priv->tab_width_item = item->data;
|
|
if (gtk_widget_get_realized (window->priv->tab_width_menu))
|
|
{
|
|
gtk_menu_shell_select_item (GTK_MENU_SHELL (window->priv->tab_width_menu), GTK_WIDGET (item->data));
|
|
}
|
|
gtk_widget_show (GTK_WIDGET(item->data));
|
|
}
|
|
else
|
|
{
|
|
gtk_widget_hide (GTK_WIDGET(item->data));
|
|
}
|
|
|
|
break;
|
|
}
|
|
}
|
|
|
|
set_tab_spaces_label (XED_VIEW (object), window, use_spaces);
|
|
|
|
g_list_free (items);
|
|
}
|
|
|
|
static void
|
|
language_changed (GObject *object,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
GtkSourceLanguage *new_language;
|
|
const gchar *label;
|
|
GtkAction *action;
|
|
|
|
new_language = gtk_source_buffer_get_language (GTK_SOURCE_BUFFER (object));
|
|
|
|
if (new_language)
|
|
{
|
|
label = gtk_source_language_get_name (new_language);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditToggleComment");
|
|
gtk_action_set_sensitive (action,
|
|
gtk_source_language_get_metadata (new_language, "line-comment-start") != NULL);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditToggleCommentBlock");
|
|
gtk_action_set_sensitive (action,
|
|
(gtk_source_language_get_metadata (new_language, "block-comment-start") != NULL
|
|
&& gtk_source_language_get_metadata (new_language, "block-comment-end") != NULL));
|
|
}
|
|
else
|
|
{
|
|
label = _("Plain Text");
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditToggleComment");
|
|
gtk_action_set_sensitive (action, FALSE);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditToggleCommentBlock");
|
|
gtk_action_set_sensitive (action, FALSE);
|
|
}
|
|
|
|
xed_status_menu_button_set_label (XED_STATUS_MENU_BUTTON (window->priv->language_button), label);
|
|
}
|
|
|
|
static void
|
|
word_wrap_changed (GObject *object,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
GtkWrapMode wrap_mode;
|
|
GtkAction *action;
|
|
gboolean wrap_enabled;
|
|
|
|
view = XED_VIEW (object);
|
|
wrap_mode = gtk_text_view_get_wrap_mode (GTK_TEXT_VIEW (view));
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewWordWrap");
|
|
|
|
if (wrap_mode == GTK_WRAP_NONE)
|
|
{
|
|
wrap_enabled = FALSE;
|
|
}
|
|
else
|
|
{
|
|
wrap_enabled = TRUE;
|
|
}
|
|
|
|
g_signal_handlers_block_by_func (action, G_CALLBACK (_xed_cmd_view_toggle_word_wrap), window);
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action), wrap_enabled);
|
|
g_signal_handlers_unblock_by_func (action, G_CALLBACK (_xed_cmd_view_toggle_word_wrap), window);
|
|
}
|
|
|
|
static void
|
|
show_overview_map_changed (GObject *object,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
GtkFrame *map_frame;
|
|
GtkAction *action;
|
|
gboolean overveiw_map_visible = FALSE;
|
|
|
|
map_frame = GTK_FRAME (object);
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewOverviewMap");
|
|
|
|
if (gtk_widget_get_visible (GTK_WIDGET (map_frame)))
|
|
{
|
|
overveiw_map_visible = TRUE;
|
|
}
|
|
|
|
g_signal_handlers_block_by_func (action, G_CALLBACK (_xed_cmd_view_toggle_overview_map), window);
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action), overveiw_map_visible);
|
|
g_signal_handlers_unblock_by_func (action, G_CALLBACK (_xed_cmd_view_toggle_overview_map), window);
|
|
}
|
|
|
|
static void
|
|
notebook_switch_page (GtkNotebook *book,
|
|
GtkWidget *pg,
|
|
gint page_num,
|
|
XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
GtkFrame *map_frame;
|
|
XedTab *tab;
|
|
GtkAction *action;
|
|
gchar *action_name;
|
|
|
|
/* CHECK: I don't know why but it seems notebook_switch_page is called
|
|
two times every time the user change the active tab */
|
|
|
|
tab = XED_TAB (gtk_notebook_get_nth_page (book, page_num));
|
|
if (tab == window->priv->active_tab)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (window->priv->active_tab)
|
|
{
|
|
if (window->priv->tab_width_id)
|
|
{
|
|
g_signal_handler_disconnect (xed_tab_get_view (window->priv->active_tab), window->priv->tab_width_id);
|
|
window->priv->tab_width_id = 0;
|
|
}
|
|
|
|
if (window->priv->spaces_instead_of_tabs_id)
|
|
{
|
|
g_signal_handler_disconnect (xed_tab_get_view (window->priv->active_tab), window->priv->spaces_instead_of_tabs_id);
|
|
window->priv->spaces_instead_of_tabs_id = 0;
|
|
}
|
|
|
|
if (window->priv->use_word_wrap_id)
|
|
{
|
|
g_signal_handler_disconnect (xed_tab_get_view (window->priv->active_tab), window->priv->use_word_wrap_id);
|
|
window->priv->use_word_wrap_id = 0;
|
|
}
|
|
|
|
if (window->priv->show_overview_map_id)
|
|
{
|
|
g_signal_handler_disconnect (xed_view_frame_get_map_frame (XED_VIEW_FRAME (_xed_tab_get_view_frame (window->priv->active_tab))), window->priv->show_overview_map_id);
|
|
window->priv->show_overview_map_id = 0;
|
|
}
|
|
}
|
|
|
|
/* set the active tab */
|
|
window->priv->active_tab = tab;
|
|
|
|
set_title (window);
|
|
set_sensitivity_according_to_tab (window, tab);
|
|
|
|
/* activate the right item in the documents menu */
|
|
action_name = g_strdup_printf ("Tab_%d", page_num);
|
|
action = gtk_action_group_get_action (window->priv->documents_list_action_group, action_name);
|
|
|
|
/* sometimes the action doesn't exist yet, and the proper action
|
|
* is set active during the documents list menu creation
|
|
* CHECK: would it be nicer if active_tab was a property and we monitored the notify signal?
|
|
*/
|
|
if (action != NULL)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action), TRUE);
|
|
}
|
|
|
|
g_free (action_name);
|
|
|
|
view = xed_tab_get_view (tab);
|
|
map_frame = xed_view_frame_get_map_frame (XED_VIEW_FRAME (_xed_tab_get_view_frame (tab)));
|
|
|
|
/* sync the statusbar */
|
|
update_cursor_position_statusbar (GTK_TEXT_BUFFER (xed_tab_get_document (tab)), window);
|
|
xed_statusbar_set_overwrite (XED_STATUSBAR (window->priv->statusbar),
|
|
gtk_text_view_get_overwrite (GTK_TEXT_VIEW(view)));
|
|
|
|
gtk_widget_show (window->priv->tab_width_button);
|
|
gtk_widget_show (window->priv->language_button);
|
|
|
|
window->priv->tab_width_id = g_signal_connect(view, "notify::tab-width", G_CALLBACK (tab_width_changed), window);
|
|
|
|
window->priv->spaces_instead_of_tabs_id = g_signal_connect(view, "notify::insert-spaces-instead-of-tabs",
|
|
G_CALLBACK (spaces_instead_of_tabs_changed), window);
|
|
|
|
window->priv->language_changed_id = g_signal_connect(xed_tab_get_document (tab), "notify::language",
|
|
G_CALLBACK (language_changed), window);
|
|
|
|
window->priv->use_word_wrap_id = g_signal_connect (view, "notify::wrap-mode",
|
|
G_CALLBACK (word_wrap_changed), window);
|
|
|
|
window->priv->show_overview_map_id = g_signal_connect (map_frame, "notify::visible",
|
|
G_CALLBACK (show_overview_map_changed), window);
|
|
|
|
/* call it for the first time */
|
|
tab_width_changed (G_OBJECT (view), NULL, window);
|
|
spaces_instead_of_tabs_changed (G_OBJECT (view), NULL, window);
|
|
language_changed (G_OBJECT (xed_tab_get_document (tab)), NULL, window);
|
|
word_wrap_changed (G_OBJECT (view), NULL, window);
|
|
show_overview_map_changed (G_OBJECT (map_frame), NULL, window);
|
|
|
|
g_signal_emit (G_OBJECT (window), signals[ACTIVE_TAB_CHANGED], 0, window->priv->active_tab);
|
|
}
|
|
|
|
static void
|
|
set_sensitivity_according_to_window_state (XedWindow *window)
|
|
{
|
|
GtkAction *action;
|
|
|
|
/* We disable File->Quit/SaveAll/CloseAll while printing to avoid to have two
|
|
operations (save and print/print preview) that uses the message area at
|
|
the same time (may be we can remove this limitation in the future) */
|
|
/* We disable File->Quit/CloseAll if state is saving since saving cannot be
|
|
cancelled (may be we can remove this limitation in the future) */
|
|
gtk_action_group_set_sensitive (
|
|
window->priv->quit_action_group,
|
|
!(window->priv->state & XED_WINDOW_STATE_SAVING)
|
|
&& !(window->priv->state & XED_WINDOW_STATE_PRINTING));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileCloseAll");
|
|
gtk_action_set_sensitive (
|
|
action,
|
|
!(window->priv->state & XED_WINDOW_STATE_SAVING)
|
|
&& !(window->priv->state & XED_WINDOW_STATE_PRINTING));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileSaveAll");
|
|
gtk_action_set_sensitive (action, !(window->priv->state & XED_WINDOW_STATE_PRINTING));
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "FileNew");
|
|
gtk_action_set_sensitive (action, !(window->priv->state & XED_WINDOW_STATE_SAVING_SESSION));
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "FileOpen");
|
|
gtk_action_set_sensitive (action, !(window->priv->state & XED_WINDOW_STATE_SAVING_SESSION));
|
|
|
|
gtk_action_group_set_sensitive (window->priv->recents_action_group,
|
|
!(window->priv->state & XED_WINDOW_STATE_SAVING_SESSION));
|
|
|
|
xed_notebook_set_close_buttons_sensitive (XED_NOTEBOOK(window->priv->notebook),
|
|
!(window->priv->state & XED_WINDOW_STATE_SAVING_SESSION));
|
|
|
|
xed_notebook_set_tab_drag_and_drop_enabled (XED_NOTEBOOK(window->priv->notebook),
|
|
!(window->priv->state & XED_WINDOW_STATE_SAVING_SESSION));
|
|
|
|
if ((window->priv->state & XED_WINDOW_STATE_SAVING_SESSION) != 0)
|
|
{
|
|
/* TODO: If we really care, Find could be active
|
|
* when in SAVING_SESSION state */
|
|
if (gtk_action_group_get_sensitive (window->priv->action_group))
|
|
{
|
|
gtk_action_group_set_sensitive (window->priv->action_group, FALSE);
|
|
}
|
|
if (gtk_action_group_get_sensitive (window->priv->quit_action_group))
|
|
{
|
|
gtk_action_group_set_sensitive (window->priv->quit_action_group, FALSE);
|
|
}
|
|
if (gtk_action_group_get_sensitive (window->priv->close_action_group))
|
|
{
|
|
gtk_action_group_set_sensitive (window->priv->close_action_group, FALSE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if (!gtk_action_group_get_sensitive (window->priv->action_group))
|
|
{
|
|
gtk_action_group_set_sensitive (window->priv->action_group, window->priv->num_tabs > 0);
|
|
}
|
|
if (!gtk_action_group_get_sensitive (window->priv->quit_action_group))
|
|
{
|
|
gtk_action_group_set_sensitive (window->priv->quit_action_group, window->priv->num_tabs > 0);
|
|
}
|
|
if (!gtk_action_group_get_sensitive (window->priv->close_action_group))
|
|
{
|
|
gtk_action_group_set_sensitive (window->priv->close_action_group, window->priv->num_tabs > 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
analyze_tab_state (XedTab *tab,
|
|
XedWindow *window)
|
|
{
|
|
XedTabState ts;
|
|
|
|
ts = xed_tab_get_state (tab);
|
|
|
|
switch (ts)
|
|
{
|
|
case XED_TAB_STATE_LOADING:
|
|
case XED_TAB_STATE_REVERTING:
|
|
window->priv->state |= XED_WINDOW_STATE_LOADING;
|
|
break;
|
|
case XED_TAB_STATE_SAVING:
|
|
window->priv->state |= XED_WINDOW_STATE_SAVING;
|
|
break;
|
|
case XED_TAB_STATE_PRINTING:
|
|
case XED_TAB_STATE_PRINT_PREVIEWING:
|
|
window->priv->state |= XED_WINDOW_STATE_PRINTING;
|
|
break;
|
|
case XED_TAB_STATE_LOADING_ERROR:
|
|
case XED_TAB_STATE_REVERTING_ERROR:
|
|
case XED_TAB_STATE_SAVING_ERROR:
|
|
case XED_TAB_STATE_GENERIC_ERROR:
|
|
window->priv->state |= XED_WINDOW_STATE_ERROR;
|
|
++window->priv->num_tabs_with_error;
|
|
break;
|
|
default:
|
|
/* NOP */
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
update_window_state (XedWindow *window)
|
|
{
|
|
XedWindowState old_ws;
|
|
gint old_num_of_errors;
|
|
|
|
xed_debug_message (DEBUG_WINDOW, "Old state: %x", window->priv->state);
|
|
|
|
old_ws = window->priv->state;
|
|
old_num_of_errors = window->priv->num_tabs_with_error;
|
|
|
|
window->priv->state = old_ws & XED_WINDOW_STATE_SAVING_SESSION;
|
|
|
|
window->priv->num_tabs_with_error = 0;
|
|
|
|
gtk_container_foreach (GTK_CONTAINER(window->priv->notebook), (GtkCallback) analyze_tab_state, window);
|
|
|
|
xed_debug_message (DEBUG_WINDOW, "New state: %x", window->priv->state);
|
|
|
|
if (old_ws != window->priv->state)
|
|
{
|
|
set_sensitivity_according_to_window_state (window);
|
|
xed_statusbar_set_window_state (XED_STATUSBAR(window->priv->statusbar), window->priv->state,
|
|
window->priv->num_tabs_with_error);
|
|
g_object_notify (G_OBJECT(window), "state");
|
|
}
|
|
else if (old_num_of_errors != window->priv->num_tabs_with_error)
|
|
{
|
|
xed_statusbar_set_window_state (XED_STATUSBAR(window->priv->statusbar), window->priv->state,
|
|
window->priv->num_tabs_with_error);
|
|
}
|
|
}
|
|
|
|
static void
|
|
update_can_close (XedWindow *window)
|
|
{
|
|
XedWindowPrivate *priv = window->priv;
|
|
GList *tabs;
|
|
GList *l;
|
|
gboolean can_close = TRUE;
|
|
|
|
tabs = xed_notebook_get_all_tabs (XED_NOTEBOOK (priv->notebook));
|
|
|
|
for (l = tabs; l != NULL; l = g_list_next (l))
|
|
{
|
|
XedTab *tab = l->data;
|
|
|
|
if (!_xed_tab_get_can_close (tab))
|
|
{
|
|
can_close = FALSE;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (can_close && (priv->inhibition_cookie != 0))
|
|
{
|
|
gtk_application_uninhibit (GTK_APPLICATION (g_application_get_default ()), priv->inhibition_cookie);
|
|
priv->inhibition_cookie = 0;
|
|
}
|
|
else if (!can_close && (priv->inhibition_cookie == 0))
|
|
{
|
|
priv->inhibition_cookie = gtk_application_inhibit (GTK_APPLICATION (g_application_get_default ()),
|
|
GTK_WINDOW (window),
|
|
GTK_APPLICATION_INHIBIT_LOGOUT,
|
|
_("There are unsaved documents"));
|
|
}
|
|
|
|
g_list_free (tabs);
|
|
}
|
|
|
|
static void
|
|
sync_state (XedTab *tab,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
xed_debug (DEBUG_WINDOW);
|
|
update_window_state (window);
|
|
if (tab != window->priv->active_tab)
|
|
{
|
|
return;
|
|
}
|
|
set_sensitivity_according_to_tab (window, tab);
|
|
g_signal_emit (G_OBJECT(window), signals[ACTIVE_TAB_STATE_CHANGED], 0);
|
|
}
|
|
|
|
static void
|
|
sync_name (XedTab *tab,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
GtkAction *action;
|
|
gchar *action_name;
|
|
gchar *tab_name;
|
|
gchar *escaped_name;
|
|
gchar *tip;
|
|
gint n;
|
|
XedDocument *doc;
|
|
|
|
if (tab == window->priv->active_tab)
|
|
{
|
|
set_title (window);
|
|
doc = xed_tab_get_document (tab);
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileRevert");
|
|
gtk_action_set_sensitive (action, !xed_document_is_untitled (doc));
|
|
}
|
|
|
|
/* sync the item in the documents list menu */
|
|
n = gtk_notebook_page_num (GTK_NOTEBOOK(window->priv->notebook), GTK_WIDGET(tab));
|
|
action_name = g_strdup_printf ("Tab_%d", n);
|
|
action = gtk_action_group_get_action (window->priv->documents_list_action_group, action_name);
|
|
g_free (action_name);
|
|
g_return_if_fail(action != NULL);
|
|
|
|
tab_name = _xed_tab_get_name (tab);
|
|
escaped_name = xed_utils_escape_underscores (tab_name, -1);
|
|
tip = get_menu_tip_for_tab (tab);
|
|
|
|
g_object_set (action, "label", escaped_name, NULL);
|
|
g_object_set (action, "tooltip", tip, NULL);
|
|
|
|
g_free (tab_name);
|
|
g_free (escaped_name);
|
|
g_free (tip);
|
|
|
|
peas_extension_set_call (window->priv->extensions, "update_state");
|
|
}
|
|
|
|
static void
|
|
sync_can_close (XedTab *tab,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
update_can_close (window);
|
|
}
|
|
|
|
static XedWindow *
|
|
get_drop_window (GtkWidget *widget)
|
|
{
|
|
GtkWidget *target_window;
|
|
target_window = gtk_widget_get_toplevel (widget);
|
|
g_return_val_if_fail(XED_IS_WINDOW (target_window), NULL);
|
|
|
|
if ((XED_WINDOW(target_window)->priv->state & XED_WINDOW_STATE_SAVING_SESSION) != 0)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
return XED_WINDOW(target_window);
|
|
}
|
|
|
|
static void
|
|
load_uris_from_drop (XedWindow *window,
|
|
gchar **uri_list)
|
|
{
|
|
GSList *locations = NULL;
|
|
gint i;
|
|
GSList *loaded;
|
|
|
|
if (uri_list == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
for (i = 0; uri_list[i] != NULL; ++i)
|
|
{
|
|
locations = g_slist_prepend (locations, g_file_new_for_uri (uri_list[i]));
|
|
}
|
|
|
|
locations = g_slist_reverse (locations);
|
|
loaded = xed_commands_load_locations (window, locations, NULL, 0);
|
|
|
|
g_slist_free (loaded);
|
|
|
|
g_slist_foreach (locations, (GFunc) g_object_unref, NULL);
|
|
g_slist_free (locations);
|
|
}
|
|
|
|
/* Handle drops on the XedWindow */
|
|
static void
|
|
drag_data_received_cb (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
GtkSelectionData *selection_data,
|
|
guint info,
|
|
guint timestamp,
|
|
gpointer data)
|
|
{
|
|
XedWindow *window;
|
|
gchar **uri_list;
|
|
|
|
window = get_drop_window (widget);
|
|
if (window == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
if (info == TARGET_URI_LIST)
|
|
{
|
|
uri_list = xed_utils_drop_get_uris (selection_data);
|
|
load_uris_from_drop (window, uri_list);
|
|
g_strfreev (uri_list);
|
|
}
|
|
}
|
|
|
|
/* Handle drops on the XedView */
|
|
static void
|
|
drop_uris_cb (GtkWidget *widget,
|
|
gchar **uri_list)
|
|
{
|
|
XedWindow *window;
|
|
window = get_drop_window (widget);
|
|
if (window == NULL)
|
|
{
|
|
return;
|
|
}
|
|
load_uris_from_drop (window, uri_list);
|
|
}
|
|
|
|
static gboolean
|
|
on_fullscreen_controls_enter_notify_event (GtkWidget *widget,
|
|
GdkEventCrossing *event,
|
|
XedWindow *window)
|
|
{
|
|
gtk_revealer_set_reveal_child (GTK_REVEALER (window->priv->fullscreen_controls), TRUE);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
on_fullscreen_controls_leave_notify_event (GtkWidget *widget,
|
|
GdkEventCrossing *event,
|
|
XedWindow *window)
|
|
{
|
|
GdkWindow *gdk_window;
|
|
GdkDevice *device;
|
|
GdkSeat *seat;
|
|
gint x, y;
|
|
|
|
seat = gdk_display_get_default_seat (gdk_display_get_default ());
|
|
device = gdk_seat_get_pointer (seat);
|
|
gdk_window = gtk_widget_get_parent_window (widget);
|
|
|
|
gdk_window_get_device_position (gdk_window, device, &x, &y, NULL);
|
|
|
|
// sometimes the leave event is erroneously triggered when y = 0
|
|
if (y != 0)
|
|
{
|
|
gtk_revealer_set_reveal_child (GTK_REVEALER (window->priv->fullscreen_controls), FALSE);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
fullscreen_controls_build (XedWindow *window)
|
|
{
|
|
XedWindowPrivate *priv = window->priv;
|
|
GtkAction *action;
|
|
GtkWidget *box;
|
|
GtkWidget *toolbar;
|
|
GtkToolItem *toolitem;
|
|
GtkWidget *toolbox;
|
|
GtkWidget *fullscreen_btn;
|
|
GtkWidget *separator;
|
|
GtkWidget *button;
|
|
|
|
if (priv->fullscreen_controls != NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
priv->fullscreen_controls = gtk_revealer_new ();
|
|
gtk_widget_set_valign (priv->fullscreen_controls, GTK_ALIGN_START);
|
|
gtk_container_add (GTK_CONTAINER (priv->fullscreen_eventbox), priv->fullscreen_controls);
|
|
|
|
toolbar = gtk_toolbar_new ();
|
|
toolitem = gtk_tool_item_new ();
|
|
gtk_tool_item_set_expand (toolitem, TRUE);
|
|
gtk_toolbar_insert (GTK_TOOLBAR (toolbar), toolitem, 0);
|
|
|
|
toolbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (toolbar), "primary-toolbar");
|
|
gtk_container_add (GTK_CONTAINER (toolitem), toolbox);
|
|
gtk_container_add (GTK_CONTAINER (priv->fullscreen_controls), toolbar);
|
|
|
|
box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
|
|
gtk_widget_set_vexpand (box, FALSE);
|
|
gtk_box_pack_start (GTK_BOX (toolbox), box, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "FileNew");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "FileOpen");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "FileSave");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
separator = gtk_separator_new (GTK_ORIENTATION_VERTICAL);
|
|
gtk_box_pack_start (GTK_BOX (box), separator, FALSE, FALSE, 6);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditUndo");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditRedo");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
separator = gtk_separator_new (GTK_ORIENTATION_VERTICAL);
|
|
gtk_box_pack_start (GTK_BOX (box), separator, FALSE, FALSE, 6);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCut");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCopy");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditPaste");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
separator = gtk_separator_new (GTK_ORIENTATION_VERTICAL);
|
|
gtk_box_pack_start (GTK_BOX (box), separator, FALSE, FALSE, 6);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchFind");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchReplace");
|
|
button = create_toolbar_button (action);
|
|
gtk_box_pack_start (GTK_BOX (box), button, FALSE, FALSE, 0);
|
|
|
|
action = gtk_action_group_get_action (priv->always_sensitive_action_group, "LeaveFullscreen");
|
|
g_object_set (action, "is-important", TRUE, NULL);
|
|
fullscreen_btn = create_toolbar_button (action);
|
|
gtk_box_pack_end (GTK_BOX (box), fullscreen_btn, FALSE, FALSE, 0);
|
|
|
|
gtk_widget_show_all (toolbox);
|
|
|
|
g_signal_connect (priv->fullscreen_eventbox, "enter-notify-event",
|
|
G_CALLBACK (on_fullscreen_controls_enter_notify_event), window);
|
|
|
|
g_signal_connect (priv->fullscreen_eventbox, "leave-notify-event",
|
|
G_CALLBACK (on_fullscreen_controls_leave_notify_event), window);
|
|
|
|
gtk_widget_set_size_request (priv->fullscreen_eventbox, -1, 1);
|
|
}
|
|
|
|
static void
|
|
search_text_notify_cb (XedDocument *doc,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
gboolean sensitive;
|
|
GtkAction *action;
|
|
|
|
if (doc != xed_window_get_active_document (window))
|
|
{
|
|
return;
|
|
}
|
|
|
|
sensitive = TRUE;
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchFindNext");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "SearchFindPrevious");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
}
|
|
|
|
static void
|
|
can_undo (XedDocument *doc,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
GtkAction *action;
|
|
gboolean sensitive;
|
|
|
|
sensitive = gtk_source_buffer_can_undo (GTK_SOURCE_BUFFER(doc));
|
|
|
|
if (doc != xed_window_get_active_document (window))
|
|
{
|
|
return;
|
|
}
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditUndo");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
}
|
|
|
|
static void
|
|
can_redo (XedDocument *doc,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
GtkAction *action;
|
|
gboolean sensitive;
|
|
|
|
sensitive = gtk_source_buffer_can_redo (GTK_SOURCE_BUFFER(doc));
|
|
|
|
if (doc != xed_window_get_active_document (window))
|
|
{
|
|
return;
|
|
}
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditRedo");
|
|
gtk_action_set_sensitive (action, sensitive);
|
|
}
|
|
|
|
static void
|
|
selection_changed (XedDocument *doc,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
XedTab *tab;
|
|
XedView *view;
|
|
GtkAction *action;
|
|
XedTabState state;
|
|
gboolean state_normal;
|
|
gboolean editable;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
if (doc != xed_window_get_active_document (window))
|
|
{
|
|
return;
|
|
}
|
|
|
|
tab = xed_tab_get_from_document (doc);
|
|
state = xed_tab_get_state (tab);
|
|
state_normal = (state == XED_TAB_STATE_NORMAL);
|
|
|
|
view = xed_tab_get_view (tab);
|
|
editable = gtk_text_view_get_editable (GTK_TEXT_VIEW(view));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCut");
|
|
gtk_action_set_sensitive (action,
|
|
state_normal && editable && gtk_text_buffer_get_has_selection (GTK_TEXT_BUFFER(doc)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditCopy");
|
|
gtk_action_set_sensitive (action,
|
|
(state_normal || state == XED_TAB_STATE_EXTERNALLY_MODIFIED_NOTIFICATION)
|
|
&& gtk_text_buffer_get_has_selection (GTK_TEXT_BUFFER(doc)));
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "EditDelete");
|
|
gtk_action_set_sensitive (action,
|
|
state_normal && editable && gtk_text_buffer_get_has_selection (GTK_TEXT_BUFFER(doc)));
|
|
|
|
peas_extension_set_call (window->priv->extensions, "update_state");
|
|
}
|
|
|
|
static void
|
|
sync_languages_menu (XedDocument *doc,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
peas_extension_set_call (window->priv->extensions, "update_state");
|
|
}
|
|
|
|
static void
|
|
readonly_changed (XedDocument *doc,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
set_sensitivity_according_to_tab (window, window->priv->active_tab);
|
|
sync_name (window->priv->active_tab, NULL, window);
|
|
peas_extension_set_call (window->priv->extensions, "update_state");
|
|
}
|
|
|
|
static void
|
|
editable_changed (XedView *view,
|
|
GParamSpec *arg1,
|
|
XedWindow *window)
|
|
{
|
|
peas_extension_set_call (window->priv->extensions, "update_state");
|
|
}
|
|
|
|
static void
|
|
update_sensitivity_according_to_open_tabs (XedWindow *window)
|
|
{
|
|
GtkAction *action;
|
|
|
|
/* Set sensitivity */
|
|
gtk_action_group_set_sensitive (window->priv->action_group, window->priv->num_tabs != 0);
|
|
|
|
action = gtk_action_group_get_action (window->priv->action_group, "DocumentsMoveToNewWindow");
|
|
gtk_action_set_sensitive (action, window->priv->num_tabs > 1);
|
|
gtk_action_group_set_sensitive (window->priv->close_action_group, window->priv->num_tabs != 0);
|
|
}
|
|
|
|
static void
|
|
notebook_tab_added (XedNotebook *notebook,
|
|
XedTab *tab,
|
|
XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
XedDocument *doc;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
g_return_if_fail((window->priv->state & XED_WINDOW_STATE_SAVING_SESSION) == 0);
|
|
|
|
++window->priv->num_tabs;
|
|
|
|
update_sensitivity_according_to_open_tabs (window);
|
|
|
|
view = xed_tab_get_view (tab);
|
|
doc = xed_tab_get_document (tab);
|
|
|
|
/* IMPORTANT: remember to disconnect the signal in notebook_tab_removed
|
|
* if a new signal is connected here */
|
|
|
|
g_signal_connect (tab, "notify::name", G_CALLBACK (sync_name), window);
|
|
g_signal_connect (tab, "notify::state", G_CALLBACK (sync_state), window);
|
|
g_signal_connect (tab, "notify::can-close", G_CALLBACK (sync_can_close), window);
|
|
|
|
g_signal_connect (doc, "cursor-moved", G_CALLBACK (update_cursor_position_statusbar), window);
|
|
g_signal_connect (doc, "notify::search-text", G_CALLBACK (search_text_notify_cb), window);
|
|
g_signal_connect (doc, "notify::can-undo", G_CALLBACK (can_undo), window);
|
|
g_signal_connect (doc, "notify::can-redo", G_CALLBACK (can_redo), window);
|
|
g_signal_connect (doc, "notify::has-selection", G_CALLBACK (selection_changed), window);
|
|
g_signal_connect (doc, "notify::language", G_CALLBACK (sync_languages_menu), window);
|
|
g_signal_connect (doc, "notify::read-only", G_CALLBACK (readonly_changed), window);
|
|
g_signal_connect (view, "toggle_overwrite", G_CALLBACK (update_overwrite_mode_statusbar), window);
|
|
g_signal_connect (view, "notify::editable", G_CALLBACK (editable_changed), window);
|
|
|
|
update_documents_list_menu (window);
|
|
|
|
g_signal_connect(view, "drop_uris", G_CALLBACK (drop_uris_cb), NULL);
|
|
|
|
update_window_state (window);
|
|
update_can_close (window);
|
|
|
|
g_signal_emit (G_OBJECT(window), signals[TAB_ADDED], 0, tab);
|
|
}
|
|
|
|
static void
|
|
notebook_tab_removed (XedNotebook *notebook,
|
|
XedTab *tab,
|
|
XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
XedViewFrame *frame;
|
|
XedDocument *doc;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
g_return_if_fail((window->priv->state & XED_WINDOW_STATE_SAVING_SESSION) == 0);
|
|
|
|
--window->priv->num_tabs;
|
|
|
|
view = xed_tab_get_view (tab);
|
|
frame = XED_VIEW_FRAME (_xed_tab_get_view_frame (tab));
|
|
doc = xed_tab_get_document (tab);
|
|
|
|
g_signal_handlers_disconnect_by_func (tab, G_CALLBACK (sync_name), window);
|
|
g_signal_handlers_disconnect_by_func (tab, G_CALLBACK (sync_state), window);
|
|
g_signal_handlers_disconnect_by_func (tab, G_CALLBACK (sync_can_close), window);
|
|
g_signal_handlers_disconnect_by_func (doc, G_CALLBACK (update_cursor_position_statusbar), window);
|
|
g_signal_handlers_disconnect_by_func (doc, G_CALLBACK (search_text_notify_cb), window);
|
|
g_signal_handlers_disconnect_by_func (doc, G_CALLBACK (can_undo), window);
|
|
g_signal_handlers_disconnect_by_func (doc, G_CALLBACK (can_redo), window);
|
|
g_signal_handlers_disconnect_by_func (doc, G_CALLBACK (selection_changed), window);
|
|
g_signal_handlers_disconnect_by_func (doc, G_CALLBACK (sync_languages_menu), window);
|
|
g_signal_handlers_disconnect_by_func (doc, G_CALLBACK (readonly_changed), window);
|
|
g_signal_handlers_disconnect_by_func (view, G_CALLBACK (update_overwrite_mode_statusbar), window);
|
|
g_signal_handlers_disconnect_by_func (view, G_CALLBACK (editable_changed), window);
|
|
g_signal_handlers_disconnect_by_func (view, G_CALLBACK (drop_uris_cb), NULL);
|
|
|
|
if (window->priv->tab_width_id && tab == xed_window_get_active_tab (window))
|
|
{
|
|
g_signal_handler_disconnect (view, window->priv->tab_width_id);
|
|
window->priv->tab_width_id = 0;
|
|
}
|
|
|
|
if (window->priv->spaces_instead_of_tabs_id && tab == xed_window_get_active_tab (window))
|
|
{
|
|
g_signal_handler_disconnect (view, window->priv->spaces_instead_of_tabs_id);
|
|
window->priv->spaces_instead_of_tabs_id = 0;
|
|
}
|
|
|
|
if (window->priv->language_changed_id && tab == xed_window_get_active_tab (window))
|
|
{
|
|
g_signal_handler_disconnect (doc, window->priv->language_changed_id);
|
|
window->priv->language_changed_id = 0;
|
|
}
|
|
|
|
if (window->priv->use_word_wrap_id && tab == xed_window_get_active_tab (window))
|
|
{
|
|
g_signal_handler_disconnect (view, window->priv->use_word_wrap_id);
|
|
window->priv->use_word_wrap_id = 0;
|
|
}
|
|
|
|
if (window->priv->show_overview_map_id && tab == xed_window_get_active_tab (window))
|
|
{
|
|
g_signal_handler_disconnect (xed_view_frame_get_map_frame (frame), window->priv->show_overview_map_id);
|
|
window->priv->show_overview_map_id = 0;
|
|
}
|
|
|
|
g_return_if_fail(window->priv->num_tabs >= 0);
|
|
if (window->priv->num_tabs == 0)
|
|
{
|
|
window->priv->active_tab = NULL;
|
|
set_title (window);
|
|
|
|
/* Remove line and col info */
|
|
xed_statusbar_set_cursor_position (XED_STATUSBAR(window->priv->statusbar), -1, -1);
|
|
|
|
xed_statusbar_clear_overwrite (XED_STATUSBAR(window->priv->statusbar));
|
|
|
|
/* hide the combos */
|
|
gtk_widget_hide (window->priv->tab_width_button);
|
|
gtk_widget_hide (window->priv->language_button);
|
|
}
|
|
|
|
if (!window->priv->removing_tabs || window->priv->num_tabs == 0)
|
|
{
|
|
update_documents_list_menu (window);
|
|
update_next_prev_doc_sensitivity_per_window (window);
|
|
}
|
|
|
|
update_sensitivity_according_to_open_tabs (window);
|
|
|
|
if (window->priv->num_tabs == 0)
|
|
{
|
|
peas_extension_set_call (window->priv->extensions, "update_state");
|
|
}
|
|
|
|
update_window_state (window);
|
|
update_can_close (window);
|
|
|
|
g_signal_emit (G_OBJECT(window), signals[TAB_REMOVED], 0, tab);
|
|
}
|
|
|
|
static void
|
|
notebook_tabs_reordered (XedNotebook *notebook,
|
|
XedWindow *window)
|
|
{
|
|
update_documents_list_menu (window);
|
|
update_next_prev_doc_sensitivity_per_window (window);
|
|
g_signal_emit (G_OBJECT(window), signals[TABS_REORDERED], 0);
|
|
}
|
|
|
|
static void
|
|
notebook_tab_detached (XedNotebook *notebook,
|
|
XedTab *tab,
|
|
XedWindow *window)
|
|
{
|
|
XedWindow *new_window;
|
|
|
|
new_window = clone_window (window);
|
|
|
|
xed_notebook_move_tab (notebook, XED_NOTEBOOK(_xed_window_get_notebook (new_window)), tab, 0);
|
|
|
|
gtk_window_set_position (GTK_WINDOW(new_window), GTK_WIN_POS_MOUSE);
|
|
|
|
gtk_widget_show (GTK_WIDGET(new_window));
|
|
}
|
|
|
|
static void
|
|
notebook_tab_close_request (XedNotebook *notebook,
|
|
XedTab *tab,
|
|
GtkWindow *window)
|
|
{
|
|
/* Note: we are destroying the tab before the default handler
|
|
* seems to be ok, but we need to keep an eye on this. */
|
|
_xed_cmd_file_close_tab (tab, XED_WINDOW(window));
|
|
}
|
|
|
|
static gboolean
|
|
show_notebook_popup_menu (GtkNotebook *notebook,
|
|
XedWindow *window,
|
|
GdkEventButton *event)
|
|
{
|
|
GtkWidget *menu;
|
|
// GtkAction *action;
|
|
|
|
menu = gtk_ui_manager_get_widget (window->priv->manager, "/NotebookPopup");
|
|
g_return_val_if_fail(menu != NULL, FALSE);
|
|
|
|
// CHECK do we need this?
|
|
#if 0
|
|
/* allow extensions to sync when showing the popup */
|
|
action = gtk_action_group_get_action (window->priv->action_group,
|
|
"NotebookPopupAction");
|
|
g_return_val_if_fail (action != NULL, FALSE);
|
|
gtk_action_activate (action);
|
|
#endif
|
|
if (event != NULL)
|
|
{
|
|
gtk_menu_popup_at_pointer (GTK_MENU(menu), (GdkEvent *) event);
|
|
|
|
}
|
|
else
|
|
{
|
|
GtkWidget *tab;
|
|
GtkWidget *tab_label;
|
|
|
|
tab = GTK_WIDGET (xed_window_get_active_tab (window));
|
|
g_return_val_if_fail (tab != NULL, FALSE);
|
|
|
|
tab_label = gtk_notebook_get_tab_label (notebook, tab);
|
|
|
|
gtk_menu_popup_at_widget (GTK_MENU (menu), tab_label, GDK_GRAVITY_SOUTH_WEST, GDK_GRAVITY_NORTH_WEST, NULL);
|
|
|
|
gtk_menu_shell_select_first (GTK_MENU_SHELL(menu), FALSE);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
notebook_button_press_event (GtkNotebook *notebook,
|
|
GdkEventButton *event,
|
|
XedWindow *window)
|
|
{
|
|
if (GDK_BUTTON_PRESS == event->type && 3 == event->button)
|
|
{
|
|
return show_notebook_popup_menu (notebook, window, event);
|
|
}
|
|
else if (GDK_BUTTON_PRESS == event->type && 2 == event->button)
|
|
{
|
|
XedTab *tab;
|
|
tab = xed_window_get_active_tab (window);
|
|
notebook_tab_close_request (XED_NOTEBOOK(notebook), tab, GTK_WINDOW(window));
|
|
return FALSE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static gboolean
|
|
notebook_popup_menu (GtkNotebook *notebook,
|
|
XedWindow *window)
|
|
{
|
|
/* Only respond if the notebook is the actual focus */
|
|
if (XED_IS_NOTEBOOK(gtk_window_get_focus (GTK_WINDOW (window))))
|
|
{
|
|
return show_notebook_popup_menu (notebook, window, NULL);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
side_panel_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation,
|
|
XedWindow *window)
|
|
{
|
|
if (!xed_paned_get_is_animating (XED_PANED (window->priv->hpaned)))
|
|
{
|
|
window->priv->side_panel_size = allocation->width;
|
|
}
|
|
}
|
|
|
|
static void
|
|
bottom_panel_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation,
|
|
XedWindow *window)
|
|
{
|
|
if (!xed_paned_get_is_animating (XED_PANED (window->priv->vpaned)))
|
|
{
|
|
window->priv->bottom_panel_size = allocation->height;
|
|
}
|
|
}
|
|
|
|
static void
|
|
hpaned_restore_position (GtkWidget *widget,
|
|
XedWindow *window)
|
|
{
|
|
gint pos;
|
|
xed_debug_message (DEBUG_WINDOW, "Restoring hpaned position: side panel size %d", window->priv->side_panel_size);
|
|
|
|
pos = MAX(100, window->priv->side_panel_size);
|
|
gtk_paned_set_position (GTK_PANED(window->priv->hpaned), pos);
|
|
|
|
/* start monitoring the size */
|
|
g_signal_connect(window->priv->side_panel, "size-allocate", G_CALLBACK (side_panel_size_allocate), window);
|
|
|
|
/* run this only once */
|
|
g_signal_handlers_disconnect_by_func(widget, hpaned_restore_position, window);
|
|
}
|
|
|
|
static void
|
|
vpaned_restore_position (GtkWidget *widget,
|
|
XedWindow *window)
|
|
{
|
|
GtkAllocation allocation;
|
|
gint pos;
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
xed_debug_message (DEBUG_WINDOW, "Restoring vpaned position: bottom panel size %d",
|
|
window->priv->bottom_panel_size);
|
|
|
|
pos = allocation.height - MAX(50, window->priv->bottom_panel_size);
|
|
gtk_paned_set_position (GTK_PANED(window->priv->vpaned), pos);
|
|
|
|
/* start monitoring the size */
|
|
g_signal_connect(window->priv->bottom_panel, "size-allocate", G_CALLBACK (bottom_panel_size_allocate), window);
|
|
|
|
/* run this only once */
|
|
g_signal_handlers_disconnect_by_func(widget, vpaned_restore_position, window);
|
|
}
|
|
|
|
static void
|
|
side_panel_visibility_changed (GtkWidget *side_panel,
|
|
XedWindow *window)
|
|
{
|
|
gboolean visible;
|
|
GtkAction *action;
|
|
|
|
visible = gtk_widget_get_visible (side_panel);
|
|
|
|
g_settings_set_boolean (window->priv->ui_settings, XED_SETTINGS_SIDE_PANEL_VISIBLE, visible);
|
|
|
|
action = gtk_action_group_get_action (window->priv->panes_action_group, "ViewSidePane");
|
|
|
|
if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION(action)) != visible)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION(action), visible);
|
|
}
|
|
|
|
/* focus the document */
|
|
if (!visible && window->priv->active_tab != NULL)
|
|
{
|
|
gtk_widget_grab_focus (GTK_WIDGET(xed_tab_get_view (XED_TAB (window->priv->active_tab))));
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_side_panel (XedWindow *window)
|
|
{
|
|
GtkWidget *documents_panel;
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
window->priv->side_panel = xed_panel_new (GTK_ORIENTATION_VERTICAL);
|
|
|
|
gtk_paned_pack1 (GTK_PANED (window->priv->hpaned), window->priv->side_panel, FALSE, TRUE);
|
|
|
|
g_signal_connect_after (window->priv->side_panel, "show", G_CALLBACK (side_panel_visibility_changed), window);
|
|
g_signal_connect_after (window->priv->side_panel, "hide", G_CALLBACK (side_panel_visibility_changed), window);
|
|
|
|
documents_panel = xed_documents_panel_new (window);
|
|
xed_panel_add_item (XED_PANEL (window->priv->side_panel), documents_panel, _("Documents"), "text-x-generic");
|
|
}
|
|
|
|
static void
|
|
bottom_panel_visibility_changed (XedPanel *bottom_panel,
|
|
XedWindow *window)
|
|
{
|
|
gboolean visible;
|
|
GtkAction *action;
|
|
|
|
visible = gtk_widget_get_visible (GTK_WIDGET (bottom_panel));
|
|
|
|
g_settings_set_boolean (window->priv->ui_settings, XED_SETTINGS_BOTTOM_PANEL_VISIBLE, visible);
|
|
|
|
action = gtk_action_group_get_action (window->priv->panes_action_group, "ViewBottomPane");
|
|
|
|
if (gtk_toggle_action_get_active (GTK_TOGGLE_ACTION(action)) != visible)
|
|
{
|
|
gtk_toggle_action_set_active (GTK_TOGGLE_ACTION(action), visible);
|
|
}
|
|
|
|
/* focus the document */
|
|
if (!visible && window->priv->active_tab != NULL)
|
|
{
|
|
gtk_widget_grab_focus (GTK_WIDGET(xed_tab_get_view (XED_TAB (window->priv->active_tab))));
|
|
}
|
|
}
|
|
|
|
static void
|
|
bottom_panel_item_removed (XedPanel *panel,
|
|
GtkWidget *item,
|
|
XedWindow *window)
|
|
{
|
|
if (xed_panel_get_n_items (panel) == 0)
|
|
{
|
|
GtkAction *action;
|
|
|
|
xed_paned_close (XED_PANED (window->priv->vpaned), 2);
|
|
gtk_revealer_set_reveal_child (GTK_REVEALER (window->priv->bottom_pane_button_revealer), FALSE);
|
|
action = gtk_action_group_get_action (window->priv->panes_action_group, "ViewBottomPane");
|
|
gtk_action_set_sensitive (action, FALSE);
|
|
}
|
|
}
|
|
|
|
static void
|
|
bottom_panel_item_added (XedPanel *panel,
|
|
GtkWidget *item,
|
|
XedWindow *window)
|
|
{
|
|
/* if it's the first item added, set the menu item
|
|
* sensitive and if needed show the panel */
|
|
if (xed_panel_get_n_items (panel) == 1)
|
|
{
|
|
GtkAction *action;
|
|
gboolean show;
|
|
|
|
gtk_revealer_set_reveal_child (GTK_REVEALER (window->priv->bottom_pane_button_revealer), TRUE);
|
|
|
|
action = gtk_action_group_get_action (window->priv->panes_action_group, "ViewBottomPane");
|
|
gtk_action_set_sensitive (action, TRUE);
|
|
show = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action));
|
|
if (show)
|
|
{
|
|
gtk_widget_show (GTK_WIDGET(panel));
|
|
}
|
|
}
|
|
}
|
|
|
|
static void
|
|
create_bottom_panel (XedWindow *window)
|
|
{
|
|
xed_debug (DEBUG_WINDOW);
|
|
window->priv->bottom_panel = xed_panel_new (GTK_ORIENTATION_HORIZONTAL);
|
|
gtk_paned_pack2 (GTK_PANED (window->priv->vpaned), window->priv->bottom_panel, FALSE, TRUE);
|
|
g_signal_connect_after(window->priv->bottom_panel, "show", G_CALLBACK (bottom_panel_visibility_changed), window);
|
|
g_signal_connect_after(window->priv->bottom_panel, "hide", G_CALLBACK (bottom_panel_visibility_changed), window);
|
|
}
|
|
|
|
static void
|
|
init_panels_visibility (XedWindow *window)
|
|
{
|
|
gint active_page;
|
|
gboolean side_panel_visible;
|
|
gboolean bottom_panel_visible;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
/* side pane */
|
|
active_page = g_settings_get_int (window->priv->window_settings, XED_SETTINGS_SIDE_PANEL_ACTIVE_PAGE);
|
|
_xed_panel_set_active_item_by_id (XED_PANEL (window->priv->side_panel), active_page);
|
|
|
|
side_panel_visible = g_settings_get_boolean (window->priv->ui_settings, XED_SETTINGS_SIDE_PANEL_VISIBLE);
|
|
bottom_panel_visible = g_settings_get_boolean (window->priv->ui_settings, XED_SETTINGS_BOTTOM_PANEL_VISIBLE);
|
|
|
|
if (side_panel_visible)
|
|
{
|
|
gtk_widget_show (window->priv->side_panel);
|
|
}
|
|
|
|
/* bottom pane, it can be empty */
|
|
if (xed_panel_get_n_items (XED_PANEL (window->priv->bottom_panel)) > 0)
|
|
{
|
|
active_page = g_settings_get_int (window->priv->window_settings, XED_SETTINGS_BOTTOM_PANEL_ACTIVE_PAGE);
|
|
_xed_panel_set_active_item_by_id (XED_PANEL(window->priv->bottom_panel), active_page);
|
|
|
|
if (bottom_panel_visible)
|
|
{
|
|
gtk_widget_show (window->priv->bottom_panel);
|
|
}
|
|
|
|
gtk_revealer_set_reveal_child (GTK_REVEALER (window->priv->bottom_pane_button_revealer), TRUE);
|
|
}
|
|
else
|
|
{
|
|
GtkAction *action;
|
|
|
|
action = gtk_action_group_get_action (window->priv->panes_action_group, "ViewBottomPane");
|
|
gtk_action_set_sensitive (action, FALSE);
|
|
gtk_revealer_set_reveal_child (GTK_REVEALER (window->priv->bottom_pane_button_revealer), FALSE);
|
|
}
|
|
|
|
/* start track sensitivity after the initial state is set */
|
|
window->priv->bottom_panel_item_removed_handler_id =
|
|
g_signal_connect (window->priv->bottom_panel, "item_removed",
|
|
G_CALLBACK (bottom_panel_item_removed), window);
|
|
|
|
g_signal_connect(window->priv->bottom_panel, "item_added", G_CALLBACK (bottom_panel_item_added), window);
|
|
}
|
|
|
|
static void
|
|
clipboard_owner_change (GtkClipboard *clipboard,
|
|
GdkEventOwnerChange *event,
|
|
XedWindow *window)
|
|
{
|
|
set_paste_sensitivity_according_to_clipboard (window, clipboard);
|
|
}
|
|
|
|
static void
|
|
window_realized (GtkWidget *window,
|
|
gpointer *data)
|
|
{
|
|
GtkClipboard *clipboard;
|
|
clipboard = gtk_widget_get_clipboard (window, GDK_SELECTION_CLIPBOARD);
|
|
g_signal_connect(clipboard, "owner_change", G_CALLBACK (clipboard_owner_change), window);
|
|
}
|
|
|
|
static void
|
|
window_unrealized (GtkWidget *window,
|
|
gpointer *data)
|
|
{
|
|
GtkClipboard *clipboard;
|
|
clipboard = gtk_widget_get_clipboard (window, GDK_SELECTION_CLIPBOARD);
|
|
g_signal_handlers_disconnect_by_func(clipboard, G_CALLBACK (clipboard_owner_change), window);
|
|
}
|
|
|
|
static void
|
|
check_window_is_active (XedWindow *window,
|
|
GParamSpec *property,
|
|
gpointer useless)
|
|
{
|
|
if (window->priv->window_state & GDK_WINDOW_STATE_FULLSCREEN)
|
|
{
|
|
gtk_widget_set_visible (window->priv->fullscreen_eventbox, gtk_window_is_active (GTK_WINDOW (window)));
|
|
}
|
|
}
|
|
|
|
static void
|
|
connect_notebook_signals (XedWindow *window,
|
|
GtkWidget *notebook)
|
|
{
|
|
g_signal_connect(notebook, "switch-page", G_CALLBACK (notebook_switch_page), window);
|
|
g_signal_connect(notebook, "tab-added", G_CALLBACK (notebook_tab_added), window);
|
|
g_signal_connect(notebook, "tab-removed", G_CALLBACK (notebook_tab_removed), window);
|
|
g_signal_connect(notebook, "tabs-reordered", G_CALLBACK (notebook_tabs_reordered), window);
|
|
g_signal_connect(notebook, "tab-detached", G_CALLBACK (notebook_tab_detached), window);
|
|
g_signal_connect(notebook, "tab-close-request", G_CALLBACK (notebook_tab_close_request), window);
|
|
g_signal_connect(notebook, "button-press-event", G_CALLBACK (notebook_button_press_event), window);
|
|
g_signal_connect(notebook, "popup-menu", G_CALLBACK (notebook_popup_menu), window);
|
|
}
|
|
|
|
static void
|
|
add_notebook (XedWindow *window,
|
|
GtkWidget *notebook)
|
|
{
|
|
gtk_paned_pack1 (GTK_PANED(window->priv->vpaned), notebook, TRUE, TRUE);
|
|
gtk_widget_show (notebook);
|
|
|
|
// xed_notebook_set_tab_scrolling_enabled (XED_NOTEBOOK (notebook), xed_prefs_manager_get_enable_tab_scrolling ());
|
|
|
|
connect_notebook_signals (window, notebook);
|
|
}
|
|
|
|
static void
|
|
extension_added (PeasExtensionSet *extensions,
|
|
PeasPluginInfo *info,
|
|
PeasExtension *exten,
|
|
XedWindow *window)
|
|
{
|
|
peas_extension_call (exten, "activate");
|
|
}
|
|
|
|
static void
|
|
extension_removed (PeasExtensionSet *extensions,
|
|
PeasPluginInfo *info,
|
|
PeasExtension *exten,
|
|
XedWindow *window)
|
|
{
|
|
peas_extension_call (exten, "deactivate");
|
|
|
|
/* Ensure update of the ui manager, because we suspect it does something with expected static strings in the
|
|
* type module (when unloaded the strings don't exist anymore, and the ui manager update in a idle func) */
|
|
gtk_ui_manager_ensure_update (window->priv->manager);
|
|
}
|
|
|
|
static void
|
|
xed_window_init (XedWindow *window)
|
|
{
|
|
GtkWidget *main_box;
|
|
GtkTargetList *tl;
|
|
|
|
xed_debug (DEBUG_WINDOW);
|
|
|
|
window->priv = xed_window_get_instance_private (window);
|
|
window->priv->active_tab = NULL;
|
|
window->priv->num_tabs = 0;
|
|
window->priv->removing_tabs = FALSE;
|
|
window->priv->state = XED_WINDOW_STATE_NORMAL;
|
|
window->priv->inhibition_cookie = 0;
|
|
window->priv->dispose_has_run = FALSE;
|
|
window->priv->fullscreen_controls = NULL;
|
|
window->priv->editor_settings = g_settings_new ("org.x.editor.preferences.editor");
|
|
window->priv->ui_settings = g_settings_new ("org.x.editor.preferences.ui");
|
|
|
|
/* Window settings are applied only once the window is closed. We do not want
|
|
to keep writing to disk when the window is dragged around */
|
|
window->priv->window_settings = g_settings_new ("org.x.editor.state.window");
|
|
g_settings_delay (window->priv->window_settings);
|
|
|
|
window->priv->message_bus = xed_message_bus_new ();
|
|
|
|
window->priv->window_group = gtk_window_group_new ();
|
|
gtk_window_group_add_window (window->priv->window_group, GTK_WINDOW (window));
|
|
|
|
gtk_style_context_add_class (gtk_widget_get_style_context (GTK_WIDGET (window)), "xed-window");
|
|
|
|
main_box = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
|
|
window->priv->fullscreen_overlay = gtk_overlay_new ();
|
|
gtk_container_add (GTK_CONTAINER (window->priv->fullscreen_overlay), main_box);
|
|
gtk_container_add (GTK_CONTAINER (window), window->priv->fullscreen_overlay);
|
|
gtk_widget_show (window->priv->fullscreen_overlay);
|
|
gtk_widget_show (main_box);
|
|
|
|
window->priv->fullscreen_eventbox = gtk_event_box_new ();
|
|
gtk_widget_set_valign (window->priv->fullscreen_eventbox, GTK_ALIGN_START);
|
|
gtk_overlay_add_overlay (GTK_OVERLAY (window->priv->fullscreen_overlay), window->priv->fullscreen_eventbox);
|
|
|
|
/* Add menu bar and toolbar bar */
|
|
create_menu_bar_and_toolbar (window, main_box);
|
|
|
|
/* If we're running as root, add and infobar to warn about elevated privileges */
|
|
if (geteuid () == 0)
|
|
{
|
|
GtkWidget *root_bar = gtk_info_bar_new ();
|
|
gtk_info_bar_set_message_type (GTK_INFO_BAR (root_bar), GTK_MESSAGE_ERROR);
|
|
GtkWidget *content_area = gtk_info_bar_get_content_area (GTK_INFO_BAR (root_bar));
|
|
GtkWidget *label = gtk_label_new (_("Elevated Privileges"));
|
|
gtk_widget_show (label);
|
|
gtk_container_add (GTK_CONTAINER (content_area), label);
|
|
gtk_box_pack_start (GTK_BOX (main_box), root_bar, FALSE, FALSE, 0);
|
|
gtk_widget_set_visible (root_bar, TRUE);
|
|
}
|
|
|
|
/* Add status bar */
|
|
create_statusbar (window, main_box);
|
|
|
|
/* Add the main area */
|
|
xed_debug_message (DEBUG_WINDOW, "Add main area");
|
|
window->priv->hpaned = xed_paned_new (GTK_ORIENTATION_HORIZONTAL);
|
|
gtk_box_pack_start (GTK_BOX (main_box), window->priv->hpaned, TRUE, TRUE, 0);
|
|
|
|
window->priv->vpaned = xed_paned_new (GTK_ORIENTATION_VERTICAL);
|
|
gtk_paned_pack2 (GTK_PANED (window->priv->hpaned), window->priv->vpaned, TRUE, FALSE);
|
|
|
|
xed_debug_message (DEBUG_WINDOW, "Create xed notebook");
|
|
window->priv->notebook = xed_notebook_new ();
|
|
add_notebook (window, window->priv->notebook);
|
|
|
|
/* side and bottom panels */
|
|
create_side_panel (window);
|
|
create_bottom_panel (window);
|
|
|
|
/* panes' state must be restored after panels have been mapped,
|
|
* since the bottom pane position depends on the size of the vpaned. */
|
|
window->priv->side_panel_size = g_settings_get_int (window->priv->window_settings, XED_SETTINGS_SIDE_PANEL_SIZE);
|
|
window->priv->bottom_panel_size = g_settings_get_int (window->priv->window_settings, XED_SETTINGS_BOTTOM_PANEL_SIZE);
|
|
|
|
g_signal_connect_after(window->priv->hpaned, "map", G_CALLBACK (hpaned_restore_position), window);
|
|
g_signal_connect_after(window->priv->vpaned, "map", G_CALLBACK (vpaned_restore_position), window);
|
|
|
|
gtk_widget_show (window->priv->hpaned);
|
|
gtk_widget_show (window->priv->vpaned);
|
|
|
|
/* Drag and drop support, set targets to NULL because we add the
|
|
default uri_targets below */
|
|
gtk_drag_dest_set (GTK_WIDGET(window), GTK_DEST_DEFAULT_MOTION | GTK_DEST_DEFAULT_HIGHLIGHT | GTK_DEST_DEFAULT_DROP,
|
|
NULL, 0, GDK_ACTION_COPY);
|
|
|
|
/* Add uri targets */
|
|
tl = gtk_drag_dest_get_target_list (GTK_WIDGET (window));
|
|
|
|
if (tl == NULL)
|
|
{
|
|
tl = gtk_target_list_new (NULL, 0);
|
|
gtk_drag_dest_set_target_list (GTK_WIDGET (window), tl);
|
|
gtk_target_list_unref (tl);
|
|
}
|
|
|
|
gtk_target_list_add_uri_targets (tl, TARGET_URI_LIST);
|
|
|
|
/* connect instead of override, so that we can share the cb code with the view */
|
|
g_signal_connect (window, "drag_data_received", G_CALLBACK (drag_data_received_cb), NULL);
|
|
|
|
/* we can get the clipboard only after the widget is realized */
|
|
g_signal_connect (window, "realize", G_CALLBACK (window_realized), NULL);
|
|
g_signal_connect (window, "unrealize", G_CALLBACK (window_unrealized), NULL);
|
|
|
|
/* Check if the window is active for fullscreen */
|
|
g_signal_connect (window, "notify::is-active", G_CALLBACK (check_window_is_active), NULL);
|
|
|
|
g_signal_connect (GTK_WIDGET (window), "key-press-event", G_CALLBACK (on_key_pressed), window);
|
|
|
|
xed_debug_message (DEBUG_WINDOW, "Update plugins ui");
|
|
|
|
window->priv->extensions = peas_extension_set_new (PEAS_ENGINE (xed_plugins_engine_get_default ()),
|
|
XED_TYPE_WINDOW_ACTIVATABLE, "window", window, NULL);
|
|
|
|
g_signal_connect (window->priv->extensions, "extension-added", G_CALLBACK (extension_added), window);
|
|
g_signal_connect (window->priv->extensions, "extension-removed", G_CALLBACK (extension_removed), window);
|
|
|
|
peas_extension_set_call (window->priv->extensions, "activate");
|
|
|
|
/* set visibility of panes.
|
|
* This needs to be done after plugins activatation */
|
|
init_panels_visibility (window);
|
|
|
|
update_sensitivity_according_to_open_tabs (window);
|
|
|
|
xed_debug_message (DEBUG_WINDOW, "END");
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_active_view:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the active #XedView.
|
|
*
|
|
* Returns: (transfer none): the active #XedView
|
|
*/
|
|
XedView *
|
|
xed_window_get_active_view (XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
|
|
if (window == NULL)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
if (window->priv->active_tab == NULL)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
view = xed_tab_get_view (XED_TAB (window->priv->active_tab));
|
|
|
|
return view;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_active_document:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the active #XedDocument.
|
|
*
|
|
* Returns: (transfer none): the active #XedDocument
|
|
*/
|
|
XedDocument *
|
|
xed_window_get_active_document (XedWindow *window)
|
|
{
|
|
XedView *view;
|
|
|
|
if (window == NULL)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
view = xed_window_get_active_view (window);
|
|
if (view == NULL)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
return XED_DOCUMENT (gtk_text_view_get_buffer (GTK_TEXT_VIEW (view)));
|
|
}
|
|
|
|
GtkWidget *
|
|
_xed_window_get_notebook (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
|
|
return window->priv->notebook;
|
|
}
|
|
|
|
/**
|
|
* xed_window_create_tab:
|
|
* @window: a #XedWindow
|
|
* @jump_to: %TRUE to set the new #XedTab as active
|
|
*
|
|
* Creates a new #XedTab and adds the new tab to the #XedNotebook.
|
|
* In case @jump_to is %TRUE the #XedNotebook switches to that new #XedTab.
|
|
*
|
|
* Returns: (transfer none): a new #XedTab
|
|
*/
|
|
XedTab *
|
|
xed_window_create_tab (XedWindow *window,
|
|
gboolean jump_to)
|
|
{
|
|
XedTab *tab;
|
|
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
|
|
tab = XED_TAB(_xed_tab_new ());
|
|
gtk_widget_show (GTK_WIDGET (tab));
|
|
|
|
xed_notebook_add_tab (XED_NOTEBOOK (window->priv->notebook), tab, -1, jump_to);
|
|
|
|
if (!gtk_widget_get_visible (GTK_WIDGET (window)))
|
|
{
|
|
gtk_window_present (GTK_WINDOW (window));
|
|
}
|
|
|
|
return tab;
|
|
}
|
|
|
|
static XedTab *
|
|
process_create_tab (XedWindow *window,
|
|
XedTab *tab,
|
|
gboolean jump_to)
|
|
{
|
|
if (tab == NULL)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
gtk_widget_show (GTK_WIDGET (tab));
|
|
|
|
xed_notebook_add_tab (XED_NOTEBOOK (window->priv->notebook), tab, -1, jump_to);
|
|
|
|
if (!gtk_widget_get_visible (GTK_WIDGET (window)))
|
|
{
|
|
gtk_window_present (GTK_WINDOW (window));
|
|
}
|
|
|
|
return tab;
|
|
}
|
|
|
|
/**
|
|
* xed_window_create_tab_from_location:
|
|
* @window: a #XedWindow
|
|
* @location: the location of the document
|
|
* @encoding: (allow-none): a #GtkSourceEncoding
|
|
* @line_pos: the line position to visualize
|
|
* @create: %TRUE to create a new document in case @uri does exist
|
|
* @jump_to: %TRUE to set the new #XedTab as active
|
|
*
|
|
* Creates a new #XedTab loading the document specified by @uri.
|
|
* In case @jump_to is %TRUE the #XedNotebook swithes to that new #XedTab.
|
|
* Whether @create is %TRUE, creates a new empty document if location does
|
|
* not refer to an existing file
|
|
*
|
|
* Returns: (transfer none): a new #XedTab
|
|
*/
|
|
XedTab *
|
|
xed_window_create_tab_from_location (XedWindow *window,
|
|
GFile *location,
|
|
const GtkSourceEncoding *encoding,
|
|
gint line_pos,
|
|
gboolean create,
|
|
gboolean jump_to)
|
|
{
|
|
GtkWidget *tab;
|
|
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
g_return_val_if_fail(G_IS_FILE (location), NULL);
|
|
|
|
tab = _xed_tab_new_from_location (location, encoding, line_pos, create);
|
|
|
|
return process_create_tab (window, XED_TAB (tab), jump_to);
|
|
}
|
|
|
|
/**
|
|
* xed_window_create_tab_from_stream:
|
|
* @window: a #XedWindow
|
|
* @stream: the #GInputStream
|
|
* @encoding: the encoding to use in the stream
|
|
* @line_pos: the position
|
|
* @jump_to: whether to jump to the new tab
|
|
*
|
|
* Returns: (transfer none): the active #XedTab in the @window.
|
|
*/
|
|
XedTab *
|
|
xed_window_create_tab_from_stream (XedWindow *window,
|
|
GInputStream *stream,
|
|
const GtkSourceEncoding *encoding,
|
|
gint line_pos,
|
|
gboolean jump_to)
|
|
{
|
|
GtkWidget *tab;
|
|
|
|
g_return_val_if_fail (XED_IS_WINDOW (window), NULL);
|
|
g_return_val_if_fail (G_IS_INPUT_STREAM (stream), NULL);
|
|
|
|
tab = _xed_tab_new_from_stream (stream, encoding, line_pos);
|
|
|
|
return process_create_tab (window, XED_TAB (tab), jump_to);
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_active_tab:
|
|
* @window: a XedWindow
|
|
*
|
|
* Gets the active #XedTab in the @window.
|
|
*
|
|
* Returns: (transfer none): the active #XedTab in the @window.
|
|
*/
|
|
XedTab *
|
|
xed_window_get_active_tab (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
|
|
if (window->priv->active_tab == NULL)
|
|
{
|
|
return NULL;
|
|
}
|
|
else
|
|
{
|
|
return XED_TAB(window->priv->active_tab);
|
|
}
|
|
}
|
|
|
|
static void
|
|
add_document (XedTab *tab,
|
|
GList **res)
|
|
{
|
|
XedDocument *doc;
|
|
|
|
doc = xed_tab_get_document (tab);
|
|
|
|
*res = g_list_prepend (*res, doc);
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_documents:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets a newly allocated list with all the documents in the window.
|
|
* This list must be freed.
|
|
*
|
|
* Returns: (element-type Xed.Document) (transfer container): a newly
|
|
* allocated list with all the documents in the window
|
|
*/
|
|
GList *
|
|
xed_window_get_documents (XedWindow *window)
|
|
{
|
|
GList *res = NULL;
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
gtk_container_foreach (GTK_CONTAINER(window->priv->notebook), (GtkCallback) add_document, &res);
|
|
res = g_list_reverse (res);
|
|
return res;
|
|
}
|
|
|
|
static void
|
|
add_view (XedTab *tab,
|
|
GList **res)
|
|
{
|
|
XedView *view;
|
|
view = xed_tab_get_view (tab);
|
|
*res = g_list_prepend (*res, view);
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_views:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets a list with all the views in the window. This list must be freed.
|
|
*
|
|
* Returns: (element-type Xed.View) (transfer container): a newly allocated
|
|
* list with all the views in the window
|
|
*/
|
|
GList *
|
|
xed_window_get_views (XedWindow *window)
|
|
{
|
|
GList *res = NULL;
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
gtk_container_foreach (GTK_CONTAINER(window->priv->notebook), (GtkCallback) add_view, &res);
|
|
res = g_list_reverse (res);
|
|
return res;
|
|
}
|
|
|
|
/**
|
|
* xed_window_close_tab:
|
|
* @window: a #XedWindow
|
|
* @tab: the #XedTab to close
|
|
*
|
|
* Closes the @tab.
|
|
*/
|
|
void
|
|
xed_window_close_tab (XedWindow *window,
|
|
XedTab *tab)
|
|
{
|
|
g_return_if_fail (XED_IS_WINDOW (window));
|
|
g_return_if_fail (XED_IS_TAB (tab));
|
|
g_return_if_fail ((xed_tab_get_state (tab) != XED_TAB_STATE_SAVING)
|
|
&& (xed_tab_get_state (tab) != XED_TAB_STATE_SHOWING_PRINT_PREVIEW));
|
|
|
|
xed_notebook_remove_tab (XED_NOTEBOOK (window->priv->notebook), tab);
|
|
}
|
|
|
|
/**
|
|
* xed_window_close_all_tabs:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Closes all opened tabs.
|
|
*/
|
|
void
|
|
xed_window_close_all_tabs (XedWindow *window)
|
|
{
|
|
g_return_if_fail(XED_IS_WINDOW (window));
|
|
g_return_if_fail(!(window->priv->state & XED_WINDOW_STATE_SAVING)
|
|
&& !(window->priv->state & XED_WINDOW_STATE_SAVING_SESSION));
|
|
|
|
window->priv->removing_tabs = TRUE;
|
|
xed_notebook_remove_all_tabs (XED_NOTEBOOK(window->priv->notebook));
|
|
window->priv->removing_tabs = FALSE;
|
|
}
|
|
|
|
/**
|
|
* xed_window_close_tabs:
|
|
* @window: a #XedWindow
|
|
* @tabs: (element-type Xed.Tab): a list of #XedTab
|
|
*
|
|
* Closes all tabs specified by @tabs.
|
|
*/
|
|
void
|
|
xed_window_close_tabs (XedWindow *window,
|
|
const GList *tabs)
|
|
{
|
|
g_return_if_fail(XED_IS_WINDOW (window));
|
|
g_return_if_fail(!(window->priv->state & XED_WINDOW_STATE_SAVING)
|
|
&& !(window->priv->state & XED_WINDOW_STATE_SAVING_SESSION));
|
|
|
|
if (tabs == NULL)
|
|
{
|
|
return;
|
|
}
|
|
|
|
window->priv->removing_tabs = TRUE;
|
|
|
|
while (tabs != NULL)
|
|
{
|
|
if (tabs->next == NULL)
|
|
{
|
|
window->priv->removing_tabs = FALSE;
|
|
}
|
|
|
|
xed_notebook_remove_tab (XED_NOTEBOOK(window->priv->notebook), XED_TAB(tabs->data));
|
|
tabs = g_list_next(tabs);
|
|
}
|
|
|
|
g_return_if_fail(window->priv->removing_tabs == FALSE);
|
|
}
|
|
|
|
XedWindow *
|
|
_xed_window_move_tab_to_new_window (XedWindow *window,
|
|
XedTab *tab)
|
|
{
|
|
XedWindow *new_window;
|
|
|
|
g_return_val_if_fail (XED_IS_WINDOW (window), NULL);
|
|
g_return_val_if_fail (XED_IS_TAB (tab), NULL);
|
|
g_return_val_if_fail (gtk_notebook_get_n_pages (GTK_NOTEBOOK (window->priv->notebook)) > 1, NULL);
|
|
|
|
new_window = clone_window (window);
|
|
xed_notebook_move_tab (XED_NOTEBOOK (window->priv->notebook), XED_NOTEBOOK (new_window->priv->notebook), tab, -1);
|
|
gtk_widget_show (GTK_WIDGET (new_window));
|
|
|
|
return new_window;
|
|
}
|
|
|
|
/**
|
|
* xed_window_set_active_tab:
|
|
* @window: a #XedWindow
|
|
* @tab: a #XedTab
|
|
*
|
|
* Switches to the tab that matches with @tab.
|
|
*/
|
|
void
|
|
xed_window_set_active_tab (XedWindow *window,
|
|
XedTab *tab)
|
|
{
|
|
gint page_num;
|
|
|
|
g_return_if_fail(XED_IS_WINDOW (window));
|
|
g_return_if_fail(XED_IS_TAB (tab));
|
|
|
|
page_num = gtk_notebook_page_num (GTK_NOTEBOOK(window->priv->notebook), GTK_WIDGET(tab));
|
|
g_return_if_fail(page_num != -1);
|
|
gtk_notebook_set_current_page (GTK_NOTEBOOK(window->priv->notebook), page_num);
|
|
|
|
/* Make sure to grab focus if the page didn't change */
|
|
gtk_widget_grab_focus (GTK_WIDGET (xed_tab_get_view (tab)));
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_group:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the #GtkWindowGroup in which @window resides.
|
|
*
|
|
* Returns: (transfer none): the #GtkWindowGroup
|
|
*/
|
|
GtkWindowGroup *
|
|
xed_window_get_group (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
return window->priv->window_group;
|
|
}
|
|
|
|
gboolean
|
|
_xed_window_is_removing_tabs (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), FALSE);
|
|
return window->priv->removing_tabs;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_ui_manager:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the #GtkUIManager associated with the @window.
|
|
*
|
|
* Returns: (transfer none): the #GtkUIManager of the @window.
|
|
*/
|
|
GtkUIManager *
|
|
xed_window_get_ui_manager (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
return window->priv->manager;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_side_panel:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the side #XedPanel of the @window.
|
|
*
|
|
* Returns: (transfer none): the side #XedPanel.
|
|
*/
|
|
XedPanel *
|
|
xed_window_get_side_panel (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
return XED_PANEL(window->priv->side_panel);
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_bottom_panel:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the bottom #XedPanel of the @window.
|
|
*
|
|
* Returns: (transfer none): the bottom #XedPanel.
|
|
*/
|
|
XedPanel *
|
|
xed_window_get_bottom_panel (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
return XED_PANEL(window->priv->bottom_panel);
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_statusbar:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the #XedStatusbar of the @window.
|
|
*
|
|
* Returns: (transfer none): the #XedStatusbar of the @window.
|
|
*/
|
|
GtkWidget *
|
|
xed_window_get_statusbar (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), 0);
|
|
return window->priv->statusbar;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_searchbar:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the #XedSearchbar of the @window.
|
|
*
|
|
* Returns: (transfer none): the #XedSearchbar of the @window.
|
|
*/
|
|
GtkWidget *
|
|
xed_window_get_searchbar (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), 0);
|
|
return window->priv->searchbar;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_state:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Retrieves the state of the @window.
|
|
*
|
|
* Returns: the current #XedWindowState of the @window.
|
|
*/
|
|
XedWindowState
|
|
xed_window_get_state (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), XED_WINDOW_STATE_NORMAL);
|
|
return window->priv->state;
|
|
}
|
|
|
|
GFile *
|
|
_xed_window_get_default_location (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
if (window->priv->default_location != NULL)
|
|
{
|
|
return g_object_ref (window->priv->default_location);
|
|
}
|
|
else
|
|
{
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
void
|
|
_xed_window_set_default_location (XedWindow *window,
|
|
GFile *location)
|
|
{
|
|
GFile *dir;
|
|
|
|
g_return_if_fail(XED_IS_WINDOW (window));
|
|
g_return_if_fail(G_IS_FILE (location));
|
|
|
|
dir = g_file_get_parent (location);
|
|
g_return_if_fail(dir != NULL);
|
|
|
|
if (window->priv->default_location != NULL)
|
|
{
|
|
g_object_unref (window->priv->default_location);
|
|
}
|
|
|
|
window->priv->default_location = dir;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_unsaved_documents:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the list of documents that need to be saved before closing the window.
|
|
*
|
|
* Returns: (element-type Xed.Document) (transfer container): a list of
|
|
* #XedDocument that need to be saved before closing the window
|
|
*/
|
|
GList *
|
|
xed_window_get_unsaved_documents (XedWindow *window)
|
|
{
|
|
GList *unsaved_docs = NULL;
|
|
GList *tabs;
|
|
GList *l;
|
|
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
|
|
tabs = gtk_container_get_children (GTK_CONTAINER(window->priv->notebook));
|
|
|
|
l = tabs;
|
|
while (l != NULL)
|
|
{
|
|
XedTab *tab;
|
|
tab = XED_TAB(l->data);
|
|
if (!_xed_tab_get_can_close (tab))
|
|
{
|
|
XedDocument *doc;
|
|
doc = xed_tab_get_document (tab);
|
|
unsaved_docs = g_list_prepend (unsaved_docs, doc);
|
|
}
|
|
l = g_list_next(l);
|
|
}
|
|
|
|
g_list_free (tabs);
|
|
|
|
return g_list_reverse (unsaved_docs);
|
|
}
|
|
|
|
void
|
|
_xed_window_set_saving_session_state (XedWindow *window,
|
|
gboolean saving_session)
|
|
{
|
|
XedWindowState old_state;
|
|
|
|
g_return_if_fail(XED_IS_WINDOW (window));
|
|
|
|
old_state = window->priv->state;
|
|
|
|
if (saving_session)
|
|
{
|
|
window->priv->state |= XED_WINDOW_STATE_SAVING_SESSION;
|
|
}
|
|
else
|
|
{
|
|
window->priv->state &= ~XED_WINDOW_STATE_SAVING_SESSION;
|
|
}
|
|
|
|
if (old_state != window->priv->state)
|
|
{
|
|
set_sensitivity_according_to_window_state (window);
|
|
g_object_notify (G_OBJECT(window), "state");
|
|
}
|
|
}
|
|
|
|
static void
|
|
hide_notebook_tabs_on_fullscreen (GtkNotebook *notebook,
|
|
GParamSpec *pspec,
|
|
XedWindow *window)
|
|
{
|
|
gtk_notebook_set_show_tabs (notebook, FALSE);
|
|
}
|
|
|
|
void
|
|
_xed_window_fullscreen (XedWindow *window)
|
|
{
|
|
g_return_if_fail (XED_IS_WINDOW (window));
|
|
|
|
if (_xed_window_is_fullscreen (window))
|
|
{
|
|
return;
|
|
}
|
|
|
|
/* Go to fullscreen mode and hide bars */
|
|
gtk_window_fullscreen (GTK_WINDOW (&window->window));
|
|
|
|
gtk_widget_hide (window->priv->menubar);
|
|
|
|
g_signal_handlers_block_by_func (window->priv->toolbar, toolbar_visibility_changed, window);
|
|
gtk_widget_hide (window->priv->toolbar);
|
|
|
|
g_signal_handlers_block_by_func (window->priv->statusbar, statusbar_visibility_changed, window);
|
|
gtk_widget_hide (window->priv->statusbar);
|
|
|
|
fullscreen_controls_build (window);
|
|
|
|
gtk_widget_show_all (window->priv->fullscreen_eventbox);
|
|
}
|
|
|
|
void
|
|
_xed_window_unfullscreen (XedWindow *window)
|
|
{
|
|
gboolean visible;
|
|
GtkAction *action;
|
|
|
|
g_return_if_fail (XED_IS_WINDOW (window));
|
|
|
|
if (!_xed_window_is_fullscreen (window))
|
|
{
|
|
return;
|
|
}
|
|
|
|
/* Unfullscreen and show bars */
|
|
gtk_window_unfullscreen (GTK_WINDOW (&window->window));
|
|
g_signal_handlers_disconnect_by_func (window->priv->notebook, hide_notebook_tabs_on_fullscreen, window);
|
|
gtk_widget_show (window->priv->menubar);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewToolbar");
|
|
visible = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action));
|
|
if (visible)
|
|
{
|
|
gtk_widget_show (window->priv->toolbar);
|
|
}
|
|
g_signal_handlers_unblock_by_func (window->priv->toolbar, toolbar_visibility_changed, window);
|
|
|
|
action = gtk_action_group_get_action (window->priv->always_sensitive_action_group, "ViewStatusbar");
|
|
visible = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action));
|
|
if (visible)
|
|
{
|
|
gtk_widget_show (window->priv->statusbar);
|
|
}
|
|
g_signal_handlers_unblock_by_func (window->priv->statusbar, statusbar_visibility_changed, window);
|
|
|
|
gtk_widget_hide (window->priv->fullscreen_eventbox);
|
|
}
|
|
|
|
gboolean
|
|
_xed_window_is_fullscreen (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), FALSE);
|
|
return window->priv->window_state & GDK_WINDOW_STATE_FULLSCREEN;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_tab_from_location:
|
|
* @window: a #XedWindow
|
|
* @location: a #GFile
|
|
*
|
|
* Gets the #XedTab that matches with the given @location.
|
|
*
|
|
* Returns: (transfer none): the #XedTab that matches with the given @location.
|
|
*/
|
|
XedTab *
|
|
xed_window_get_tab_from_location (XedWindow *window,
|
|
GFile *location)
|
|
{
|
|
GList *tabs;
|
|
GList *l;
|
|
XedTab *ret = NULL;
|
|
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
g_return_val_if_fail(G_IS_FILE (location), NULL);
|
|
|
|
tabs = gtk_container_get_children (GTK_CONTAINER(window->priv->notebook));
|
|
|
|
for (l = tabs; l != NULL; l = g_list_next(l))
|
|
{
|
|
XedDocument *doc;
|
|
GtkSourceFile *file;
|
|
XedTab *tab;
|
|
GFile *cur_location;
|
|
|
|
tab = XED_TAB (l->data);
|
|
doc = xed_tab_get_document (tab);
|
|
file = xed_document_get_file (doc);
|
|
cur_location = gtk_source_file_get_location (file);
|
|
|
|
if (cur_location != NULL)
|
|
{
|
|
gboolean found = g_file_equal (location, cur_location);
|
|
|
|
if (found)
|
|
{
|
|
ret = tab;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
g_list_free (tabs);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* xed_window_get_message_bus:
|
|
* @window: a #XedWindow
|
|
*
|
|
* Gets the #XedMessageBus associated with @window. The returned reference
|
|
* is owned by the window and should not be unreffed.
|
|
*
|
|
* Return value: (transfer none): the #XedMessageBus associated with @window
|
|
*/
|
|
XedMessageBus *
|
|
xed_window_get_message_bus (XedWindow *window)
|
|
{
|
|
g_return_val_if_fail(XED_IS_WINDOW (window), NULL);
|
|
return window->priv->message_bus;
|
|
}
|
|
|
|
void
|
|
_xed_window_get_default_size (gint *width,
|
|
gint *height)
|
|
{
|
|
g_return_if_fail (width != NULL && height != NULL);
|
|
|
|
*width = XED_WINDOW_DEFAULT_WIDTH;
|
|
*height = XED_WINDOW_DEFAULT_HEIGHT;
|
|
}
|
|
|
|
XedPaned *
|
|
_xed_window_get_hpaned (XedWindow *window)
|
|
{
|
|
return XED_PANED (window->priv->hpaned);
|
|
}
|
|
|
|
XedPaned *
|
|
_xed_window_get_vpaned (XedWindow *window)
|
|
{
|
|
return XED_PANED (window->priv->vpaned);
|
|
}
|
|
|
|
gint
|
|
_xed_window_get_side_panel_size (XedWindow *window)
|
|
{
|
|
return window->priv->side_panel_size;
|
|
}
|
|
|
|
gint
|
|
_xed_window_get_bottom_panel_size (XedWindow *window)
|
|
{
|
|
return window->priv->bottom_panel_size;
|
|
}
|