9d8863f79e
macro. Set a tooltip for the Split Window plugin's Show Current tool button. Add utils_strdupa() macro. git-svn-id: https://geany.svn.sourceforge.net/svnroot/geany/trunk@3922 ea778897-0a13-0410-b9d1-a72fbfd435f5
526 lines
14 KiB
C
526 lines
14 KiB
C
/*
|
|
* splitwindow.c - this file is part of Geany, a fast and lightweight IDE
|
|
*
|
|
* Copyright 2008-2009 Nick Treleaven <nick(dot)treleaven(at)btinternet(dot)com>
|
|
* Copyright 2008-2009 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston,
|
|
* MA 02110-1301, USA.
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
/* Split Window plugin. */
|
|
|
|
#include "geany.h"
|
|
#include <string.h>
|
|
|
|
#include "support.h"
|
|
#include "Scintilla.h"
|
|
#include "ScintillaWidget.h"
|
|
#include "SciLexer.h"
|
|
|
|
#include "ui_utils.h"
|
|
#include "document.h"
|
|
#include "editor.h"
|
|
#include "plugindata.h"
|
|
#include "keybindings.h"
|
|
#include "utils.h"
|
|
|
|
#include "geanyfunctions.h"
|
|
|
|
|
|
PLUGIN_VERSION_CHECK(GEANY_API_VERSION)
|
|
PLUGIN_SET_INFO(_("Split Window"), _("Splits the editor view into two windows."),
|
|
VERSION, _("The Geany developer team"))
|
|
|
|
|
|
GeanyData *geany_data;
|
|
GeanyFunctions *geany_functions;
|
|
|
|
|
|
/* Keybinding(s) */
|
|
enum
|
|
{
|
|
KB_SPLIT_HORIZONTAL,
|
|
KB_SPLIT_VERTICAL,
|
|
KB_SPLIT_UNSPLIT,
|
|
KB_COUNT
|
|
};
|
|
|
|
PLUGIN_KEY_GROUP(split_window, KB_COUNT);
|
|
|
|
|
|
enum State
|
|
{
|
|
STATE_SPLIT_HORIZONTAL,
|
|
STATE_SPLIT_VERTICAL,
|
|
STATE_UNSPLIT,
|
|
STATE_COUNT
|
|
};
|
|
|
|
static struct
|
|
{
|
|
GtkWidget *main;
|
|
GtkWidget *horizontal;
|
|
GtkWidget *vertical;
|
|
GtkWidget *unsplit;
|
|
}
|
|
menu_items;
|
|
|
|
static enum State plugin_state;
|
|
|
|
|
|
typedef struct EditWindow
|
|
{
|
|
GeanyEditor *editor; /* original editor for split view */
|
|
ScintillaObject *sci; /* new editor widget */
|
|
GtkWidget *vbox;
|
|
GtkWidget *name_label;
|
|
}
|
|
EditWindow;
|
|
|
|
static EditWindow edit_window = {NULL, NULL, NULL, NULL};
|
|
|
|
|
|
static void on_unsplit(GtkMenuItem *menuitem, gpointer user_data);
|
|
|
|
|
|
static gint sci_get_value(ScintillaObject *sci, gint message_id, gint param)
|
|
{
|
|
return scintilla_send_message(sci, message_id, param, 0);
|
|
}
|
|
|
|
|
|
static void set_styles(ScintillaObject *oldsci, ScintillaObject *newsci)
|
|
{
|
|
gint style_id;
|
|
|
|
for (style_id = 0; style_id <= 127; style_id++)
|
|
{
|
|
gint val;
|
|
|
|
val = sci_get_value(oldsci, SCI_STYLEGETFORE, style_id);
|
|
scintilla_send_message(newsci, SCI_STYLESETFORE, style_id, val);
|
|
val = sci_get_value(oldsci, SCI_STYLEGETBACK, style_id);
|
|
scintilla_send_message(newsci, SCI_STYLESETBACK, style_id, val);
|
|
val = sci_get_value(oldsci, SCI_STYLEGETBOLD, style_id);
|
|
scintilla_send_message(newsci, SCI_STYLESETBOLD, style_id, val);
|
|
val = sci_get_value(oldsci, SCI_STYLEGETITALIC, style_id);
|
|
scintilla_send_message(newsci, SCI_STYLESETITALIC, style_id, val);
|
|
}
|
|
}
|
|
|
|
|
|
static void sci_set_font(ScintillaObject *sci, gint style, const gchar *font,
|
|
gint size)
|
|
{
|
|
scintilla_send_message(sci, SCI_STYLESETFONT, style, (sptr_t) font);
|
|
scintilla_send_message(sci, SCI_STYLESETSIZE, style, size);
|
|
}
|
|
|
|
|
|
static void update_font(ScintillaObject *current, ScintillaObject *sci)
|
|
{
|
|
gint style_id;
|
|
gint size;
|
|
gchar font_name[1024]; /* should be big enough */
|
|
|
|
scintilla_send_message(current, SCI_STYLEGETFONT, 0, (sptr_t)font_name);
|
|
size = sci_get_value(current, SCI_STYLEGETSIZE, 0);
|
|
|
|
for (style_id = 0; style_id <= 127; style_id++)
|
|
{
|
|
sci_set_font(sci, style_id, font_name, size);
|
|
}
|
|
|
|
/* line number and braces */
|
|
sci_set_font(sci, STYLE_LINENUMBER, font_name, size);
|
|
sci_set_font(sci, STYLE_BRACELIGHT, font_name, size);
|
|
sci_set_font(sci, STYLE_BRACEBAD, font_name, size);
|
|
}
|
|
|
|
|
|
/* line numbers visibility */
|
|
static void set_line_numbers(ScintillaObject * sci, gboolean set, gint extra_width)
|
|
{
|
|
if (set)
|
|
{
|
|
gchar tmp_str[15];
|
|
gint len = scintilla_send_message(sci, SCI_GETLINECOUNT, 0, 0);
|
|
gint width;
|
|
g_snprintf(tmp_str, 15, "_%d%d", len, extra_width);
|
|
width = scintilla_send_message(sci, SCI_TEXTWIDTH, STYLE_LINENUMBER, (sptr_t) tmp_str);
|
|
scintilla_send_message(sci, SCI_SETMARGINWIDTHN, 0, width);
|
|
scintilla_send_message(sci, SCI_SETMARGINSENSITIVEN, 0, FALSE); /* use default behaviour */
|
|
}
|
|
else
|
|
{
|
|
scintilla_send_message(sci, SCI_SETMARGINWIDTHN, 0, 0 );
|
|
}
|
|
}
|
|
|
|
|
|
static void sync_to_current(ScintillaObject *sci, ScintillaObject *current)
|
|
{
|
|
gpointer sdoc;
|
|
gint lexer;
|
|
gint pos;
|
|
|
|
/* set the new sci widget to view the existing Scintilla document */
|
|
sdoc = (gpointer) scintilla_send_message(current, SCI_GETDOCPOINTER, 0, 0);
|
|
scintilla_send_message(sci, SCI_SETDOCPOINTER, 0, (sptr_t) sdoc);
|
|
|
|
update_font(current, sci);
|
|
lexer = scintilla_send_message(current, SCI_GETLEXER, 0, 0);
|
|
scintilla_send_message(sci, SCI_SETLEXER, lexer, 0);
|
|
set_styles(current, sci);
|
|
|
|
pos = sci_get_current_position(current);
|
|
sci_set_current_position(sci, pos, TRUE);
|
|
|
|
/* override some defaults */
|
|
set_line_numbers(sci, TRUE, 0);
|
|
scintilla_send_message(sci, SCI_SETMARGINWIDTHN, 1, 0 ); /* hide marker margin */
|
|
}
|
|
|
|
|
|
static void set_editor(EditWindow *editwin, GeanyEditor *editor)
|
|
{
|
|
editwin->editor = editor;
|
|
|
|
/* first destroy any widget, otherwise its signals will have an
|
|
* invalid document as user_data */
|
|
if (editwin->sci != NULL)
|
|
gtk_widget_destroy(GTK_WIDGET(editwin->sci));
|
|
|
|
editwin->sci = editor_create_widget(editor);
|
|
gtk_widget_show(GTK_WIDGET(editwin->sci));
|
|
gtk_container_add(GTK_CONTAINER(editwin->vbox), GTK_WIDGET(editwin->sci));
|
|
|
|
sync_to_current(editwin->sci, editor->sci);
|
|
|
|
gtk_label_set_text(GTK_LABEL(editwin->name_label), DOC_FILENAME(editor->document));
|
|
}
|
|
|
|
|
|
static void set_state(enum State id)
|
|
{
|
|
gtk_widget_set_sensitive(menu_items.horizontal,
|
|
(id != STATE_SPLIT_HORIZONTAL) && (id != STATE_SPLIT_VERTICAL));
|
|
gtk_widget_set_sensitive(menu_items.vertical,
|
|
(id != STATE_SPLIT_HORIZONTAL) && (id != STATE_SPLIT_VERTICAL));
|
|
gtk_widget_set_sensitive(menu_items.unsplit,
|
|
id != STATE_UNSPLIT);
|
|
|
|
plugin_state = id;
|
|
}
|
|
|
|
|
|
/* Avoid adding new string translations which are the same but without an underscore.
|
|
* @warning Heavy use may cause stack exhaustion. */
|
|
#define no_underscore(str)\
|
|
utils_str_remove_chars(utils_strdupa(str), "_")
|
|
|
|
/* Like strcpy, but can handle overlapping src and dest. */
|
|
static gchar *utils_str_copy(gchar *dest, const gchar *src)
|
|
{
|
|
gchar *cpy;
|
|
|
|
/* strcpy might not handle overlaps, so make a copy */
|
|
cpy = utils_strdupa(src);
|
|
return strcpy(dest, cpy);
|
|
}
|
|
|
|
|
|
/* Remove characters from a string, in place.
|
|
* @param chars Characters to remove.
|
|
* @return @a str */
|
|
/* Note: Could be more efficient by writing non-chars into new string then using only one strcpy. */
|
|
static gchar *utils_str_remove_chars(gchar *str, const gchar *chars)
|
|
{
|
|
gchar *ptr;
|
|
gsize len;
|
|
|
|
g_return_val_if_fail(str, NULL);
|
|
|
|
len = strlen(str);
|
|
ptr = str;
|
|
|
|
while (ptr < str + len)
|
|
{
|
|
if (strchr(chars, *ptr))
|
|
{
|
|
utils_str_copy(ptr, ptr + 1);
|
|
len--;
|
|
}
|
|
else
|
|
ptr++;
|
|
}
|
|
return str;
|
|
}
|
|
|
|
|
|
static const gchar *ui_get_stock_label(const gchar *stock_id)
|
|
{
|
|
GtkStockItem item;
|
|
|
|
if (gtk_stock_lookup(stock_id, &item))
|
|
return item.label;
|
|
|
|
g_warning("No stock id '%s'!", stock_id);
|
|
return "";
|
|
}
|
|
|
|
|
|
/* Create a GtkToolButton with stock icon and tooltip.
|
|
* @param label can be NULL to use stock label text, without underscores.
|
|
* @param tooltip can be NULL to use label text (useful for GTK_TOOLBAR_ICONS). */
|
|
static GtkWidget *ui_tool_button_new(const gchar *stock_id, const gchar *label, const gchar *tooltip)
|
|
{
|
|
GtkToolItem *item;
|
|
gchar *dup = NULL;
|
|
|
|
if (stock_id && !label)
|
|
{
|
|
label = ui_get_stock_label(stock_id);
|
|
dup = g_strdup(label);
|
|
label = utils_str_remove_chars(dup, "_");
|
|
}
|
|
item = gtk_tool_button_new(NULL, label);
|
|
if (stock_id)
|
|
gtk_tool_button_set_icon_name(GTK_TOOL_BUTTON(item), stock_id);
|
|
|
|
if (!tooltip)
|
|
tooltip = label;
|
|
if (tooltip)
|
|
ui_widget_set_tooltip_text(GTK_WIDGET(item), tooltip);
|
|
|
|
g_free(dup);
|
|
return GTK_WIDGET(item);
|
|
}
|
|
|
|
|
|
static void on_refresh(void)
|
|
{
|
|
GeanyDocument *doc = document_get_current();
|
|
|
|
g_return_if_fail(doc);
|
|
g_return_if_fail(edit_window.sci);
|
|
|
|
set_editor(&edit_window, doc->editor);
|
|
}
|
|
|
|
|
|
static GtkWidget *create_toolbar(void)
|
|
{
|
|
GtkWidget *toolbar, *item;
|
|
GtkToolItem *tool_item;
|
|
|
|
toolbar = gtk_toolbar_new();
|
|
gtk_toolbar_set_icon_size(GTK_TOOLBAR(toolbar), GTK_ICON_SIZE_MENU);
|
|
gtk_toolbar_set_style(GTK_TOOLBAR(toolbar), GTK_TOOLBAR_ICONS);
|
|
|
|
item = ui_tool_button_new(GTK_STOCK_JUMP_TO, "",
|
|
_("Show the current document"));
|
|
gtk_container_add(GTK_CONTAINER(toolbar), item);
|
|
g_signal_connect(item, "clicked", G_CALLBACK(on_refresh), NULL);
|
|
|
|
tool_item = gtk_tool_item_new();
|
|
gtk_tool_item_set_expand(tool_item, TRUE);
|
|
gtk_container_add(GTK_CONTAINER(toolbar), GTK_WIDGET(tool_item));
|
|
|
|
item = gtk_label_new(NULL);
|
|
gtk_label_set_ellipsize(GTK_LABEL(item), PANGO_ELLIPSIZE_START);
|
|
gtk_container_add(GTK_CONTAINER(tool_item), item);
|
|
edit_window.name_label = item;
|
|
|
|
item = ui_tool_button_new(GTK_STOCK_CLOSE, no_underscore(_("_Unsplit")), NULL);
|
|
gtk_container_add(GTK_CONTAINER(toolbar), item);
|
|
g_signal_connect(item, "clicked", G_CALLBACK(on_unsplit), NULL);
|
|
|
|
return toolbar;
|
|
}
|
|
|
|
|
|
static void split_view(gboolean horizontal)
|
|
{
|
|
GtkWidget *notebook = geany_data->main_widgets->notebook;
|
|
GtkWidget *parent = gtk_widget_get_parent(notebook);
|
|
GtkWidget *pane, *toolbar, *box;
|
|
GeanyDocument *doc = document_get_current();
|
|
gint width = notebook->allocation.width / 2;
|
|
gint height = notebook->allocation.height / 2;
|
|
|
|
g_return_if_fail(doc);
|
|
g_return_if_fail(edit_window.editor == NULL);
|
|
|
|
set_state(horizontal ? STATE_SPLIT_HORIZONTAL : STATE_SPLIT_VERTICAL);
|
|
|
|
/* temporarily put document notebook in main vbox (scintilla widgets must stay
|
|
* in a visible parent window, otherwise there are X selection and scrollbar issues) */
|
|
gtk_widget_reparent(notebook,
|
|
ui_lookup_widget(geany->main_widgets->window, "vbox1"));
|
|
|
|
pane = horizontal ? gtk_hpaned_new() : gtk_vpaned_new();
|
|
gtk_container_add(GTK_CONTAINER(parent), pane);
|
|
gtk_widget_reparent(notebook, pane);
|
|
|
|
box = gtk_vbox_new(FALSE, 0);
|
|
toolbar = create_toolbar();
|
|
gtk_box_pack_start(GTK_BOX(box), toolbar, FALSE, FALSE, 0);
|
|
gtk_container_add(GTK_CONTAINER(pane), box);
|
|
edit_window.vbox = box;
|
|
|
|
set_editor(&edit_window, doc->editor);
|
|
|
|
if (horizontal)
|
|
{
|
|
gtk_paned_set_position(GTK_PANED(pane), width);
|
|
}
|
|
else
|
|
{
|
|
gtk_paned_set_position(GTK_PANED(pane), height);
|
|
}
|
|
gtk_widget_show_all(pane);
|
|
}
|
|
|
|
|
|
static void on_split_horizontally(GtkMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
split_view(TRUE);
|
|
}
|
|
|
|
|
|
static void on_split_vertically(GtkMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
split_view(FALSE);
|
|
}
|
|
|
|
|
|
static void on_unsplit(GtkMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
GtkWidget *notebook = geany_data->main_widgets->notebook;
|
|
GtkWidget *pane = gtk_widget_get_parent(notebook);
|
|
GtkWidget *parent = gtk_widget_get_parent(pane);
|
|
|
|
set_state(STATE_UNSPLIT);
|
|
|
|
g_return_if_fail(edit_window.editor);
|
|
|
|
/* temporarily put document notebook in main vbox (scintilla widgets must stay
|
|
* in a visible parent window, otherwise there are X selection and scrollbar issues) */
|
|
gtk_widget_reparent(notebook,
|
|
ui_lookup_widget(geany->main_widgets->window, "vbox1"));
|
|
|
|
gtk_widget_destroy(pane);
|
|
edit_window.editor = NULL;
|
|
edit_window.sci = NULL;
|
|
gtk_widget_reparent(notebook, parent);
|
|
}
|
|
|
|
|
|
static void kb_activate(guint key_id)
|
|
{
|
|
switch (key_id)
|
|
{
|
|
case KB_SPLIT_HORIZONTAL:
|
|
if (plugin_state == STATE_UNSPLIT)
|
|
split_view(TRUE);
|
|
break;
|
|
case KB_SPLIT_VERTICAL:
|
|
if (plugin_state == STATE_UNSPLIT)
|
|
split_view(FALSE);
|
|
break;
|
|
case KB_SPLIT_UNSPLIT:
|
|
if (plugin_state != STATE_UNSPLIT)
|
|
on_unsplit(NULL, NULL);
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void plugin_init(GeanyData *data)
|
|
{
|
|
GtkWidget *item, *menu;
|
|
|
|
menu_items.main = item = gtk_menu_item_new_with_mnemonic(_("_Split Window"));
|
|
gtk_menu_shell_append(GTK_MENU_SHELL(geany_data->main_widgets->tools_menu), item);
|
|
ui_add_document_sensitive(item);
|
|
|
|
menu = gtk_menu_new();
|
|
gtk_menu_item_set_submenu(GTK_MENU_ITEM(menu_items.main), menu);
|
|
|
|
menu_items.horizontal = item =
|
|
gtk_menu_item_new_with_mnemonic(_("_Horizontally"));
|
|
g_signal_connect(item, "activate", G_CALLBACK(on_split_horizontally), NULL);
|
|
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
|
|
|
|
menu_items.vertical = item =
|
|
gtk_menu_item_new_with_mnemonic(_("_Vertically"));
|
|
g_signal_connect(item, "activate", G_CALLBACK(on_split_vertically), NULL);
|
|
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
|
|
|
|
menu_items.unsplit = item =
|
|
gtk_menu_item_new_with_mnemonic(_("_Unsplit"));
|
|
g_signal_connect(item, "activate", G_CALLBACK(on_unsplit), NULL);
|
|
gtk_menu_shell_append(GTK_MENU_SHELL(menu), item);
|
|
|
|
gtk_widget_show_all(menu_items.main);
|
|
|
|
set_state(STATE_UNSPLIT);
|
|
|
|
/* setup keybindings */
|
|
keybindings_set_item(plugin_key_group, KB_SPLIT_HORIZONTAL, kb_activate,
|
|
0, 0, "split_horizontal", _("Split Horizontally"), menu_items.horizontal);
|
|
keybindings_set_item(plugin_key_group, KB_SPLIT_VERTICAL, kb_activate,
|
|
0, 0, "split_vertical", _("Split Vertically"), menu_items.vertical);
|
|
keybindings_set_item(plugin_key_group, KB_SPLIT_UNSPLIT, kb_activate,
|
|
0, 0, "split_unsplit", _("Unsplit"), menu_items.unsplit);
|
|
}
|
|
|
|
|
|
static void on_document_close(GObject *obj, GeanyDocument *doc, gpointer user_data)
|
|
{
|
|
/* remove the split window because the document won't exist anymore */
|
|
if (doc->editor == edit_window.editor)
|
|
on_unsplit(NULL, NULL);
|
|
}
|
|
|
|
|
|
static void on_document_save(GObject *obj, GeanyDocument *doc, gpointer user_data)
|
|
{
|
|
/* update filename */
|
|
if (doc->editor == edit_window.editor)
|
|
gtk_label_set_text(GTK_LABEL(edit_window.name_label), DOC_FILENAME(doc));
|
|
}
|
|
|
|
|
|
PluginCallback plugin_callbacks[] =
|
|
{
|
|
{ "document-close", (GCallback) &on_document_close, FALSE, NULL },
|
|
{ "document-save", (GCallback) &on_document_save, FALSE, NULL },
|
|
{ NULL, NULL, FALSE, NULL }
|
|
};
|
|
|
|
|
|
void plugin_cleanup(void)
|
|
{
|
|
if (plugin_state != STATE_UNSPLIT)
|
|
on_unsplit(NULL, NULL);
|
|
|
|
gtk_widget_destroy(menu_items.main);
|
|
}
|