1
0
mirror of https://github.com/irssi/irssi.git synced 2024-12-04 14:46:39 -05:00

Merge pull request #917 from dequis/wcwidth-wrapper

Add a wrapper of wcwidth() that picks the best implementation
This commit is contained in:
ailin-nemui 2018-09-04 09:52:21 +02:00 committed by GitHub
commit d93cd63243
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
12 changed files with 191 additions and 18 deletions

View File

@ -326,6 +326,11 @@ PKG_CHECK_MODULES([OPENSSL], [openssl], [
]) ])
]) ])
dnl **
dnl ** utf8proc
dnl **
AC_CHECK_LIB([utf8proc], [utf8proc_version])
dnl ** dnl **
dnl ** curses checks dnl ** curses checks
dnl ** dnl **
@ -815,7 +820,7 @@ echo "Building with 64bit DCC support .. : $offt_64bit"
echo "Building with true color support.. : $want_truecolor" echo "Building with true color support.. : $want_truecolor"
echo "Building with GRegex ............. : $want_gregex" echo "Building with GRegex ............. : $want_gregex"
echo "Building with Capsicum ........... : $want_capsicum" echo "Building with Capsicum ........... : $want_capsicum"
echo "Building with utf8proc ........... : $ac_cv_lib_utf8proc_utf8proc_version"
if test "x$want_otr" = "xstatic"; then if test "x$want_otr" = "xstatic"; then
echo "Building with OTR support ........ : static (in irssi binary)" echo "Building with OTR support ........ : static (in irssi binary)"
elif test "x$want_otr" = "xmodule"; then elif test "x$want_otr" = "xmodule"; then

View File

@ -6,7 +6,7 @@
#define IRSSI_GLOBAL_CONFIG "irssi.conf" /* config file name in /etc/ */ #define IRSSI_GLOBAL_CONFIG "irssi.conf" /* config file name in /etc/ */
#define IRSSI_HOME_CONFIG "config" /* config file name in ~/.irssi/ */ #define IRSSI_HOME_CONFIG "config" /* config file name in ~/.irssi/ */
#define IRSSI_ABI_VERSION 16 #define IRSSI_ABI_VERSION 17
#define DEFAULT_SERVER_ADD_PORT 6667 #define DEFAULT_SERVER_ADD_PORT 6667
#define DEFAULT_SERVER_ADD_TLS_PORT 6697 #define DEFAULT_SERVER_ADD_TLS_PORT 6697

View File

@ -53,6 +53,7 @@ libcore_a_SOURCES = \
utf8.c \ utf8.c \
$(regex_impl) \ $(regex_impl) \
wcwidth.c \ wcwidth.c \
wcwidth-wrapper.c \
tls.c \ tls.c \
write-buffer.c write-buffer.c

View File

@ -60,6 +60,9 @@ void chat_commands_deinit(void);
void log_away_init(void); void log_away_init(void);
void log_away_deinit(void); void log_away_deinit(void);
void wcwidth_wrapper_init(void);
void wcwidth_wrapper_deinit(void);
int irssi_gui; int irssi_gui;
int irssi_init_finished; int irssi_init_finished;
int reload_config; int reload_config;
@ -258,6 +261,7 @@ void core_init(void)
nicklist_init(); nicklist_init();
chat_commands_init(); chat_commands_init();
wcwidth_wrapper_init();
settings_add_str("misc", "ignore_signals", ""); settings_add_str("misc", "ignore_signals", "");
settings_add_bool("misc", "override_coredump_limit", FALSE); settings_add_bool("misc", "override_coredump_limit", FALSE);
@ -281,6 +285,7 @@ void core_deinit(void)
signal_remove("setup changed", (SIGNAL_FUNC) read_settings); signal_remove("setup changed", (SIGNAL_FUNC) read_settings);
signal_remove("irssi init finished", (SIGNAL_FUNC) sig_irssi_init_finished); signal_remove("irssi init finished", (SIGNAL_FUNC) sig_irssi_init_finished);
wcwidth_wrapper_deinit();
chat_commands_deinit(); chat_commands_deinit();
nicklist_deinit(); nicklist_deinit();

View File

