2007-06-02 16:14:07 +00:00
|
|
|
/*
|
|
|
|
* navqueue.c - this file is part of Geany, a fast and lightweight IDE
|
|
|
|
*
|
2008-01-06 18:11:57 +00:00
|
|
|
* Copyright 2007 Dave Moore <wrex006(at)gmail(dot)com>
|
|
|
|
* Copyright 2007-2008 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
|
|
|
|
* Copyright 2007-2008 Nick Treleaven <nick(dot)treleaven(at)btinternet(dot)com>
|
2007-06-02 16:14:07 +00:00
|
|
|
*
|
|
|
|
* 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.
|
|
|
|
*
|
2007-06-03 16:15:53 +00:00
|
|
|
* $Id$
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Simple code navigation
|
2007-06-02 16:14:07 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include "geany.h"
|
|
|
|
|
|
|
|
#include "navqueue.h"
|
|
|
|
#include "sciwrappers.h"
|
|
|
|
#include "document.h"
|
|
|
|
#include "utils.h"
|
2007-08-23 11:34:06 +00:00
|
|
|
#include "support.h"
|
2007-06-02 16:14:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
// for the navigation history queue
|
|
|
|
typedef struct
|
|
|
|
{
|
2007-07-17 16:11:38 +00:00
|
|
|
gchar *file; // this is the tagmanager filename, not document::file_name
|
2007-06-02 16:14:07 +00:00
|
|
|
/// TODO maybe it is better to work on positions than on lines to be more accurate when
|
|
|
|
/// switching back or forward, sci_get_position_from_line() could be used for tm_tag lines
|
2007-07-17 16:11:38 +00:00
|
|
|
gint line; // line is counted with 1 as the first line, not 0
|
2007-06-02 16:14:07 +00:00
|
|
|
} filepos;
|
|
|
|
|
2007-08-23 11:34:06 +00:00
|
|
|
static GQueue *navigation_queue;
|
|
|
|
static guint nav_queue_pos;
|
|
|
|
|
|
|
|
static GtkWidget *navigation_buttons[2];
|
|
|
|
|
2007-06-02 16:14:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
void navqueue_init()
|
|
|
|
{
|
|
|
|
navigation_queue = g_queue_new();
|
|
|
|
nav_queue_pos = 0;
|
2007-08-23 11:34:06 +00:00
|
|
|
|
|
|
|
navigation_buttons[0] = lookup_widget(app->window, "toolbutton_back");
|
|
|
|
navigation_buttons[1] = lookup_widget(app->window, "toolbutton_forward");
|
2007-06-02 16:14:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void navqueue_free()
|
|
|
|
{
|
|
|
|
while (! g_queue_is_empty(navigation_queue))
|
|
|
|
{
|
|
|
|
g_free(g_queue_pop_tail(navigation_queue));
|
|
|
|
}
|
|
|
|
g_queue_free(navigation_queue);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-20 11:24:23 +00:00
|
|
|
static void adjust_buttons(void)
|
2007-06-02 16:14:07 +00:00
|
|
|
{
|
|
|
|
if (g_queue_get_length(navigation_queue) < 2)
|
|
|
|
{
|
2007-08-23 11:34:06 +00:00
|
|
|
gtk_widget_set_sensitive(navigation_buttons[0], FALSE);
|
|
|
|
gtk_widget_set_sensitive(navigation_buttons[1], FALSE);
|
2007-06-02 16:14:07 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
if (nav_queue_pos == 0)
|
|
|
|
{
|
2007-08-23 11:34:06 +00:00
|
|
|
gtk_widget_set_sensitive(navigation_buttons[0], TRUE);
|
|
|
|
gtk_widget_set_sensitive(navigation_buttons[1], FALSE);
|
2007-06-02 16:14:07 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
// forward should be sensitive since where not at the start
|
2007-08-23 11:34:06 +00:00
|
|
|
gtk_widget_set_sensitive(navigation_buttons[1], TRUE);
|
2007-06-02 16:14:07 +00:00
|
|
|
|
|
|
|
// back should be sensitive if there's a place to go back to
|
|
|
|
(nav_queue_pos < g_queue_get_length(navigation_queue) - 1) ?
|
2007-08-23 11:34:06 +00:00
|
|
|
gtk_widget_set_sensitive(navigation_buttons[0], TRUE) :
|
|
|
|
gtk_widget_set_sensitive(navigation_buttons[0], FALSE);
|
2007-06-02 16:14:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-17 16:11:38 +00:00
|
|
|
static gboolean
|
|
|
|
queue_pos_matches(guint queue_pos, const gchar *fname, gint line)
|
|
|
|
{
|
|
|
|
if (queue_pos < g_queue_get_length(navigation_queue))
|
|
|
|
{
|
|
|
|
filepos *fpos = g_queue_peek_nth(navigation_queue, queue_pos);
|
|
|
|
|
|
|
|
return (utils_str_equal(fpos->file, fname) && fpos->line == line);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-09-25 12:39:41 +00:00
|
|
|
static void add_new_position(gchar *tm_filename, gint line)
|
2007-06-02 16:14:07 +00:00
|
|
|
{
|
|
|
|
filepos *npos;
|
|
|
|
guint i;
|
|
|
|
|
2007-07-17 16:11:38 +00:00
|
|
|
if (queue_pos_matches(nav_queue_pos, tm_filename, line))
|
|
|
|
return; // prevent duplicates
|
|
|
|
|
2007-06-02 16:14:07 +00:00
|
|
|
npos = g_new0(filepos, 1);
|
2007-07-17 16:11:38 +00:00
|
|
|
npos->file = tm_filename;
|
2007-06-02 16:14:07 +00:00
|
|
|
npos->line = line;
|
|
|
|
|
|
|
|
// if we've jumped to a new position from
|
|
|
|
// inside the queue rather than going forward
|
|
|
|
if (nav_queue_pos > 0)
|
|
|
|
{
|
|
|
|
for (i = 0; i < nav_queue_pos; i++)
|
|
|
|
{
|
|
|
|
g_free(g_queue_pop_head(navigation_queue));
|
|
|
|
}
|
|
|
|
nav_queue_pos = 0;
|
|
|
|
}
|
|
|
|
g_queue_push_head(navigation_queue, npos);
|
|
|
|
adjust_buttons();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-07-18 12:20:13 +00:00
|
|
|
/* Adds the current document position to the queue before adding the new position.
|
|
|
|
* line is counted with 1 as the first line, not 0. */
|
2008-02-10 12:34:28 +00:00
|
|
|
gboolean navqueue_goto_line(gint old_idx, gint new_idx, gint line)
|
2007-07-17 16:11:38 +00:00
|
|
|
{
|
|
|
|
g_return_val_if_fail(DOC_IDX_VALID(old_idx), FALSE);
|
|
|
|
g_return_val_if_fail(DOC_IDX_VALID(new_idx), FALSE);
|
2007-09-25 12:39:41 +00:00
|
|
|
g_return_val_if_fail(doc_list[new_idx].tm_file, FALSE);
|
2007-07-17 16:11:38 +00:00
|
|
|
g_return_val_if_fail(line >= 1, FALSE);
|
|
|
|
|
|
|
|
// first add old file as old position
|
|
|
|
if (doc_list[old_idx].tm_file)
|
|
|
|
{
|
2007-09-25 16:44:33 +00:00
|
|
|
gint cur_line = sci_get_current_line(doc_list[old_idx].sci);
|
2007-07-17 16:11:38 +00:00
|
|
|
|
2007-09-25 12:39:41 +00:00
|
|
|
add_new_position(doc_list[old_idx].tm_file->file_name, cur_line + 1);
|
2007-07-17 16:11:38 +00:00
|
|
|
}
|
|
|
|
|
2007-09-25 12:39:41 +00:00
|
|
|
add_new_position(doc_list[new_idx].tm_file->file_name, line);
|
|
|
|
return utils_goto_line(new_idx, line);
|
2007-07-17 16:11:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-06-02 16:14:07 +00:00
|
|
|
void navqueue_go_back()
|
|
|
|
{
|
|
|
|
filepos *fprev;
|
|
|
|
|
|
|
|
// return if theres no place to go back to
|
|
|
|
if (g_queue_is_empty(navigation_queue) ||
|
|
|
|
nav_queue_pos >= g_queue_get_length(navigation_queue) - 1)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// jump back
|
|
|
|
fprev = g_queue_peek_nth(navigation_queue, nav_queue_pos + 1);
|
|
|
|
if (utils_goto_file_line(fprev->file, TRUE, fprev->line))
|
|
|
|
{
|
|
|
|
nav_queue_pos++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// TODO: add option to re open the file
|
2007-07-18 12:20:13 +00:00
|
|
|
g_free(g_queue_pop_nth(navigation_queue, nav_queue_pos + 1));
|
2007-06-02 16:14:07 +00:00
|
|
|
}
|
|
|
|
adjust_buttons();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void navqueue_go_forward()
|
|
|
|
{
|
|
|
|
filepos *fnext;
|
|
|
|
|
2007-07-18 12:20:13 +00:00
|
|
|
if (nav_queue_pos < 1 ||
|
2007-07-18 12:25:18 +00:00
|
|
|
nav_queue_pos >= g_queue_get_length(navigation_queue))
|
2007-06-02 16:14:07 +00:00
|
|
|
return;
|
|
|
|
|
|
|
|
// jump forward
|
|
|
|
fnext = g_queue_peek_nth(navigation_queue, nav_queue_pos - 1);
|
|
|
|
if (utils_goto_file_line(fnext->file, TRUE, fnext->line))
|
|
|
|
{
|
|
|
|
nav_queue_pos--;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/// TODO: add option to re open the file
|
2007-07-18 12:20:13 +00:00
|
|
|
g_free(g_queue_pop_nth(navigation_queue, nav_queue_pos - 1));
|
2007-06-02 16:14:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
adjust_buttons();
|
|
|
|
}
|
|
|
|
|