1
0
mirror of https://github.com/profanity-im/profanity.git synced 2024-11-03 19:37:16 -05:00
profanity/src/ui/inputwin.c

785 lines
21 KiB
C
Raw Normal View History

/*
2013-02-02 14:57:46 -05:00
* inputwin.c
2012-02-20 15:07:38 -05:00
*
2014-03-08 20:18:19 -05:00
* Copyright (C) 2012 - 2014 James Booth <boothj5@gmail.com>
*
2012-02-20 15:07:38 -05:00
* This file is part of Profanity.
*
* Profanity 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 3 of the License, or
* (at your option) any later version.
*
* Profanity 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 Profanity. If not, see <http://www.gnu.org/licenses/>.
*
* In addition, as a special exception, the copyright holders give permission to
* link the code of portions of this program with the OpenSSL library under
* certain conditions as described in each individual source file, and
* distribute linked combinations including the two.
*
* You must obey the GNU General Public License in all respects for all of the
* code used other than OpenSSL. If you modify file(s) with this exception, you
* may extend this exception to your version of the file(s), but you are not
* obligated to do so. If you do not wish to do so, delete this exception
* statement from your version. If you delete this exception statement from all
* source files in the program, then also delete it here.
*
2012-02-20 15:07:38 -05:00
*/
2013-01-02 19:16:39 -05:00
#define _XOPEN_SOURCE_EXTENDED
2012-09-08 11:51:09 -04:00
#include "config.h"
#include <stdlib.h>
2012-08-25 20:50:50 -04:00
#include <string.h>
2013-01-02 19:16:39 -05:00
#include <wchar.h>
2012-05-10 18:51:06 -04:00
2013-01-02 15:27:37 -05:00
#ifdef HAVE_NCURSESW_NCURSES_H
#include <ncursesw/ncurses.h>
#elif HAVE_NCURSES_H
#include <ncurses.h>
2012-09-08 11:51:09 -04:00
#endif
2012-05-10 18:51:06 -04:00
#include "command/command.h"
2013-02-02 15:55:58 -05:00
#include "common.h"
#include "config/accounts.h"
2013-02-02 16:59:29 -05:00
#include "config/preferences.h"
#include "config/theme.h"
#include "tools/history.h"
2012-10-30 21:36:52 -04:00
#include "log.h"
#include "muc.h"
2012-11-19 18:15:42 -05:00
#include "profanity.h"
#include "roster_list.h"
2013-02-02 15:55:58 -05:00
#include "ui/ui.h"
2014-04-07 16:12:30 -04:00
#include "ui/statusbar.h"
2014-04-07 16:50:28 -04:00
#include "ui/inputwin.h"
#include "ui/windows.h"
#include "xmpp/xmpp.h"
2012-02-08 18:55:11 -05:00
#define _inp_win_update_virtual() pnoutrefresh(inp_win, 0, pad_start, rows-1, 0, rows-1, cols-1)
#define KEY_CTRL_A 0001
#define KEY_CTRL_B 0002
#define KEY_CTRL_D 0004
#define KEY_CTRL_E 0005
#define KEY_CTRL_F 0006
#define KEY_CTRL_N 0016
#define KEY_CTRL_P 0020
#define KEY_CTRL_U 0025
#define KEY_CTRL_W 0027
#define MAX_HISTORY 100
2015-01-15 18:43:22 -05:00
#define INP_WIN_MAX 1000
2012-02-08 18:55:11 -05:00
static WINDOW *inp_win;
2015-01-16 18:32:57 -05:00
static History history;
static char input[INP_WIN_MAX];
static int input_len_bytes;
static int pad_start = 0;
static int rows, cols;
2012-02-08 18:55:11 -05:00
static int _handle_edit(int key_type, const wint_t ch);
static int _handle_alt_key(int key);
static void _handle_backspace(void);
2013-01-02 19:16:39 -05:00
static int _printable(const wint_t ch);
static void _clear_input(void);
static void _go_to_end(void);
static void _delete_previous_word(void);
2012-02-29 18:15:27 -05:00
2014-04-07 16:50:28 -04:00
void
create_input_window(void)
2012-02-08 18:55:11 -05:00
{
2012-10-22 20:31:19 -04:00
#ifdef NCURSES_REENTRANT
set_escdelay(25);
#else
ESCDELAY = 25;
#endif
2012-02-08 18:55:11 -05:00
getmaxyx(stdscr, rows, cols);
inp_win = newpad(1, INP_WIN_MAX);
2014-11-16 15:40:19 -05:00
wbkgd(inp_win, theme_attrs(THEME_INPUT_TEXT));;
2012-02-08 18:55:11 -05:00
keypad(inp_win, TRUE);
wmove(inp_win, 0, 0);
_inp_win_update_virtual();
history = history_new(MAX_HISTORY);
2012-02-08 18:55:11 -05:00
}
2014-04-07 16:50:28 -04:00
void
2014-11-19 11:32:26 -05:00
inp_win_resize(void)
2012-04-22 15:59:36 -04:00
{
int inp_x;
2012-04-22 15:59:36 -04:00
getmaxyx(stdscr, rows, cols);
inp_x = getcurx(inp_win);
2012-07-07 22:31:54 -04:00
// if lost cursor off screen, move contents to show it
if (inp_x >= pad_start + cols) {
pad_start = inp_x - (cols / 2);
if (pad_start < 0) {
pad_start = 0;
}
2012-07-07 22:31:54 -04:00
}
2014-11-19 11:32:26 -05:00
wbkgd(inp_win, theme_attrs(THEME_INPUT_TEXT));;
_inp_win_update_virtual();
2012-02-08 18:55:11 -05:00
}
2014-04-07 16:50:28 -04:00
void
inp_non_block(gint timeout)
2012-02-08 18:55:11 -05:00
{
wtimeout(inp_win, timeout);
2012-02-08 18:55:11 -05:00
}
2014-04-07 16:50:28 -04:00
void
inp_block(void)
2012-02-16 19:42:41 -05:00
{
wtimeout(inp_win, -1);
}
2015-01-15 18:43:22 -05:00
char *
2015-01-15 18:46:54 -05:00
inp_read(int *key_type, wint_t *ch)
2012-02-08 18:55:11 -05:00
{
2015-01-17 16:09:40 -05:00
int display_size = utf8_display_len(input);
2012-07-07 16:24:39 -04:00
// echo off, and get some more input
2012-02-08 18:55:11 -05:00
noecho();
2015-01-15 18:29:48 -05:00
*key_type = wget_wch(inp_win, ch);
2012-02-08 18:55:11 -05:00
2015-01-11 15:20:17 -05:00
gboolean in_command = FALSE;
2015-01-16 18:32:57 -05:00
if ((display_size > 0 && input[0] == '/') ||
2015-01-15 18:29:48 -05:00
(display_size == 0 && *ch == '/')) {
2015-01-11 15:20:17 -05:00
in_command = TRUE;
}
2015-01-15 18:29:48 -05:00
if (*key_type == ERR) {
2015-01-11 15:20:17 -05:00
prof_handle_idle();
}
2015-01-15 18:29:48 -05:00
if ((*key_type != ERR) && (*key_type != KEY_CODE_YES) && !in_command && _printable(*ch)) {
2015-01-11 15:20:17 -05:00
prof_handle_activity();
}
2013-01-17 14:43:49 -05:00
// if it wasn't an arrow key etc
if (!_handle_edit(*key_type, *ch)) {
2015-01-15 18:29:48 -05:00
if (_printable(*ch) && *key_type != KEY_CODE_YES) {
2015-01-16 18:32:57 -05:00
if (input_len_bytes >= INP_WIN_MAX) {
2015-01-15 18:29:48 -05:00
*ch = ERR;
2015-01-15 18:43:22 -05:00
return NULL;
2014-04-02 17:19:27 -04:00
}
2014-04-25 19:36:36 -04:00
int inp_x = getcurx(inp_win);
// handle insert if not at end of input
2013-01-02 19:16:39 -05:00
if (inp_x < display_size) {
2013-01-03 19:57:02 -05:00
char bytes[MB_CUR_MAX];
2015-01-15 18:29:48 -05:00
size_t utf_len = wcrtomb(bytes, *ch, NULL);
2015-01-16 18:32:57 -05:00
char *next_ch = g_utf8_offset_to_pointer(input, inp_x);
2013-01-03 17:41:03 -05:00
char *offset;
2015-01-16 18:32:57 -05:00
for (offset = &input[input_len_bytes - 1]; offset >= next_ch; offset--) {
2013-01-03 17:41:03 -05:00
*(offset + utf_len) = *offset;
}
2014-04-25 19:36:36 -04:00
int i;
2013-01-03 17:41:03 -05:00
for (i = 0; i < utf_len; i++) {
*(next_ch + i) = bytes[i];
}
2015-01-16 18:32:57 -05:00
input_len_bytes += utf_len;
input[input_len_bytes] = '\0';
2013-02-09 15:39:52 -05:00
waddstr(inp_win, next_ch);
2013-01-05 19:20:34 -05:00
wmove(inp_win, 0, inp_x + 1);
if (inp_x - pad_start > cols-3) {
pad_start++;
_inp_win_update_virtual();
}
// otherwise just append
} else {
char bytes[MB_CUR_MAX+1];
2015-01-15 18:29:48 -05:00
size_t utf_len = wcrtomb(bytes, *ch, NULL);
// wcrtomb can return (size_t) -1
if (utf_len < MB_CUR_MAX) {
2014-04-25 19:36:36 -04:00
int i;
for (i = 0 ; i < utf_len; i++) {
2015-01-16 18:32:57 -05:00
input[input_len_bytes++] = bytes[i];
}
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
bytes[utf_len] = '\0';
2013-02-09 15:39:52 -05:00
waddstr(inp_win, bytes);
display_size++;
2013-01-05 19:24:11 -05:00
// if gone over screen size follow input
int rows, cols;
getmaxyx(stdscr, rows, cols);
if (display_size - pad_start > cols-2) {
pad_start++;
_inp_win_update_virtual();
}
}
2012-02-29 18:15:27 -05:00
}
cmd_reset_autocomplete();
2012-02-29 18:15:27 -05:00
}
}
echo();
2013-01-02 19:16:39 -05:00
2015-01-15 18:29:48 -05:00
if (*ch == '\n') {
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
input_len_bytes = 0;
return strdup(input);
2015-01-15 18:43:22 -05:00
} else {
return NULL;
2015-01-15 17:51:05 -05:00
}
2012-02-29 18:15:27 -05:00
}
2014-04-07 16:50:28 -04:00
void
inp_get_password(char *passwd)
2012-02-29 18:15:27 -05:00
{
_clear_input();
_inp_win_update_virtual();
doupdate();
2012-02-29 18:15:27 -05:00
noecho();
mvwgetnstr(inp_win, 0, 1, passwd, MAX_PASSWORD_SIZE);
wmove(inp_win, 0, 0);
2012-02-29 18:15:27 -05:00
echo();
status_bar_clear();
}
2014-04-07 16:50:28 -04:00
void
inp_put_back(void)
2012-02-29 18:15:27 -05:00
{
_inp_win_update_virtual();
2012-02-29 18:15:27 -05:00
}
2014-04-07 16:50:28 -04:00
void
inp_replace_input(const char * const new_input)
2012-10-27 20:08:04 -04:00
{
2015-01-16 18:32:57 -05:00
strncpy(input, new_input, INP_WIN_MAX);
input_len_bytes = strlen(input);
inp_win_reset();
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
waddstr(inp_win, input);
_go_to_end();
2012-10-27 20:08:04 -04:00
}
2014-04-07 16:50:28 -04:00
void
inp_win_reset(void)
{
_clear_input();
pad_start = 0;
_inp_win_update_virtual();
}
void
inp_history_append(char *inp)
{
history_append(history, inp);
}
static void
_clear_input(void)
{
werase(inp_win);
wmove(inp_win, 0, 0);
}
2012-10-27 20:08:04 -04:00
2012-02-29 18:15:27 -05:00
/*
* Deal with command editing, return 1 if ch was an edit
* key press: up, down, left, right or backspace
* return 0 if it wasn't
2012-02-29 18:15:27 -05:00
*/
2012-07-24 18:19:48 -04:00
static int
_handle_edit(int key_type, const wint_t ch)
2012-02-29 18:15:27 -05:00
{
char *prev = NULL;
char *next = NULL;
int inp_x = getcurx(inp_win);
2012-11-23 20:57:24 -05:00
int next_ch;
2015-01-17 16:09:40 -05:00
int display_size = utf8_display_len(input);
2012-02-29 18:15:27 -05:00
2013-01-16 19:28:44 -05:00
// CTRL-LEFT
if ((key_type == KEY_CODE_YES) && (ch == 547 || ch == 545 || ch == 544 || ch == 540 || ch == 539) && (inp_x > 0)) {
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
gchar *curr_ch = g_utf8_offset_to_pointer(input, inp_x);
curr_ch = g_utf8_find_prev_char(input, curr_ch);
2013-01-16 19:59:40 -05:00
gchar *prev_ch;
gunichar curr_uni;
gunichar prev_uni;
2013-01-16 19:59:40 -05:00
while (curr_ch != NULL) {
curr_uni = g_utf8_get_char(curr_ch);
2013-01-16 19:28:44 -05:00
2013-01-16 19:59:40 -05:00
if (g_unichar_isspace(curr_uni)) {
2015-01-16 18:32:57 -05:00
curr_ch = g_utf8_find_prev_char(input, curr_ch);
2013-01-16 19:59:40 -05:00
} else {
2015-01-16 18:32:57 -05:00
prev_ch = g_utf8_find_prev_char(input, curr_ch);
2013-01-16 19:59:40 -05:00
if (prev_ch == NULL) {
curr_ch = NULL;
2012-11-23 20:57:24 -05:00
break;
2013-01-16 19:59:40 -05:00
} else {
prev_uni = g_utf8_get_char(prev_ch);
if (g_unichar_isspace(prev_uni)) {
break;
} else {
curr_ch = prev_ch;
}
2013-01-16 19:28:44 -05:00
}
}
2013-01-16 19:59:40 -05:00
}
2013-01-16 19:28:44 -05:00
2013-01-16 19:59:40 -05:00
if (curr_ch == NULL) {
inp_x = 0;
wmove(inp_win, 0, inp_x);
} else {
2015-01-16 18:32:57 -05:00
glong offset = g_utf8_pointer_to_offset(input, curr_ch);
2013-01-16 19:59:40 -05:00
inp_x = offset;
wmove(inp_win, 0, inp_x);
2012-11-23 20:57:24 -05:00
}
2013-01-16 19:59:40 -05:00
2013-01-16 19:28:44 -05:00
// if gone off screen to left, jump left (half a screen worth)
if (inp_x <= pad_start) {
pad_start = pad_start - (cols / 2);
if (pad_start < 0) {
pad_start = 0;
}
_inp_win_update_virtual();
2013-01-16 19:28:44 -05:00
}
return 1;
2013-01-16 19:59:40 -05:00
// CTRL-RIGHT
} else if ((key_type == KEY_CODE_YES) && (ch == 562 || ch == 560 || ch == 555 || ch == 559 || ch == 554) && (inp_x < display_size)) {
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
gchar *curr_ch = g_utf8_offset_to_pointer(input, inp_x);
2013-01-16 21:21:00 -05:00
gchar *next_ch = g_utf8_find_next_char(curr_ch, NULL);
gunichar curr_uni;
gunichar next_uni;
gboolean moved = FALSE;
2015-01-16 18:32:57 -05:00
while (g_utf8_pointer_to_offset(input, next_ch) < display_size) {
2013-01-16 21:21:00 -05:00
curr_uni = g_utf8_get_char(curr_ch);
next_uni = g_utf8_get_char(next_ch);
curr_ch = next_ch;
next_ch = g_utf8_find_next_char(next_ch, NULL);
if (!g_unichar_isspace(curr_uni) && g_unichar_isspace(next_uni) && moved) {
break;
} else {
moved = TRUE;
}
}
if (next_ch == NULL) {
inp_x = display_size;
wmove(inp_win, 0, inp_x);
} else {
2015-01-16 18:32:57 -05:00
glong offset = g_utf8_pointer_to_offset(input, curr_ch);
if (offset == display_size - 1) {
inp_x = offset + 1;
} else {
inp_x = offset;
}
2013-01-16 21:21:00 -05:00
wmove(inp_win, 0, inp_x);
}
2013-01-16 19:59:40 -05:00
2013-01-16 21:21:00 -05:00
// if gone off screen to right, jump right (half a screen worth)
if (inp_x > pad_start + cols) {
pad_start = pad_start + (cols / 2);
_inp_win_update_virtual();
2013-01-16 19:59:40 -05:00
}
2013-01-16 21:21:00 -05:00
2013-01-16 19:28:44 -05:00
return 1;
2013-01-16 19:59:40 -05:00
// ALT-LEFT
} else if ((key_type == KEY_CODE_YES) && (ch == 537 || ch == 542)) {
ui_previous_win();
return 1;
// ALT-RIGHT
} else if ((key_type == KEY_CODE_YES) && (ch == 552 || ch == 557)) {
ui_next_win();
return 1;
2013-01-16 19:59:40 -05:00
// other editing keys
2013-01-16 19:28:44 -05:00
} else {
switch(ch) {
case 27: // ESC
2013-03-02 18:01:12 -05:00
// check for ALT-key
2013-01-16 19:28:44 -05:00
next_ch = wgetch(inp_win);
if (next_ch != ERR) {
return _handle_alt_key(next_ch);
2013-01-16 19:28:44 -05:00
} else {
2015-01-16 18:32:57 -05:00
input_len_bytes = 0;
2013-01-16 19:28:44 -05:00
inp_win_reset();
return 1;
}
case 127:
_handle_backspace();
return 1;
2013-01-16 19:28:44 -05:00
case KEY_BACKSPACE:
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
_handle_backspace();
2013-01-16 19:28:44 -05:00
return 1;
2013-01-16 19:28:44 -05:00
case KEY_DC: // DEL
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
2014-12-31 01:36:54 -05:00
case KEY_CTRL_D:
2013-01-16 19:28:44 -05:00
if (inp_x == display_size-1) {
2015-01-16 18:32:57 -05:00
gchar *start = g_utf8_substring(input, 0, inp_x);
for (input_len_bytes = 0; input_len_bytes < strlen(start); input_len_bytes++) {
input[input_len_bytes] = start[input_len_bytes];
}
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
g_free(start);
_clear_input();
2015-01-16 18:32:57 -05:00
waddstr(inp_win, input);
2013-01-16 19:28:44 -05:00
} else if (inp_x < display_size-1) {
2015-01-16 18:32:57 -05:00
gchar *start = g_utf8_substring(input, 0, inp_x);
gchar *end = g_utf8_substring(input, inp_x+1, input_len_bytes);
GString *new = g_string_new(start);
g_string_append(new, end);
2015-01-16 18:32:57 -05:00
for (input_len_bytes = 0; input_len_bytes < strlen(new->str); input_len_bytes++) {
input[input_len_bytes] = new->str[input_len_bytes];
}
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
g_free(start);
g_free(end);
g_string_free(new, FALSE);
_clear_input();
2015-01-16 18:32:57 -05:00
waddstr(inp_win, input);
2013-01-16 19:28:44 -05:00
wmove(inp_win, 0, inp_x);
}
2013-01-16 19:28:44 -05:00
return 1;
2013-01-16 19:28:44 -05:00
case KEY_LEFT:
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
2014-12-31 01:36:54 -05:00
case KEY_CTRL_B:
2013-01-16 19:28:44 -05:00
if (inp_x > 0) {
wmove(inp_win, 0, inp_x-1);
2012-02-08 18:55:11 -05:00
2013-01-16 19:28:44 -05:00
// current position off screen to left
if (inp_x - 1 < pad_start) {
pad_start--;
_inp_win_update_virtual();
2013-01-16 19:28:44 -05:00
}
2013-01-03 19:19:18 -05:00
}
2013-01-16 19:28:44 -05:00
return 1;
2013-01-16 19:28:44 -05:00
case KEY_RIGHT:
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
2014-12-31 01:36:54 -05:00
case KEY_CTRL_F:
2013-01-16 19:28:44 -05:00
if (inp_x < display_size) {
wmove(inp_win, 0, inp_x+1);
2013-01-03 19:19:18 -05:00
2013-01-16 19:28:44 -05:00
// current position off screen to right
if ((inp_x + 1 - pad_start) >= cols) {
pad_start++;
_inp_win_update_virtual();
2013-01-16 19:28:44 -05:00
}
2013-01-03 19:19:18 -05:00
}
2013-01-16 19:28:44 -05:00
return 1;
2013-01-03 19:19:18 -05:00
2013-01-16 19:28:44 -05:00
case KEY_UP:
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
2014-12-31 01:36:54 -05:00
case KEY_CTRL_P:
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
prev = history_previous(history, input);
2013-01-16 19:28:44 -05:00
if (prev) {
inp_replace_input(prev);
2013-01-16 19:28:44 -05:00
}
return 1;
2012-04-09 18:40:26 -04:00
2013-01-16 19:28:44 -05:00
case KEY_DOWN:
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
2014-12-31 01:36:54 -05:00
case KEY_CTRL_N:
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
next = history_next(history, input);
2013-01-16 19:28:44 -05:00
if (next) {
inp_replace_input(next);
2015-01-16 18:32:57 -05:00
} else if (input_len_bytes != 0) {
input[input_len_bytes] = '\0';
history_append(history, input);
inp_replace_input("");
2013-01-16 19:28:44 -05:00
}
return 1;
2013-01-16 19:28:44 -05:00
case KEY_HOME:
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
2014-12-31 01:36:54 -05:00
case KEY_CTRL_A:
2013-01-16 19:28:44 -05:00
wmove(inp_win, 0, 0);
pad_start = 0;
_inp_win_update_virtual();
2013-01-16 19:28:44 -05:00
return 1;
2013-01-16 19:28:44 -05:00
case KEY_END:
if (key_type != KEY_CODE_YES) {
2013-09-04 18:14:35 -04:00
return 0;
}
2014-12-31 01:36:54 -05:00
case KEY_CTRL_E:
_go_to_end();
2013-01-16 19:28:44 -05:00
return 1;
2012-02-29 18:15:27 -05:00
2013-01-16 19:28:44 -05:00
case 9: // tab
2015-01-16 18:32:57 -05:00
if (input_len_bytes != 0) {
input[input_len_bytes] = '\0';
if ((strncmp(input, "/", 1) != 0) && (ui_current_win_type() == WIN_MUC)) {
char *result = muc_autocomplete(input);
if (result) {
inp_replace_input(result);
free(result);
}
2015-01-16 18:32:57 -05:00
} else if (strncmp(input, "/", 1) == 0) {
char *result = cmd_autocomplete(input);
if (result) {
inp_replace_input(result);
free(result);
}
}
}
2013-01-16 19:28:44 -05:00
return 1;
2012-07-07 21:21:39 -04:00
2014-12-31 01:36:54 -05:00
case KEY_CTRL_W:
_delete_previous_word();
2014-12-01 16:45:08 -05:00
return 1;
break;
2014-12-31 01:36:54 -05:00
case KEY_CTRL_U:
2015-01-01 17:44:21 -05:00
while (getcurx(inp_win) > 0) {
_delete_previous_word();
2015-01-01 17:44:21 -05:00
}
2014-12-31 01:36:54 -05:00
return 1;
break;
2013-01-16 19:28:44 -05:00
default:
return 0;
2012-07-07 22:18:39 -04:00
}
2012-02-08 18:55:11 -05:00
}
}
static void
_handle_backspace(void)
{
int inp_x = getcurx(inp_win);
2015-01-17 16:09:40 -05:00
int display_size = utf8_display_len(input);
roster_reset_search_attempts();
if (display_size > 0) {
// if at end, delete last char
if (inp_x >= display_size) {
2015-01-16 18:32:57 -05:00
gchar *start = g_utf8_substring(input, 0, inp_x-1);
for (input_len_bytes = 0; input_len_bytes < strlen(start); input_len_bytes++) {
input[input_len_bytes] = start[input_len_bytes];
}
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
g_free(start);
_clear_input();
2015-01-16 18:32:57 -05:00
waddstr(inp_win, input);
wmove(inp_win, 0, inp_x -1);
// if in middle, delete and shift chars left
} else if (inp_x > 0 && inp_x < display_size) {
2015-01-16 18:32:57 -05:00
gchar *start = g_utf8_substring(input, 0, inp_x - 1);
gchar *end = g_utf8_substring(input, inp_x, input_len_bytes);
GString *new = g_string_new(start);
g_string_append(new, end);
2015-01-16 18:32:57 -05:00
for (input_len_bytes = 0; input_len_bytes < strlen(new->str); input_len_bytes++) {
input[input_len_bytes] = new->str[input_len_bytes];
}
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
g_free(start);
g_free(end);
g_string_free(new, FALSE);
_clear_input();
2015-01-16 18:32:57 -05:00
waddstr(inp_win, input);
wmove(inp_win, 0, inp_x -1);
}
// if gone off screen to left, jump left (half a screen worth)
if (inp_x <= pad_start) {
pad_start = pad_start - (cols / 2);
if (pad_start < 0) {
pad_start = 0;
}
_inp_win_update_virtual();
}
}
}
2013-03-02 18:01:12 -05:00
static int
_handle_alt_key(int key)
2013-03-02 18:01:12 -05:00
{
switch (key)
{
case '1':
ui_switch_win(1);
2013-03-02 18:01:12 -05:00
break;
case '2':
ui_switch_win(2);
2013-03-02 18:01:12 -05:00
break;
case '3':
ui_switch_win(3);
2013-03-02 18:01:12 -05:00
break;
case '4':
ui_switch_win(4);
2013-03-02 18:01:12 -05:00
break;
case '5':
ui_switch_win(5);
2013-03-02 18:01:12 -05:00
break;
case '6':
ui_switch_win(6);
2013-03-02 18:01:12 -05:00
break;
case '7':
ui_switch_win(7);
2013-03-02 18:01:12 -05:00
break;
case '8':
ui_switch_win(8);
2013-03-02 18:01:12 -05:00
break;
case '9':
ui_switch_win(9);
2013-03-02 18:01:12 -05:00
break;
case '0':
ui_switch_win(0);
2013-03-02 18:01:12 -05:00
break;
case KEY_LEFT:
ui_previous_win();
break;
case KEY_RIGHT:
ui_next_win();
break;
case 263:
2013-03-02 18:01:12 -05:00
case 127:
_delete_previous_word();
2014-12-01 16:45:08 -05:00
break;
default:
break;
}
return 1;
}
2013-03-02 21:23:16 -05:00
2014-12-01 16:45:08 -05:00
static void
_delete_previous_word(void)
2014-12-01 16:45:08 -05:00
{
int end_del = getcurx(inp_win);
int start_del = end_del;
2013-03-02 21:23:16 -05:00
2015-01-16 18:32:57 -05:00
input[input_len_bytes] = '\0';
gchar *curr_ch = g_utf8_offset_to_pointer(input, end_del);
curr_ch = g_utf8_find_prev_char(input, curr_ch);
2014-12-01 16:45:08 -05:00
gchar *prev_ch;
gunichar curr_uni;
gunichar prev_uni;
2013-03-02 21:23:16 -05:00
2014-12-01 16:45:08 -05:00
while (curr_ch != NULL) {
curr_uni = g_utf8_get_char(curr_ch);
if (g_unichar_isspace(curr_uni)) {
2015-01-16 18:32:57 -05:00
curr_ch = g_utf8_find_prev_char(input, curr_ch);
2014-12-01 16:45:08 -05:00
} else {
2015-01-16 18:32:57 -05:00
prev_ch = g_utf8_find_prev_char(input, curr_ch);
2014-12-01 16:45:08 -05:00
if (prev_ch == NULL) {
curr_ch = NULL;
break;
2013-03-02 21:23:16 -05:00
} else {
2014-12-01 16:45:08 -05:00
prev_uni = g_utf8_get_char(prev_ch);
if (g_unichar_isspace(prev_uni)) {
break;
} else {
curr_ch = prev_ch;
}
2013-03-02 21:23:16 -05:00
}
2014-12-01 16:45:08 -05:00
}
}
2013-03-02 21:23:16 -05:00
2014-12-01 16:45:08 -05:00
if (curr_ch == NULL) {
start_del = 0;
} else {
2015-01-16 18:32:57 -05:00
start_del = g_utf8_pointer_to_offset(input, curr_ch);
2014-12-01 16:45:08 -05:00
}
2013-03-02 21:23:16 -05:00
2015-01-16 18:32:57 -05:00
gint len = g_utf8_strlen(input, -1);
gchar *start_string = g_utf8_substring(input, 0, start_del);
gchar *end_string = g_utf8_substring(input, end_del, len);
2013-03-02 21:23:16 -05:00
2014-12-01 16:45:08 -05:00
int i;
for (i = 0; i < strlen(start_string); i++) {
2015-01-16 18:32:57 -05:00
input[i] = start_string[i];
2014-12-01 16:45:08 -05:00
}
for (i = 0; i < strlen(end_string); i++) {
2015-01-16 18:32:57 -05:00
input[strlen(start_string)+i] = end_string[i];
2014-12-01 16:45:08 -05:00
}
2013-03-02 21:23:16 -05:00
2015-01-16 18:32:57 -05:00
input_len_bytes = strlen(start_string)+i;
input[input_len_bytes] = '\0';
2013-03-02 21:23:16 -05:00
2014-12-01 16:45:08 -05:00
_clear_input();
2015-01-16 18:32:57 -05:00
waddstr(inp_win, input);
2014-12-01 16:45:08 -05:00
wmove(inp_win, 0, start_del);
2013-03-02 21:23:16 -05:00
2014-12-01 16:45:08 -05:00
// if gone off screen to left, jump left (half a screen worth)
if (start_del <= pad_start) {
pad_start = pad_start - (cols / 2);
if (pad_start < 0) {
pad_start = 0;
}
2013-03-02 21:23:16 -05:00
2014-12-01 16:45:08 -05:00
_inp_win_update_virtual();
2013-03-02 18:01:12 -05:00
}
}
2013-01-05 17:45:34 -05:00
static void
_go_to_end(void)
2013-01-05 17:45:34 -05:00
{
2015-01-17 16:09:40 -05:00
int display_size = utf8_display_len(input);
2013-01-05 19:20:34 -05:00
wmove(inp_win, 0, display_size);
2013-01-05 17:45:34 -05:00
if (display_size > cols-2) {
pad_start = display_size - cols + 1;
_inp_win_update_virtual();
2013-01-05 17:45:34 -05:00
}
}
2012-07-24 18:19:48 -04:00
static int
2013-01-02 19:16:39 -05:00
_printable(const wint_t ch)
2012-02-08 18:55:11 -05:00
{
char bytes[MB_CUR_MAX+1];
size_t utf_len = wcrtomb(bytes, ch, NULL);
bytes[utf_len] = '\0';
gunichar unichar = g_utf8_get_char(bytes);
return g_unichar_isprint(unichar) && (ch != KEY_MOUSE);
2014-04-25 19:36:36 -04:00
}