@ -36,7 +36,7 @@ int string_advance(char const **str, int policy)
c = g_utf8_get_char(*str); c = g_utf8_get_char(*str);
*str = g_utf8_next_char(*str); *str = g_utf8_next_char(*str);
return unichar_isprint(c) ? mk_wcwidth(c) : 1; return unichar_isprint(c) ? i_wcwidth(c) : 1;
} else { } else {
/* Assume TREAT_STRING_AS_BYTES: */ /* Assume TREAT_STRING_AS_BYTES: */
*str += 1; *str += 1;

View File

@ -12,8 +12,14 @@
typedef guint32 unichar; typedef guint32 unichar;
/* Returns width for character (0-2). */ /* Returns width for character (0-2). */
int i_wcwidth(unichar c);
/* Older variant of the above */
int mk_wcwidth(unichar c); int mk_wcwidth(unichar c);
/* Signature for wcwidth implementations */
typedef int (*WCWIDTH_FUNC) (unichar ucs);
/* Advance the str pointer one character further; return the number of columns /* Advance the str pointer one character further; return the number of columns
* occupied by the skipped character. * occupied by the skipped character.
*/ */

141
src/core/wcwidth-wrapper.c Normal file
View File

@ -0,0 +1,141 @@
/*
wcwidth-wrapper.c : irssi
Copyright (C) 2018 dequis
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 2 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
#define _GNU_SOURCE
#include <wchar.h>
#include "module.h"
#include "signals.h"
#include "settings.h"
#include "utf8.h"
#ifdef HAVE_LIBUTF8PROC
#include <utf8proc.h>
#endif
/* wcwidth=2 since unicode 5.2.0 */
#define UNICODE_SQUARE_HIRAGANA_HOKA 0x1F200
/* wcwidth=2 since unicode 9.0.0 */
#define UNICODE_IRSSI_LOGO 0x1F525
enum {
WCWIDTH_IMPL_OLD,
WCWIDTH_IMPL_SYSTEM
#ifdef HAVE_LIBUTF8PROC
,WCWIDTH_IMPL_JULIA
#endif
};
WCWIDTH_FUNC wcwidth_impl_func = mk_wcwidth;
int i_wcwidth(unichar ucs)
{
return (*wcwidth_impl_func)(ucs);
}
static int system_wcwidth(unichar ucs)
{
int retval = wcwidth((wchar_t) ucs);
if (retval < 0) {
/* Treat all unknown characters as taking one cell. This is
* the reason mk_wcwidth and other outdated implementations
* mostly worked with newer unicode, while glibc's wcwidth
* needs updating to recognize new characters.
*
* Instead of relying on that, we keep the behavior of assuming
* one cell even for glibc's implementation, which is still
* highly accurate and less of a headache overall.
*/
return 1;
}
return retval;
}
#ifdef HAVE_LIBUTF8PROC
/* wrapper because the function signatures are different
* (the parameter is unsigned for us, signed for them) */
static int julia_wcwidth(unichar ucs)
{
return utf8proc_charwidth(ucs);
}
#endif
static void read_settings(void)
{
static int choice = -1;
int newchoice;
newchoice = settings_get_choice("wcwidth_implementation");
if (choice == newchoice) {
return;
}
choice = newchoice;
switch (choice) {
case WCWIDTH_IMPL_OLD:
wcwidth_impl_func = &mk_wcwidth;
break;
case WCWIDTH_IMPL_SYSTEM:
wcwidth_impl_func = &system_wcwidth;
break;
#ifdef HAVE_LIBUTF8PROC
case WCWIDTH_IMPL_JULIA:
wcwidth_impl_func = &julia_wcwidth;
break;
#endif
}
}
void wcwidth_wrapper_init(void)
{
int wcwidth_impl_default = 0;
/* Test against characters that have wcwidth=2
* since unicode 5.2 and 9.0 respectively */
if (system_wcwidth(UNICODE_SQUARE_HIRAGANA_HOKA) == 2 ||
system_wcwidth(UNICODE_IRSSI_LOGO) == 2) {
wcwidth_impl_default = WCWIDTH_IMPL_SYSTEM;
} else {
/* Fall back to our own (which implements 5.0) */
wcwidth_impl_default = WCWIDTH_IMPL_OLD;
}
#ifdef HAVE_LIBUTF8PROC
settings_add_choice("misc", "wcwidth_implementation", wcwidth_impl_default, "old;system;julia");
#else
settings_add_choice("misc", "wcwidth_implementation", wcwidth_impl_default, "old;system");
#endif
read_settings();
signal_add_first("setup changed", (SIGNAL_FUNC) read_settings);
}
void wcwidth_wrapper_deinit(void)
{
signal_remove("setup changed", (SIGNAL_FUNC) read_settings);
}

View File

@ -51,7 +51,7 @@ static unichar i_tolower(unichar c)
static int i_isalnum(unichar c) static int i_isalnum(unichar c)
{ {
if (term_type == TERM_TYPE_UTF8) if (term_type == TERM_TYPE_UTF8)
return (g_unichar_isalnum(c) || mk_wcwidth(c) == 0); return (g_unichar_isalnum(c) || i_wcwidth(c) == 0);
return c <= 255 ? isalnum(c) : 0; return c <= 255 ? isalnum(c) : 0;
} }
@ -219,7 +219,7 @@ static int pos2scrpos(GUI_ENTRY_REC *entry, int pos, int cursor)
if (term_type == TERM_TYPE_BIG5) if (term_type == TERM_TYPE_BIG5)
xpos += big5_width(c); xpos += big5_width(c);
else if (entry->utf8) else if (entry->utf8)
xpos += unichar_isprint(c) ? mk_wcwidth(c) : 1; xpos += unichar_isprint(c) ? i_wcwidth(c) : 1;
else else
xpos++; xpos++;
@ -246,7 +246,7 @@ static int scrpos2pos(GUI_ENTRY_REC *entry, int pos)
if (term_type == TERM_TYPE_BIG5) if (term_type == TERM_TYPE_BIG5)
width = big5_width(c); width = big5_width(c);
else if (entry->utf8) else if (entry->utf8)
width = unichar_isprint(c) ? mk_wcwidth(c) : 1; width = unichar_isprint(c) ? i_wcwidth(c) : 1;
else else
width = 1; width = 1;
@ -373,7 +373,7 @@ static void gui_entry_draw_from(GUI_ENTRY_REC *entry, int pos)
else if (term_type == TERM_TYPE_BIG5) else if (term_type == TERM_TYPE_BIG5)
new_xpos += big5_width(c); new_xpos += big5_width(c);
else if (entry->utf8) else if (entry->utf8)
new_xpos += unichar_isprint(c) ? mk_wcwidth(c) : 1; new_xpos += unichar_isprint(c) ? i_wcwidth(c) : 1;
else else
new_xpos++; new_xpos++;
@ -647,7 +647,7 @@ void gui_entry_insert_char(GUI_ENTRY_REC *entry, unichar chr)
if (chr == 0 || chr == 13 || chr == 10) if (chr == 0 || chr == 13 || chr == 10)
return; /* never insert NUL, CR or LF characters */ return; /* never insert NUL, CR or LF characters */
if (entry->utf8 && entry->pos == 0 && mk_wcwidth(chr) == 0) if (entry->utf8 && entry->pos == 0 && i_wcwidth(chr) == 0)
return; return;
gui_entry_redraw_from(entry, entry->pos); gui_entry_redraw_from(entry, entry->pos);
@ -829,7 +829,7 @@ void gui_entry_erase(GUI_ENTRY_REC *entry, int size, CUTBUFFER_UPDATE_OP update_
if (entry->utf8) if (entry->utf8)
while (entry->pos-size-w > 0 && while (entry->pos-size-w > 0 &&
mk_wcwidth(entry->text[entry->pos-size-w]) == 0) w++; i_wcwidth(entry->text[entry->pos-size-w]) == 0) w++;
g_memmove(entry->text + entry->pos - size, entry->text + entry->pos, g_memmove(entry->text + entry->pos - size, entry->text + entry->pos,
(entry->text_len-entry->pos+1) * sizeof(unichar)); (entry->text_len-entry->pos+1) * sizeof(unichar));
@ -867,7 +867,7 @@ void gui_entry_erase_cell(GUI_ENTRY_REC *entry)
if (entry->utf8) if (entry->utf8)
while (entry->pos+size < entry->text_len && while (entry->pos+size < entry->text_len &&
mk_wcwidth(entry->text[entry->pos+size]) == 0) size++; i_wcwidth(entry->text[entry->pos+size]) == 0) size++;
g_memmove(entry->text + entry->pos, entry->text + entry->pos + size, g_memmove(entry->text + entry->pos, entry->text + entry->pos + size,
(entry->text_len-entry->pos-size+1) * sizeof(unichar)); (entry->text_len-entry->pos-size+1) * sizeof(unichar));
@ -1188,7 +1188,7 @@ void gui_entry_move_pos(GUI_ENTRY_REC *entry, int pos)
if (entry->utf8) { if (entry->utf8) {
int step = pos < 0 ? -1 : 1; int step = pos < 0 ? -1 : 1;
while(mk_wcwidth(entry->text[entry->pos]) == 0 && while(i_wcwidth(entry->text[entry->pos]) == 0 &&
entry->pos + step >= 0 && entry->pos + step <= entry->text_len) entry->pos + step >= 0 && entry->pos + step <= entry->text_len)
entry->pos += step; entry->pos += step;
} }

View File

@ -31,6 +31,7 @@
#include "gui-printtext.h" #include "gui-printtext.h"
static int window_create_override; static int window_create_override;
static int wcwidth_impl;
static GUI_WINDOW_REC *gui_window_init(WINDOW_REC *window, static GUI_WINDOW_REC *gui_window_init(WINDOW_REC *window,
MAIN_WINDOW_REC *parent) MAIN_WINDOW_REC *parent)
@ -51,6 +52,7 @@ static GUI_WINDOW_REC *gui_window_init(WINDOW_REC *window,
!settings_get_bool("indent_always"), !settings_get_bool("indent_always"),
get_default_indent_func()); get_default_indent_func());
textbuffer_view_set_break_wide(gui->view, settings_get_bool("break_wide")); textbuffer_view_set_break_wide(gui->view, settings_get_bool("break_wide"));
wcwidth_impl = settings_get_choice("wcwidth_implementation");
textbuffer_view_set_hidden_level(gui->view, MSGLEVEL_HIDDEN); textbuffer_view_set_hidden_level(gui->view, MSGLEVEL_HIDDEN);
if (parent->active == window) if (parent->active == window)
textbuffer_view_set_window(gui->view, parent->screen_win); textbuffer_view_set_window(gui->view, parent->screen_win);
@ -202,11 +204,18 @@ void gui_window_reparent(WINDOW_REC *window, MAIN_WINDOW_REC *parent)
void gui_windows_reset_settings(void) void gui_windows_reset_settings(void)
{ {
GSList *tmp; GSList *tmp;
int old_wcwidth_impl = wcwidth_impl;
wcwidth_impl = settings_get_choice("wcwidth_implementation");
for (tmp = windows; tmp != NULL; tmp = tmp->next) { for (tmp = windows; tmp != NULL; tmp = tmp->next) {
WINDOW_REC *rec = tmp->data; WINDOW_REC *rec = tmp->data;
GUI_WINDOW_REC *gui = WINDOW_GUI(rec); GUI_WINDOW_REC *gui = WINDOW_GUI(rec);
if (old_wcwidth_impl != wcwidth_impl) {
textbuffer_view_reset_cache(gui->view);
}
textbuffer_view_set_break_wide(gui->view, settings_get_bool("break_wide")); textbuffer_view_set_break_wide(gui->view, settings_get_bool("break_wide"));
textbuffer_view_set_default_indent(gui->view, textbuffer_view_set_default_indent(gui->view,
@ -217,6 +226,10 @@ void gui_windows_reset_settings(void)
textbuffer_view_set_scroll(gui->view, textbuffer_view_set_scroll(gui->view,
gui->use_scroll ? gui->scroll : gui->use_scroll ? gui->scroll :
settings_get_bool("scroll")); settings_get_bool("scroll"));
if (old_wcwidth_impl != wcwidth_impl) {
textbuffer_view_redraw(gui->view);
}
} }
} }

View File

@ -515,7 +515,7 @@ void term_add_unichar(TERM_WINDOW *window, unichar chr)
switch (term_type) { switch (term_type) {
case TERM_TYPE_UTF8: case TERM_TYPE_UTF8:
term_printed_text(unichar_isprint(chr) ? mk_wcwidth(chr) : 1); term_printed_text(unichar_isprint(chr) ? i_wcwidth(chr) : 1);
term_addch_utf8(window, chr); term_addch_utf8(window, chr);
break; break;
case TERM_TYPE_BIG5: case TERM_TYPE_BIG5:
@ -558,7 +558,7 @@ int term_addstr(TERM_WINDOW *window, const char *str)
len++; len++;
ptr++; ptr++;
} else { } else {
len += unichar_isprint(tmp) ? mk_wcwidth(tmp) : 1; len += unichar_isprint(tmp) ? i_wcwidth(tmp) : 1;
ptr = g_utf8_next_char(ptr); ptr = g_utf8_next_char(ptr);
} }
} }

View File

@ -197,7 +197,7 @@ static inline unichar read_unichar(const unsigned char *data, const unsigned cha
*width = 1; *width = 1;
} else { } else {
*next = (unsigned char *)g_utf8_next_char(data); *next = (unsigned char *)g_utf8_next_char(data);
*width = unichar_isprint(chr) ? mk_wcwidth(chr) : 1; *width = unichar_isprint(chr) ? i_wcwidth(chr) : 1;
} }
return chr; return chr;
} }
@ -374,7 +374,7 @@ static void view_update_cache(TEXT_BUFFER_VIEW_REC *view, LINE_REC *line,
view->cache->last_linecount = view_get_linecount(view, line); view->cache->last_linecount = view_get_linecount(view, line);
} }
static void view_reset_cache(TEXT_BUFFER_VIEW_REC *view) void textbuffer_view_reset_cache(TEXT_BUFFER_VIEW_REC *view)
{ {
GSList *tmp; GSList *tmp;
@ -691,7 +691,7 @@ void textbuffer_view_set_break_wide(TEXT_BUFFER_VIEW_REC *view,
{ {
if (view->break_wide != break_wide) { if (view->break_wide != break_wide) {
view->break_wide = break_wide; view->break_wide = break_wide;
view_reset_cache(view); textbuffer_view_reset_cache(view);
} }
} }
@ -703,7 +703,7 @@ static void view_unregister_indent_func(TEXT_BUFFER_VIEW_REC *view,
/* recreate cache so it won't contain references /* recreate cache so it won't contain references
to the indent function */ to the indent function */
view_reset_cache(view); textbuffer_view_reset_cache(view);
view->cache = textbuffer_cache_get(view->siblings, view->width); view->cache = textbuffer_cache_get(view->siblings, view->width);
} }
@ -1314,7 +1314,7 @@ void textbuffer_view_remove_all_lines(TEXT_BUFFER_VIEW_REC *view)
g_hash_table_foreach_remove(view->bookmarks, g_hash_table_foreach_remove(view->bookmarks,
(GHRFunc) g_free_true, NULL); (GHRFunc) g_free_true, NULL);
view_reset_cache(view); textbuffer_view_reset_cache(view);
textbuffer_view_clear(view); textbuffer_view_clear(view);
g_slist_foreach(view->siblings, (GFunc) textbuffer_view_clear, NULL); g_slist_foreach(view->siblings, (GFunc) textbuffer_view_clear, NULL);
} }

View File

@ -129,6 +129,8 @@ void textbuffer_view_scroll_line(TEXT_BUFFER_VIEW_REC *view, LINE_REC *line);
/* Return line cache */ /* Return line cache */
LINE_CACHE_REC *textbuffer_view_get_line_cache(TEXT_BUFFER_VIEW_REC *view, LINE_CACHE_REC *textbuffer_view_get_line_cache(TEXT_BUFFER_VIEW_REC *view,
LINE_REC *line); LINE_REC *line);
/* Reset the whole line cache */
void textbuffer_view_reset_cache(TEXT_BUFFER_VIEW_REC *view);
/* /*
Functions for manipulating the text buffer, using these commands update Functions for manipulating the text buffer, using these commands update