diff options
author | Tom Feist <shabble@metavore.org> | 2011-05-03 15:40:34 +0100 |
---|---|---|
committer | Ailin Nemui <ailin@esf51.localdomain> | 2014-06-30 00:54:16 +0200 |
commit | 2d4edc51877719c49d712271967313310f4796fb (patch) | |
tree | a427b451338dd0e7b27b462a971528acdfbaca4d /src/fe-common | |
parent | 2e6f16c0faf345245c5a224de529827a51203d1c (diff) | |
download | irssi-2d4edc51877719c49d712271967313310f4796fb.zip |
Initial implementation of 256 colour support for Irssi
This patch implements some 256 colour support for Irssi up from the
previous 16 colours. Initial parsing of the %x/%X format codes is
implemented and the parser accounts in advances the char* for
that.
The colour attributes are widened from 4 to 8 bit. The colour protocol
is changed to a new format. Some pointers to remaining work are
written in the comment in textbuffer.h.
Note that Irssi already does support requesting 256 colours from the
terminal in the original source code, so this part did not have to be
touched.
Diffstat (limited to 'src/fe-common')
-rw-r--r-- | src/fe-common/core/fe-common-core.c | 52 | ||||
-rw-r--r-- | src/fe-common/core/fe-exec.c | 4 | ||||
-rw-r--r-- | src/fe-common/core/formats.c | 120 | ||||
-rw-r--r-- | src/fe-common/core/formats.h | 6 | ||||
-rw-r--r-- | src/fe-common/core/printtext.c | 13 |
5 files changed, 164 insertions, 31 deletions
diff --git a/src/fe-common/core/fe-common-core.c b/src/fe-common/core/fe-common-core.c index dce6890e..4c5fbb84 100644 --- a/src/fe-common/core/fe-common-core.c +++ b/src/fe-common/core/fe-common-core.c @@ -55,6 +55,9 @@ static int no_autoconnect; static char *cmdline_nick; static char *cmdline_hostname; +static char *irssi_logfile = NULL; +static FILE * logfile_FILE = NULL; + void fe_core_log_init(void); void fe_core_log_deinit(void); @@ -120,13 +123,26 @@ static void sig_channel_destroyed(CHANNEL_REC *channel) void fe_common_core_register_options(void) { - static GOptionEntry options[] = { - { "connect", 'c', 0, G_OPTION_ARG_STRING, &autocon_server, "Automatically connect to server/network", "SERVER" }, - { "password", 'w', 0, G_OPTION_ARG_STRING, &autocon_password, "Autoconnect password", "PASSWORD" }, - { "port", 'p', 0, G_OPTION_ARG_INT, &autocon_port, "Autoconnect port", "PORT" }, - { "noconnect", '!', 0, G_OPTION_ARG_NONE, &no_autoconnect, "Disable autoconnecting", NULL }, + static GOptionEntry options[] + = { + { "connect", 'c', 0, G_OPTION_ARG_STRING, &autocon_server, + "Automatically connect to server/network", "SERVER" + }, + { "password", 'w', 0, G_OPTION_ARG_STRING, &autocon_password, + "Autoconnect password", "PASSWORD" + }, + { "port", 'p', 0, G_OPTION_ARG_INT, &autocon_port, + "Autoconnect port", "PORT" }, + { "noconnect", '!', 0, G_OPTION_ARG_NONE, &no_autoconnect, + "Disable autoconnecting", NULL }, { "nick", 'n', 0, G_OPTION_ARG_STRING, &cmdline_nick, "Specify nick to use", NULL }, - { "hostname", 'h', 0, G_OPTION_ARG_STRING, &cmdline_hostname, "Specify host name to use", NULL }, + { "hostname", 'h', 0, G_OPTION_ARG_STRING, &cmdline_hostname, + "Specify host name to use", NULL }, + { "logfile", 0, 0, G_OPTION_ARG_FILENAME, &irssi_logfile, + "Logfile to write debugging data which would otherwise be printed " \ + "to STDERR or as Irssi internal messages", "PATH" + }, + { NULL } }; @@ -143,6 +159,9 @@ void fe_common_core_init(void) { const char *str; + logfile_FILE = g_fopen(irssi_logfile, "a"); + if (logfile_FILE == NULL) irssi_logfile = NULL; + settings_add_bool("lookandfeel", "timestamps", TRUE); settings_add_level("lookandfeel", "timestamp_level", "ALL"); settings_add_time("lookandfeel", "timestamp_timeout", "0"); @@ -243,6 +262,10 @@ void fe_common_core_deinit(void) signal_remove("server disconnected", (SIGNAL_FUNC) sig_disconnected); signal_remove("channel created", (SIGNAL_FUNC) sig_channel_created); signal_remove("channel destroyed", (SIGNAL_FUNC) sig_channel_destroyed); + + if (irssi_logfile && logfile_FILE) { + fclose(logfile_FILE); + } } void glog_func(const char *log_domain, GLogLevelFlags log_level, @@ -257,17 +280,24 @@ void glog_func(const char *log_domain, GLogLevelFlags log_level, case G_LOG_LEVEL_CRITICAL: reason = "critical"; break; + case G_LOG_LEVEL_MESSAGE: + reason = "msg"; + break; default: reason = "error"; break; } - if (windows == NULL) - fprintf(stderr, "GLib %s: %s\n", reason, message); - else { - printformat(NULL, NULL, MSGLEVEL_CLIENTERROR, - TXT_GLIB_ERROR, reason, message); + if (irssi_logfile != NULL && logfile_FILE != NULL) { + fprintf(logfile_FILE, "GLib: %s: %s", reason, message); + fflush(logfile_FILE); + } else { // if (windows == NULL) + fprintf(stderr, "GLib %s: %s", reason, message); } + /* else { */ + /* printformat(NULL, NULL, MSGLEVEL_CLIENTERROR, */ + /* TXT_GLIB_ERROR, reason, message); */ + /* } */ } #define MSGS_WINDOW_LEVELS (MSGLEVEL_MSGS|MSGLEVEL_ACTIONS|MSGLEVEL_DCCMSGS) diff --git a/src/fe-common/core/fe-exec.c b/src/fe-common/core/fe-exec.c index 9249f432..98245a59 100644 --- a/src/fe-common/core/fe-exec.c +++ b/src/fe-common/core/fe-exec.c @@ -348,12 +348,12 @@ static void process_exec(PROCESS_REC *rec, const char *cmd) if (rec->shell) { execvp(shell_args[0], (char **) shell_args); - fprintf(stderr, "Exec: /bin/sh: %s\n", g_strerror(errno)); + g_message( "Exec: /bin/sh: %s\n", g_strerror(errno)); } else { args = g_strsplit(cmd, " ", -1); execvp(args[0], args); - fprintf(stderr, "Exec: %s: %s\n", args[0], g_strerror(errno)); + g_message( "Exec: %s: %s\n", args[0], g_strerror(errno)); } _exit(-1); diff --git a/src/fe-common/core/formats.c b/src/fe-common/core/formats.c index f2a82030..2b331fa8 100644 --- a/src/fe-common/core/formats.c +++ b/src/fe-common/core/formats.c @@ -66,6 +66,8 @@ static void format_expand_code(const char **format, GString *out, int *flags) { int set; + g_message( "format_expand_codes()\n"); + if (flags == NULL) { /* flags are being ignored - skip the code */ while (**format != ']') @@ -103,9 +105,20 @@ static void format_expand_code(const char **format, GString *out, int *flags) int format_expand_styles(GString *out, const char **format, int *flags) { + int retval = 1; + char *p, fmt; + /* storage for numerical parsing code for %x/X formats. */ + unsigned char accum = 0; + int tmp, i; + + //memset(num_buf, 0, 4); + fmt = **format; + + g_message( "format_expand_styles: fmtchar: %c\n", fmt); + switch (fmt) { case '{': case '}': @@ -121,6 +134,7 @@ int format_expand_styles(GString *out, const char **format, int *flags) case '9': case '_': /* bold on/off */ + g_message( "setting bold flag: %04x\n", FORMAT_STYLE_BOLD); g_string_append_c(out, 4); g_string_append_c(out, FORMAT_STYLE_BOLD); break; @@ -162,6 +176,46 @@ int format_expand_styles(GString *out, const char **format, int *flags) /* code */ format_expand_code(format, out, flags); break; + case 'x': + tmp = 0; + accum = 0; + for (i = 1; i < 3; i++) { + char fmtchar = (*format)[i]; + g_message("Format X: code: %c\n", fmtchar); + tmp = g_ascii_xdigit_value(fmtchar); + if (tmp != -1) { + accum = accum * 16 + tmp; + } + } + retval += i - 1; + + g_string_append_c(out, 4); + g_string_append_c(out, FORMAT_COLOR_NOCHANGE); + g_string_append_c(out, accum); + + g_message("Format x: code: %d (0x%02x)\n", accum, accum); + + break; + case 'X': + tmp = 0; + accum = 0; + for (i = 1; i < 3; i++) { + char fmtchar = (*format)[i]; + g_message("Format X: code: %c\n", fmtchar); + tmp = g_ascii_xdigit_value(fmtchar); + if (tmp != -1) { + accum = accum * 16 + tmp; + } + } + g_string_append_c(out, 4); + g_string_append_c(out, accum); + g_string_append_c(out, FORMAT_COLOR_NOCHANGE); + retval += i - 1; + + g_message("Format X: code: %d (0x%02x)\n", accum, accum); + + break; + default: /* check if it's a background color */ p = strchr(format_backs, fmt); @@ -169,6 +223,8 @@ int format_expand_styles(GString *out, const char **format, int *flags) g_string_append_c(out, 4); g_string_append_c(out, FORMAT_COLOR_NOCHANGE); g_string_append_c(out, (char) ((int) (p-format_backs)+'0')); + g_message( "BG: Printing: %d '%s'\n", + ((int) (p-format_backs)+'0'), out->str); break; } @@ -176,6 +232,7 @@ int format_expand_styles(GString *out, const char **format, int *flags) if (fmt == 'p') fmt = 'm'; p = strchr(format_fores, fmt); if (p != NULL) { + /* color code indicator for format_send_to_gui */ g_string_append_c(out, 4); g_string_append_c(out, (char) ((int) (p-format_fores)+'0')); g_string_append_c(out, FORMAT_COLOR_NOCHANGE); @@ -187,15 +244,16 @@ int format_expand_styles(GString *out, const char **format, int *flags) p = strchr(format_boldfores, fmt); if (p != NULL) { g_string_append_c(out, 4); + /* +8 selects bold version */ g_string_append_c(out, (char) (8+(int) (p-format_boldfores)+'0')); g_string_append_c(out, FORMAT_COLOR_NOCHANGE); break; } - return FALSE; + return 0; } - return TRUE; + return retval; } void format_read_arglist(va_list va, FORMAT_REC *format, @@ -303,6 +361,7 @@ int format_get_length(const char *str) GString *tmp; int len; gboolean utf8; + int adv = 0; g_return_val_if_fail(str != NULL, 0); @@ -313,11 +372,13 @@ int format_get_length(const char *str) while (*str != '\0') { if (*str == '%' && str[1] != '\0') { str++; - if (*str != '%' && - format_expand_styles(tmp, &str, NULL)) { - str++; + if (*str != '%') { + adv = format_expand_styles(tmp, &str, NULL); + str += adv; + if (adv > 1) { continue; } + } /* %% or unknown %code, written as-is */ if (*str != '%') @@ -340,7 +401,7 @@ int format_real_length(const char *str, int len) const char *start; const char *oldstr; gboolean utf8; - + int adv = 0; g_return_val_if_fail(str != NULL, 0); g_return_val_if_fail(len >= 0, 0); @@ -351,11 +412,13 @@ int format_real_length(const char *str, int len) while (*str != '\0' && len > 0) { if (*str == '%' && str[1] != '\0') { str++; - if (*str != '%' && - format_expand_styles(tmp, &str, NULL)) { - str++; + if (*str != '%') { + adv = format_expand_styles(tmp, &str, NULL); + str += adv; + if (adv > 1) { continue; } + } /* %% or unknown %code, written as-is */ if (*str != '%') { @@ -376,8 +439,11 @@ int format_real_length(const char *str, int len) char *format_string_expand(const char *text, int *flags) { + g_message( "format_string_expand: flags: %04x\n", *flags); + GString *out; char code, *ret; + int adv; g_return_val_if_fail(text != NULL, NULL); @@ -388,10 +454,13 @@ char *format_string_expand(const char *text, int *flags) while (*text != '\0') { if (code == '%') { /* color code */ - if (!format_expand_styles(out, &text, flags)) { + adv = format_expand_styles(out, &text, flags); + if (!adv) { g_string_append_c(out, '%'); g_string_append_c(out, '%'); g_string_append_c(out, *text); + } else { + text += adv -1; } code = 0; } else { @@ -415,6 +484,7 @@ static char *format_get_text_args(TEXT_DEST_REC *dest, GString *out; char code, *ret; int need_free; + int adv; out = g_string_new(NULL); @@ -422,10 +492,13 @@ static char *format_get_text_args(TEXT_DEST_REC *dest, while (*text != '\0') { if (code == '%') { /* color code */ - if (!format_expand_styles(out, &text, &dest->flags)) { + adv = format_expand_styles(out, &text, &dest->flags); + if (!adv) { g_string_append_c(out, '%'); g_string_append_c(out, '%'); g_string_append_c(out, *text); + } else { + text += adv -1; } code = 0; } else if (code == '$') { @@ -830,10 +903,14 @@ static void get_mirc_color(const char **str, int *fg_ret, int *bg_ret) if (bg_ret) *bg_ret = bg; } +/* TODO: What are these magic numbers!? + */ #define IS_COLOR_CODE(c) \ ((c) == 2 || (c) == 3 || (c) == 4 || (c) == 6 || (c) == 7 || \ (c) == 15 || (c) == 22 || (c) == 27 || (c) == 31) +//#define IS_COLOR_CODE(c) ((c) < 255) + /* Return how many characters in `str' must be skipped before `len' characters of text is skipped. */ int strip_real_length(const char *str, int len, @@ -936,7 +1013,10 @@ void format_send_to_gui(TEXT_DEST_REC *dest, const char *text) dup = str = g_strdup(text); - flags = 0; fgcolor = theme->default_color; bgcolor = -1; + flags = 0; + fgcolor = theme->default_color; + bgcolor = -1; + while (*str != '\0') { type = '\0'; for (ptr = str; *ptr != '\0'; ptr++) { @@ -958,12 +1038,17 @@ void format_send_to_gui(TEXT_DEST_REC *dest, const char *text) if (*str != '\0' || (flags & GUI_PRINT_FLAG_CLRTOEOL)) { /* send the text to gui handler */ + g_message("format_send_to_gui: sending: fg: 0x%04x, " \ + "bg: 0x%04x flags: 0x%04x\n", fgcolor, bgcolor, flags); + signal_emit_id(signal_gui_print_text, 6, dest->window, GINT_TO_POINTER(fgcolor), GINT_TO_POINTER(bgcolor), GINT_TO_POINTER(flags), str, dest); + flags &= ~(GUI_PRINT_FLAG_INDENT|GUI_PRINT_FLAG_CLRTOEOL); + /* fprintf("format_send_to_gui: resetting flags: 0x%04x\n", flags); */ } if (type == '\n') { @@ -978,12 +1063,12 @@ void format_send_to_gui(TEXT_DEST_REC *dest, const char *text) switch (type) { - case 2: + case MIRC_BOLD_MARKER: /* bold */ if (!hide_text_style) flags ^= GUI_PRINT_FLAG_BOLD; break; - case 3: + case MIRC_COLOR_MARKER: /* MIRC color */ get_mirc_color((const char **) &ptr, hide_colors ? NULL : &fgcolor, @@ -991,7 +1076,7 @@ void format_send_to_gui(TEXT_DEST_REC *dest, const char *text) if (!hide_colors) flags |= GUI_PRINT_FLAG_MIRC_COLOR; break; - case 4: + case LINE_FORMAT_MARKER: /* user specific colors */ flags &= ~GUI_PRINT_FLAG_MIRC_COLOR; switch (*ptr) { @@ -1003,7 +1088,12 @@ void format_send_to_gui(TEXT_DEST_REC *dest, const char *text) break; case FORMAT_STYLE_BOLD: flags ^= GUI_PRINT_FLAG_BOLD; + g_message( + "format bold spotted, flags now: 0x%04x\n", + flags); + break; + case FORMAT_STYLE_REVERSE: flags ^= GUI_PRINT_FLAG_REVERSE; break; diff --git a/src/fe-common/core/formats.h b/src/fe-common/core/formats.h index 6c55a068..56412937 100644 --- a/src/fe-common/core/formats.h +++ b/src/fe-common/core/formats.h @@ -4,6 +4,12 @@ #include "themes.h" #include "fe-windows.h" +/* various types of colour codings possible. */ +#define MIRC_BOLD_MARKER ('\002') +#define MIRC_COLOR_MARKER ('\003') +#define LINE_FORMAT_MARKER ('\004') + + #define GUI_PRINT_FLAG_BOLD 0x0001 #define GUI_PRINT_FLAG_REVERSE 0x0002 #define GUI_PRINT_FLAG_UNDERLINE 0x0004 diff --git a/src/fe-common/core/printtext.c b/src/fe-common/core/printtext.c index d3653f90..1ac4e4de 100644 --- a/src/fe-common/core/printtext.c +++ b/src/fe-common/core/printtext.c @@ -206,6 +206,7 @@ static char *printtext_get_args(TEXT_DEST_REC *dest, const char *str, { GString *out; char *ret; + int adv; out = g_string_new(NULL); for (; *str != '\0'; str++) { @@ -255,9 +256,12 @@ static char *printtext_get_args(TEXT_DEST_REC *dest, const char *str, break; } default: - if (!format_expand_styles(out, &str, &dest->flags)) { + adv = format_expand_styles(out, &str, &dest->flags); + if (!adv) { g_string_append_c(out, '%'); g_string_append_c(out, *str); + } else { + str += adv -1; } break; } @@ -272,7 +276,7 @@ static char *printtext_expand_formats(const char *str, int *flags) { GString *out; char *ret; - + int adv; out = g_string_new(NULL); for (; *str != '\0'; str++) { if (*str != '%') { @@ -283,9 +287,12 @@ static char *printtext_expand_formats(const char *str, int *flags) if (*++str == '\0') break; - if (!format_expand_styles(out, &str, flags)) { + adv = format_expand_styles(out, &str, flags); + if (!adv) { g_string_append_c(out, '%'); g_string_append_c(out, *str); + } else { + str += adv -1; } } |