X-Git-Url: https://git.stderr.nl/gitweb?a=blobdiff_plain;f=libchimara%2Finput.c;h=6a050005e9a555094bd09351e00e2269ef64ced8;hb=2925a5dc60b950752eb86c96ffb380ed40e4fe1b;hp=c35564593ec0a5d6990cb5d34901cb422f0a1ea7;hpb=64d6d61f49c8400e61c3db0b3e31705776d63232;p=rodin%2Fchimara.git diff --git a/libchimara/input.c b/libchimara/input.c index c355645..6a05000 100644 --- a/libchimara/input.c +++ b/libchimara/input.c @@ -153,6 +153,8 @@ text_grid_request_line_event_common(winid_t win, glui32 maxlen, gboolean insert, gtk_widget_modify_base(win->input_entry, GTK_STATE_NORMAL, &background); g_signal_connect(win->input_entry, "activate", G_CALLBACK(on_input_entry_activate), win); + g_signal_connect(win->input_entry, "key-press-event", G_CALLBACK(on_input_entry_key_press_event), win); + win->line_input_entry_changed = g_signal_connect(win->input_entry, "changed", G_CALLBACK(on_input_entry_changed), win); gtk_widget_show(win->input_entry); gtk_text_view_add_child_at_anchor(GTK_TEXT_VIEW(win->widget), win->input_entry, win->input_anchor); @@ -188,11 +190,18 @@ text_buffer_request_line_event_common(winid_t win, glui32 maxlen, gboolean inser gtk_text_buffer_apply_tag_by_name(buffer, "uneditable", &start_iter, &end_iter); /* Insert pre-entered text if needed */ - if(insert) + if(insert) { gtk_text_buffer_insert(buffer, &end_iter, inserttext, -1); + gtk_text_buffer_get_end_iter(buffer, &end_iter); /* update after text insertion */ + } /* Scroll to input point */ gtk_text_view_scroll_mark_onscreen(GTK_TEXT_VIEW(win->widget), input_position); + + /* Apply the correct style to the input prompt */ + GtkTextIter input_iter; + gtk_text_buffer_get_iter_at_mark(buffer, &input_iter, input_position); + gtk_text_buffer_apply_tag_by_name(buffer, "input", &input_iter, &end_iter); gtk_text_view_set_editable(GTK_TEXT_VIEW(win->widget), TRUE); g_signal_handler_unblock(buffer, win->insert_text_handler); @@ -610,8 +619,18 @@ finish_text_grid_line_input(winid_t win, gboolean emit_signal) g_assert(glk); g_signal_emit_by_name(glk, "line-input", win->rock, text); } + + /* Add the text to the window input history */ + if(win->history_pos != NULL) + { + g_free(win->history->data); + win->history = g_list_delete_link(win->history, win->history); + } + if(*text != 0) + win->history = g_list_prepend(win->history, g_strdup(text)); + win->history_pos = NULL; + g_free(text); - return chars_written; } @@ -623,6 +642,8 @@ pasted into the window. */ void after_window_insert_text(GtkTextBuffer *textbuffer, GtkTextIter *location, gchar *text, gint len, winid_t win) { + GtkTextBuffer *window_buffer = gtk_text_view_get_buffer( GTK_TEXT_VIEW(win->widget) ); + /* Set the history position to NULL and erase the text we were already editing */ if(win->history_pos != NULL) { @@ -633,7 +654,6 @@ after_window_insert_text(GtkTextBuffer *textbuffer, GtkTextIter *location, gchar if( strchr(text, '\n') != NULL ) { /* Remove signal handlers */ - GtkTextBuffer *window_buffer = gtk_text_view_get_buffer( GTK_TEXT_VIEW(win->widget) ); g_signal_handler_block(window_buffer, win->insert_text_handler); /* Make the window uneditable again and retrieve the text that was input */ @@ -643,6 +663,14 @@ after_window_insert_text(GtkTextBuffer *textbuffer, GtkTextIter *location, gchar ChimaraGlk *glk = CHIMARA_GLK(gtk_widget_get_ancestor(win->widget, CHIMARA_TYPE_GLK)); event_throw(glk, evtype_LineInput, win, chars_written, 0); } + + /* Apply the 'input' style to the text that was entered */ + GtkTextIter end_iter; + gtk_text_buffer_get_end_iter(window_buffer, &end_iter); + GtkTextIter input_iter; + GtkTextMark *input_position = gtk_text_buffer_get_mark(window_buffer, "input_position"); + gtk_text_buffer_get_iter_at_mark(window_buffer, &input_iter, input_position); + gtk_text_buffer_apply_tag_by_name(window_buffer, "input", &input_iter, &end_iter); } /* Internal function: Callback for signal activate on the line input GtkEntry @@ -657,6 +685,60 @@ on_input_entry_activate(GtkEntry *input_entry, winid_t win) event_throw(glk, evtype_LineInput, win, chars_written, 0); } +/* Internal function: Callback for signal key-press-event on the line input +GtkEntry in a text grid window. */ +gboolean +on_input_entry_key_press_event(GtkEntry *input_entry, GdkEventKey *event, winid_t win) +{ + if(event->keyval == GDK_Up || event->keyval == GDK_KP_Up + || event->keyval == GDK_Down || event->keyval == GDK_KP_Down) + { + /* Prevent falling off the end of the history list */ + if( (event->keyval == GDK_Up || event->keyval == GDK_KP_Up) + && win->history_pos && win->history_pos->next == NULL) + return TRUE; + if( (event->keyval == GDK_Down || event->keyval == GDK_KP_Down) + && (win->history_pos == NULL || win->history_pos->prev == NULL) ) + return TRUE; + + if(win->history_pos == NULL) + { + gchar *current_input = gtk_entry_get_text(input_entry); + win->history = g_list_prepend(win->history, current_input); + win->history_pos = win->history; + } + + if(event->keyval == GDK_Up || event->keyval == GDK_KP_Up) + { + if(win->history_pos) + win->history_pos = g_list_next(win->history_pos); + else + win->history_pos = win->history; + } + else /* down */ + win->history_pos = g_list_previous(win->history_pos); + + /* Insert the history item into the window */ + g_signal_handler_block(input_entry, win->line_input_entry_changed); + gtk_entry_set_text(input_entry, win->history_pos->data); + g_signal_handler_unblock(input_entry, win->line_input_entry_changed); + return TRUE; + } + return FALSE; +} + +void +on_input_entry_changed(GtkEditable *editable, winid_t win) +{ + /* Set the history position to NULL and erase the text we were already editing */ + if(win->history_pos != NULL) + { + g_free(win->history->data); + win->history = g_list_delete_link(win->history, win->history); + win->history_pos = NULL; + } +} + glui32 keyval_to_glk_keycode(guint keyval, gboolean unicode) {