2012-10-21 15:02:20 -04:00
|
|
|
/*
|
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-10-21 15:02:20 -04:00
|
|
|
*
|
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/>.
|
|
|
|
*
|
2014-08-24 15:57:39 -04:00
|
|
|
* 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"
|
|
|
|
|
2012-03-10 17:32:07 -05:00
|
|
|
#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>
|
2013-01-03 19:35:54 -05:00
|
|
|
#elif HAVE_NCURSES_H
|
|
|
|
#include <ncurses.h>
|
2012-09-08 11:51:09 -04:00
|
|
|
#endif
|
2012-05-10 18:51:06 -04:00
|
|
|
|
2013-02-02 16:43:59 -05:00
|
|
|
#include "command/command.h"
|
2013-02-02 15:55:58 -05:00
|
|
|
#include "common.h"
|
2013-11-07 17:15:43 -05:00
|
|
|
#include "config/accounts.h"
|
2013-02-02 16:59:29 -05:00
|
|
|
#include "config/preferences.h"
|
|
|
|
#include "config/theme.h"
|
2015-01-16 17:50:40 -05:00
|
|
|
#include "tools/history.h"
|
2012-10-30 21:36:52 -04:00
|
|
|
#include "log.h"
|
2014-07-08 19:35:43 -04:00
|
|
|
#include "muc.h"
|
2012-11-19 18:15:42 -05:00
|
|
|
#include "profanity.h"
|
2013-12-14 10:34:17 -05:00
|
|
|
#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"
|
2013-09-25 15:51:54 -04:00
|
|
|
#include "ui/windows.h"
|
2013-05-06 18:04:46 -04:00
|
|
|
#include "xmpp/xmpp.h"
|
2012-02-08 18:55:11 -05:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
#define _inp_win_update_virtual() pnoutrefresh(inp_win, 0, pad_start, wrows-1, 0, wrows-1, wcols-1)
|
2013-01-05 20:06:12 -05:00
|
|
|
|
2015-01-01 17:36:48 -05:00
|
|
|
#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
|
|
|
|
|
2015-01-16 17:50:40 -05:00
|
|
|
#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;
|
|
|
|
|
2015-01-18 17:55:51 -05:00
|
|
|
// input line
|
2015-01-18 14:55:48 -05:00
|
|
|
static char line[INP_WIN_MAX];
|
2015-01-18 17:55:51 -05:00
|
|
|
// current position in the utf8 string
|
2015-01-18 14:55:48 -05:00
|
|
|
static int line_utf8_pos;
|
2015-01-16 18:32:57 -05:00
|
|
|
|
2012-07-07 16:51:43 -04:00
|
|
|
static int pad_start = 0;
|
2015-01-18 14:55:48 -05:00
|
|
|
static int wrows, wcols;
|
2012-02-08 18:55:11 -05:00
|
|
|
|
2015-01-15 19:36:42 -05:00
|
|
|
static int _handle_edit(int key_type, const wint_t ch);
|
|
|
|
static int _handle_alt_key(int key);
|
2015-01-18 17:55:51 -05:00
|
|
|
|
|
|
|
static void _handle_delete_previous_word(void);
|
2015-01-15 19:36:42 -05:00
|
|
|
static void _handle_backspace(void);
|
2015-01-18 17:55:51 -05:00
|
|
|
|
|
|
|
static gboolean _is_ctrl_left(int key_type, const wint_t ch);
|
|
|
|
static gboolean _is_ctrl_right(int key_type, const wint_t ch);
|
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
|
2015-01-18 14:55:48 -05:00
|
|
|
getmaxyx(stdscr, wrows, wcols);
|
2012-07-09 18:48:53 -04:00
|
|
|
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);
|
2012-07-07 21:43:28 -04:00
|
|
|
wmove(inp_win, 0, 0);
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2015-01-16 17:50:40 -05:00
|
|
|
history = history_new(MAX_HISTORY);
|
2015-01-18 14:55:48 -05:00
|
|
|
line_utf8_pos = 0;
|
2015-01-18 17:55:51 -05:00
|
|
|
line[0] = '\0';
|
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
|
|
|
{
|
2015-01-18 17:55:51 -05:00
|
|
|
int col;
|
2015-01-18 14:55:48 -05:00
|
|
|
getmaxyx(stdscr, wrows, wcols);
|
2015-01-18 17:55:51 -05:00
|
|
|
col = getcurx(inp_win);
|
2012-10-21 15:02:20 -04:00
|
|
|
|
2012-07-07 22:31:54 -04:00
|
|
|
// if lost cursor off screen, move contents to show it
|
2015-01-18 17:55:51 -05:00
|
|
|
if (col >= pad_start + wcols) {
|
|
|
|
pad_start = col - (wcols / 2);
|
2012-07-09 18:48:53 -04:00
|
|
|
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));;
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2012-02-08 18:55:11 -05:00
|
|
|
}
|
|
|
|
|
2014-04-07 16:50:28 -04:00
|
|
|
void
|
2015-01-08 17:43:11 -05:00
|
|
|
inp_non_block(gint timeout)
|
2012-02-08 18:55:11 -05:00
|
|
|
{
|
2015-01-08 17:43: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
|
|
|
{
|
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-18 17:55:51 -05:00
|
|
|
int bytes_len = strlen(line);
|
|
|
|
|
2015-01-11 15:20:17 -05:00
|
|
|
gboolean in_command = FALSE;
|
2015-01-18 17:55:51 -05:00
|
|
|
if ((bytes_len > 0 && line[0] == '/') ||
|
|
|
|
(bytes_len == 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-18 17:55:51 -05:00
|
|
|
if ((*key_type != ERR) && (*key_type != KEY_CODE_YES) && !in_command && utf8_is_printable(*ch)) {
|
2015-01-11 15:20:17 -05:00
|
|
|
prof_handle_activity();
|
|
|
|
}
|
2013-01-17 14:43:49 -05:00
|
|
|
|
2012-04-17 17:43:35 -04:00
|
|
|
// if it wasn't an arrow key etc
|
2015-01-15 19:36:42 -05:00
|
|
|
if (!_handle_edit(*key_type, *ch)) {
|
2015-01-18 17:55:51 -05:00
|
|
|
if (utf8_is_printable(*ch) && *key_type != KEY_CODE_YES) {
|
|
|
|
if (bytes_len >= 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
|
|
|
}
|
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
int col = getcurx(inp_win);
|
|
|
|
int utf8_len = g_utf8_strlen(line, -1);
|
2012-10-21 15:02:20 -04:00
|
|
|
|
2012-04-17 17:43:35 -04:00
|
|
|
// handle insert if not at end of input
|
2015-01-18 14:55:48 -05:00
|
|
|
if (line_utf8_pos < utf8_len) {
|
2013-01-03 19:57:02 -05:00
|
|
|
char bytes[MB_CUR_MAX];
|
2015-01-18 14:55:48 -05:00
|
|
|
size_t utf8_ch_len = wcrtomb(bytes, *ch, NULL);
|
|
|
|
bytes[utf8_ch_len] = '\0';
|
2012-04-17 17:43:35 -04:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
gchar *start = g_utf8_substring(line, 0, line_utf8_pos);
|
|
|
|
gchar *end = g_utf8_substring(line, line_utf8_pos, utf8_len);
|
|
|
|
GString *new_line = g_string_new(start);
|
|
|
|
g_string_append(new_line, bytes);
|
|
|
|
g_string_append(new_line, end);
|
2012-04-17 17:43:35 -04:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
int old_pos = line_utf8_pos;
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, new_line->str, INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
line_utf8_pos = old_pos+1;
|
2012-04-17 17:43:35 -04:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
g_free(start);
|
|
|
|
g_free(end);
|
|
|
|
g_string_free(new_line, TRUE);
|
|
|
|
|
|
|
|
col++;
|
|
|
|
gunichar uni = g_utf8_get_char(bytes);
|
|
|
|
if (g_unichar_iswide(uni)) {
|
|
|
|
col++;
|
2013-01-05 17:30:21 -05:00
|
|
|
}
|
2015-01-18 14:55:48 -05:00
|
|
|
wmove(inp_win, 0, col);
|
2013-01-05 17:30:21 -05:00
|
|
|
|
2012-04-17 17:43:35 -04:00
|
|
|
// otherwise just append
|
|
|
|
} else {
|
2013-01-05 17:37:11 -05:00
|
|
|
char bytes[MB_CUR_MAX+1];
|
2015-01-18 14:55:48 -05:00
|
|
|
size_t utf8_ch_len = wcrtomb(bytes, *ch, NULL);
|
2012-10-21 15:02:20 -04:00
|
|
|
|
2013-01-05 17:56:59 -05:00
|
|
|
// wcrtomb can return (size_t) -1
|
2015-01-18 14:55:48 -05:00
|
|
|
if (utf8_ch_len < MB_CUR_MAX) {
|
2014-04-25 19:36:36 -04:00
|
|
|
int i;
|
2015-01-18 14:55:48 -05:00
|
|
|
for (i = 0 ; i < utf8_ch_len; i++) {
|
2015-01-18 17:55:51 -05:00
|
|
|
line[bytes_len++] = bytes[i];
|
2013-01-05 17:56:59 -05:00
|
|
|
}
|
2015-01-18 17:55:51 -05:00
|
|
|
line[bytes_len] = '\0';
|
2013-01-05 17:56:59 -05:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
bytes[utf8_ch_len] = '\0';
|
2013-02-09 15:39:52 -05:00
|
|
|
waddstr(inp_win, bytes);
|
2015-01-18 14:55:48 -05:00
|
|
|
|
|
|
|
line_utf8_pos++;
|
|
|
|
|
|
|
|
col++;
|
|
|
|
gunichar uni = g_utf8_get_char(bytes);
|
|
|
|
if (g_unichar_iswide(uni)) {
|
|
|
|
col++;
|
|
|
|
}
|
|
|
|
wmove(inp_win, 0, col);
|
2013-01-05 17:56:59 -05:00
|
|
|
|
2013-01-05 19:24:11 -05:00
|
|
|
// if gone over screen size follow input
|
2015-01-18 14:55:48 -05:00
|
|
|
int wrows, wcols;
|
|
|
|
getmaxyx(stdscr, wrows, wcols);
|
2015-01-18 17:55:51 -05:00
|
|
|
if (col - pad_start > wcols-2) {
|
2013-01-05 17:56:59 -05:00
|
|
|
pad_start++;
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2013-01-05 17:56:59 -05:00
|
|
|
}
|
2012-07-07 16:51:43 -04:00
|
|
|
}
|
2012-02-29 18:15:27 -05:00
|
|
|
}
|
2012-04-17 17:43:35 -04:00
|
|
|
|
2012-10-27 20:42:26 -04:00
|
|
|
cmd_reset_autocomplete();
|
2012-02-29 18:15:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
echo();
|
2013-01-02 19:16:39 -05:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
char *result = NULL;
|
2015-01-15 18:29:48 -05:00
|
|
|
if (*ch == '\n') {
|
2015-01-18 14:55:48 -05:00
|
|
|
result = strdup(line);
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*ch != ERR && *key_type != ERR) {
|
2015-01-18 17:55:51 -05:00
|
|
|
cons_debug("BYTE LEN = %d", bytes_len);
|
|
|
|
cons_debug("UTF8 LEN = %d", utf8_display_len(line));
|
2015-01-18 14:55:48 -05:00
|
|
|
cons_debug("CURR COL = %d", getcurx(inp_win));
|
|
|
|
cons_debug("CURR UNI = %d", line_utf8_pos);
|
|
|
|
cons_debug("");
|
2015-01-15 17:51:05 -05:00
|
|
|
}
|
2015-01-18 14:55:48 -05:00
|
|
|
|
|
|
|
return result;
|
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
|
|
|
{
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2014-02-02 18:56:50 -05:00
|
|
|
doupdate();
|
2012-02-29 18:15:27 -05:00
|
|
|
noecho();
|
2013-11-07 17:15:43 -05:00
|
|
|
mvwgetnstr(inp_win, 0, 1, passwd, MAX_PASSWORD_SIZE);
|
2012-07-07 21:43:28 -04:00
|
|
|
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
|
|
|
{
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2012-02-29 18:15:27 -05:00
|
|
|
}
|
|
|
|
|
2014-04-07 16:50:28 -04:00
|
|
|
void
|
|
|
|
inp_win_reset(void)
|
2013-01-05 20:06:12 -05:00
|
|
|
{
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2013-01-05 20:06:12 -05:00
|
|
|
}
|
|
|
|
|
2015-01-16 17:50:40 -05:00
|
|
|
void
|
|
|
|
inp_history_append(char *inp)
|
|
|
|
{
|
|
|
|
history_append(history, inp);
|
|
|
|
}
|
|
|
|
|
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
|
2014-08-27 07:26:11 -04:00
|
|
|
* return 0 if it wasn't
|
2012-02-29 18:15:27 -05:00
|
|
|
*/
|
2012-07-24 18:19:48 -04:00
|
|
|
static int
|
2015-01-15 19:36:42 -05:00
|
|
|
_handle_edit(int key_type, const wint_t ch)
|
2012-02-29 18:15:27 -05:00
|
|
|
{
|
2015-01-18 14:55:48 -05:00
|
|
|
int col = getcurx(inp_win);
|
|
|
|
int utf8_len = g_utf8_strlen(line, -1);
|
2012-02-29 18:15:27 -05:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
// CTRL-LEFT
|
2015-01-18 17:55:51 -05:00
|
|
|
if (line_utf8_pos > 0 && _is_ctrl_left(key_type, ch)) {
|
|
|
|
gchar *curr_ch = g_utf8_offset_to_pointer(line, line_utf8_pos);
|
|
|
|
gunichar curr_uni = g_utf8_get_char(curr_ch);
|
|
|
|
line_utf8_pos--;
|
|
|
|
col--;
|
|
|
|
if (g_unichar_iswide(curr_uni)) {
|
|
|
|
col--;
|
|
|
|
}
|
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
curr_ch = g_utf8_find_prev_char(line, curr_ch);
|
2015-01-18 17:55:51 -05:00
|
|
|
|
2013-01-16 19:59:40 -05:00
|
|
|
gchar *prev_ch;
|
|
|
|
gunichar prev_uni;
|
|
|
|
while (curr_ch != NULL) {
|
|
|
|
curr_uni = g_utf8_get_char(curr_ch);
|
|
|
|
if (g_unichar_isspace(curr_uni)) {
|
2015-01-18 14:55:48 -05:00
|
|
|
curr_ch = g_utf8_find_prev_char(line, curr_ch);
|
2015-01-18 17:55:51 -05:00
|
|
|
line_utf8_pos--;
|
|
|
|
col--;
|
2013-01-16 19:59:40 -05:00
|
|
|
} else {
|
2015-01-18 14:55:48 -05:00
|
|
|
prev_ch = g_utf8_find_prev_char(line, 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);
|
2015-01-18 17:55:51 -05:00
|
|
|
line_utf8_pos--;
|
|
|
|
col--;
|
|
|
|
if (g_unichar_iswide(prev_uni)) {
|
|
|
|
col--;
|
|
|
|
}
|
2013-01-16 19:59:40 -05:00
|
|
|
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) {
|
2015-01-18 14:55:48 -05:00
|
|
|
col = 0;
|
|
|
|
wmove(inp_win, 0, col);
|
2013-01-16 19:59:40 -05:00
|
|
|
} else {
|
2015-01-18 17:55:51 -05:00
|
|
|
col++;
|
|
|
|
line_utf8_pos++;
|
2015-01-18 14:55:48 -05:00
|
|
|
wmove(inp_win, 0, col);
|
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)
|
2015-01-18 14:55:48 -05:00
|
|
|
if (col <= pad_start) {
|
|
|
|
pad_start = pad_start - (wcols / 2);
|
2013-01-16 19:28:44 -05:00
|
|
|
if (pad_start < 0) {
|
|
|
|
pad_start = 0;
|
|
|
|
}
|
|
|
|
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2013-01-16 19:28:44 -05:00
|
|
|
}
|
|
|
|
return 1;
|
2013-01-16 19:59:40 -05:00
|
|
|
|
|
|
|
// CTRL-RIGHT
|
2015-01-18 17:55:51 -05:00
|
|
|
} else if (line_utf8_pos < utf8_len && _is_ctrl_right(key_type, ch)) {
|
|
|
|
gchar *curr_ch = g_utf8_offset_to_pointer(line, line_utf8_pos);
|
|
|
|
gunichar curr_uni = g_utf8_get_char(curr_ch);
|
|
|
|
|
|
|
|
// find next word if in whitespace
|
|
|
|
while (g_unichar_isspace(curr_uni)) {
|
|
|
|
col++;
|
|
|
|
line_utf8_pos++;
|
|
|
|
curr_ch = g_utf8_find_next_char(curr_ch, NULL);
|
|
|
|
if (!curr_ch) {
|
2013-01-16 21:21:00 -05:00
|
|
|
break;
|
|
|
|
}
|
2015-01-18 17:55:51 -05:00
|
|
|
curr_uni = g_utf8_get_char(curr_ch);
|
2013-01-16 21:21:00 -05:00
|
|
|
}
|
|
|
|
|
2015-01-18 17:55:51 -05:00
|
|
|
if (curr_ch) {
|
|
|
|
while (!g_unichar_isspace(curr_uni)) {
|
|
|
|
line_utf8_pos++;
|
|
|
|
col++;
|
|
|
|
if (g_unichar_iswide(curr_uni)) {
|
|
|
|
col++;
|
|
|
|
}
|
|
|
|
curr_ch = g_utf8_find_next_char(curr_ch, NULL);
|
|
|
|
if (!curr_ch || line_utf8_pos >= utf8_len) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
curr_uni = g_utf8_get_char(curr_ch);
|
2013-01-16 21:30:26 -05:00
|
|
|
}
|
2013-01-16 21:21:00 -05:00
|
|
|
}
|
2013-01-16 19:59:40 -05:00
|
|
|
|
2015-01-18 17:55:51 -05:00
|
|
|
wmove(inp_win, 0, col);
|
|
|
|
|
2013-01-16 21:21:00 -05:00
|
|
|
// if gone off screen to right, jump right (half a screen worth)
|
2015-01-18 14:55:48 -05:00
|
|
|
if (col > pad_start + wcols) {
|
|
|
|
pad_start = pad_start + (wcols / 2);
|
2014-04-01 16:52:04 -04:00
|
|
|
_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
|
|
|
|
2013-09-25 19:25:04 -04:00
|
|
|
// ALT-LEFT
|
2015-01-15 19:36:42 -05:00
|
|
|
} else if ((key_type == KEY_CODE_YES) && (ch == 537 || ch == 542)) {
|
2013-09-25 19:25:04 -04:00
|
|
|
ui_previous_win();
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
// ALT-RIGHT
|
2015-01-15 19:36:42 -05:00
|
|
|
} else if ((key_type == KEY_CODE_YES) && (ch == 552 || ch == 557)) {
|
2013-09-25 19:25:04 -04:00
|
|
|
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 {
|
2015-01-19 15:51:41 -05:00
|
|
|
int display_len;
|
2015-01-18 17:55:51 -05:00
|
|
|
int bytes_len = strlen(line);
|
|
|
|
int next_ch;
|
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
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) {
|
2015-01-15 19:36:42 -05:00
|
|
|
return _handle_alt_key(next_ch);
|
2013-01-16 19:28:44 -05:00
|
|
|
} else {
|
|
|
|
inp_win_reset();
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 127:
|
2015-01-15 19:36:42 -05:00
|
|
|
_handle_backspace();
|
2013-09-08 11:14:26 -04:00
|
|
|
return 1;
|
2013-01-16 19:28:44 -05:00
|
|
|
case KEY_BACKSPACE:
|
2015-01-15 19:36:42 -05:00
|
|
|
if (key_type != KEY_CODE_YES) {
|
2013-09-04 18:14:35 -04:00
|
|
|
return 0;
|
|
|
|
}
|
2015-01-15 19:36:42 -05:00
|
|
|
_handle_backspace();
|
2013-01-16 19:28:44 -05:00
|
|
|
return 1;
|
2012-02-26 18:33:14 -05:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
case KEY_DC: // DEL
|
2015-01-15 19:36:42 -05:00
|
|
|
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:
|
2015-01-18 17:55:51 -05:00
|
|
|
if (line_utf8_pos == utf8_len) {
|
|
|
|
return 1;
|
|
|
|
} else if (line_utf8_pos == utf8_len-1) {
|
|
|
|
gchar *curr_ch = g_utf8_offset_to_pointer(line, line_utf8_pos);
|
|
|
|
int bytes_len_ch = strlen(curr_ch);
|
|
|
|
bytes_len -= bytes_len_ch;
|
|
|
|
line[bytes_len] = '\0';
|
|
|
|
line_utf8_pos--;
|
|
|
|
wdelch(inp_win);
|
2013-01-03 18:54:12 -05:00
|
|
|
|
2015-01-18 17:55:51 -05:00
|
|
|
return 1;
|
|
|
|
} else if (line_utf8_pos < utf8_len-1) {
|
2015-01-18 14:55:48 -05:00
|
|
|
gchar *start = g_utf8_substring(line, 0, col);
|
2015-01-18 17:55:51 -05:00
|
|
|
gchar *end = g_utf8_substring(line, col+1, bytes_len);
|
2013-01-03 18:47:45 -05:00
|
|
|
GString *new = g_string_new(start);
|
|
|
|
g_string_append(new, end);
|
|
|
|
|
2015-01-18 17:55:51 -05:00
|
|
|
for (bytes_len = 0; bytes_len < strlen(new->str); bytes_len++) {
|
|
|
|
line[bytes_len] = new->str[bytes_len];
|
2013-01-03 18:47:45 -05:00
|
|
|
}
|
2015-01-18 17:55:51 -05:00
|
|
|
line[bytes_len] = '\0';
|
2013-01-03 18:47:45 -05:00
|
|
|
|
|
|
|
g_free(start);
|
|
|
|
g_free(end);
|
|
|
|
g_string_free(new, FALSE);
|
2012-02-26 18:33:14 -05:00
|
|
|
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
2015-01-18 14:55:48 -05:00
|
|
|
waddstr(inp_win, line);
|
|
|
|
wmove(inp_win, 0, col);
|
2012-02-26 18:33:14 -05:00
|
|
|
}
|
2013-01-16 19:28:44 -05:00
|
|
|
return 1;
|
2012-07-09 18:48:53 -04:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
case KEY_LEFT:
|
2015-01-15 19:36:42 -05:00
|
|
|
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:
|
2015-01-18 14:55:48 -05:00
|
|
|
if (line_utf8_pos > 0) {
|
|
|
|
col--;
|
|
|
|
gchar *curr_ch = g_utf8_offset_to_pointer(line, line_utf8_pos);
|
|
|
|
gchar *prev_ch = g_utf8_find_prev_char(line, curr_ch);
|
|
|
|
if (prev_ch) {
|
|
|
|
gunichar uni = g_utf8_get_char(prev_ch);
|
|
|
|
if (g_unichar_iswide(uni)) {
|
|
|
|
col--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
wmove(inp_win, 0, col);
|
|
|
|
line_utf8_pos--;
|
2012-02-08 18:55:11 -05:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
// current position off screen to left
|
2015-01-18 14:55:48 -05:00
|
|
|
if (col - 1 < pad_start) {
|
2013-01-16 19:28:44 -05:00
|
|
|
pad_start--;
|
2014-04-01 16:52:04 -04:00
|
|
|
_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;
|
2012-10-21 15:02:20 -04:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
case KEY_RIGHT:
|
2015-01-15 19:36:42 -05:00
|
|
|
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:
|
2015-01-18 14:55:48 -05:00
|
|
|
if (line_utf8_pos < utf8_len) {
|
|
|
|
col++;
|
|
|
|
gchar *curr_ch = g_utf8_offset_to_pointer(line, line_utf8_pos);
|
|
|
|
if (curr_ch) {
|
|
|
|
gunichar uni = g_utf8_get_char(curr_ch);
|
|
|
|
if (g_unichar_iswide(uni)) {
|
|
|
|
col++;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
wmove(inp_win, 0, col);
|
|
|
|
line_utf8_pos++;
|
2013-01-03 19:19:18 -05:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
// current position off screen to right
|
2015-01-18 14:55:48 -05:00
|
|
|
if ((col + 1 - pad_start) >= wcols) {
|
2013-01-16 19:28:44 -05:00
|
|
|
pad_start++;
|
2014-04-01 16:52:04 -04:00
|
|
|
_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:
|
2015-01-15 19:36:42 -05:00
|
|
|
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-18 17:55:51 -05:00
|
|
|
line[bytes_len] = '\0';
|
|
|
|
char *prev = history_previous(history, line);
|
2013-01-16 19:28:44 -05:00
|
|
|
if (prev) {
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, prev, INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
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:
|
2015-01-15 19:36:42 -05:00
|
|
|
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-18 17:55:51 -05:00
|
|
|
line[bytes_len] = '\0';
|
|
|
|
char *next = history_next(history, line);
|
2013-01-16 19:28:44 -05:00
|
|
|
if (next) {
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, next, INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
2015-01-18 17:55:51 -05:00
|
|
|
} else if (bytes_len != 0) {
|
|
|
|
line[bytes_len] = '\0';
|
2015-01-18 14:55:48 -05:00
|
|
|
history_append(history, line);
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, "", INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
2013-01-16 19:28:44 -05:00
|
|
|
}
|
|
|
|
return 1;
|
2012-10-21 15:02:20 -04:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
case KEY_HOME:
|
2015-01-15 19:36:42 -05:00
|
|
|
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;
|
2015-01-18 17:55:51 -05:00
|
|
|
line_utf8_pos = 0;
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2013-01-16 19:28:44 -05:00
|
|
|
return 1;
|
2012-10-21 15:02:20 -04:00
|
|
|
|
2013-01-16 19:28:44 -05:00
|
|
|
case KEY_END:
|
2015-01-15 19:36:42 -05:00
|
|
|
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:
|
2015-01-19 15:51:41 -05:00
|
|
|
display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
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-18 17:55:51 -05:00
|
|
|
if (bytes_len != 0) {
|
|
|
|
line[bytes_len] = '\0';
|
2015-01-18 14:55:48 -05:00
|
|
|
if ((strncmp(line, "/", 1) != 0) && (ui_current_win_type() == WIN_MUC)) {
|
|
|
|
char *result = muc_autocomplete(line);
|
2015-01-16 17:50:40 -05:00
|
|
|
if (result) {
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, result, INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
|
|
|
|
2015-01-16 17:50:40 -05:00
|
|
|
free(result);
|
|
|
|
}
|
2015-01-18 14:55:48 -05:00
|
|
|
} else if (strncmp(line, "/", 1) == 0) {
|
|
|
|
char *result = cmd_autocomplete(line);
|
2015-01-16 17:50:40 -05:00
|
|
|
if (result) {
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, result, INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
|
|
|
|
2015-01-16 17:50:40 -05:00
|
|
|
free(result);
|
|
|
|
}
|
2014-07-08 20:04:54 -04:00
|
|
|
}
|
2014-07-08 19:35:43 -04:00
|
|
|
}
|
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:
|
2015-01-18 17:55:51 -05:00
|
|
|
_handle_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) {
|
2015-01-18 17:55:51 -05:00
|
|
|
_handle_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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-09-08 11:14:26 -04:00
|
|
|
static void
|
2015-01-15 19:36:42 -05:00
|
|
|
_handle_backspace(void)
|
2013-09-08 11:14:26 -04:00
|
|
|
{
|
2015-01-17 17:03:22 -05:00
|
|
|
int col = getcurx(inp_win);
|
2015-01-18 14:55:48 -05:00
|
|
|
int utf8_len = g_utf8_strlen(line, -1);
|
2013-09-08 11:14:26 -04:00
|
|
|
roster_reset_search_attempts();
|
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
if (utf8_len > 0) {
|
2013-09-08 11:14:26 -04:00
|
|
|
// if at end, delete last char
|
2015-01-18 14:55:48 -05:00
|
|
|
if (line_utf8_pos >= utf8_len) {
|
|
|
|
gchar *new_line = g_utf8_substring(line, 0, utf8_len-1);
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, new_line, INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
2013-09-08 11:14:26 -04:00
|
|
|
|
|
|
|
// if in middle, delete and shift chars left
|
2015-01-18 14:55:48 -05:00
|
|
|
} else if (line_utf8_pos > 0 && line_utf8_pos < utf8_len) {
|
|
|
|
gchar *del_char = g_utf8_offset_to_pointer(line, line_utf8_pos-1);
|
|
|
|
gunichar uni = g_utf8_get_char(del_char);
|
|
|
|
|
|
|
|
gchar *start = g_utf8_substring(line, 0, line_utf8_pos-1);
|
|
|
|
gchar *end = g_utf8_substring(line, line_utf8_pos, utf8_len);
|
|
|
|
GString *new_line = g_string_new(start);
|
|
|
|
g_string_append(new_line, end);
|
|
|
|
|
|
|
|
int old_pos = line_utf8_pos;
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
strncpy(line, new_line->str, INP_WIN_MAX);
|
|
|
|
waddstr(inp_win, line);
|
|
|
|
|
|
|
|
int display_len = utf8_display_len(line);
|
|
|
|
wmove(inp_win, 0, display_len);
|
|
|
|
line_utf8_pos = g_utf8_strlen(line, -1);
|
|
|
|
|
|
|
|
if (display_len > wcols-2) {
|
|
|
|
pad_start = display_len - wcols + 1;
|
|
|
|
_inp_win_update_virtual();
|
|
|
|
}
|
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
line_utf8_pos = old_pos-1;
|
2013-09-08 11:14:26 -04:00
|
|
|
|
|
|
|
g_free(start);
|
|
|
|
g_free(end);
|
2015-01-18 14:55:48 -05:00
|
|
|
g_string_free(new_line, TRUE);
|
|
|
|
|
|
|
|
col--;
|
|
|
|
if (g_unichar_iswide(uni)) {
|
|
|
|
col--;
|
|
|
|
}
|
2013-09-08 11:14:26 -04:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
wmove(inp_win, 0, col);
|
2013-09-08 11:14:26 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
// if gone off screen to left, jump left (half a screen worth)
|
2015-01-17 17:03:22 -05:00
|
|
|
if (col <= pad_start) {
|
2015-01-18 14:55:48 -05:00
|
|
|
pad_start = pad_start - (wcols / 2);
|
2013-09-08 11:14:26 -04:00
|
|
|
if (pad_start < 0) {
|
|
|
|
pad_start = 0;
|
|
|
|
}
|
|
|
|
|
2014-04-01 16:52:04 -04:00
|
|
|
_inp_win_update_virtual();
|
2013-09-08 11:14:26 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2013-03-02 18:01:12 -05:00
|
|
|
static int
|
2015-01-15 19:36:42 -05:00
|
|
|
_handle_alt_key(int key)
|
2013-03-02 18:01:12 -05:00
|
|
|
{
|
|
|
|
switch (key)
|
|
|
|
{
|
|
|
|
case '1':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(1);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '2':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(2);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '3':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(3);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '4':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(4);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '5':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(5);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '6':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(6);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '7':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(7);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '8':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(8);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '9':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(9);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
|
|
|
case '0':
|
2013-08-27 19:50:15 -04:00
|
|
|
ui_switch_win(0);
|
2013-03-02 18:01:12 -05:00
|
|
|
break;
|
2013-09-25 15:51:54 -04:00
|
|
|
case KEY_LEFT:
|
2013-09-25 19:25:04 -04:00
|
|
|
ui_previous_win();
|
2013-09-25 15:51:54 -04:00
|
|
|
break;
|
|
|
|
case KEY_RIGHT:
|
2013-09-25 19:25:04 -04:00
|
|
|
ui_next_win();
|
2013-09-25 15:51:54 -04:00
|
|
|
break;
|
2013-03-04 03:31:02 -05:00
|
|
|
case 263:
|
2013-03-02 18:01:12 -05:00
|
|
|
case 127:
|
2015-01-18 17:55:51 -05:00
|
|
|
_handle_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
|
2015-01-18 17:55:51 -05:00
|
|
|
_handle_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-18 14:55:48 -05:00
|
|
|
gchar *curr_ch = g_utf8_offset_to_pointer(line, end_del);
|
|
|
|
curr_ch = g_utf8_find_prev_char(line, 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-18 14:55:48 -05:00
|
|
|
curr_ch = g_utf8_find_prev_char(line, curr_ch);
|
2014-12-01 16:45:08 -05:00
|
|
|
} else {
|
2015-01-18 14:55:48 -05:00
|
|
|
prev_ch = g_utf8_find_prev_char(line, 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-18 14:55:48 -05:00
|
|
|
start_del = g_utf8_pointer_to_offset(line, curr_ch);
|
2014-12-01 16:45:08 -05:00
|
|
|
}
|
2013-03-02 21:23:16 -05:00
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
gint len = g_utf8_strlen(line, -1);
|
|
|
|
gchar *start_string = g_utf8_substring(line, 0, start_del);
|
|
|
|
gchar *end_string = g_utf8_substring(line, 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-18 14:55:48 -05:00
|
|
|
line[i] = start_string[i];
|
2014-12-01 16:45:08 -05:00
|
|
|
}
|
|
|
|
for (i = 0; i < strlen(end_string); i++) {
|
2015-01-18 14:55:48 -05:00
|
|
|
line[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-18 17:55:51 -05:00
|
|
|
int bytes_len = strlen(start_string)+i;
|
|
|
|
line[bytes_len] = '\0';
|
2013-03-02 21:23:16 -05:00
|
|
|
|
2015-01-19 15:51:41 -05:00
|
|
|
werase(inp_win);
|
|
|
|
wmove(inp_win, 0, 0);
|
|
|
|
pad_start = 0;
|
|
|
|
line[0] = '\0';
|
|
|
|
line_utf8_pos = 0;
|
|
|
|
|
2015-01-18 14:55:48 -05:00
|
|
|
waddstr(inp_win, line);
|
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) {
|
2015-01-18 14:55:48 -05:00
|
|
|
pad_start = pad_start - (wcols / 2);
|
2014-12-01 16:45:08 -05:00
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-18 17:55:51 -05:00
|
|
|
static gboolean
|
|
|
|
_is_ctrl_left(int key_type, const wint_t ch)
|
2012-02-08 18:55:11 -05:00
|
|
|
{
|
2015-01-18 17:55:51 -05:00
|
|
|
return ((key_type == KEY_CODE_YES)
|
|
|
|
&& (ch == 547 || ch == 545 || ch == 544 || ch == 540 || ch == 539));
|
2014-04-25 19:36:36 -04:00
|
|
|
}
|
2015-01-18 17:55:51 -05:00
|
|
|
|
|
|
|
static gboolean
|
|
|
|
_is_ctrl_right(int key_type, const wint_t ch)
|
|
|
|
{
|
|
|
|
return ((key_type == KEY_CODE_YES)
|
|
|
|
&& (ch == 562 || ch == 560 || ch == 555 || ch == 559 || ch == 554));
|
|
|
|
}
|