Rewrite tab switching queue
There was one more bug related to the tab switching. When we switch so many times that we return back to the original document (so we actually don't switch at all) then the following switch attempt doesn't switch immediately to the next document. After spending two hours thinking what is wrong, I gave up and rewrote the whole thing in a different way. The problem with the previous implementation was that before you couldn't just look what's in the queue "now" - you had to imagine what will be inserted there in the next step because the switch_in_progress variable was set after the first switch (this is also why I put the long comment why mru_pos = 2 - that is not clear at all when you first look at it). Also there were some not very nice "workarounds" like the idle function that was executed after the switch and removed the double entry on top of the queue. So with the new implementation things are much simpler IMO. The queue starts with the current document and the previously opened documments follow. It's *always* like that, no exceptions. The idle function is gone and cb_func_switch_tablastused() is simplified too. The rest of the functionality should be clear from the code. Signed-off-by: Jiří Techet <techet@gmail.com>
This commit is contained in:
parent
d0892b95d1
commit
0acb273c55
@ -597,17 +597,12 @@ static void init_default_kb(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* before the tab changes, add the current document to the MRU list */
|
static void update_mru_docs_head(GeanyDocument *doc)
|
||||||
static void on_notebook_switch_page(void)
|
|
||||||
{
|
{
|
||||||
GeanyDocument *old = document_get_current();
|
if (doc)
|
||||||
|
|
||||||
/* when closing current doc, old is NULL.
|
|
||||||
* Don't add to the mru list when switch dialog is visible. */
|
|
||||||
if (old && !switch_in_progress)
|
|
||||||
{
|
{
|
||||||
g_queue_remove(mru_docs, old);
|
g_queue_remove(mru_docs, doc);
|
||||||
g_queue_push_head(mru_docs, old);
|
g_queue_push_head(mru_docs, doc);
|
||||||
|
|
||||||
if (g_queue_get_length(mru_docs) > MAX_MRU_DOCS)
|
if (g_queue_get_length(mru_docs) > MAX_MRU_DOCS)
|
||||||
g_queue_pop_tail(mru_docs);
|
g_queue_pop_tail(mru_docs);
|
||||||
@ -615,16 +610,21 @@ static void on_notebook_switch_page(void)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* really this should be just after a document was closed, not idle */
|
/* before the tab changes, add the current document to the MRU list */
|
||||||
static gboolean on_idle_close(gpointer data)
|
static void on_notebook_switch_page(GtkNotebook *notebook,
|
||||||
|
GtkNotebookPage *page, guint page_num, gpointer user_data)
|
||||||
{
|
{
|
||||||
GeanyDocument *current;
|
GeanyDocument *new;
|
||||||
|
|
||||||
current = document_get_current();
|
new = document_get_from_page(page_num);
|
||||||
if (current && g_queue_peek_head(mru_docs) == current)
|
|
||||||
g_queue_pop_head(mru_docs);
|
|
||||||
|
|
||||||
return FALSE;
|
/* insert the very first document (when adding the second document
|
||||||
|
* and switching to it) */
|
||||||
|
if (g_queue_get_length(mru_docs) == 0 && gtk_notebook_get_n_pages(notebook) == 2)
|
||||||
|
update_mru_docs_head(document_get_current());
|
||||||
|
|
||||||
|
if (!switch_in_progress)
|
||||||
|
update_mru_docs_head(new);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -634,7 +634,7 @@ static void on_document_close(GObject *obj, GeanyDocument *doc)
|
|||||||
{
|
{
|
||||||
GeanyDocument *last_doc;
|
GeanyDocument *last_doc;
|
||||||
|
|
||||||
last_doc = g_queue_peek_head(mru_docs);
|
last_doc = g_queue_peek_nth(mru_docs, 1);
|
||||||
|
|
||||||
if (DOC_VALID(last_doc) && document_get_current() == doc)
|
if (DOC_VALID(last_doc) && document_get_current() == doc)
|
||||||
{
|
{
|
||||||
@ -642,8 +642,10 @@ static void on_document_close(GObject *obj, GeanyDocument *doc)
|
|||||||
document_get_notebook_page(last_doc));
|
document_get_notebook_page(last_doc));
|
||||||
}
|
}
|
||||||
g_queue_remove(mru_docs, doc);
|
g_queue_remove(mru_docs, doc);
|
||||||
|
/* this prevents the pop up window from showing when there's a single
|
||||||
g_idle_add(on_idle_close, NULL);
|
* document */
|
||||||
|
if (gtk_notebook_get_n_pages(GTK_NOTEBOOK(main_widgets.notebook)) == 2)
|
||||||
|
g_queue_clear(mru_docs);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1768,6 +1770,7 @@ static gboolean on_key_release_event(GtkWidget *widget, GdkEventKey *ev, gpointe
|
|||||||
switch_dialog = NULL;
|
switch_dialog = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
update_mru_docs_head(document_get_current());
|
||||||
mru_pos = 0;
|
mru_pos = 0;
|
||||||
}
|
}
|
||||||
return FALSE;
|
return FALSE;
|
||||||
@ -1845,7 +1848,11 @@ static gboolean on_switch_timeout(G_GNUC_UNUSED gpointer data)
|
|||||||
|
|
||||||
static void cb_func_switch_tablastused(G_GNUC_UNUSED guint key_id)
|
static void cb_func_switch_tablastused(G_GNUC_UNUSED guint key_id)
|
||||||
{
|
{
|
||||||
GeanyDocument *last_doc = g_queue_peek_nth(mru_docs, mru_pos);
|
GeanyDocument *last_doc;
|
||||||
|
gboolean switch_start = !switch_in_progress;
|
||||||
|
|
||||||
|
mru_pos += 1;
|
||||||
|
last_doc = g_queue_peek_nth(mru_docs, mru_pos);
|
||||||
|
|
||||||
if (! DOC_VALID(last_doc))
|
if (! DOC_VALID(last_doc))
|
||||||
{
|
{
|
||||||
@ -1856,29 +1863,15 @@ static void cb_func_switch_tablastused(G_GNUC_UNUSED guint key_id)
|
|||||||
if (! DOC_VALID(last_doc))
|
if (! DOC_VALID(last_doc))
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
switch_in_progress = TRUE;
|
||||||
document_show_tab(last_doc);
|
document_show_tab(last_doc);
|
||||||
|
|
||||||
/* if there's a modifier key, we can switch back in MRU order each time unless
|
/* if there's a modifier key, we can switch back in MRU order each time unless
|
||||||
* the key is released */
|
* the key is released */
|
||||||
if (!switch_in_progress)
|
if (switch_start)
|
||||||
{
|
|
||||||
switch_in_progress = TRUE;
|
|
||||||
|
|
||||||
/* because switch_in_progress was not set when we called
|
|
||||||
* gtk_notebook_set_current_page() above, this function inserted last_doc
|
|
||||||
* into the queue => on mru_pos = 0 there is last_doc, on mru_pos = 1
|
|
||||||
* there is the currently displayed doc, so we want to continue from 2
|
|
||||||
* next time this function is called */
|
|
||||||
mru_pos = 2;
|
|
||||||
|
|
||||||
/* delay showing dialog to give user time to let go of any modifier keys */
|
|
||||||
g_timeout_add(600, on_switch_timeout, NULL);
|
g_timeout_add(600, on_switch_timeout, NULL);
|
||||||
}
|
|
||||||
else
|
else
|
||||||
{
|
|
||||||
update_filename_label();
|
update_filename_label();
|
||||||
mru_pos += 1;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user