4 Copyright (C) 1999-2000 Timo Sirainen
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License along
17 with this program; if not, write to the Free Software Foundation, Inc.,
18 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 #include "module-formats.h"
24 #include "special-vars.h"
30 #include "fe-windows.h"
31 #include "window-items.h"
37 static const char *format_backs = "04261537";
38 static const char *format_fores = "kbgcrmyw";
39 static const char *format_boldfores = "KBGCRMYW";
41 static int signal_gui_print_text;
42 static int hide_text_style, hide_server_tags, hide_colors;
44 static int timestamp_level;
45 static int timestamp_timeout;
47 int format_find_tag(const char *module, const char *tag)
52 formats = g_hash_table_lookup(default_formats, module);
56 for (n = 0; formats[n].def != NULL; n++) {
57 if (formats[n].tag != NULL &&
58 g_strcasecmp(formats[n].tag, tag) == 0)
65 static void format_expand_code(const char **format, GString *out, int *flags)
70 /* flags are being ignored - skip the code */
71 while (**format != ']')
78 while (**format != ']' && **format != '\0') {
81 else if (**format == '-')
83 else switch (**format) {
86 *flags |= !set ? PRINT_FLAG_UNSET_LINE_START :
87 **format == 's' ? PRINT_FLAG_SET_LINE_START :
88 PRINT_FLAG_SET_LINE_START_IRSSI;
91 *flags |= set ? PRINT_FLAG_SET_TIMESTAMP :
92 PRINT_FLAG_UNSET_TIMESTAMP;
95 *flags |= set ? PRINT_FLAG_SET_SERVERTAG :
96 PRINT_FLAG_UNSET_SERVERTAG;
104 int format_expand_styles(GString *out, const char **format, int *flags)
114 g_string_append_c(out, fmt);
117 /* Underline on/off */
118 g_string_append_c(out, 4);
119 g_string_append_c(out, FORMAT_STYLE_UNDERLINE);
124 g_string_append_c(out, 4);
125 g_string_append_c(out, FORMAT_STYLE_BOLD);
129 g_string_append_c(out, 4);
130 g_string_append_c(out, FORMAT_STYLE_REVERSE);
134 g_string_append_c(out, '\n');
138 g_string_append_c(out, 4);
139 g_string_append_c(out, FORMAT_STYLE_INDENT);
143 g_string_append_c(out, 4);
144 g_string_append_c(out, FORMAT_STYLE_BLINK);
149 g_string_append_c(out, 4);
150 g_string_append_c(out, FORMAT_STYLE_DEFAULTS);
153 /* clear to end of line */
154 g_string_append_c(out, 4);
155 g_string_append_c(out, FORMAT_STYLE_CLRTOEOL);
158 g_string_append_c(out, 4);
159 g_string_append_c(out, FORMAT_STYLE_MONOSPACE);
163 format_expand_code(format, out, flags);
166 /* check if it's a background color */
167 p = strchr(format_backs, fmt);
169 g_string_append_c(out, 4);
170 g_string_append_c(out, FORMAT_COLOR_NOCHANGE);
171 g_string_append_c(out, (char) ((int) (p-format_backs)+'0'));
175 /* check if it's a foreground color */
176 if (fmt == 'p') fmt = 'm';
177 p = strchr(format_fores, fmt);
179 g_string_append_c(out, 4);
180 g_string_append_c(out, (char) ((int) (p-format_fores)+'0'));
181 g_string_append_c(out, FORMAT_COLOR_NOCHANGE);
185 /* check if it's a bold foreground color */
186 if (fmt == 'P') fmt = 'M';
187 p = strchr(format_boldfores, fmt);
189 g_string_append_c(out, 4);
190 g_string_append_c(out, (char) (8+(int) (p-format_boldfores)+'0'));
191 g_string_append_c(out, FORMAT_COLOR_NOCHANGE);
201 void format_read_arglist(va_list va, FORMAT_REC *format,
202 char **arglist, int arglist_size,
203 char *buffer, int buffer_size)
205 int num, len, bufpos;
207 g_return_if_fail(format->params < arglist_size);
210 arglist[format->params] = NULL;
211 for (num = 0; num < format->params; num++) {
212 switch (format->paramtypes[num]) {
214 arglist[num] = (char *) va_arg(va, char *);
215 if (arglist[num] == NULL)
219 int d = (int) va_arg(va, int);
221 if (bufpos >= buffer_size) {
226 arglist[num] = buffer+bufpos;
227 len = g_snprintf(buffer+bufpos, buffer_size-bufpos,
233 long l = (long) va_arg(va, long);
235 if (bufpos >= buffer_size) {
240 arglist[num] = buffer+bufpos;
241 len = g_snprintf(buffer+bufpos, buffer_size-bufpos,
247 double f = (double) va_arg(va, double);
249 if (bufpos >= buffer_size) {
254 arglist[num] = buffer+bufpos;
255 len = g_snprintf(buffer+bufpos, buffer_size-bufpos,
263 void format_create_dest(TEXT_DEST_REC *dest,
264 void *server, const char *target,
265 int level, WINDOW_REC *window)
267 format_create_dest_tag(dest, server, NULL, target, level, window);
270 void format_create_dest_tag(TEXT_DEST_REC *dest, void *server,
271 const char *server_tag, const char *target,
272 int level, WINDOW_REC *window)
274 memset(dest, 0, sizeof(TEXT_DEST_REC));
276 dest->server = server;
277 dest->server_tag = server != NULL ? SERVER(server)->tag : server_tag;
278 dest->target = target;
280 dest->window = window != NULL ? window :
281 window_find_closest(server, target, level);
284 static int advance (char const **str, gboolean utf8)
289 c = g_utf8_get_char(*str);
290 *str = g_utf8_next_char(*str);
292 return unichar_isprint(c) ? mk_wcwidth(c) : 1;
300 /* Return length of text part in string (ie. without % codes) */
301 int format_get_length(const char *str)
307 g_return_val_if_fail(str != NULL, 0);
309 utf8 = is_utf8() && g_utf8_validate(str, -1, NULL);
311 tmp = g_string_new(NULL);
313 while (*str != '\0') {
314 if (*str == '%' && str[1] != '\0') {
317 format_expand_styles(tmp, &str, NULL)) {
322 /* %% or unknown %code, written as-is */
327 len += advance(&str, utf8);
330 g_string_free(tmp, TRUE);
334 /* Return how many characters in `str' must be skipped before `len'
335 characters of text is skipped. Like strip_real_length(), except this
337 int format_real_length(const char *str, int len)
344 g_return_val_if_fail(str != NULL, 0);
345 g_return_val_if_fail(len >= 0, 0);
347 utf8 = is_utf8() && g_utf8_validate(str, -1, NULL);
350 tmp = g_string_new(NULL);
351 while (*str != '\0' && len > 0) {
352 if (*str == '%' && str[1] != '\0') {
355 format_expand_styles(tmp, &str, NULL)) {
360 /* %% or unknown %code, written as-is */
368 len -= advance(&str, utf8);
373 g_string_free(tmp, TRUE);
374 return (int) (str-start);
377 char *format_string_expand(const char *text, int *flags)
382 g_return_val_if_fail(text != NULL, NULL);
384 out = g_string_new(NULL);
386 if (flags != NULL) *flags = 0;
388 while (*text != '\0') {
391 if (!format_expand_styles(out, &text, flags)) {
392 g_string_append_c(out, '%');
393 g_string_append_c(out, '%');
394 g_string_append_c(out, *text);
401 g_string_append_c(out, *text);
408 g_string_free(out, FALSE);
412 static char *format_get_text_args(TEXT_DEST_REC *dest,
413 const char *text, char **arglist)
419 out = g_string_new(NULL);
422 while (*text != '\0') {
425 if (!format_expand_styles(out, &text, &dest->flags)) {
426 g_string_append_c(out, '%');
427 g_string_append_c(out, '%');
428 g_string_append_c(out, *text);
431 } else if (code == '$') {
435 ret = parse_special((char **) &text, dest->server,
436 dest->target == NULL ? NULL :
437 window_item_find(dest->server, dest->target),
438 arglist, &need_free, NULL, 0);
441 /* string shouldn't end with \003 or it could
442 mess up the next one or two characters */
444 int len = strlen(ret);
445 while (len > 0 && ret[len-1] == 3) len--;
446 diff = strlen(ret)-len;
448 g_string_append(out, ret);
450 g_string_truncate(out, out->len-diff);
451 if (need_free) g_free(ret);
455 if (*text == '%' || *text == '$')
458 g_string_append_c(out, *text);
465 g_string_free(out, FALSE);
469 char *format_get_text_theme(THEME_REC *theme, const char *module,
470 TEXT_DEST_REC *dest, int formatnum, ...)
476 theme = window_get_theme(dest->window);
478 va_start(va, formatnum);
479 str = format_get_text_theme_args(theme, module, dest, formatnum, va);
485 char *format_get_text_theme_args(THEME_REC *theme, const char *module,
486 TEXT_DEST_REC *dest, int formatnum,
489 char *arglist[MAX_FORMAT_PARAMS];
490 char buffer[DEFAULT_FORMAT_ARGLIST_SIZE];
493 formats = g_hash_table_lookup(default_formats, module);
494 format_read_arglist(va, &formats[formatnum],
495 arglist, sizeof(arglist)/sizeof(char *),
496 buffer, sizeof(buffer));
498 return format_get_text_theme_charargs(theme, module, dest,
502 char *format_get_text_theme_charargs(THEME_REC *theme, const char *module,
503 TEXT_DEST_REC *dest, int formatnum,
506 MODULE_THEME_REC *module_theme;
509 module_theme = g_hash_table_lookup(theme->modules, module);
510 if (module_theme == NULL)
513 text = module_theme->expanded_formats[formatnum];
514 return format_get_text_args(dest, text, args);
517 char *format_get_text(const char *module, WINDOW_REC *window,
518 void *server, const char *target,
526 format_create_dest(&dest, server, target, 0, window);
527 theme = window_get_theme(dest.window);
529 va_start(va, formatnum);
530 str = format_get_text_theme_args(theme, module, &dest, formatnum, va);
536 /* add `linestart' to start of each line in `text'. `text' may contain
537 multiple lines separated with \n. */
538 char *format_add_linestart(const char *text, const char *linestart)
543 if (linestart == NULL)
544 return g_strdup(text);
546 if (strchr(text, '\n') == NULL)
547 return g_strconcat(linestart, text, NULL);
549 str = g_string_new(linestart);
550 while (*text != '\0') {
551 g_string_append_c(str, *text);
553 g_string_append(str, linestart);
558 g_string_free(str, FALSE);
562 char *format_add_lineend(const char *text, const char *linestart)
567 if (linestart == NULL)
568 return g_strdup(text);
570 if (strchr(text, '\n') == NULL)
571 return g_strconcat(text, linestart, NULL);
573 str = g_string_new(NULL);
574 while (*text != '\0') {
576 g_string_append(str, linestart);
577 g_string_append_c(str, *text);
580 g_string_append(str, linestart);
583 g_string_free(str, FALSE);
587 #define LINE_START_IRSSI_LEVEL \
588 (MSGLEVEL_CLIENTERROR | MSGLEVEL_CLIENTNOTICE)
590 #define NOT_LINE_START_LEVEL \
591 (MSGLEVEL_NEVER | MSGLEVEL_LASTLOG | MSGLEVEL_CLIENTCRAP | \
592 MSGLEVEL_MSGS | MSGLEVEL_PUBLIC | MSGLEVEL_DCC | MSGLEVEL_DCCMSGS | \
593 MSGLEVEL_ACTIONS | MSGLEVEL_NOTICES | MSGLEVEL_SNOTES | MSGLEVEL_CTCPS)
595 /* return the "-!- " text at the start of the line */
596 char *format_get_level_tag(THEME_REC *theme, TEXT_DEST_REC *dest)
600 /* check for flags if we want to override defaults */
601 if (dest->flags & PRINT_FLAG_UNSET_LINE_START)
604 if (dest->flags & PRINT_FLAG_SET_LINE_START)
605 format = TXT_LINE_START;
606 else if (dest->flags & PRINT_FLAG_SET_LINE_START_IRSSI)
607 format = TXT_LINE_START_IRSSI;
610 if (dest->level & LINE_START_IRSSI_LEVEL)
611 format = TXT_LINE_START_IRSSI;
612 else if ((dest->level & NOT_LINE_START_LEVEL) == 0)
613 format = TXT_LINE_START;
618 return format_get_text_theme(theme, MODULE_NAME, dest, format);
621 static char *get_timestamp(THEME_REC *theme, TEXT_DEST_REC *dest, time_t t)
623 char *format, str[256];
627 if ((timestamp_level & dest->level) == 0)
630 /* check for flags if we want to override defaults */
631 if (dest->flags & PRINT_FLAG_UNSET_TIMESTAMP)
634 if ((dest->flags & PRINT_FLAG_SET_TIMESTAMP) == 0 &&
635 (dest->level & (MSGLEVEL_NEVER|MSGLEVEL_LASTLOG)) != 0)
639 if (timestamp_timeout > 0) {
640 diff = t - dest->window->last_timestamp;
641 dest->window->last_timestamp = t;
642 if (diff < timestamp_timeout)
647 format = format_get_text_theme(theme, MODULE_NAME, dest,
649 if (strftime(str, sizeof(str), format, tm) <= 0)
652 return g_strdup(str);
655 static char *get_server_tag(THEME_REC *theme, TEXT_DEST_REC *dest)
659 if (dest->server_tag == NULL || hide_server_tags)
662 /* check for flags if we want to override defaults */
663 if (dest->flags & PRINT_FLAG_UNSET_SERVERTAG)
666 if ((dest->flags & PRINT_FLAG_SET_SERVERTAG) == 0) {
667 if (dest->window->active != NULL &&
668 dest->window->active->server == dest->server)
671 if (servers != NULL) {
673 if (servers->next != NULL)
676 if (count < 2 && lookup_servers != NULL) {
678 if (lookup_servers->next != NULL)
686 return format_get_text_theme(theme, MODULE_NAME, dest,
687 TXT_SERVERTAG, dest->server_tag);
690 char *format_get_line_start(THEME_REC *theme, TEXT_DEST_REC *dest, time_t t)
692 char *timestamp, *servertag;
695 timestamp = get_timestamp(theme, dest, t);
696 servertag = get_server_tag(theme, dest);
698 if (timestamp == NULL && servertag == NULL)
701 linestart = g_strconcat(timestamp != NULL ? timestamp : "",
704 g_free_not_null(timestamp);
705 g_free_not_null(servertag);
709 void format_newline(WINDOW_REC *window)
711 g_return_if_fail(window != NULL);
713 signal_emit_id(signal_gui_print_text, 6, window,
714 GINT_TO_POINTER(-1), GINT_TO_POINTER(-1),
715 GINT_TO_POINTER(GUI_PRINT_FLAG_NEWLINE),
719 /* parse ANSI color string */
720 static const char *get_ansi_color(THEME_REC *theme, const char *str,
721 int *fg_ret, int *bg_ret, int *flags_ret)
723 static char ansitab[8] = { 0, 4, 2, 6, 1, 5, 3, 7 };
725 int fg, bg, flags, num;
731 fg = fg_ret == NULL || *fg_ret < 0 ? theme->default_color : *fg_ret;
732 bg = bg_ret == NULL || *bg_ret < 0 ? -1 : *bg_ret;
733 flags = flags_ret == NULL ? 0 : *flags_ret;
737 if (*str == '\0') return start;
739 if (i_isdigit(*str)) {
740 num = num*10 + (*str-'0');
744 if (*str != ';' && *str != 'm')
749 /* reset colors back to default */
750 fg = theme->default_color;
752 flags &= ~GUI_PRINT_FLAG_INDENT;
756 flags |= GUI_PRINT_FLAG_BOLD;
760 flags |= GUI_PRINT_FLAG_BLINK;
764 flags |= GUI_PRINT_FLAG_REVERSE;
767 if (num >= 30 && num <= 37) {
768 if (fg == -1) fg = 0;
769 fg = (fg & 0xf8) | ansitab[num-30];
771 if (num >= 40 && num <= 47) {
772 if (bg == -1) bg = 0;
773 bg = (bg & 0xf8) | ansitab[num-40];
780 if (fg_ret != NULL) *fg_ret = fg;
781 if (bg_ret != NULL) *bg_ret = bg;
782 if (flags_ret != NULL) *flags_ret = flags;
792 /* parse MIRC color string */
793 static void get_mirc_color(const char **str, int *fg_ret, int *bg_ret)
797 fg = fg_ret == NULL ? -1 : *fg_ret;
798 bg = bg_ret == NULL ? -1 : *bg_ret;
800 if (!i_isdigit(**str) && **str != ',') {
804 /* foreground color */
808 if (i_isdigit(**str)) {
809 fg = fg*10 + (**str-'0');
814 /* background color */
815 if (!i_isdigit((*str)[1]))
821 if (i_isdigit(**str)) {
822 bg = bg*10 + (**str-'0');
829 if (fg_ret) *fg_ret = fg;
830 if (bg_ret) *bg_ret = bg;
833 #define IS_COLOR_CODE(c) \
834 ((c) == 2 || (c) == 3 || (c) == 4 || (c) == 6 || (c) == 7 || \
835 (c) == 15 || (c) == 22 || (c) == 27 || (c) == 31)
837 /* Return how many characters in `str' must be skipped before `len'
838 characters of text is skipped. */
839 int strip_real_length(const char *str, int len,
840 int *last_color_pos, int *last_color_len)
842 const char *start = str;
844 if (last_color_pos != NULL)
845 *last_color_pos = -1;
846 if (last_color_len != NULL)
847 *last_color_len = -1;
849 while (*str != '\0') {
851 const char *mircstart = str;
853 if (last_color_pos != NULL)
854 *last_color_pos = (int) (str-start);
856 get_mirc_color(&str, NULL, NULL);
857 if (last_color_len != NULL)
858 *last_color_len = (int) (str-mircstart);
860 } else if (*str == 4 && str[1] != '\0') {
861 if (str[1] < FORMAT_STYLE_SPECIAL && str[2] != '\0') {
862 if (last_color_pos != NULL)
863 *last_color_pos = (int) (str-start);
864 if (last_color_len != NULL)
867 } else if (str[1] == FORMAT_STYLE_DEFAULTS) {
868 if (last_color_pos != NULL)
869 *last_color_pos = (int) (str-start);
870 if (last_color_len != NULL)
875 if (!IS_COLOR_CODE(*str)) {
883 return (int) (str-start);
886 char *strip_codes(const char *input)
891 out = str = g_strdup(input);
892 for (p = input; *p != '\0'; p++) {
897 get_mirc_color(&p, NULL, NULL);
902 if (*p == 4 && p[1] != '\0') {
903 if (p[1] >= FORMAT_STYLE_SPECIAL) {
915 if (*p == 27 && p[1] != '\0') {
917 p = get_ansi_color(current_theme, p, NULL, NULL, NULL);
919 } else if (!IS_COLOR_CODE(*p))
927 /* send a fully parsed text string for GUI to print */
928 void format_send_to_gui(TEXT_DEST_REC *dest, const char *text)
931 char *dup, *str, *ptr, type;
932 int fgcolor, bgcolor;
935 theme = window_get_theme(dest->window);
937 dup = str = g_strdup(text);
939 flags = 0; fgcolor = theme->default_color; bgcolor = -1;
940 while (*str != '\0') {
942 for (ptr = str; *ptr != '\0'; ptr++) {
943 if (IS_COLOR_CODE(*ptr) || *ptr == '\n') {
952 if (settings_get_bool("bell_beeps"))
953 signal_emit("beep", 0);
954 } else if (type == 4 && *ptr == FORMAT_STYLE_CLRTOEOL) {
955 /* clear to end of line */
956 flags |= GUI_PRINT_FLAG_CLRTOEOL;
959 if (*str != '\0' || (flags & GUI_PRINT_FLAG_CLRTOEOL)) {
960 /* send the text to gui handler */
961 signal_emit_id(signal_gui_print_text, 6, dest->window,
962 GINT_TO_POINTER(fgcolor),
963 GINT_TO_POINTER(bgcolor),
964 GINT_TO_POINTER(flags), str,
966 flags &= ~(GUI_PRINT_FLAG_INDENT|GUI_PRINT_FLAG_CLRTOEOL);
970 format_newline(dest->window);
971 fgcolor = theme->default_color;
973 flags &= GUI_PRINT_FLAG_INDENT|GUI_PRINT_FLAG_MONOSPACE;
983 if (!hide_text_style)
984 flags ^= GUI_PRINT_FLAG_BOLD;
988 get_mirc_color((const char **) &ptr,
989 hide_colors ? NULL : &fgcolor,
990 hide_colors ? NULL : &bgcolor);
992 flags |= GUI_PRINT_FLAG_MIRC_COLOR;
995 /* user specific colors */
996 flags &= ~GUI_PRINT_FLAG_MIRC_COLOR;
998 case FORMAT_STYLE_BLINK:
999 flags ^= GUI_PRINT_FLAG_BLINK;
1001 case FORMAT_STYLE_UNDERLINE:
1002 flags ^= GUI_PRINT_FLAG_UNDERLINE;
1004 case FORMAT_STYLE_BOLD:
1005 flags ^= GUI_PRINT_FLAG_BOLD;
1007 case FORMAT_STYLE_REVERSE:
1008 flags ^= GUI_PRINT_FLAG_REVERSE;
1010 case FORMAT_STYLE_MONOSPACE:
1011 flags ^= GUI_PRINT_FLAG_MONOSPACE;
1013 case FORMAT_STYLE_INDENT:
1014 flags |= GUI_PRINT_FLAG_INDENT;
1016 case FORMAT_STYLE_DEFAULTS:
1017 fgcolor = theme->default_color;
1019 flags &= GUI_PRINT_FLAG_INDENT|GUI_PRINT_FLAG_MONOSPACE;
1021 case FORMAT_STYLE_CLRTOEOL:
1024 if (*ptr != FORMAT_COLOR_NOCHANGE) {
1025 fgcolor = (unsigned char) *ptr-'0';
1031 if (*ptr != FORMAT_COLOR_NOCHANGE) {
1039 if (!hide_text_style)
1040 flags ^= GUI_PRINT_FLAG_BLINK;
1043 /* remove all styling */
1044 fgcolor = theme->default_color;
1046 flags &= GUI_PRINT_FLAG_INDENT|GUI_PRINT_FLAG_MONOSPACE;
1050 if (!hide_text_style)
1051 flags ^= GUI_PRINT_FLAG_REVERSE;
1055 if (!hide_text_style)
1056 flags ^= GUI_PRINT_FLAG_UNDERLINE;
1059 /* ansi color code */
1061 get_ansi_color(theme, ptr,
1062 hide_colors ? NULL : &fgcolor,
1063 hide_colors ? NULL : &bgcolor,
1064 hide_colors ? NULL : &flags);
1074 static void read_settings(void)
1076 timestamp_level = settings_get_bool("timestamps") ? MSGLEVEL_ALL : 0;
1077 if (timestamp_level > 0)
1078 timestamp_level = settings_get_level("timestamp_level");
1079 timestamp_timeout = settings_get_time("timestamp_timeout")/1000;
1081 hide_server_tags = settings_get_bool("hide_server_tags");
1082 hide_text_style = settings_get_bool("hide_text_style");
1083 hide_colors = hide_text_style || settings_get_bool("hide_colors");
1086 void formats_init(void)
1088 signal_gui_print_text = signal_get_uniq_id("gui print text");
1091 signal_add("setup changed", (SIGNAL_FUNC) read_settings);
1094 void formats_deinit(void)
1096 signal_remove("setup changed", (SIGNAL_FUNC) read_settings);