2013-07-31 20:20:16 +02:00
|
|
|
/*
|
|
|
|
* Toxic -- Tox Curses Client
|
|
|
|
*/
|
|
|
|
|
2013-08-23 09:50:04 +02:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2013-07-31 19:20:03 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2013-08-03 01:44:32 +02:00
|
|
|
#include <time.h>
|
2013-07-31 19:20:03 +02:00
|
|
|
|
2013-08-23 00:18:53 +02:00
|
|
|
#include "toxic_windows.h"
|
2013-11-10 03:43:56 +01:00
|
|
|
#include "execute.h"
|
2013-09-21 02:35:03 +02:00
|
|
|
#include "misc_tools.h"
|
2013-11-29 23:48:08 +01:00
|
|
|
#include "friendlist.h"
|
2013-12-10 09:03:45 +01:00
|
|
|
#include "toxic_strings.h"
|
2013-07-31 19:20:03 +02:00
|
|
|
|
2013-09-04 08:05:36 +02:00
|
|
|
extern char *DATA_FILE;
|
|
|
|
extern int store_data(Tox *m, char *path);
|
|
|
|
|
2013-11-30 00:52:21 +01:00
|
|
|
extern FileSender file_senders[MAX_FILES];
|
|
|
|
extern ToxicFriend friends[MAX_FRIENDS_NUM];
|
|
|
|
|
2013-12-10 01:25:09 +01:00
|
|
|
#define AC_NUM_CHAT_COMMANDS 17
|
2013-12-09 23:56:20 +01:00
|
|
|
|
|
|
|
/* Array of chat command names used for tab completion. */
|
|
|
|
static const uint8_t chat_cmd_list[AC_NUM_CHAT_COMMANDS][MAX_CMDNAME_SIZE] = {
|
|
|
|
{ "/accept" },
|
|
|
|
{ "/add" },
|
|
|
|
{ "/clear" },
|
|
|
|
{ "/close" },
|
|
|
|
{ "/connect" },
|
|
|
|
{ "/exit" },
|
|
|
|
{ "/groupchat" },
|
|
|
|
{ "/help" },
|
|
|
|
{ "/invite" },
|
|
|
|
{ "/join" },
|
|
|
|
{ "/myid" },
|
|
|
|
{ "/nick" },
|
|
|
|
{ "/note" },
|
|
|
|
{ "/quit" },
|
|
|
|
{ "/savefile" },
|
|
|
|
{ "/sendfile" },
|
|
|
|
{ "/status" },
|
|
|
|
};
|
2013-12-09 00:14:57 +01:00
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
static void chat_onMessage(ToxWindow *self, Tox *m, int num, uint8_t *msg, uint16_t len)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-09-15 22:38:38 +02:00
|
|
|
if (self->num != num)
|
2013-09-06 00:24:58 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-09-06 00:24:58 +02:00
|
|
|
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
|
2013-11-29 16:14:59 +01:00
|
|
|
tox_get_name(m, num, nick);
|
2013-10-19 05:08:37 +02:00
|
|
|
nick[TOXIC_MAX_NAME_LENGTH] = '\0';
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-10-11 07:11:43 +02:00
|
|
|
print_time(ctx->history);
|
2013-12-04 00:01:17 +01:00
|
|
|
wattron(ctx->history, COLOR_PAIR(CYAN));
|
2013-08-16 19:11:09 +02:00
|
|
|
wprintw(ctx->history, "%s: ", nick);
|
2013-12-04 00:01:17 +01:00
|
|
|
wattroff(ctx->history, COLOR_PAIR(CYAN));
|
2013-09-26 06:33:51 +02:00
|
|
|
|
|
|
|
if (msg[0] == '>') {
|
|
|
|
wattron(ctx->history, COLOR_PAIR(GREEN));
|
|
|
|
wprintw(ctx->history, "%s\n", msg);
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(GREEN));
|
|
|
|
} else
|
|
|
|
wprintw(ctx->history, "%s\n", msg);
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-11-29 01:45:28 +01:00
|
|
|
alert_window(self, WINDOW_ALERT_1, true);
|
2013-07-31 19:20:03 +02:00
|
|
|
}
|
|
|
|
|
2013-09-21 02:35:03 +02:00
|
|
|
static void chat_onConnectionChange(ToxWindow *self, Tox *m, int num, uint8_t status)
|
2013-09-05 03:25:59 +02:00
|
|
|
{
|
2013-09-15 22:38:38 +02:00
|
|
|
if (self->num != num)
|
2013-09-05 03:25:59 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
StatusBar *statusbar = self->stb;
|
2013-09-06 00:24:58 +02:00
|
|
|
statusbar->is_online = status == 1 ? true : false;
|
2013-09-05 03:25:59 +02:00
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
static void chat_onAction(ToxWindow *self, Tox *m, int num, uint8_t *action, uint16_t len)
|
2013-08-08 21:01:33 +02:00
|
|
|
{
|
2013-09-15 22:38:38 +02:00
|
|
|
if (self->num != num)
|
2013-09-06 00:24:58 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-08-08 21:01:33 +02:00
|
|
|
|
2013-09-06 00:24:58 +02:00
|
|
|
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
|
2013-11-29 16:14:59 +01:00
|
|
|
tox_get_name(m, num, nick);
|
2013-10-19 05:08:37 +02:00
|
|
|
nick[TOXIC_MAX_NAME_LENGTH] = '\0';
|
2013-08-31 08:22:07 +02:00
|
|
|
|
2013-10-11 07:11:43 +02:00
|
|
|
print_time(ctx->history);
|
2013-09-05 06:47:33 +02:00
|
|
|
wattron(ctx->history, COLOR_PAIR(YELLOW));
|
2013-08-31 08:22:07 +02:00
|
|
|
wprintw(ctx->history, "* %s %s\n", nick, action);
|
2013-09-05 06:47:33 +02:00
|
|
|
wattroff(ctx->history, COLOR_PAIR(YELLOW));
|
2013-08-08 21:01:33 +02:00
|
|
|
|
2013-11-29 01:45:28 +01:00
|
|
|
alert_window(self, WINDOW_ALERT_1, true);
|
2013-08-08 21:01:33 +02:00
|
|
|
}
|
|
|
|
|
2013-09-09 07:08:06 +02:00
|
|
|
static void chat_onNickChange(ToxWindow *self, int num, uint8_t *nick, uint16_t len)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-09-15 22:38:38 +02:00
|
|
|
if (self->num != num)
|
2013-08-16 19:11:09 +02:00
|
|
|
return;
|
|
|
|
|
2013-10-19 05:08:37 +02:00
|
|
|
nick[TOXIC_MAX_NAME_LENGTH] = '\0';
|
|
|
|
len = strlen(nick) + 1;
|
2013-09-23 07:22:21 +02:00
|
|
|
memcpy(self->name, nick, len);
|
2013-07-31 19:20:03 +02:00
|
|
|
}
|
|
|
|
|
2013-09-05 03:25:59 +02:00
|
|
|
static void chat_onStatusChange(ToxWindow *self, Tox *m, int num, TOX_USERSTATUS status)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-09-15 22:38:38 +02:00
|
|
|
if (self->num != num)
|
2013-08-16 19:11:09 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
StatusBar *statusbar = self->stb;
|
2013-09-06 00:24:58 +02:00
|
|
|
statusbar->status = status;
|
2013-09-05 03:25:59 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void chat_onStatusMessageChange(ToxWindow *self, int num, uint8_t *status, uint16_t len)
|
|
|
|
{
|
2013-09-15 22:38:38 +02:00
|
|
|
if (self->num != num)
|
2013-09-05 03:25:59 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
StatusBar *statusbar = self->stb;
|
2013-09-10 10:04:13 +02:00
|
|
|
statusbar->statusmsg_len = len;
|
2013-09-23 07:22:21 +02:00
|
|
|
memcpy(statusbar->statusmsg, status, len);
|
2013-07-31 19:20:03 +02:00
|
|
|
}
|
|
|
|
|
2013-10-11 10:42:30 +02:00
|
|
|
static void chat_onFileSendRequest(ToxWindow *self, Tox *m, int num, uint8_t filenum,
|
2013-10-14 01:09:20 +02:00
|
|
|
uint64_t filesize, uint8_t *pathname, uint16_t path_len)
|
2013-10-11 06:23:39 +02:00
|
|
|
{
|
2013-10-17 12:13:28 +02:00
|
|
|
if (self->num != num)
|
2013-10-11 06:23:39 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-10-11 06:23:39 +02:00
|
|
|
|
2013-11-27 20:25:56 +01:00
|
|
|
int idx = strlen(pathname) - 1;
|
2013-11-30 22:14:09 +01:00
|
|
|
while (idx >= 0 && pathname[idx] == '/') {
|
2013-11-27 20:25:56 +01:00
|
|
|
pathname[idx--] = 0;
|
|
|
|
}
|
|
|
|
|
2013-10-17 12:13:28 +02:00
|
|
|
/* try to get file name from path */
|
|
|
|
uint8_t *filename = strrchr(pathname, '/'); // Try unix style paths
|
|
|
|
|
|
|
|
if (filename != NULL) {
|
2013-10-18 01:53:29 +02:00
|
|
|
if (!strlen(++filename))
|
|
|
|
filename = pathname;
|
2013-10-17 12:13:28 +02:00
|
|
|
} else {
|
|
|
|
filename = strrchr(pathname, '\\'); // Try windows style paths
|
|
|
|
|
|
|
|
if (filename == NULL)
|
|
|
|
filename = pathname;
|
|
|
|
}
|
|
|
|
|
|
|
|
wprintw(ctx->history, "File transfer request for '%s' (%llu bytes).\n", filename,
|
2013-10-11 10:42:30 +02:00
|
|
|
(long long unsigned int)filesize);
|
2013-10-11 06:23:39 +02:00
|
|
|
|
2013-10-18 07:35:08 +02:00
|
|
|
if (filenum >= MAX_FILES) {
|
2013-10-11 06:23:39 +02:00
|
|
|
wprintw(ctx->history, "Too many pending file requests; discarding.\n");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-11-12 22:53:41 +01:00
|
|
|
/* Append a number to duplicate file names */
|
2013-10-18 04:20:40 +02:00
|
|
|
FILE *filecheck = NULL;
|
2013-11-12 08:18:43 +01:00
|
|
|
int count = 1;
|
|
|
|
int len = strlen(filename);
|
|
|
|
|
|
|
|
while ((filecheck = fopen(filename, "r"))) {
|
|
|
|
filename[len] = '\0';
|
|
|
|
char d[9];
|
|
|
|
sprintf(d,"(%d)", count++);
|
|
|
|
strcat(filename, d);
|
|
|
|
filename[len + strlen(d)] = '\0';
|
|
|
|
|
2013-11-12 23:26:13 +01:00
|
|
|
if (count > 999999) {
|
2013-11-12 08:18:43 +01:00
|
|
|
wprintw(ctx->history, "Error saving file to disk.\n");
|
|
|
|
return;
|
|
|
|
}
|
2013-10-18 04:20:40 +02:00
|
|
|
}
|
|
|
|
|
2013-10-14 01:09:20 +02:00
|
|
|
wprintw(ctx->history, "Type '/savefile %d' to accept the file transfer.\n", filenum);
|
2013-10-11 06:23:39 +02:00
|
|
|
|
2013-10-18 07:35:08 +02:00
|
|
|
friends[num].file_receiver.pending[filenum] = true;
|
2013-10-18 01:53:29 +02:00
|
|
|
strcpy(friends[num].file_receiver.filenames[filenum], filename);
|
2013-10-11 06:23:39 +02:00
|
|
|
|
2013-11-29 01:45:28 +01:00
|
|
|
alert_window(self, WINDOW_ALERT_2, true);
|
2013-10-11 06:23:39 +02:00
|
|
|
}
|
|
|
|
|
2013-10-11 10:42:30 +02:00
|
|
|
static void chat_onFileControl(ToxWindow *self, Tox *m, int num, uint8_t receive_send,
|
|
|
|
uint8_t filenum, uint8_t control_type, uint8_t *data, uint16_t length)
|
2013-10-11 06:23:39 +02:00
|
|
|
{
|
|
|
|
if (self->num != num)
|
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-10-18 04:20:40 +02:00
|
|
|
uint8_t *filename;
|
2013-10-11 06:23:39 +02:00
|
|
|
|
2013-10-18 04:20:40 +02:00
|
|
|
if (receive_send == 0)
|
|
|
|
filename = friends[num].file_receiver.filenames[filenum];
|
|
|
|
else
|
|
|
|
filename = file_senders[filenum].pathname;
|
2013-10-18 01:53:29 +02:00
|
|
|
|
2013-11-12 07:50:04 +01:00
|
|
|
switch (control_type) {
|
2013-10-17 12:13:28 +02:00
|
|
|
case TOX_FILECONTROL_ACCEPT:
|
2013-10-18 01:53:29 +02:00
|
|
|
wprintw(ctx->history, "File transfer for '%s' accepted.\n", filename);
|
2013-10-11 06:23:39 +02:00
|
|
|
break;
|
2013-10-17 12:13:28 +02:00
|
|
|
case TOX_FILECONTROL_PAUSE:
|
2013-11-30 22:09:45 +01:00
|
|
|
// wprintw(ctx->history, "File transfer for '%s' paused.\n", filename);
|
2013-10-17 12:13:28 +02:00
|
|
|
break;
|
|
|
|
case TOX_FILECONTROL_KILL:
|
2013-10-18 01:53:29 +02:00
|
|
|
wprintw(ctx->history, "File transfer for '%s' failed.\n", filename);
|
2013-11-30 01:26:59 +01:00
|
|
|
|
2013-11-30 01:31:47 +01:00
|
|
|
if (receive_send == 0)
|
2013-11-30 01:26:59 +01:00
|
|
|
friends[num].file_receiver.pending[filenum] = false;
|
|
|
|
else
|
|
|
|
close_file_sender(filenum);
|
|
|
|
|
2013-10-11 06:23:39 +02:00
|
|
|
break;
|
2013-10-17 12:13:28 +02:00
|
|
|
case TOX_FILECONTROL_FINISHED:
|
2013-10-18 01:53:29 +02:00
|
|
|
wprintw(ctx->history, "File transfer for '%s' complete.\n", filename);
|
2013-10-11 06:23:39 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-11-29 01:45:28 +01:00
|
|
|
alert_window(self, WINDOW_ALERT_2, true);
|
2013-10-11 06:23:39 +02:00
|
|
|
}
|
|
|
|
|
2013-10-11 10:42:30 +02:00
|
|
|
static void chat_onFileData(ToxWindow *self, Tox *m, int num, uint8_t filenum, uint8_t *data,
|
|
|
|
uint16_t length)
|
2013-10-11 06:23:39 +02:00
|
|
|
{
|
|
|
|
if (self->num != num)
|
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-10-11 06:23:39 +02:00
|
|
|
|
2013-10-18 01:53:29 +02:00
|
|
|
uint8_t *filename = friends[num].file_receiver.filenames[filenum];
|
2013-10-17 12:13:28 +02:00
|
|
|
FILE *file_to_save = fopen(filename, "a");
|
2013-10-11 06:23:39 +02:00
|
|
|
|
2013-11-27 20:31:10 +01:00
|
|
|
// we have a problem here, but don't let it segfault
|
|
|
|
if (file_to_save == NULL) {
|
2013-11-29 00:56:56 +01:00
|
|
|
wattron(ctx->history, COLOR_PAIR(RED));
|
|
|
|
wprintw(ctx->history, "* Error writing to file.\n");
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(RED));
|
2013-11-30 01:26:59 +01:00
|
|
|
tox_file_send_control(m, num, 1, filenum, TOX_FILECONTROL_KILL, 0, 0);
|
2013-11-27 20:31:10 +01:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-10-11 06:23:39 +02:00
|
|
|
if (fwrite(data, length, 1, file_to_save) != 1) {
|
|
|
|
wattron(ctx->history, COLOR_PAIR(RED));
|
|
|
|
wprintw(ctx->history, "* Error writing to file.\n");
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(RED));
|
2013-11-30 01:26:59 +01:00
|
|
|
tox_file_send_control(m, num, 1, filenum, TOX_FILECONTROL_KILL, 0, 0);
|
2013-10-11 06:23:39 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
fclose(file_to_save);
|
|
|
|
}
|
|
|
|
|
2013-11-10 03:43:56 +01:00
|
|
|
static void chat_onGroupInvite(ToxWindow *self, Tox *m, int friendnumber, uint8_t *group_pub_key)
|
2013-10-14 01:09:20 +02:00
|
|
|
{
|
2013-11-12 23:26:13 +01:00
|
|
|
if (self->num != friendnumber)
|
2013-10-14 01:09:20 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-11-10 03:43:56 +01:00
|
|
|
uint8_t name[TOX_MAX_NAME_LENGTH] = {'\0'};
|
2013-10-14 01:09:20 +02:00
|
|
|
|
2013-11-29 16:14:59 +01:00
|
|
|
if (tox_get_name(m, friendnumber, name) == -1)
|
2013-10-14 01:09:20 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-11 05:17:46 +01:00
|
|
|
wprintw(ctx->history, "%s has invited you to a group chat.\n", name);
|
2013-10-14 01:09:20 +02:00
|
|
|
|
2013-11-12 23:26:13 +01:00
|
|
|
memcpy(friends[friendnumber].pending_groupchat, group_pub_key, TOX_CLIENT_ID_SIZE);
|
|
|
|
wprintw(ctx->history, "Type \"/join\" to join the chat.\n");
|
2013-11-29 01:45:28 +01:00
|
|
|
alert_window(self, WINDOW_ALERT_2, true);
|
2013-08-04 10:42:17 +02:00
|
|
|
}
|
2013-08-03 16:00:48 +02:00
|
|
|
|
2013-09-19 12:37:42 +02:00
|
|
|
static void send_action(ToxWindow *self, ChatContext *ctx, Tox *m, uint8_t *action) {
|
|
|
|
if (action == NULL) {
|
|
|
|
wprintw(ctx->history, "Invalid syntax.\n");
|
|
|
|
return;
|
2013-09-02 07:14:51 +02:00
|
|
|
}
|
|
|
|
|
2013-09-19 12:37:42 +02:00
|
|
|
uint8_t selfname[TOX_MAX_NAME_LENGTH];
|
2013-11-29 16:14:59 +01:00
|
|
|
tox_get_self_name(m, selfname, TOX_MAX_NAME_LENGTH);
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-10-11 07:11:43 +02:00
|
|
|
print_time(ctx->history);
|
2013-09-19 12:37:42 +02:00
|
|
|
wattron(ctx->history, COLOR_PAIR(YELLOW));
|
|
|
|
wprintw(ctx->history, "* %s %s\n", selfname, action);
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(YELLOW));
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-11-29 16:14:59 +01:00
|
|
|
if (tox_send_action(m, self->num, action, strlen(action) + 1) == 0) {
|
2013-09-19 12:37:42 +02:00
|
|
|
wattron(ctx->history, COLOR_PAIR(RED));
|
|
|
|
wprintw(ctx->history, " * Failed to send action\n");
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(RED));
|
2013-08-04 10:42:17 +02:00
|
|
|
}
|
2013-07-31 19:20:03 +02:00
|
|
|
}
|
|
|
|
|
2013-09-10 10:04:13 +02:00
|
|
|
static void chat_onKey(ToxWindow *self, Tox *m, wint_t key)
|
|
|
|
{
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
|
|
|
StatusBar *statusbar = self->stb;
|
2013-09-10 10:04:13 +02:00
|
|
|
|
|
|
|
int x, y, y2, x2;
|
|
|
|
getyx(self->window, y, x);
|
|
|
|
getmaxyx(self->window, y2, x2);
|
2013-10-23 09:24:08 +02:00
|
|
|
|
2013-12-01 08:58:21 +01:00
|
|
|
if (key == 0x107 || key == 0x8 || key == 0x7f) { /* BACKSPACE key: Remove character behind pos */
|
2013-09-26 13:52:17 +02:00
|
|
|
if (ctx->pos > 0) {
|
2013-12-01 08:58:21 +01:00
|
|
|
del_char_buf_bck(ctx->line, &ctx->pos, &ctx->len);
|
2013-09-10 10:04:13 +02:00
|
|
|
|
2013-09-26 13:52:17 +02:00
|
|
|
if (x == 0)
|
2013-12-01 04:12:43 +01:00
|
|
|
wmove(self->window, y-1, x2-1);
|
2013-09-26 13:52:17 +02:00
|
|
|
else
|
2013-12-01 04:12:43 +01:00
|
|
|
wmove(self->window, y, x-1);
|
2013-09-26 13:52:17 +02:00
|
|
|
}
|
2013-12-01 22:57:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
else if (key == KEY_DC) { /* DEL key: Remove character at pos */
|
2013-12-04 22:14:33 +01:00
|
|
|
if (ctx->pos != ctx->len)
|
|
|
|
del_char_buf_frnt(ctx->line, &ctx->pos, &ctx->len);
|
2013-12-04 22:21:32 +01:00
|
|
|
}
|
2013-12-01 22:57:05 +01:00
|
|
|
|
2013-12-06 04:55:14 +01:00
|
|
|
else if (key == T_KEY_DISCARD) { /* CTRL-U: Delete entire line behind pos */
|
|
|
|
if (ctx->pos > 0) {
|
|
|
|
discard_buf(ctx->line, &ctx->pos, &ctx->len);
|
|
|
|
wmove(self->window, y2 - CURS_Y_OFFSET, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (key == T_KEY_KILL) { /* CTRL-K: Delete entire line in front of pos */
|
|
|
|
if (ctx->pos != ctx->len)
|
|
|
|
kill_buf(ctx->line, &ctx->pos, &ctx->len);
|
|
|
|
}
|
|
|
|
|
2013-12-01 22:57:05 +01:00
|
|
|
else if (key == KEY_HOME) { /* HOME key: Move cursor to beginning of line */
|
2013-12-04 22:21:32 +01:00
|
|
|
if (ctx->pos > 0) {
|
|
|
|
ctx->pos = 0;
|
|
|
|
wmove(self->window, y2 - CURS_Y_OFFSET, 0);
|
|
|
|
}
|
2013-12-01 22:57:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
else if (key == KEY_END) { /* END key: move cursor to end of line */
|
2013-12-04 22:21:32 +01:00
|
|
|
if (ctx->pos != ctx->len) {
|
|
|
|
ctx->pos = ctx->len;
|
2013-12-11 06:10:09 +01:00
|
|
|
mv_curs_end(self->window, ctx->len, y2, x2);
|
2013-12-04 22:21:32 +01:00
|
|
|
}
|
|
|
|
}
|
2013-12-01 22:57:05 +01:00
|
|
|
|
|
|
|
else if (key == KEY_LEFT) {
|
2013-12-01 19:25:03 +01:00
|
|
|
if (ctx->pos > 0) {
|
|
|
|
--ctx->pos;
|
2013-12-01 04:12:43 +01:00
|
|
|
|
2013-12-01 19:25:03 +01:00
|
|
|
if (x == 0)
|
|
|
|
wmove(self->window, y-1, x2-1);
|
|
|
|
else
|
|
|
|
wmove(self->window, y, x-1);
|
|
|
|
}
|
2013-12-01 22:57:05 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
else if (key == KEY_RIGHT) {
|
2013-12-01 19:25:03 +01:00
|
|
|
if (ctx->pos < ctx->len) {
|
|
|
|
++ctx->pos;
|
2013-12-01 04:12:43 +01:00
|
|
|
|
2013-12-01 19:25:03 +01:00
|
|
|
if (x == x2-1)
|
|
|
|
wmove(self->window, y+1, 0);
|
|
|
|
else
|
|
|
|
wmove(self->window, y, x+1);
|
|
|
|
}
|
2013-12-09 00:14:57 +01:00
|
|
|
}
|
|
|
|
|
2013-12-11 06:10:09 +01:00
|
|
|
else if (key == KEY_UP) { /* fetches previous item in history */
|
|
|
|
if (ctx->hst_pos >= 0) {
|
|
|
|
fetch_hist_item(ctx->line, &ctx->pos, &ctx->len, ctx->ln_history, &ctx->hst_tot,
|
|
|
|
&ctx->hst_pos, LN_HIST_MV_UP);
|
|
|
|
mv_curs_end(self->window, ctx->len, y2, x2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (key == KEY_DOWN) { /* fetches next item in history */
|
|
|
|
if (ctx->hst_pos < ctx->hst_tot) {
|
|
|
|
fetch_hist_item(ctx->line, &ctx->pos, &ctx->len, ctx->ln_history, &ctx->hst_tot,
|
|
|
|
&ctx->hst_pos, LN_HIST_MV_DWN);
|
|
|
|
mv_curs_end(self->window, ctx->len, y2, x2);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-09 00:14:57 +01:00
|
|
|
else if (key == '\t') { /* TAB key: command */
|
|
|
|
if (ctx->len > 1 && ctx->line[0] == '/') {
|
2013-12-09 23:56:20 +01:00
|
|
|
int diff = complete_line(ctx->line, &ctx->pos, &ctx->len, chat_cmd_list, AC_NUM_CHAT_COMMANDS,
|
2013-12-09 00:14:57 +01:00
|
|
|
MAX_CMDNAME_SIZE);
|
|
|
|
|
|
|
|
if (diff != -1) {
|
|
|
|
if (x + diff > x2 - 1) {
|
|
|
|
int ofst = (x + diff - 1) - (x2 - 1);
|
|
|
|
wmove(self->window, y+1, ofst);
|
|
|
|
} else {
|
|
|
|
wmove(self->window, y, x+diff);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
2013-09-10 10:04:13 +02:00
|
|
|
#if HAVE_WIDECHAR
|
2013-11-29 02:23:37 +01:00
|
|
|
if (iswprint(key))
|
2013-09-10 10:04:13 +02:00
|
|
|
#else
|
2013-11-29 02:23:37 +01:00
|
|
|
if (isprint(key))
|
2013-09-10 10:04:13 +02:00
|
|
|
#endif
|
2013-12-01 04:12:43 +01:00
|
|
|
{ /* prevents buffer overflows and strange behaviour when cursor goes past the window */
|
|
|
|
if ( (ctx->len < MAX_STR_SIZE-1) && (ctx->len < (x2 * (CHATBOX_HEIGHT - 1)-1)) ) {
|
2013-12-03 00:23:04 +01:00
|
|
|
add_char_to_buf(ctx->line, &ctx->pos, &ctx->len, key);
|
2013-12-01 04:12:43 +01:00
|
|
|
|
|
|
|
if (x == x2-1)
|
|
|
|
wmove(self->window, y+1, 0);
|
|
|
|
else
|
|
|
|
wmove(self->window, y, x+1);
|
2013-09-10 10:04:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
/* RETURN key: Execute command or print line */
|
|
|
|
else if (key == '\n') {
|
2013-12-08 10:16:49 +01:00
|
|
|
uint8_t line[MAX_STR_SIZE];
|
|
|
|
|
|
|
|
if (wcs_to_mbs_buf(line, ctx->line, MAX_STR_SIZE) == -1)
|
|
|
|
memset(&line, 0, sizeof(line));
|
|
|
|
|
2013-09-10 10:04:13 +02:00
|
|
|
wclear(ctx->linewin);
|
|
|
|
wmove(self->window, y2 - CURS_Y_OFFSET, 0);
|
|
|
|
wclrtobot(self->window);
|
|
|
|
bool close_win = false;
|
2013-10-23 09:24:08 +02:00
|
|
|
|
2013-12-11 06:10:09 +01:00
|
|
|
if (!string_is_empty(line))
|
|
|
|
add_line_to_hist(ctx->line, ctx->len, ctx->ln_history, &ctx->hst_tot, &ctx->hst_pos);
|
|
|
|
|
2013-09-10 10:04:13 +02:00
|
|
|
if (line[0] == '/') {
|
2013-11-19 21:32:35 +01:00
|
|
|
if (close_win = !strcmp(line, "/close")) {
|
2013-09-15 22:38:38 +02:00
|
|
|
int f_num = self->num;
|
2013-09-10 10:04:13 +02:00
|
|
|
delwin(ctx->linewin);
|
|
|
|
delwin(statusbar->topline);
|
|
|
|
del_window(self);
|
|
|
|
disable_chatwin(f_num);
|
2013-09-19 12:37:42 +02:00
|
|
|
} else if (!strncmp(line, "/me ", strlen("/me ")))
|
2013-10-14 01:09:20 +02:00
|
|
|
send_action(self, ctx, m, line + strlen("/me "));
|
2013-09-19 12:37:42 +02:00
|
|
|
else
|
2013-11-19 21:32:35 +01:00
|
|
|
execute(ctx->history, self, m, line, CHAT_COMMAND_MODE);
|
2013-09-10 10:04:13 +02:00
|
|
|
} else {
|
|
|
|
/* make sure the string has at least non-space character */
|
|
|
|
if (!string_is_empty(line)) {
|
|
|
|
uint8_t selfname[TOX_MAX_NAME_LENGTH];
|
2013-11-29 16:14:59 +01:00
|
|
|
tox_get_self_name(m, selfname, TOX_MAX_NAME_LENGTH);
|
2013-09-10 10:04:13 +02:00
|
|
|
|
2013-10-11 07:11:43 +02:00
|
|
|
print_time(ctx->history);
|
2013-09-10 10:04:13 +02:00
|
|
|
wattron(ctx->history, COLOR_PAIR(GREEN));
|
|
|
|
wprintw(ctx->history, "%s: ", selfname);
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(GREEN));
|
2013-09-26 06:33:51 +02:00
|
|
|
|
|
|
|
if (line[0] == '>') {
|
|
|
|
wattron(ctx->history, COLOR_PAIR(GREEN));
|
|
|
|
wprintw(ctx->history, "%s\n", line);
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(GREEN));
|
|
|
|
} else
|
|
|
|
wprintw(ctx->history, "%s\n", line);
|
2013-09-10 10:04:13 +02:00
|
|
|
|
|
|
|
if (!statusbar->is_online
|
2013-11-29 16:14:59 +01:00
|
|
|
|| tox_send_message(m, self->num, line, strlen(line) + 1) == 0) {
|
2013-09-10 10:04:13 +02:00
|
|
|
wattron(ctx->history, COLOR_PAIR(RED));
|
|
|
|
wprintw(ctx->history, " * Failed to send message.\n");
|
|
|
|
wattroff(ctx->history, COLOR_PAIR(RED));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (close_win) {
|
|
|
|
free(ctx);
|
|
|
|
free(statusbar);
|
|
|
|
} else {
|
2013-12-01 04:12:43 +01:00
|
|
|
reset_buf(ctx->line, &ctx->pos, &ctx->len);
|
2013-09-10 10:04:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
static void chat_onDraw(ToxWindow *self, Tox *m)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
curs_set(1);
|
|
|
|
int x, y;
|
|
|
|
getmaxyx(self->window, y, x);
|
2013-09-06 00:24:58 +02:00
|
|
|
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-09-10 10:04:13 +02:00
|
|
|
|
2013-12-01 04:12:43 +01:00
|
|
|
wclear(ctx->linewin);
|
2013-12-08 10:16:49 +01:00
|
|
|
|
|
|
|
uint8_t line[MAX_STR_SIZE];
|
|
|
|
|
|
|
|
if (wcs_to_mbs_buf(line, ctx->line, MAX_STR_SIZE) == -1)
|
|
|
|
memset(&line, 0, sizeof(line));
|
|
|
|
|
|
|
|
mvwprintw(ctx->linewin, 1, 0, "%s", line);
|
2013-12-01 04:12:43 +01:00
|
|
|
|
2013-09-06 00:24:58 +02:00
|
|
|
/* Draw status bar */
|
2013-11-29 23:48:08 +01:00
|
|
|
StatusBar *statusbar = self->stb;
|
2013-11-24 05:46:46 +01:00
|
|
|
mvwhline(statusbar->topline, 1, 0, ACS_HLINE, x);
|
2013-09-06 00:24:58 +02:00
|
|
|
wmove(statusbar->topline, 0, 0);
|
|
|
|
|
|
|
|
/* Draw name, status and note in statusbar */
|
|
|
|
if (statusbar->is_online) {
|
|
|
|
char *status_text = "Unknown";
|
|
|
|
int colour = WHITE;
|
|
|
|
|
|
|
|
TOX_USERSTATUS status = statusbar->status;
|
|
|
|
|
2013-11-12 07:50:04 +01:00
|
|
|
switch (status) {
|
2013-09-06 00:24:58 +02:00
|
|
|
case TOX_USERSTATUS_NONE:
|
|
|
|
status_text = "Online";
|
|
|
|
colour = GREEN;
|
|
|
|
break;
|
|
|
|
case TOX_USERSTATUS_AWAY:
|
|
|
|
status_text = "Away";
|
|
|
|
colour = YELLOW;
|
|
|
|
break;
|
|
|
|
case TOX_USERSTATUS_BUSY:
|
|
|
|
status_text = "Busy";
|
|
|
|
colour = RED;
|
|
|
|
break;
|
|
|
|
}
|
2013-09-07 01:59:45 +02:00
|
|
|
|
2013-09-06 00:24:58 +02:00
|
|
|
wattron(statusbar->topline, A_BOLD);
|
2013-09-06 06:56:55 +02:00
|
|
|
wprintw(statusbar->topline, " %s ", self->name);
|
2013-09-06 00:24:58 +02:00
|
|
|
wattroff(statusbar->topline, A_BOLD);
|
|
|
|
wattron(statusbar->topline, COLOR_PAIR(colour) | A_BOLD);
|
2013-09-06 06:56:55 +02:00
|
|
|
wprintw(statusbar->topline, "[%s]", status_text);
|
2013-09-06 00:24:58 +02:00
|
|
|
wattroff(statusbar->topline, COLOR_PAIR(colour) | A_BOLD);
|
|
|
|
} else {
|
|
|
|
wattron(statusbar->topline, A_BOLD);
|
2013-09-06 06:56:55 +02:00
|
|
|
wprintw(statusbar->topline, " %s ", self->name);
|
2013-09-06 00:24:58 +02:00
|
|
|
wattroff(statusbar->topline, A_BOLD);
|
2013-09-06 06:56:55 +02:00
|
|
|
wprintw(statusbar->topline, "[Offline]");
|
2013-09-06 00:24:58 +02:00
|
|
|
}
|
|
|
|
|
2013-09-13 08:02:49 +02:00
|
|
|
/* Reset statusbar->statusmsg on window resize */
|
|
|
|
if (x != self->x) {
|
|
|
|
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
2013-11-29 16:14:59 +01:00
|
|
|
tox_get_status_message(m, self->num, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
2013-09-13 08:02:49 +02:00
|
|
|
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
2013-11-29 16:14:59 +01:00
|
|
|
statusbar->statusmsg_len = tox_get_status_message_size(m, self->num);
|
2013-09-13 08:02:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
self->x = x;
|
|
|
|
|
2013-09-10 10:04:13 +02:00
|
|
|
/* Truncate note if it doesn't fit in statusbar */
|
2013-11-24 05:46:46 +01:00
|
|
|
uint16_t maxlen = x - getcurx(statusbar->topline) - 4;
|
2013-09-10 10:04:13 +02:00
|
|
|
if (statusbar->statusmsg_len > maxlen) {
|
|
|
|
statusbar->statusmsg[maxlen] = '\0';
|
|
|
|
statusbar->statusmsg_len = maxlen;
|
|
|
|
}
|
|
|
|
|
2013-09-13 08:02:49 +02:00
|
|
|
if (statusbar->statusmsg[0]) {
|
|
|
|
wattron(statusbar->topline, A_BOLD);
|
2013-11-24 05:46:46 +01:00
|
|
|
wprintw(statusbar->topline, " - %s ", statusbar->statusmsg);
|
2013-09-13 08:02:49 +02:00
|
|
|
wattroff(statusbar->topline, A_BOLD);
|
|
|
|
}
|
2013-09-06 00:24:58 +02:00
|
|
|
|
|
|
|
wprintw(statusbar->topline, "\n");
|
2013-11-24 05:46:46 +01:00
|
|
|
mvwhline(ctx->linewin, 0, 0, ACS_HLINE, x);
|
2013-07-31 19:20:03 +02:00
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
static void chat_onInit(ToxWindow *self, Tox *m)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
int x, y;
|
|
|
|
getmaxyx(self->window, y, x);
|
2013-09-13 08:02:49 +02:00
|
|
|
self->x = x;
|
2013-09-06 00:24:58 +02:00
|
|
|
|
|
|
|
/* Init statusbar info */
|
2013-11-29 23:48:08 +01:00
|
|
|
StatusBar *statusbar = self->stb;
|
2013-11-29 16:14:59 +01:00
|
|
|
statusbar->status = tox_get_user_status(m, self->num);
|
|
|
|
statusbar->is_online = tox_get_friend_connection_status(m, self->num) == 1;
|
2013-09-06 00:24:58 +02:00
|
|
|
|
2013-09-08 09:18:34 +02:00
|
|
|
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
2013-11-29 16:14:59 +01:00
|
|
|
tox_get_status_message(m, self->num, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
2013-09-08 09:18:34 +02:00
|
|
|
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
2013-11-29 16:14:59 +01:00
|
|
|
statusbar->statusmsg_len = tox_get_status_message_size(m, self->num);
|
2013-09-06 00:24:58 +02:00
|
|
|
|
|
|
|
/* Init subwindows */
|
2013-11-29 23:48:08 +01:00
|
|
|
ChatContext *ctx = self->chatwin;
|
2013-09-06 00:24:58 +02:00
|
|
|
statusbar->topline = subwin(self->window, 2, x, 0, 0);
|
2013-11-26 23:39:11 +01:00
|
|
|
ctx->history = subwin(self->window, y-CHATBOX_HEIGHT+1, x, 0, 0);
|
2013-08-16 19:11:09 +02:00
|
|
|
scrollok(ctx->history, 1);
|
2013-11-26 23:39:11 +01:00
|
|
|
ctx->linewin = subwin(self->window, CHATBOX_HEIGHT, x, y-CHATBOX_HEIGHT, 0);
|
2013-09-06 00:36:46 +02:00
|
|
|
wprintw(ctx->history, "\n\n");
|
2013-11-19 21:32:35 +01:00
|
|
|
execute(ctx->history, self, m, "/help", CHAT_COMMAND_MODE);
|
2013-08-16 19:11:09 +02:00
|
|
|
wmove(self->window, y - CURS_Y_OFFSET, 0);
|
2013-07-31 19:20:03 +02:00
|
|
|
}
|
|
|
|
|
2013-11-19 21:32:35 +01:00
|
|
|
ToxWindow new_chat(Tox *m, int friendnum)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
ToxWindow ret;
|
|
|
|
memset(&ret, 0, sizeof(ret));
|
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
ret.active = true;
|
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
ret.onKey = &chat_onKey;
|
|
|
|
ret.onDraw = &chat_onDraw;
|
|
|
|
ret.onInit = &chat_onInit;
|
|
|
|
ret.onMessage = &chat_onMessage;
|
2013-09-05 03:25:59 +02:00
|
|
|
ret.onConnectionChange = &chat_onConnectionChange;
|
2013-11-10 03:43:56 +01:00
|
|
|
ret.onGroupInvite = &chat_onGroupInvite;
|
2013-08-16 19:11:09 +02:00
|
|
|
ret.onNickChange = &chat_onNickChange;
|
2013-09-05 03:25:59 +02:00
|
|
|
ret.onStatusChange = &chat_onStatusChange;
|
2013-09-03 05:27:34 +02:00
|
|
|
ret.onStatusMessageChange = &chat_onStatusMessageChange;
|
2013-08-16 19:11:09 +02:00
|
|
|
ret.onAction = &chat_onAction;
|
2013-10-11 06:23:39 +02:00
|
|
|
ret.onFileSendRequest = &chat_onFileSendRequest;
|
|
|
|
ret.onFileControl = &chat_onFileControl;
|
|
|
|
ret.onFileData = &chat_onFileData;
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-09-06 00:24:58 +02:00
|
|
|
uint8_t name[TOX_MAX_NAME_LENGTH] = {'\0'};
|
2013-11-29 16:14:59 +01:00
|
|
|
uint16_t len = tox_get_name(m, friendnum, name);
|
2013-10-19 05:46:58 +02:00
|
|
|
memcpy(ret.name, name, len);
|
|
|
|
ret.name[TOXIC_MAX_NAME_LENGTH] = '\0';
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-09-13 08:02:49 +02:00
|
|
|
ChatContext *chatwin = calloc(1, sizeof(ChatContext));
|
|
|
|
StatusBar *stb = calloc(1, sizeof(StatusBar));
|
2013-09-12 00:07:26 +02:00
|
|
|
|
2013-09-13 08:02:49 +02:00
|
|
|
if (stb != NULL && chatwin != NULL) {
|
|
|
|
ret.chatwin = chatwin;
|
|
|
|
ret.stb = stb;
|
2013-09-12 00:07:26 +02:00
|
|
|
} else {
|
|
|
|
endwin();
|
2013-09-12 07:33:41 +02:00
|
|
|
fprintf(stderr, "calloc() failed. Aborting...\n");
|
2013-09-12 00:07:26 +02:00
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
2013-09-07 01:59:45 +02:00
|
|
|
|
2013-09-15 22:38:38 +02:00
|
|
|
ret.num = friendnum;
|
2013-09-06 00:24:58 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
return ret;
|
2013-07-31 19:20:03 +02:00
|
|
|
}
|