Overhauled the whole reverse video thing.
[projects/chimara/chimara.git] / libchimara / garglk.c
index aacd1d54a72b9d0dcc18278fde5738f1171b4987..4eb59a5fad97816fb17c869111a937513dde8cd7 100644 (file)
@@ -1,4 +1,6 @@
+#include <glib.h>
 #include <glib/gi18n.h>
+#include <glib/gprintf.h>
 #include <libchimara/glk.h>
 #include "chimara-glk-private.h"
 #include "stream.h"
@@ -164,11 +166,17 @@ garglk_unput_string_uni(glui32 *str)
 void
 garglk_set_zcolors_stream(strid_t str, glui32 fg, glui32 bg)
 {
+#ifdef DEBUG_STYLES
+       g_printf("garglk_set_zcolors_stream(str->rock=%d, fg=%08X, bg=%08X)\n", str->rock, fg, bg);
+#endif
+
        VALID_STREAM(str, return);
        g_return_if_fail(str->window != NULL);
 
        winid_t window = str->window;
 
+       gdk_threads_enter();
+
        GtkTextBuffer *buffer = gtk_text_view_get_buffer( GTK_TEXT_VIEW(window->widget) );
        GtkTextTagTable *tags = gtk_text_buffer_get_tag_table(buffer);
        GdkColor fore, back;
@@ -183,7 +191,7 @@ garglk_set_zcolors_stream(strid_t str, glui32 fg, glui32 bg)
                WARNING(_("zcolor_Transparent, zcolor_Cursor not implemented"));
                // Fallthrough to default
        case zcolor_Default:
-               fore_name = "default";
+               fore_name = g_strdup("default");
                break;
        case zcolor_Current:
        {
@@ -199,7 +207,7 @@ garglk_set_zcolors_stream(strid_t str, glui32 fg, glui32 bg)
                        fore.blue = current_color->blue;
                        fore_pointer = &fore;
                } else {
-                       fore_name = "default";
+                       fore_name = g_strdup("default");
                }
                break;
        }
@@ -219,7 +227,7 @@ garglk_set_zcolors_stream(strid_t str, glui32 fg, glui32 bg)
                WARNING(_("zcolor_Transparent, zcolor_Cursor not implemented"));
                // Fallthrough to default
        case zcolor_Default:
-               back_name = "default";
+               back_name = g_strdup("default");
                break;
        case zcolor_Current:
        {
@@ -235,7 +243,7 @@ garglk_set_zcolors_stream(strid_t str, glui32 fg, glui32 bg)
                        back.blue = current_color->blue;
                        back_pointer = &back;
                } else {
-                       back_name = "default";
+                       back_name = g_strdup("default");
                }
                break;
        }
@@ -249,12 +257,14 @@ garglk_set_zcolors_stream(strid_t str, glui32 fg, glui32 bg)
                );
        }
 
-       char *name = g_strdup_printf("zcolor:#%s/#%s", fore_name, back_name);
-
        if(fore_pointer == NULL && back_pointer == NULL) {
                // NULL value means to ignore the zcolor property altogether
                window->zcolor = NULL;
        } else {
+               char *name = g_strdup_printf("zcolor:#%s/#%s", fore_name, back_name);
+               g_free(fore_name);
+               g_free(back_name);
+
                // See if we have used this color combination before
                GtkTextTag *tag = gtk_text_tag_table_lookup(tags, name);
 
@@ -274,6 +284,8 @@ garglk_set_zcolors_stream(strid_t str, glui32 fg, glui32 bg)
                // From now on, text will be drawn in the specified colors
                window->zcolor = tag;
        }
+
+       gdk_threads_leave();
 }
 
 /**
@@ -298,25 +310,69 @@ garglk_set_zcolors(glui32 fg, glui32 bg)
        garglk_set_zcolors_stream(glk_data->current_stream, fg, bg);
 }
 
-static void
-apply_reverse_color(GtkTextTag *tag, gpointer data)
-{
-       const gchar *tag_name;
-       g_object_get(tag, "name", &tag_name, NULL);
-
-       if( g_str_has_prefix(tag_name, "glk-") )
-               g_object_set_data( G_OBJECT(tag), "reverse_color", data );
-}
-
 /* TODO document */
 void
 garglk_set_reversevideo_stream(strid_t str, glui32 reverse)
 {
+#ifdef DEBUG_STYLES
+       g_printf("garglk_set_reversevideo_stream(str->rock=%d, reverse=%d)\n", str->rock, reverse);
+#endif
+
        VALID_STREAM(str, return);
+       g_return_if_fail(str->window != NULL);
+       g_return_if_fail(str->window->type != wintype_TextBuffer || str->window->type != wintype_TextGrid);
+
+       // Determine the current colors
        
-       GtkTextBuffer *buffer = gtk_text_view_get_buffer( GTK_TEXT_VIEW(str->window->widget) );
+       // If all fails, use black/white
+       // FIXME: Use system theme here
+       GdkColor foreground, background;
+       gdk_color_parse("black", &foreground);
+       gdk_color_parse("white", &background);
+       GdkColor *current_foreground = &foreground;
+       GdkColor *current_background = &background;
+
+       gdk_threads_enter();
+
+       style_stream_colors(str, &current_foreground, &current_background);
+
+       if(reverse) {
+               GdkColor *temp = current_foreground;
+               current_foreground = current_background;
+               current_background = temp;
+       }
+
+       // Name the color
+       gchar *name = g_strdup_printf(
+               "zcolor:#%04X%04X%04X/#%04X%04X%04X",
+               current_foreground->red,
+               current_foreground->green,
+               current_foreground->blue,
+               current_background->red,
+               current_background->green,
+               current_background->blue
+       );
+
+       // Create a tag for the new colors if it doesn't exist yet
+       GtkTextBuffer *buffer = gtk_text_view_get_buffer( GTK_TEXT_VIEW(str->window->widget) ); 
        GtkTextTagTable *tags = gtk_text_buffer_get_tag_table(buffer);
-       gtk_text_tag_table_foreach( tags, apply_reverse_color, GINT_TO_POINTER(reverse) );
+       GtkTextTag *tag = gtk_text_tag_table_lookup(tags, name);
+       if(tag == NULL) {
+               tag = gtk_text_buffer_create_tag(
+                       buffer,
+                       name,
+                       "foreground-gdk", current_foreground,
+                       "foreground-set", TRUE,
+                       "background-gdk", current_background,
+                       "background-set", TRUE,
+                       NULL
+               );
+       }
+
+       // From now on, text will be drawn in the specified colors
+       str->window->zcolor_reversed = tag;
+       
+       gdk_threads_leave();
 }
 
 /**