rofi/source/widgets/textbox.c

828 lines
24 KiB
C
Raw Normal View History

2015-03-08 14:43:31 +00:00
/**
* MIT/X11 License
* Copyright (c) 2012 Sean Pringle <sean.pringle@gmail.com>
2015-12-31 23:27:00 +00:00
* Modified (c) 2013-2016 Qball Cow <qball@gmpclient.org>
2015-03-08 14:43:31 +00:00
*
* Permission is hereby granted, free of charge, to any person obtaining
* a copy of this software and associated documentation files (the
* "Software"), to deal in the Software without restriction, including
* without limitation the rights to use, copy, modify, merge, publish,
* distribute, sublicense, and/or sell copies of the Software, and to
* permit persons to whom the Software is furnished to do so, subject to
* the following conditions:
*
* The above copyright notice and this permission notice shall be
* included in all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
* OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY
* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,
* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE
* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
2014-03-22 20:04:19 +00:00
*/
2012-08-24 01:28:39 +00:00
#include <config.h>
#include <xcb/xcb.h>
#include <ctype.h>
2016-01-07 15:01:56 +00:00
#include <string.h>
2015-09-27 10:57:54 +00:00
#include <glib.h>
2016-07-18 21:49:26 +00:00
#include <math.h>
2016-01-07 15:01:56 +00:00
#include "settings.h"
#include "widgets/textbox.h"
#include "keyb.h"
2015-09-27 10:57:54 +00:00
#include "x11-helper.h"
2016-02-10 21:12:49 +00:00
#include "mode.h"
#include "view.h"
2015-09-27 10:57:54 +00:00
#define DOT_OFFSET 15
2016-07-18 21:49:26 +00:00
static void textbox_draw ( widget *, cairo_t * );
static void textbox_free ( widget * );
static int textbox_get_width ( widget * );
static int _textbox_get_height ( widget * );
/**
* @param tb Handle to the textbox
*
* Move the cursor to the end of the string.
*/
static void textbox_cursor_end ( textbox *tb );
2014-05-26 07:00:14 +00:00
/**
* Font + font color cache.
* Avoid re-loading font on every change on every textbox.
*/
2016-03-24 21:13:19 +00:00
typedef struct
2015-04-06 15:13:26 +00:00
{
2015-09-26 18:34:34 +00:00
Color fg;
Color bg;
Color bgalt;
Color hlfg;
Color hlbg;
2015-04-06 15:13:26 +00:00
} RowColor;
2016-10-14 16:56:09 +00:00
/** Number of states */
2015-04-06 15:13:26 +00:00
#define num_states 3
2016-10-14 16:56:09 +00:00
/**
* Different colors for the different states
*/
2016-10-15 13:39:08 +00:00
RowColor colors[num_states];
2014-08-02 18:02:37 +00:00
2016-10-14 16:56:09 +00:00
/** Default pango context */
2016-07-31 21:28:31 +00:00
static PangoContext *p_context = NULL;
2016-10-14 16:56:09 +00:00
/** The pango font metrics */
static PangoFontMetrics *p_metrics = NULL;
2016-10-14 16:56:09 +00:00
2016-02-06 12:06:58 +00:00
static gboolean textbox_blink ( gpointer data )
{
2016-02-06 12:06:58 +00:00
textbox *tb = (textbox *) data;
if ( tb->blink < 2 ) {
tb->blink = !tb->blink;
tb->update = TRUE;
widget_queue_redraw ( WIDGET ( tb ) );
2016-02-06 12:06:58 +00:00
rofi_view_queue_redraw ( );
}
else {
tb->blink--;
}
return TRUE;
}
2014-08-02 18:02:37 +00:00
static void textbox_resize ( widget *wid, short w, short h )
{
textbox *tb = (textbox *) wid;
textbox_moveresize ( tb, tb->widget.x, tb->widget.y, w, h );
}
2015-09-26 18:34:34 +00:00
textbox* textbox_create ( TextboxFlags flags, short x, short y, short w, short h,
2015-09-19 10:57:48 +00:00
TextBoxFontType tbft, const char *text )
2012-08-24 01:28:39 +00:00
{
2016-02-19 18:29:06 +00:00
textbox *tb = g_slice_new0 ( textbox );
2012-08-24 01:28:39 +00:00
tb->widget.draw = textbox_draw;
tb->widget.free = textbox_free;
tb->widget.resize = textbox_resize;
tb->widget.get_width = textbox_get_width;
tb->widget.get_height = _textbox_get_height;
tb->flags = flags;
2012-08-24 01:28:39 +00:00
tb->widget.x = x;
tb->widget.y = y;
tb->widget.w = MAX ( 1, w );
tb->widget.h = MAX ( 1, h );
2012-11-07 04:16:59 +00:00
tb->changed = FALSE;
tb->main_surface = cairo_image_surface_create ( CAIRO_FORMAT_ARGB32, tb->widget.w, tb->widget.h );
2015-09-26 18:34:34 +00:00
tb->main_draw = cairo_create ( tb->main_surface );
tb->layout = pango_layout_new ( p_context );
textbox_font ( tb, tbft );
2012-08-24 01:28:39 +00:00
2015-09-23 18:44:24 +00:00
if ( ( flags & TB_WRAP ) == TB_WRAP ) {
pango_layout_set_wrap ( tb->layout, PANGO_WRAP_WORD_CHAR );
}
2015-09-22 20:23:52 +00:00
textbox_text ( tb, text ? text : "" );
2014-03-22 20:04:19 +00:00
textbox_cursor_end ( tb );
2012-08-24 01:28:39 +00:00
2014-01-10 09:35:38 +00:00
// auto height/width modes get handled here
textbox_moveresize ( tb, tb->widget.x, tb->widget.y, tb->widget.w, tb->widget.h );
2012-08-24 01:28:39 +00:00
tb->blink_timeout = 0;
tb->blink = 1;
if ( ( flags & TB_EDITABLE ) == TB_EDITABLE ) {
2016-02-06 12:06:58 +00:00
tb->blink_timeout = g_timeout_add ( 1200, textbox_blink, tb );
}
2016-06-26 13:48:12 +00:00
// Enabled by default
tb->widget.enabled = TRUE;
2014-01-10 09:35:38 +00:00
return tb;
2012-08-24 01:28:39 +00:00
}
void textbox_font ( textbox *tb, TextBoxFontType tbft )
2012-08-24 01:28:39 +00:00
{
2015-10-26 13:22:58 +00:00
TextBoxFontType t = tbft & STATE_MASK;
if ( tb == NULL ) {
return;
}
2015-10-26 13:22:58 +00:00
// ACTIVE has priority over URGENT if both set.
if ( t == ( URGENT | ACTIVE ) ) {
t = ACTIVE;
}
RowColor *color = &( colors[t] );
2015-04-06 15:13:26 +00:00
switch ( ( tbft & FMOD_MASK ) )
2014-03-22 20:04:19 +00:00
{
case HIGHLIGHT:
2015-04-06 15:13:26 +00:00
tb->color_bg = color->hlbg;
tb->color_fg = color->hlfg;
break;
case ALT:
2015-04-06 15:13:26 +00:00
tb->color_bg = color->bgalt;
tb->color_fg = color->fg;
break;
default:
2015-04-06 15:13:26 +00:00
tb->color_bg = color->bg;
tb->color_fg = color->fg;
break;
}
2015-09-26 18:34:34 +00:00
if ( tb->tbft != tbft ) {
tb->update = TRUE;
widget_queue_redraw ( WIDGET ( tb ) );
2015-09-26 18:34:34 +00:00
}
tb->tbft = tbft;
2012-08-24 01:28:39 +00:00
}
/**
* @param tb The textbox object.
*
* Update the pango layout's text. It does this depending on the
* textbox flags.
*/
static void __textbox_update_pango_text ( textbox *tb )
{
if ( ( tb->flags & TB_PASSWORD ) == TB_PASSWORD ) {
size_t l = g_utf8_strlen ( tb->text, -1 );
char string [l + 1];
memset ( string, '*', l );
string[l] = '\0';
pango_layout_set_attributes ( tb->layout, NULL );
pango_layout_set_text ( tb->layout, string, l );
}
2016-02-09 20:25:29 +00:00
else if ( tb->flags & TB_MARKUP || tb->tbft & MARKUP ) {
2016-01-11 20:25:38 +00:00
pango_layout_set_markup ( tb->layout, tb->text, -1 );
}
else {
pango_layout_set_attributes ( tb->layout, NULL );
pango_layout_set_text ( tb->layout, tb->text, -1 );
}
}
2016-08-23 22:39:56 +00:00
const char *textbox_get_visible_text ( const textbox *tb )
{
return pango_layout_get_text ( tb->layout );
}
PangoAttrList *textbox_get_pango_attributes ( textbox *tb )
{
return pango_layout_get_attributes ( tb->layout );
}
void textbox_set_pango_attributes ( textbox *tb, PangoAttrList *list )
{
pango_layout_set_attributes ( tb->layout, list );
}
2012-08-24 01:28:39 +00:00
// set the default text to display
2015-02-03 07:21:59 +00:00
void textbox_text ( textbox *tb, const char *text )
2012-08-24 01:28:39 +00:00
{
2015-09-26 18:34:34 +00:00
tb->update = TRUE;
g_free ( tb->text );
const gchar *last_pointer = NULL;
2015-09-26 18:34:34 +00:00
if ( g_utf8_validate ( text, -1, &last_pointer ) ) {
tb->text = g_strdup ( text );
}
else {
if ( last_pointer != NULL ) {
// Copy string up to invalid character.
tb->text = g_strndup ( text, ( last_pointer - text ) );
}
else {
tb->text = g_strdup ( "Invalid UTF-8 string." );
}
}
__textbox_update_pango_text ( tb );
if ( tb->flags & TB_AUTOWIDTH ) {
textbox_moveresize ( tb, tb->widget.x, tb->widget.y, tb->widget.w, tb->widget.h );
widget_update ( WIDGET ( tb ) );
}
tb->cursor = MAX ( 0, MIN ( ( int ) g_utf8_strlen ( tb->text, -1 ), tb->cursor ) );
widget_queue_redraw ( WIDGET ( tb ) );
2012-08-24 01:28:39 +00:00
}
2015-03-05 19:26:52 +00:00
// within the parent handled auto width/height modes
2014-03-22 20:04:19 +00:00
void textbox_moveresize ( textbox *tb, int x, int y, int w, int h )
2012-08-24 01:28:39 +00:00
{
2016-07-18 21:49:26 +00:00
unsigned int offset = ( tb->flags & TB_INDICATOR ) ? DOT_OFFSET : 0;
2014-06-04 19:29:23 +00:00
if ( tb->flags & TB_AUTOWIDTH ) {
pango_layout_set_width ( tb->layout, -1 );
unsigned int offset = ( tb->flags & TB_INDICATOR ) ? DOT_OFFSET : 0;
w = textbox_get_font_width ( tb ) + 2 * config.line_padding + offset;
2014-03-22 20:04:19 +00:00
}
2014-08-02 18:02:37 +00:00
else {
// set ellipsize
if ( ( tb->flags & TB_EDITABLE ) == TB_EDITABLE ) {
pango_layout_set_ellipsize ( tb->layout, PANGO_ELLIPSIZE_MIDDLE );
}
2015-09-23 18:44:24 +00:00
else if ( ( tb->flags & TB_WRAP ) != TB_WRAP ) {
pango_layout_set_ellipsize ( tb->layout, PANGO_ELLIPSIZE_END );
}
2014-08-02 18:02:37 +00:00
}
2014-01-10 09:35:38 +00:00
if ( tb->flags & TB_AUTOHEIGHT ) {
// Width determines height!
int tw = MAX ( 1, w );
pango_layout_set_width ( tb->layout, PANGO_SCALE * ( tw - 2 * config.line_padding - offset ) );
h = textbox_get_height ( tb );
}
if ( x != tb->widget.x || y != tb->widget.y || w != tb->widget.w || h != tb->widget.h ) {
tb->widget.x = x;
tb->widget.y = y;
tb->widget.h = MAX ( 1, h );
tb->widget.w = MAX ( 1, w );
2014-01-10 09:35:38 +00:00
}
2015-09-26 18:34:34 +00:00
// We always want to update this
pango_layout_set_width ( tb->layout, PANGO_SCALE * ( tb->widget.w - 2 * config.line_padding - offset ) );
tb->update = TRUE;
widget_queue_redraw ( WIDGET ( tb ) );
2012-08-24 01:28:39 +00:00
}
// will also unmap the window if still displayed
static void textbox_free ( widget *wid )
2012-08-24 01:28:39 +00:00
{
textbox *tb = (textbox *) wid;
if ( tb->blink_timeout > 0 ) {
g_source_remove ( tb->blink_timeout );
tb->blink_timeout = 0;
}
2014-01-10 09:35:38 +00:00
g_free ( tb->text );
if ( tb->layout != NULL ) {
2014-08-02 18:02:37 +00:00
g_object_unref ( tb->layout );
}
2015-09-26 18:34:34 +00:00
if ( tb->main_draw ) {
cairo_destroy ( tb->main_draw );
tb->main_draw = NULL;
}
if ( tb->main_surface ) {
cairo_surface_destroy ( tb->main_surface );
tb->main_surface = NULL;
}
2014-01-10 09:35:38 +00:00
2016-02-19 18:29:06 +00:00
g_slice_free ( textbox, tb );
2012-08-24 01:28:39 +00:00
}
2015-09-26 18:34:34 +00:00
static void texbox_update ( textbox *tb )
2012-08-24 01:28:39 +00:00
{
2015-09-26 18:34:34 +00:00
if ( tb->update ) {
unsigned int offset = ( tb->flags & TB_INDICATOR ) ? DOT_OFFSET : 0;
2015-09-26 18:34:34 +00:00
if ( tb->main_surface ) {
cairo_destroy ( tb->main_draw );
cairo_surface_destroy ( tb->main_surface );
tb->main_draw = NULL;
tb->main_surface = NULL;
}
tb->main_surface = cairo_image_surface_create ( CAIRO_FORMAT_ARGB32, tb->widget.w, tb->widget.h );
2015-09-26 18:34:34 +00:00
tb->main_draw = cairo_create ( tb->main_surface );
cairo_set_operator ( tb->main_draw, CAIRO_OPERATOR_SOURCE );
pango_cairo_update_layout ( tb->main_draw, tb->layout );
int font_height = textbox_get_font_height ( tb );
2015-09-26 18:34:34 +00:00
int cursor_x = 0;
int cursor_width = MAX ( 2, font_height / 10 );
2015-09-26 18:34:34 +00:00
if ( tb->changed ) {
__textbox_update_pango_text ( tb );
}
2014-08-02 18:02:37 +00:00
2015-09-26 18:34:34 +00:00
if ( tb->flags & TB_EDITABLE ) {
// We want to place the cursor based on the text shown.
const char *text = pango_layout_get_text ( tb->layout );
// Clamp the position, should not be needed, but we are paranoid.
int cursor_offset = MIN ( tb->cursor, g_utf8_strlen ( text, -1 ) );
2015-09-26 18:34:34 +00:00
PangoRectangle pos;
// convert to byte location.
char *offset = g_utf8_offset_to_pointer ( text, cursor_offset );
pango_layout_get_cursor_pos ( tb->layout, offset - text, &pos, NULL );
2015-09-26 18:34:34 +00:00
cursor_x = pos.x / PANGO_SCALE;
}
2012-08-24 01:28:39 +00:00
2015-09-26 18:34:34 +00:00
// Skip the side MARGIN on the X axis.
int x = config.line_padding + offset;
2015-09-26 18:34:34 +00:00
int y = 0;
2012-08-24 01:28:39 +00:00
2015-09-26 18:34:34 +00:00
if ( tb->flags & TB_RIGHT ) {
int line_width = 0;
// Get actual width.
pango_layout_get_pixel_size ( tb->layout, &line_width, NULL );
x = ( tb->widget.w - line_width - config.line_padding - offset );
2015-09-26 18:34:34 +00:00
}
else if ( tb->flags & TB_CENTER ) {
int tw = textbox_get_font_width ( tb );
x = ( ( tb->widget.w - tw - 2 * config.line_padding - offset ) ) / 2;
2015-09-26 18:34:34 +00:00
}
y = config.line_padding + ( pango_font_metrics_get_ascent ( p_metrics ) - pango_layout_get_baseline ( tb->layout ) ) / PANGO_SCALE;
2015-09-26 18:34:34 +00:00
// Set ARGB
2016-07-18 21:49:26 +00:00
Color col = tb->color_bg;
cairo_set_source_rgba ( tb->main_draw, col.red, col.green, col.blue, col.alpha );
2015-09-26 18:34:34 +00:00
cairo_paint ( tb->main_draw );
col = tb->color_fg;
2016-07-18 21:49:26 +00:00
cairo_set_source_rgba ( tb->main_draw, col.red, col.green, col.blue, col.alpha );
2015-09-26 18:34:34 +00:00
// draw the cursor
if ( tb->flags & TB_EDITABLE && tb->blink ) {
2015-09-26 18:34:34 +00:00
cairo_rectangle ( tb->main_draw, x + cursor_x, y, cursor_width, font_height );
cairo_fill ( tb->main_draw );
}
2012-08-24 01:28:39 +00:00
// Set ARGB
// We need to set over, otherwise subpixel hinting wont work.
cairo_set_operator ( tb->main_draw, CAIRO_OPERATOR_OVER );
cairo_move_to ( tb->main_draw, x, y );
pango_cairo_show_layout ( tb->main_draw, tb->layout );
2016-07-18 21:49:26 +00:00
if ( ( tb->flags & TB_INDICATOR ) == TB_INDICATOR && ( tb->tbft & ( SELECTED | HIGHLIGHT ) ) ) {
if ( ( tb->tbft & SELECTED ) == SELECTED ) {
cairo_set_source_rgba ( tb->main_draw, col.red, col.green, col.blue, col.alpha );
}
else if ( ( tb->tbft & HIGHLIGHT ) == HIGHLIGHT ) {
cairo_set_source_rgba ( tb->main_draw, col.red, col.green, col.blue, col.alpha * 0.2 );
}
cairo_arc ( tb->main_draw, DOT_OFFSET / 2.0, tb->widget.h / 2.0, 2.0, 0, 2.0 * M_PI );
cairo_fill ( tb->main_draw );
}
2015-09-26 18:34:34 +00:00
tb->update = FALSE;
2014-03-22 20:04:19 +00:00
}
2015-09-26 18:34:34 +00:00
}
static void textbox_draw ( widget *wid, cairo_t *draw )
2015-09-26 18:34:34 +00:00
{
textbox *tb = (textbox *) wid;
2015-09-26 18:34:34 +00:00
texbox_update ( tb );
2012-08-24 01:28:39 +00:00
2015-09-26 18:34:34 +00:00
/* Write buffer */
2012-08-24 01:28:39 +00:00
cairo_set_source_surface ( draw, tb->main_surface, tb->widget.x, tb->widget.y );
cairo_rectangle ( draw, tb->widget.x, tb->widget.y, tb->widget.w, tb->widget.h );
2015-09-26 18:34:34 +00:00
cairo_fill ( draw );
2012-08-24 01:28:39 +00:00
}
// cursor handling for edit mode
2014-03-22 20:04:19 +00:00
void textbox_cursor ( textbox *tb, int pos )
2012-08-24 01:28:39 +00:00
{
int length = ( tb->text == NULL ) ? 0 : g_utf8_strlen ( tb->text, -1 );
2015-02-13 14:37:55 +00:00
tb->cursor = MAX ( 0, MIN ( length, pos ) );
2015-09-26 18:34:34 +00:00
tb->update = TRUE;
// Stop blink!
tb->blink = 3;
widget_queue_redraw ( WIDGET ( tb ) );
2012-08-24 01:28:39 +00:00
}
/**
* @param tb Handle to the textbox
*
* Move cursor one position forward.
*/
static void textbox_cursor_inc ( textbox *tb )
2012-08-24 01:28:39 +00:00
{
textbox_cursor ( tb, tb->cursor + 1 );
2012-08-24 01:28:39 +00:00
}
/**
* @param tb Handle to the textbox
*
* Move cursor one position backward.
*/
static void textbox_cursor_dec ( textbox *tb )
2012-08-24 01:28:39 +00:00
{
textbox_cursor ( tb, tb->cursor - 1 );
2012-08-24 01:28:39 +00:00
}
// Move word right
static void textbox_cursor_inc_word ( textbox *tb )
{
if ( tb->text == NULL ) {
2015-02-13 14:37:55 +00:00
return;
}
// Find word boundaries, with pango_Break?
gchar *c = g_utf8_offset_to_pointer ( tb->text, tb->cursor );
while ( ( c = g_utf8_next_char ( c ) ) ) {
gunichar uc = g_utf8_get_char ( c );
GUnicodeBreakType bt = g_unichar_break_type ( uc );
2015-09-19 18:59:50 +00:00
if ( ( bt == G_UNICODE_BREAK_ALPHABETIC || bt == G_UNICODE_BREAK_HEBREW_LETTER ||
bt == G_UNICODE_BREAK_NUMERIC || bt == G_UNICODE_BREAK_QUOTATION ) ) {
break;
}
}
2016-10-22 19:34:19 +00:00
if ( c == NULL || *c == '\0' ) {
2015-02-13 14:37:55 +00:00
return;
}
while ( ( c = g_utf8_next_char ( c ) ) ) {
gunichar uc = g_utf8_get_char ( c );
GUnicodeBreakType bt = g_unichar_break_type ( uc );
2015-09-19 18:59:50 +00:00
if ( !( bt == G_UNICODE_BREAK_ALPHABETIC || bt == G_UNICODE_BREAK_HEBREW_LETTER ||
bt == G_UNICODE_BREAK_NUMERIC || bt == G_UNICODE_BREAK_QUOTATION ) ) {
break;
}
}
int index = g_utf8_pointer_to_offset ( tb->text, c );
textbox_cursor ( tb, index );
}
// move word left
static void textbox_cursor_dec_word ( textbox *tb )
{
// Find word boundaries, with pango_Break?
gchar *n;
gchar *c = g_utf8_offset_to_pointer ( tb->text, tb->cursor );
while ( ( c = g_utf8_prev_char ( c ) ) && c != tb->text ) {
gunichar uc = g_utf8_get_char ( c );
GUnicodeBreakType bt = g_unichar_break_type ( uc );
2015-09-19 18:59:50 +00:00
if ( ( bt == G_UNICODE_BREAK_ALPHABETIC || bt == G_UNICODE_BREAK_HEBREW_LETTER ||
bt == G_UNICODE_BREAK_NUMERIC || bt == G_UNICODE_BREAK_QUOTATION ) ) {
break;
}
}
if ( c != tb->text ) {
while ( ( n = g_utf8_prev_char ( c ) ) ) {
gunichar uc = g_utf8_get_char ( n );
GUnicodeBreakType bt = g_unichar_break_type ( uc );
2015-09-19 18:59:50 +00:00
if ( !( bt == G_UNICODE_BREAK_ALPHABETIC || bt == G_UNICODE_BREAK_HEBREW_LETTER ||
bt == G_UNICODE_BREAK_NUMERIC || bt == G_UNICODE_BREAK_QUOTATION ) ) {
break;
}
c = n;
if ( n == tb->text ) {
break;
}
}
}
int index = g_utf8_pointer_to_offset ( tb->text, c );
textbox_cursor ( tb, index );
}
2012-08-24 01:28:39 +00:00
// end of line
static void textbox_cursor_end ( textbox *tb )
2012-08-24 01:28:39 +00:00
{
2015-12-23 16:59:03 +00:00
if ( tb->text == NULL ) {
tb->cursor = 0;
tb->update = TRUE;
widget_queue_redraw ( WIDGET ( tb ) );
2015-12-23 16:59:03 +00:00
return;
}
tb->cursor = ( int ) g_utf8_strlen ( tb->text, -1 );
2015-09-26 18:34:34 +00:00
tb->update = TRUE;
widget_queue_redraw ( WIDGET ( tb ) );
// Stop blink!
tb->blink = 2;
2012-08-24 01:28:39 +00:00
}
// insert text
2016-08-23 22:39:56 +00:00
void textbox_insert ( textbox *tb, const int char_pos, const char *str, const int slen )
2012-08-24 01:28:39 +00:00
{
char *c = g_utf8_offset_to_pointer ( tb->text, char_pos );
int pos = c - tb->text;
int len = ( int ) strlen ( tb->text );
2014-03-22 20:04:19 +00:00
pos = MAX ( 0, MIN ( len, pos ) );
2014-01-10 09:35:38 +00:00
// expand buffer
tb->text = g_realloc ( tb->text, len + slen + 1 );
2014-01-10 09:35:38 +00:00
// move everything after cursor upward
char *at = tb->text + pos;
2014-03-22 20:04:19 +00:00
memmove ( at + slen, at, len - pos + 1 );
2014-01-10 09:35:38 +00:00
// insert new str
2014-03-22 20:04:19 +00:00
memmove ( at, str, slen );
// Set modified, lay out need te be redrawn
// Stop blink!
tb->blink = 2;
tb->changed = TRUE;
2015-09-26 18:34:34 +00:00
tb->update = TRUE;
2012-08-24 01:28:39 +00:00
}
// remove text
2014-03-22 20:04:19 +00:00
void textbox_delete ( textbox *tb, int pos, int dlen )
2012-08-24 01:28:39 +00:00
{
int len = g_utf8_strlen ( tb->text, -1 );
if ( len == pos ) {
return;
}
2014-03-22 20:04:19 +00:00
pos = MAX ( 0, MIN ( len, pos ) );
if ( ( pos + dlen ) > len ) {
dlen = len - dlen;
}
2014-01-10 09:35:38 +00:00
// move everything after pos+dlen down
char *start = g_utf8_offset_to_pointer ( tb->text, pos );
char *end = g_utf8_offset_to_pointer ( tb->text, pos + dlen );
// Move remainder + closing \0
memmove ( start, end, ( tb->text + strlen ( tb->text ) ) - end + 1 );
if ( tb->cursor >= pos && tb->cursor < ( pos + dlen ) ) {
tb->cursor = pos;
}
else if ( tb->cursor >= ( pos + dlen ) ) {
tb->cursor -= dlen;
}
// Set modified, lay out need te be redrawn
// Stop blink!
tb->blink = 2;
tb->changed = TRUE;
2015-09-26 18:34:34 +00:00
tb->update = TRUE;
2012-08-24 01:28:39 +00:00
}
/**
* @param tb Handle to the textbox
*
* Delete character after cursor.
*/
static void textbox_cursor_del ( textbox *tb )
2012-08-24 01:28:39 +00:00
{
if ( tb->text == NULL ) {
2015-02-13 14:37:55 +00:00
return;
}
textbox_delete ( tb, tb->cursor, 1 );
2012-08-24 01:28:39 +00:00
}
/**
* @param tb Handle to the textbox
*
* Delete character before cursor.
*/
static void textbox_cursor_bkspc ( textbox *tb )
2012-08-24 01:28:39 +00:00
{
2014-06-04 19:29:23 +00:00
if ( tb->cursor > 0 ) {
2014-03-22 20:04:19 +00:00
textbox_cursor_dec ( tb );
textbox_cursor_del ( tb );
2014-01-10 09:35:38 +00:00
}
2012-08-24 01:28:39 +00:00
}
static void textbox_cursor_bkspc_word ( textbox *tb )
{
if ( tb->cursor > 0 ) {
int cursor = tb->cursor;
textbox_cursor_dec_word ( tb );
if ( cursor > tb->cursor ) {
textbox_delete ( tb, tb->cursor, cursor - tb->cursor );
}
}
}
2016-07-25 09:32:30 +00:00
static void textbox_cursor_del_eol ( textbox *tb )
{
if ( tb->cursor >= 0 ) {
int length = g_utf8_strlen ( tb->text, -1 ) - tb->cursor;
2016-07-25 09:32:30 +00:00
if ( length >= 0 ) {
textbox_delete ( tb, tb->cursor, length );
}
}
}
2016-07-27 06:10:55 +00:00
static void textbox_cursor_del_sol ( textbox *tb )
{
if ( tb->cursor >= 0 ) {
int length = tb->cursor;
if ( length >= 0 ) {
textbox_delete ( tb, 0, length );
}
}
}
static void textbox_cursor_del_word ( textbox *tb )
{
if ( tb->cursor >= 0 ) {
int cursor = tb->cursor;
textbox_cursor_inc_word ( tb );
if ( cursor < tb->cursor ) {
textbox_delete ( tb, cursor, tb->cursor - cursor );
}
}
}
2012-08-24 01:28:39 +00:00
// handle a keypress in edit mode
2015-08-29 21:02:30 +00:00
// 2 = nav
2012-08-24 01:28:39 +00:00
// 0 = unhandled
// 1 = handled
// -1 = handled and return pressed (finished)
int textbox_keybinding ( textbox *tb, KeyBindingAction action )
2012-08-24 01:28:39 +00:00
{
2014-06-04 19:29:23 +00:00
if ( !( tb->flags & TB_EDITABLE ) ) {
return 0;
2014-03-22 20:04:19 +00:00
}
switch ( action )
{
// Left or Ctrl-b
case MOVE_CHAR_BACK:
textbox_cursor_dec ( tb );
return 2;
// Right or Ctrl-F
case MOVE_CHAR_FORWARD:
textbox_cursor_inc ( tb );
return 2;
// Ctrl-U: Kill from the beginning to the end of the line.
case CLEAR_LINE:
textbox_text ( tb, "" );
return 1;
// Ctrl-A
case MOVE_FRONT:
textbox_cursor ( tb, 0 );
return 2;
// Ctrl-E
case MOVE_END:
textbox_cursor_end ( tb );
return 2;
// Ctrl-Alt-h
case REMOVE_WORD_BACK:
textbox_cursor_bkspc_word ( tb );
return 1;
// Ctrl-Alt-d
case REMOVE_WORD_FORWARD:
textbox_cursor_del_word ( tb );
return 1;
2016-07-25 09:32:30 +00:00
case REMOVE_TO_EOL:
textbox_cursor_del_eol ( tb );
return 1;
2016-07-27 06:10:55 +00:00
case REMOVE_TO_SOL:
textbox_cursor_del_sol ( tb );
return 1;
// Delete or Ctrl-D
case REMOVE_CHAR_FORWARD:
textbox_cursor_del ( tb );
return 1;
// Alt-B
case MOVE_WORD_BACK:
textbox_cursor_dec_word ( tb );
return 2;
// Alt-F
case MOVE_WORD_FORWARD:
textbox_cursor_inc_word ( tb );
return 2;
// BackSpace, Ctrl-h
case REMOVE_CHAR_BACK:
textbox_cursor_bkspc ( tb );
return 1;
default:
2016-05-08 09:13:11 +00:00
g_return_val_if_reached ( 0 );
}
}
2016-08-23 22:39:56 +00:00
gboolean textbox_append_char ( textbox *tb, const char *pad, const int pad_len )
{
if ( !( tb->flags & TB_EDITABLE ) ) {
return FALSE;
2014-03-22 20:04:19 +00:00
}
// Filter When alt/ctrl is pressed do not accept the character.
if ( !g_unichar_iscntrl ( g_utf8_get_char ( pad ) ) ) {
textbox_insert ( tb, tb->cursor, pad, pad_len );
textbox_cursor ( tb, tb->cursor + 1 );
return TRUE;
2014-01-10 09:35:38 +00:00
}
return FALSE;
2012-08-24 01:28:39 +00:00
}
/***
* Font setup.
*/
static void textbox_parse_string ( const char *str, RowColor *color )
2015-04-06 15:13:26 +00:00
{
if ( str == NULL ) {
return;
}
char *cstr = g_strdup ( str );
char *endp = NULL;
char *token;
int index = 0;
const char *const sep = ",";
for ( token = strtok_r ( cstr, sep, &endp ); token != NULL; token = strtok_r ( NULL, sep, &endp ) ) {
2015-04-06 15:13:26 +00:00
switch ( index )
{
case 0:
color->bg = color_get ( g_strstrip ( token ) );
2015-04-06 15:13:26 +00:00
break;
case 1:
color->fg = color_get ( g_strstrip ( token ) );
2015-04-06 15:13:26 +00:00
break;
case 2:
color->bgalt = color_get ( g_strstrip ( token ) );
2015-04-06 15:13:26 +00:00
break;
case 3:
color->hlbg = color_get ( g_strstrip ( token ) );
2015-04-06 15:13:26 +00:00
break;
case 4:
color->hlfg = color_get ( g_strstrip ( token ) );
2015-04-06 15:13:26 +00:00
break;
}
index++;
}
g_free ( cstr );
}
void textbox_setup ( void )
{
2016-03-05 17:28:39 +00:00
textbox_parse_string ( config.color_normal, &( colors[NORMAL] ) );
textbox_parse_string ( config.color_urgent, &( colors[URGENT] ) );
textbox_parse_string ( config.color_active, &( colors[ACTIVE] ) );
}
void textbox_set_pango_context ( PangoContext *p )
{
textbox_cleanup ();
p_context = g_object_ref ( p );
p_metrics = pango_context_get_metrics ( p_context, NULL, NULL );
2015-04-06 15:13:26 +00:00
}
void textbox_cleanup ( void )
{
if ( p_metrics ) {
pango_font_metrics_unref ( p_metrics );
p_metrics = NULL;
}
2014-08-02 18:02:37 +00:00
if ( p_context ) {
g_object_unref ( p_context );
2015-09-26 18:34:34 +00:00
p_context = NULL;
}
}
2014-08-02 18:02:37 +00:00
int textbox_get_width ( widget *wid )
2014-08-02 18:02:37 +00:00
{
textbox *tb = (textbox *) wid;
if ( !wid->expand ) {
if ( tb->flags & TB_AUTOWIDTH ) {
unsigned int offset = ( tb->flags & TB_INDICATOR ) ? DOT_OFFSET : 0;
return textbox_get_font_width ( tb ) + 2 * config.line_padding + offset;
}
return tb->widget.w;
}
return tb->widget.w;
2014-08-02 18:02:37 +00:00
}
int _textbox_get_height ( widget *wid )
{
textbox *tb = (textbox *) wid;
if ( !wid->expand ) {
if ( tb->flags & TB_AUTOHEIGHT ) {
return textbox_get_height ( tb );
}
return tb->widget.h;
}
return tb->widget.h;
}
2016-08-23 22:39:56 +00:00
int textbox_get_height ( const textbox *tb )
2014-08-02 18:02:37 +00:00
{
return textbox_get_font_height ( tb ) + 2 * config.line_padding;
2014-08-02 18:02:37 +00:00
}
2016-08-23 22:39:56 +00:00
int textbox_get_font_height ( const textbox *tb )
2014-08-02 18:02:37 +00:00
{
int height;
pango_layout_get_pixel_size ( tb->layout, NULL, &height );
return height;
}
2016-08-23 22:39:56 +00:00
int textbox_get_font_width ( const textbox *tb )
2014-08-02 18:02:37 +00:00
{
int width;
pango_layout_get_pixel_size ( tb->layout, &width, NULL );
return width;
}
double textbox_get_estimated_char_width ( void )
{
2016-07-31 21:28:31 +00:00
int width = pango_font_metrics_get_approximate_char_width ( p_metrics );
return ( width ) / (double) PANGO_SCALE;
}
int textbox_get_estimated_char_height ( void )
{
2016-07-31 21:28:31 +00:00
int height = pango_font_metrics_get_ascent ( p_metrics ) + pango_font_metrics_get_descent ( p_metrics );
return ( height ) / PANGO_SCALE + 2 * config.line_padding;
}