mirror of
https://github.com/Tha14/toxic.git
synced 2025-10-24 08:06:51 +02:00
@@ -11,7 +11,9 @@ toxic_SOURCES = $(top_srcdir)/src/main.c \
|
|||||||
$(top_srcdir)/src/friendlist.h \
|
$(top_srcdir)/src/friendlist.h \
|
||||||
$(top_srcdir)/src/friendlist.c \
|
$(top_srcdir)/src/friendlist.c \
|
||||||
$(top_srcdir)/src/toxic_windows.h \
|
$(top_srcdir)/src/toxic_windows.h \
|
||||||
$(top_srcdir)/src/windows.c
|
$(top_srcdir)/src/windows.c \
|
||||||
|
$(top_srcdir)/src/groupchat.c \
|
||||||
|
$(top_srcdir)/src/groupchat.h
|
||||||
|
|
||||||
toxic_CFLAGS = -I$(top_srcdir) \
|
toxic_CFLAGS = -I$(top_srcdir) \
|
||||||
$(NCURSES_CFLAGS) \
|
$(NCURSES_CFLAGS) \
|
||||||
|
49
src/chat.c
49
src/chat.c
@@ -18,18 +18,9 @@
|
|||||||
#include "friendlist.h"
|
#include "friendlist.h"
|
||||||
#include "chat.h"
|
#include "chat.h"
|
||||||
|
|
||||||
#define CURS_Y_OFFSET 3
|
|
||||||
|
|
||||||
extern char *DATA_FILE;
|
extern char *DATA_FILE;
|
||||||
extern int store_data(Tox *m, char *path);
|
extern int store_data(Tox *m, char *path);
|
||||||
|
|
||||||
typedef struct {
|
|
||||||
wchar_t line[MAX_STR_SIZE];
|
|
||||||
size_t pos;
|
|
||||||
WINDOW *history;
|
|
||||||
WINDOW *linewin;
|
|
||||||
} ChatContext;
|
|
||||||
|
|
||||||
struct tm *get_time(void)
|
struct tm *get_time(void)
|
||||||
{
|
{
|
||||||
struct tm *timeinfo;
|
struct tm *timeinfo;
|
||||||
@@ -41,11 +32,10 @@ struct tm *get_time(void)
|
|||||||
|
|
||||||
static void chat_onMessage(ToxWindow *self, Tox *m, int num, uint8_t *msg, uint16_t len)
|
static void chat_onMessage(ToxWindow *self, Tox *m, int num, uint8_t *msg, uint16_t len)
|
||||||
{
|
{
|
||||||
if (self->friendnum != num)
|
if (self->num != num)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
ChatContext *ctx = (ChatContext *) self->chatwin;
|
ChatContext *ctx = (ChatContext *) self->chatwin;
|
||||||
|
|
||||||
struct tm *timeinfo = get_time();
|
struct tm *timeinfo = get_time();
|
||||||
|
|
||||||
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
|
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
|
||||||
@@ -65,7 +55,7 @@ static void chat_onMessage(ToxWindow *self, Tox *m, int num, uint8_t *msg, uint1
|
|||||||
|
|
||||||
void chat_onConnectionChange(ToxWindow *self, Tox *m, int num, uint8_t status)
|
void chat_onConnectionChange(ToxWindow *self, Tox *m, int num, uint8_t status)
|
||||||
{
|
{
|
||||||
if (self->friendnum != num)
|
if (self->num != num)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
StatusBar *statusbar = (StatusBar *) self->stb;
|
StatusBar *statusbar = (StatusBar *) self->stb;
|
||||||
@@ -74,7 +64,7 @@ void chat_onConnectionChange(ToxWindow *self, Tox *m, int num, uint8_t status)
|
|||||||
|
|
||||||
static void chat_onAction(ToxWindow *self, Tox *m, int num, uint8_t *action, uint16_t len)
|
static void chat_onAction(ToxWindow *self, Tox *m, int num, uint8_t *action, uint16_t len)
|
||||||
{
|
{
|
||||||
if (self->friendnum != num)
|
if (self->num != num)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
ChatContext *ctx = (ChatContext *) self->chatwin;
|
ChatContext *ctx = (ChatContext *) self->chatwin;
|
||||||
@@ -97,7 +87,7 @@ static void chat_onAction(ToxWindow *self, Tox *m, int num, uint8_t *action, uin
|
|||||||
|
|
||||||
static void chat_onNickChange(ToxWindow *self, int num, uint8_t *nick, uint16_t len)
|
static void chat_onNickChange(ToxWindow *self, int num, uint8_t *nick, uint16_t len)
|
||||||
{
|
{
|
||||||
if (self->friendnum != num)
|
if (self->num != num)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
snprintf(self->name, sizeof(self->name), "%s", nick);
|
snprintf(self->name, sizeof(self->name), "%s", nick);
|
||||||
@@ -105,7 +95,7 @@ static void chat_onNickChange(ToxWindow *self, int num, uint8_t *nick, uint16_t
|
|||||||
|
|
||||||
static void chat_onStatusChange(ToxWindow *self, Tox *m, int num, TOX_USERSTATUS status)
|
static void chat_onStatusChange(ToxWindow *self, Tox *m, int num, TOX_USERSTATUS status)
|
||||||
{
|
{
|
||||||
if (self->friendnum != num)
|
if (self->num != num)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
StatusBar *statusbar = (StatusBar *) self->stb;
|
StatusBar *statusbar = (StatusBar *) self->stb;
|
||||||
@@ -114,7 +104,7 @@ static void chat_onStatusChange(ToxWindow *self, Tox *m, int num, TOX_USERSTATUS
|
|||||||
|
|
||||||
static void chat_onStatusMessageChange(ToxWindow *self, int num, uint8_t *status, uint16_t len)
|
static void chat_onStatusMessageChange(ToxWindow *self, int num, uint8_t *status, uint16_t len)
|
||||||
{
|
{
|
||||||
if (self->friendnum != num)
|
if (self->num != num)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
StatusBar *statusbar = (StatusBar *) self->stb;
|
StatusBar *statusbar = (StatusBar *) self->stb;
|
||||||
@@ -133,7 +123,7 @@ int string_is_empty(char *string)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* convert wide characters to null terminated string */
|
/* convert wide characters to null terminated string */
|
||||||
static uint8_t *wcs_to_char(wchar_t *string)
|
uint8_t *wcs_to_char(wchar_t *string)
|
||||||
{
|
{
|
||||||
size_t len = 0;
|
size_t len = 0;
|
||||||
char *ret = NULL;
|
char *ret = NULL;
|
||||||
@@ -162,7 +152,7 @@ static uint8_t *wcs_to_char(wchar_t *string)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* convert a wide char to null terminated string */
|
/* convert a wide char to null terminated string */
|
||||||
static char *wc_to_char(wchar_t ch)
|
char *wc_to_char(wchar_t ch)
|
||||||
{
|
{
|
||||||
int len = 0;
|
int len = 0;
|
||||||
static char ret[MB_LEN_MAX + 1];
|
static char ret[MB_LEN_MAX + 1];
|
||||||
@@ -239,7 +229,7 @@ static void execute(ToxWindow *self, ChatContext *ctx, StatusBar *statusbar, Tox
|
|||||||
wattroff(ctx->history, COLOR_PAIR(YELLOW));
|
wattroff(ctx->history, COLOR_PAIR(YELLOW));
|
||||||
|
|
||||||
if (!statusbar->is_online
|
if (!statusbar->is_online
|
||||||
|| tox_sendaction(m, self->friendnum, action, strlen(action) + 1) == 0) {
|
|| tox_sendaction(m, self->num, action, strlen(action) + 1) == 0) {
|
||||||
wattron(ctx->history, COLOR_PAIR(RED));
|
wattron(ctx->history, COLOR_PAIR(RED));
|
||||||
wprintw(ctx->history, " * Failed to send action\n");
|
wprintw(ctx->history, " * Failed to send action\n");
|
||||||
wattroff(ctx->history, COLOR_PAIR(RED));
|
wattroff(ctx->history, COLOR_PAIR(RED));
|
||||||
@@ -324,7 +314,7 @@ static void execute(ToxWindow *self, ChatContext *ctx, StatusBar *statusbar, Tox
|
|||||||
}
|
}
|
||||||
|
|
||||||
tox_setname(m, nick, len+1);
|
tox_setname(m, nick, len+1);
|
||||||
prompt_update_nick(self->prompt, nick);
|
prompt_update_nick(self->prompt, nick, len+1);
|
||||||
wprintw(ctx->history, "Nickname set to: %s\n", nick);
|
wprintw(ctx->history, "Nickname set to: %s\n", nick);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -393,7 +383,7 @@ static void chat_onKey(ToxWindow *self, Tox *m, wint_t key)
|
|||||||
|
|
||||||
if (line[0] == '/') {
|
if (line[0] == '/') {
|
||||||
if (close_win = !strncmp(line, "/close", strlen("/close"))) {
|
if (close_win = !strncmp(line, "/close", strlen("/close"))) {
|
||||||
int f_num = self->friendnum;
|
int f_num = self->num;
|
||||||
delwin(ctx->linewin);
|
delwin(ctx->linewin);
|
||||||
delwin(statusbar->topline);
|
delwin(statusbar->topline);
|
||||||
del_window(self);
|
del_window(self);
|
||||||
@@ -415,7 +405,7 @@ static void chat_onKey(ToxWindow *self, Tox *m, wint_t key)
|
|||||||
wprintw(ctx->history, "%s\n", line);
|
wprintw(ctx->history, "%s\n", line);
|
||||||
|
|
||||||
if (!statusbar->is_online
|
if (!statusbar->is_online
|
||||||
|| tox_sendmessage(m, self->friendnum, line, strlen(line) + 1) == 0) {
|
|| tox_sendmessage(m, self->num, line, strlen(line) + 1) == 0) {
|
||||||
wattron(ctx->history, COLOR_PAIR(RED));
|
wattron(ctx->history, COLOR_PAIR(RED));
|
||||||
wprintw(ctx->history, " * Failed to send message.\n");
|
wprintw(ctx->history, " * Failed to send message.\n");
|
||||||
wattroff(ctx->history, COLOR_PAIR(RED));
|
wattroff(ctx->history, COLOR_PAIR(RED));
|
||||||
@@ -438,7 +428,6 @@ static void chat_onKey(ToxWindow *self, Tox *m, wint_t key)
|
|||||||
static void chat_onDraw(ToxWindow *self, Tox *m)
|
static void chat_onDraw(ToxWindow *self, Tox *m)
|
||||||
{
|
{
|
||||||
curs_set(1);
|
curs_set(1);
|
||||||
|
|
||||||
int x, y;
|
int x, y;
|
||||||
getmaxyx(self->window, y, x);
|
getmaxyx(self->window, y, x);
|
||||||
|
|
||||||
@@ -487,9 +476,9 @@ static void chat_onDraw(ToxWindow *self, Tox *m)
|
|||||||
/* Reset statusbar->statusmsg on window resize */
|
/* Reset statusbar->statusmsg on window resize */
|
||||||
if (x != self->x) {
|
if (x != self->x) {
|
||||||
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
||||||
tox_copy_statusmessage(m, self->friendnum, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
tox_copy_statusmessage(m, self->num, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
||||||
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
||||||
statusbar->statusmsg_len = tox_get_statusmessage_size(m, self->friendnum);
|
statusbar->statusmsg_len = tox_get_statusmessage_size(m, self->num);
|
||||||
}
|
}
|
||||||
|
|
||||||
self->x = x;
|
self->x = x;
|
||||||
@@ -520,13 +509,13 @@ static void chat_onInit(ToxWindow *self, Tox *m)
|
|||||||
|
|
||||||
/* Init statusbar info */
|
/* Init statusbar info */
|
||||||
StatusBar *statusbar = (StatusBar *) self->stb;
|
StatusBar *statusbar = (StatusBar *) self->stb;
|
||||||
statusbar->status = tox_get_userstatus(m, self->friendnum);
|
statusbar->status = tox_get_userstatus(m, self->num);
|
||||||
statusbar->is_online = tox_get_friend_connectionstatus(m, self->friendnum) == 1;
|
statusbar->is_online = tox_get_friend_connectionstatus(m, self->num) == 1;
|
||||||
|
|
||||||
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
||||||
tox_copy_statusmessage(m, self->friendnum, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
tox_copy_statusmessage(m, self->num, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
||||||
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
||||||
statusbar->statusmsg_len = tox_get_statusmessage_size(m, self->friendnum);
|
statusbar->statusmsg_len = tox_get_statusmessage_size(m, self->num);
|
||||||
|
|
||||||
/* Init subwindows */
|
/* Init subwindows */
|
||||||
ChatContext *ctx = (ChatContext *) self->chatwin;
|
ChatContext *ctx = (ChatContext *) self->chatwin;
|
||||||
@@ -571,7 +560,7 @@ ToxWindow new_chat(Tox *m, ToxWindow *prompt, int friendnum)
|
|||||||
}
|
}
|
||||||
|
|
||||||
ret.prompt = prompt;
|
ret.prompt = prompt;
|
||||||
ret.friendnum = friendnum;
|
ret.num = friendnum;
|
||||||
|
|
||||||
return ret;
|
return ret;
|
||||||
}
|
}
|
||||||
|
@@ -1,6 +1,10 @@
|
|||||||
#ifndef CHAT_H_6489PZ13
|
#ifndef CHAT_H_6489PZ13
|
||||||
#define CHAT_H_6489PZ13
|
#define CHAT_H_6489PZ13
|
||||||
|
|
||||||
|
struct tm *get_time(void);
|
||||||
|
char *wc_to_char(wchar_t ch);
|
||||||
|
uint8_t *wcs_to_char(wchar_t *string);
|
||||||
|
int string_is_empty(char *string);
|
||||||
ToxWindow new_chat(Tox *m, ToxWindow *prompt, int friendnum);
|
ToxWindow new_chat(Tox *m, ToxWindow *prompt, int friendnum);
|
||||||
|
|
||||||
#endif /* end of include guard: CHAT_H_6489PZ13 */
|
#endif /* end of include guard: CHAT_H_6489PZ13 */
|
||||||
|
@@ -21,6 +21,7 @@ extern ToxWindow *prompt;
|
|||||||
|
|
||||||
typedef struct {
|
typedef struct {
|
||||||
uint8_t name[TOX_MAX_NAME_LENGTH];
|
uint8_t name[TOX_MAX_NAME_LENGTH];
|
||||||
|
uint16_t namelength;
|
||||||
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH];
|
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH];
|
||||||
uint16_t statusmsg_len;
|
uint16_t statusmsg_len;
|
||||||
int num;
|
int num;
|
||||||
@@ -49,10 +50,7 @@ void friendlist_onConnectionChange(ToxWindow *self, Tox *m, int num, uint8_t sta
|
|||||||
if (num < 0 || num >= num_friends)
|
if (num < 0 || num >= num_friends)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
if (status == 1)
|
friends[num].online = status == 1 ? true : false;
|
||||||
friends[num].online = true;
|
|
||||||
else
|
|
||||||
friends[num].online = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void friendlist_onNickChange(ToxWindow *self, int num, uint8_t *str, uint16_t len)
|
void friendlist_onNickChange(ToxWindow *self, int num, uint8_t *str, uint16_t len)
|
||||||
@@ -61,6 +59,7 @@ void friendlist_onNickChange(ToxWindow *self, int num, uint8_t *str, uint16_t le
|
|||||||
return;
|
return;
|
||||||
|
|
||||||
memcpy((char *) &friends[num].name, (char *) str, len);
|
memcpy((char *) &friends[num].name, (char *) str, len);
|
||||||
|
friends[num].namelength = len;
|
||||||
}
|
}
|
||||||
|
|
||||||
void friendlist_onStatusChange(ToxWindow *self, Tox *m, int num, TOX_USERSTATUS status)
|
void friendlist_onStatusChange(ToxWindow *self, Tox *m, int num, TOX_USERSTATUS status)
|
||||||
@@ -94,9 +93,12 @@ int friendlist_onFriendAdded(Tox *m, int num)
|
|||||||
friends[i].chatwin = -1;
|
friends[i].chatwin = -1;
|
||||||
friends[i].online = false;
|
friends[i].online = false;
|
||||||
friends[i].status = TOX_USERSTATUS_NONE;
|
friends[i].status = TOX_USERSTATUS_NONE;
|
||||||
|
friends[i].namelength = tox_getname(m, num, friends[i].name);
|
||||||
|
|
||||||
if (tox_getname(m, num, friends[i].name) == -1 || friends[i].name[0] == '\0')
|
if (friends[i].namelength == -1 || friends[i].name[0] == '\0') {
|
||||||
strcpy((char *) friends[i].name, UNKNOWN_NAME);
|
strcpy((char *) friends[i].name, UNKNOWN_NAME);
|
||||||
|
friends[i].namelength = strlen(UNKNOWN_NAME) + 1;
|
||||||
|
}
|
||||||
|
|
||||||
if (i == num_friends)
|
if (i == num_friends)
|
||||||
++num_friends;
|
++num_friends;
|
||||||
@@ -227,7 +229,7 @@ static void friendlist_onDraw(ToxWindow *self, Tox *m)
|
|||||||
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
|
||||||
tox_copy_statusmessage(m, friends[i].num, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
tox_copy_statusmessage(m, friends[i].num, statusmsg, TOX_MAX_STATUSMESSAGE_LENGTH);
|
||||||
snprintf(friends[i].statusmsg, sizeof(friends[i].statusmsg), "%s", statusmsg);
|
snprintf(friends[i].statusmsg, sizeof(friends[i].statusmsg), "%s", statusmsg);
|
||||||
friends[i].statusmsg_len = tox_get_statusmessage_size(m, self->friendnum);
|
friends[i].statusmsg_len = tox_get_statusmessage_size(m, self->num);
|
||||||
}
|
}
|
||||||
|
|
||||||
self->x = x;
|
self->x = x;
|
||||||
@@ -254,6 +256,20 @@ void disable_chatwin(int f_num)
|
|||||||
friends[f_num].chatwin = -1;
|
friends[f_num].chatwin = -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Returns the respective friend number of name. Returns -1 on no match */
|
||||||
|
int get_friendnum(uint8_t *name)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
int len = strlen(name);
|
||||||
|
|
||||||
|
for (i = 0; i < num_friends; ++i) {
|
||||||
|
if (strncmp(friends[i].name, name, len) == 0)
|
||||||
|
return friends[i].num;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
static void friendlist_onInit(ToxWindow *self, Tox *m)
|
static void friendlist_onInit(ToxWindow *self, Tox *m)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
@@ -7,5 +7,6 @@
|
|||||||
ToxWindow new_friendlist();
|
ToxWindow new_friendlist();
|
||||||
int friendlist_onFriendAdded(Tox *m, int num);
|
int friendlist_onFriendAdded(Tox *m, int num);
|
||||||
void disable_chatwin(int f_num);
|
void disable_chatwin(int f_num);
|
||||||
|
int get_friendnum(uint8_t *name);
|
||||||
|
|
||||||
#endif /* end of include guard: FRIENDLIST_H_53I41IM */
|
#endif /* end of include guard: FRIENDLIST_H_53I41IM */
|
||||||
|
233
src/groupchat.c
Normal file
233
src/groupchat.c
Normal file
@@ -0,0 +1,233 @@
|
|||||||
|
/*
|
||||||
|
* Toxic -- Tox Curses Client
|
||||||
|
*/
|
||||||
|
|
||||||
|
#ifdef HAVE_CONFIG_H
|
||||||
|
#include "config.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdint.h>
|
||||||
|
#include <stdbool.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
#include <time.h>
|
||||||
|
|
||||||
|
#include "toxic_windows.h"
|
||||||
|
#include "chat.h"
|
||||||
|
|
||||||
|
static GroupChat groupchats[MAX_GROUPCHAT_NUM];
|
||||||
|
static int group_chat_index = 0;
|
||||||
|
|
||||||
|
ToxWindow new_group_chat(Tox *m, ToxWindow *prompt, int groupnum);
|
||||||
|
|
||||||
|
extern char *DATA_FILE;
|
||||||
|
extern int store_data(Tox *m, char *path);
|
||||||
|
|
||||||
|
int get_num_groupchats(void)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i <= group_chat_index; ++i) {
|
||||||
|
if (!groupchats[i].active)
|
||||||
|
return i;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
int init_groupchat_win(ToxWindow *prompt, Tox *m, int groupnum)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i <= group_chat_index; ++i) {
|
||||||
|
if (!groupchats[i].active) {
|
||||||
|
groupchats[i].chatwin = add_window(m, new_group_chat(m, prompt, groupnum));
|
||||||
|
groupchats[i].active = true;
|
||||||
|
set_active_window(groupchats[i].chatwin);
|
||||||
|
|
||||||
|
if (i == group_chat_index)
|
||||||
|
++group_chat_index;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void close_groupchatwin(Tox *m, int groupnum)
|
||||||
|
{
|
||||||
|
tox_del_groupchat(m, groupnum);
|
||||||
|
memset(&(groupchats[groupnum]), 0, sizeof(GroupChat));
|
||||||
|
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = group_chat_index; i > 0; --i) {
|
||||||
|
if (groupchats[i-1].active)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
group_chat_index = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void groupchat_onGroupMessage(ToxWindow *self, Tox *m, int groupnum, int peernum, uint8_t *msg, uint16_t len)
|
||||||
|
{
|
||||||
|
if (self->num != groupnum)
|
||||||
|
return;
|
||||||
|
|
||||||
|
ChatContext *ctx = (ChatContext *) self->chatwin;
|
||||||
|
struct tm *timeinfo = get_time();
|
||||||
|
|
||||||
|
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
|
||||||
|
tox_group_peername(m, groupnum, peernum, nick);
|
||||||
|
nick[TOXIC_MAX_NAME_LENGTH] = '\0'; /* enforce client max name length */
|
||||||
|
|
||||||
|
wattron(ctx->history, COLOR_PAIR(CYAN));
|
||||||
|
wprintw(ctx->history, "[%02d:%02d:%02d] ", timeinfo->tm_hour, timeinfo->tm_min, timeinfo->tm_sec);
|
||||||
|
wattroff(ctx->history, COLOR_PAIR(CYAN));
|
||||||
|
wattron(ctx->history, COLOR_PAIR(4));
|
||||||
|
wprintw(ctx->history, "%s: ", nick);
|
||||||
|
wattroff(ctx->history, COLOR_PAIR(4));
|
||||||
|
wprintw(ctx->history, "%s\n", msg);
|
||||||
|
|
||||||
|
self->blink = true;
|
||||||
|
beep();
|
||||||
|
}
|
||||||
|
|
||||||
|
static void groupchat_onKey(ToxWindow *self, Tox *m, wint_t key)
|
||||||
|
{
|
||||||
|
ChatContext *ctx = (ChatContext *) self->chatwin;
|
||||||
|
struct tm *timeinfo = get_time();
|
||||||
|
|
||||||
|
int x, y, y2, x2;
|
||||||
|
getyx(self->window, y, x);
|
||||||
|
getmaxyx(self->window, y2, x2);
|
||||||
|
|
||||||
|
/* Add printable chars to buffer and print on input space */
|
||||||
|
#if HAVE_WIDECHAR
|
||||||
|
if (iswprint(key)) {
|
||||||
|
#else
|
||||||
|
if (isprint(key)) {
|
||||||
|
#endif
|
||||||
|
if (ctx->pos < (MAX_STR_SIZE-1)) {
|
||||||
|
mvwaddstr(self->window, y, x, wc_to_char(key));
|
||||||
|
ctx->line[ctx->pos++] = key;
|
||||||
|
ctx->line[ctx->pos] = L'\0';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* BACKSPACE key: Remove one character from line */
|
||||||
|
else if (key == 0x107 || key == 0x8 || key == 0x7f) {
|
||||||
|
if (ctx->pos > 0) {
|
||||||
|
ctx->line[--ctx->pos] = L'\0';
|
||||||
|
|
||||||
|
if (x == 0)
|
||||||
|
mvwdelch(self->window, y - 1, x2 - 1);
|
||||||
|
else
|
||||||
|
mvwdelch(self->window, y, x - 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/* RETURN key: Execute command or print line */
|
||||||
|
else if (key == '\n') {
|
||||||
|
uint8_t *line = wcs_to_char(ctx->line);
|
||||||
|
wclear(ctx->linewin);
|
||||||
|
wmove(self->window, y2 - CURS_Y_OFFSET, 0);
|
||||||
|
wclrtobot(self->window);
|
||||||
|
bool close_win = false;
|
||||||
|
|
||||||
|
if (line[0] == '/') {
|
||||||
|
if (close_win = !strncmp(line, "/close", strlen("/close"))) {
|
||||||
|
set_active_window(0);
|
||||||
|
int groupnum = self->num;
|
||||||
|
delwin(ctx->linewin);
|
||||||
|
del_window(self);
|
||||||
|
close_groupchatwin(m, groupnum);
|
||||||
|
} //else
|
||||||
|
//execute(self, ctx, statusbar, m, line);
|
||||||
|
} else {
|
||||||
|
/* make sure the string has at least non-space character */
|
||||||
|
if (!string_is_empty(line)) {
|
||||||
|
// uint8_t selfname[TOX_MAX_NAME_LENGTH];
|
||||||
|
// tox_getselfname(m, selfname, TOX_MAX_NAME_LENGTH);
|
||||||
|
|
||||||
|
// wattron(ctx->history, COLOR_PAIR(CYAN));
|
||||||
|
// wprintw(ctx->history, "[%02d:%02d:%02d] ", timeinfo->tm_hour, timeinfo->tm_min, timeinfo->tm_sec);
|
||||||
|
// wattroff(ctx->history, COLOR_PAIR(CYAN));
|
||||||
|
// wattron(ctx->history, COLOR_PAIR(GREEN));
|
||||||
|
// wprintw(ctx->history, "%s: ", selfname);
|
||||||
|
// wattroff(ctx->history, COLOR_PAIR(GREEN));
|
||||||
|
// wprintw(ctx->history, "%s\n", line);
|
||||||
|
|
||||||
|
if (tox_group_message_send(m, self->num, line, strlen(line) + 1) == -1) {
|
||||||
|
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);
|
||||||
|
else {
|
||||||
|
ctx->line[0] = L'\0';
|
||||||
|
ctx->pos = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
free(line);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void groupchat_onDraw(ToxWindow *self, Tox *m)
|
||||||
|
{
|
||||||
|
curs_set(1);
|
||||||
|
int x, y;
|
||||||
|
getmaxyx(self->window, y, x);
|
||||||
|
ChatContext *ctx = (ChatContext *) self->chatwin;
|
||||||
|
mvwhline(ctx->linewin, 0, 0, '_', x);
|
||||||
|
wrefresh(self->window);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void groupchat_onInit(ToxWindow *self, Tox *m)
|
||||||
|
{
|
||||||
|
int x, y;
|
||||||
|
ChatContext *ctx = (ChatContext *) self->chatwin;
|
||||||
|
getmaxyx(self->window, y, x);
|
||||||
|
ctx->history = subwin(self->window, y-3, x, 0, 0);
|
||||||
|
scrollok(ctx->history, 1);
|
||||||
|
ctx->linewin = subwin(self->window, 2, x, y-4, 0);
|
||||||
|
// print_help(ctx);
|
||||||
|
wmove(self->window, y - CURS_Y_OFFSET, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
ToxWindow new_group_chat(Tox *m, ToxWindow *prompt, int groupnum)
|
||||||
|
{
|
||||||
|
ToxWindow ret;
|
||||||
|
memset(&ret, 0, sizeof(ret));
|
||||||
|
|
||||||
|
ret.onKey = &groupchat_onKey;
|
||||||
|
ret.onDraw = &groupchat_onDraw;
|
||||||
|
ret.onInit = &groupchat_onInit;
|
||||||
|
ret.onGroupMessage = &groupchat_onGroupMessage;
|
||||||
|
// ret.onNickChange = &groupchat_onNickChange;
|
||||||
|
// ret.onStatusChange = &groupchat_onStatusChange;
|
||||||
|
// ret.onAction = &groupchat_onAction;
|
||||||
|
|
||||||
|
snprintf(ret.name, sizeof(ret.name), "Room #%d", groupnum);
|
||||||
|
|
||||||
|
ChatContext *chatwin = calloc(1, sizeof(ChatContext));
|
||||||
|
|
||||||
|
if (chatwin != NULL)
|
||||||
|
ret.chatwin = chatwin;
|
||||||
|
else {
|
||||||
|
endwin();
|
||||||
|
fprintf(stderr, "calloc() failed. Aborting...\n");
|
||||||
|
exit(EXIT_FAILURE);
|
||||||
|
}
|
||||||
|
|
||||||
|
ret.prompt = prompt;
|
||||||
|
ret.num = groupnum;
|
||||||
|
|
||||||
|
return ret;
|
||||||
|
}
|
6
src/groupchat.h
Normal file
6
src/groupchat.h
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
/*
|
||||||
|
* Toxic -- Tox Curses Client
|
||||||
|
*/
|
||||||
|
|
||||||
|
int init_groupchat_win(ToxWindow *prompt, Tox *m, int groupnum);
|
||||||
|
int get_num_groupchats(void);
|
@@ -102,6 +102,8 @@ static Tox *init_tox()
|
|||||||
tox_callback_userstatus(m, on_statuschange, NULL);
|
tox_callback_userstatus(m, on_statuschange, NULL);
|
||||||
tox_callback_statusmessage(m, on_statusmessagechange, NULL);
|
tox_callback_statusmessage(m, on_statusmessagechange, NULL);
|
||||||
tox_callback_action(m, on_action, NULL);
|
tox_callback_action(m, on_action, NULL);
|
||||||
|
tox_callback_group_invite(m, on_groupinvite, NULL);
|
||||||
|
tox_callback_group_message(m, on_groupmessage, NULL);
|
||||||
#ifdef __linux__
|
#ifdef __linux__
|
||||||
tox_setname(m, (uint8_t *) "Cool guy", sizeof("Cool guy"));
|
tox_setname(m, (uint8_t *) "Cool guy", sizeof("Cool guy"));
|
||||||
#elif defined(_WIN32)
|
#elif defined(_WIN32)
|
||||||
|
273
src/prompt.c
273
src/prompt.c
@@ -11,11 +11,15 @@
|
|||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
|
|
||||||
#include "prompt.h"
|
#include "prompt.h"
|
||||||
|
#include "groupchat.h"
|
||||||
|
|
||||||
extern char *DATA_FILE;
|
extern char *DATA_FILE;
|
||||||
|
|
||||||
uint8_t pending_requests[MAX_STR_SIZE][TOX_CLIENT_ID_SIZE]; // XXX
|
uint8_t pending_frnd_requests[MAX_FRIENDS_NUM][TOX_CLIENT_ID_SIZE];
|
||||||
uint8_t num_requests = 0; // XXX
|
uint8_t num_frnd_requests = 0;
|
||||||
|
|
||||||
|
uint8_t pending_grp_requests[MAX_GROUPCHAT_NUM][TOX_CLIENT_ID_SIZE];
|
||||||
|
uint8_t num_grp_requests = 0;
|
||||||
|
|
||||||
static char prompt_buf[MAX_STR_SIZE] = {'\0'};
|
static char prompt_buf[MAX_STR_SIZE] = {'\0'};
|
||||||
static int prompt_buf_pos = 0;
|
static int prompt_buf_pos = 0;
|
||||||
@@ -25,7 +29,10 @@ void cmd_accept(ToxWindow *, Tox *m, int, char **);
|
|||||||
void cmd_add(ToxWindow *, Tox *m, int, char **);
|
void cmd_add(ToxWindow *, Tox *m, int, char **);
|
||||||
void cmd_clear(ToxWindow *, Tox *m, int, char **);
|
void cmd_clear(ToxWindow *, Tox *m, int, char **);
|
||||||
void cmd_connect(ToxWindow *, Tox *m, int, char **);
|
void cmd_connect(ToxWindow *, Tox *m, int, char **);
|
||||||
|
void cmd_groupchat(ToxWindow *, Tox *m, int, char **);
|
||||||
void cmd_help(ToxWindow *, Tox *m, int, char **);
|
void cmd_help(ToxWindow *, Tox *m, int, char **);
|
||||||
|
void cmd_invite(ToxWindow *, Tox *m, int, char **);
|
||||||
|
void cmd_join(ToxWindow *, Tox *m, int, char **);
|
||||||
void cmd_msg(ToxWindow *, Tox *m, int, char **);
|
void cmd_msg(ToxWindow *, Tox *m, int, char **);
|
||||||
void cmd_myid(ToxWindow *, Tox *m, int, char **);
|
void cmd_myid(ToxWindow *, Tox *m, int, char **);
|
||||||
void cmd_nick(ToxWindow *, Tox *m, int, char **);
|
void cmd_nick(ToxWindow *, Tox *m, int, char **);
|
||||||
@@ -33,7 +40,7 @@ void cmd_quit(ToxWindow *, Tox *m, int, char **);
|
|||||||
void cmd_status(ToxWindow *, Tox *m, int, char **);
|
void cmd_status(ToxWindow *, Tox *m, int, char **);
|
||||||
void cmd_note(ToxWindow *, Tox *m, int, char **);
|
void cmd_note(ToxWindow *, Tox *m, int, char **);
|
||||||
|
|
||||||
#define NUM_COMMANDS 13
|
#define NUM_COMMANDS 16
|
||||||
|
|
||||||
static struct {
|
static struct {
|
||||||
char *name;
|
char *name;
|
||||||
@@ -44,7 +51,10 @@ static struct {
|
|||||||
{ "clear", cmd_clear },
|
{ "clear", cmd_clear },
|
||||||
{ "connect", cmd_connect },
|
{ "connect", cmd_connect },
|
||||||
{ "exit", cmd_quit },
|
{ "exit", cmd_quit },
|
||||||
|
{ "groupchat", cmd_groupchat },
|
||||||
{ "help", cmd_help },
|
{ "help", cmd_help },
|
||||||
|
{ "invite", cmd_invite },
|
||||||
|
{ "join", cmd_join },
|
||||||
{ "msg", cmd_msg },
|
{ "msg", cmd_msg },
|
||||||
{ "myid", cmd_myid },
|
{ "myid", cmd_myid },
|
||||||
{ "nick", cmd_nick },
|
{ "nick", cmd_nick },
|
||||||
@@ -55,17 +65,19 @@ static struct {
|
|||||||
};
|
};
|
||||||
|
|
||||||
/* Updates own nick in prompt statusbar */
|
/* Updates own nick in prompt statusbar */
|
||||||
void prompt_update_nick(ToxWindow *prompt, uint8_t *nick)
|
void prompt_update_nick(ToxWindow *prompt, uint8_t *nick, uint16_t len)
|
||||||
{
|
{
|
||||||
StatusBar *statusbar = (StatusBar *) prompt->stb;
|
StatusBar *statusbar = (StatusBar *) prompt->stb;
|
||||||
snprintf(statusbar->nick, sizeof(statusbar->nick), "%s", nick);
|
snprintf(statusbar->nick, sizeof(statusbar->nick), "%s", nick);
|
||||||
|
statusbar->nick_len = len;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Updates own statusmessage in prompt statusbar */
|
/* Updates own statusmessage in prompt statusbar */
|
||||||
void prompt_update_statusmessage(ToxWindow *prompt, uint8_t *statusmsg)
|
void prompt_update_statusmessage(ToxWindow *prompt, uint8_t *statusmsg, uint16_t len)
|
||||||
{
|
{
|
||||||
StatusBar *statusbar = (StatusBar *) prompt->stb;
|
StatusBar *statusbar = (StatusBar *) prompt->stb;
|
||||||
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
snprintf(statusbar->statusmsg, sizeof(statusbar->statusmsg), "%s", statusmsg);
|
||||||
|
statusbar->statusmsg_len = len;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Updates own status in prompt statusbar */
|
/* Updates own status in prompt statusbar */
|
||||||
@@ -82,30 +94,28 @@ void prompt_update_connectionstatus(ToxWindow *prompt, bool is_connected)
|
|||||||
statusbar->is_online = is_connected;
|
statusbar->is_online = is_connected;
|
||||||
}
|
}
|
||||||
|
|
||||||
void prompt_onFriendRequest(ToxWindow *prompt, uint8_t *key, uint8_t *data, uint16_t length)
|
/* Adds friend request to pending friend requests.
|
||||||
|
Returns friend number on success, -1 if queue is full or other error. */
|
||||||
|
int add_friend_req(uint8_t *public_key)
|
||||||
{
|
{
|
||||||
int n = add_req(key);
|
if (num_frnd_requests < MAX_FRIENDS_NUM) {
|
||||||
wprintw(prompt->window, "\nFriend request from:\n");
|
memcpy(pending_frnd_requests[num_frnd_requests++], public_key, TOX_CLIENT_ID_SIZE);
|
||||||
|
return num_frnd_requests - 1;
|
||||||
size_t i;
|
|
||||||
|
|
||||||
for (i = 0; i < KEY_SIZE_BYTES; ++i) {
|
|
||||||
wprintw(prompt->window, "%02x", key[i] & 0xff);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
wprintw(prompt->window, "\n\nWith the message: %s\n\n", data);
|
return -1;
|
||||||
wprintw(prompt->window, "Type \"accept %d\" to accept it.\n", n);
|
|
||||||
|
|
||||||
prompt->blink = true;
|
|
||||||
beep();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// XXX:
|
/* Adds group chat invite to pending group chat requests.
|
||||||
int add_req(uint8_t *public_key)
|
Returns group number on success, -1 if queue is full or other error. */
|
||||||
|
int add_group_req(uint8_t *group_pub_key)
|
||||||
{
|
{
|
||||||
memcpy(pending_requests[num_requests], public_key, TOX_CLIENT_ID_SIZE);
|
if (num_grp_requests < MAX_GROUPCHAT_NUM) {
|
||||||
++num_requests;
|
memcpy(pending_grp_requests[num_grp_requests++], group_pub_key, TOX_CLIENT_ID_SIZE);
|
||||||
return num_requests - 1;
|
return num_grp_requests - 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// XXX: FIX
|
// XXX: FIX
|
||||||
@@ -132,28 +142,26 @@ unsigned char *hex_string_to_bin(char hex_string[])
|
|||||||
/* command functions */
|
/* command functions */
|
||||||
void cmd_accept(ToxWindow *self, Tox *m, int argc, char **argv)
|
void cmd_accept(ToxWindow *self, Tox *m, int argc, char **argv)
|
||||||
{
|
{
|
||||||
int num;
|
|
||||||
|
|
||||||
/* check arguments */
|
/* check arguments */
|
||||||
if (argc != 1) {
|
if (argc != 1) {
|
||||||
wprintw(self->window, "Invalid syntax.\n");
|
wprintw(self->window, "Invalid syntax.\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
num = atoi(argv[1]);
|
int num = atoi(argv[1]);
|
||||||
|
|
||||||
if (num < 0 || num >= num_requests) {
|
if (num < 0 || num >= num_frnd_requests) {
|
||||||
wprintw(self->window, "No pending request with that number.\n");
|
wprintw(self->window, "No pending friend request with that number.\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
num = tox_addfriend_norequest(m, pending_requests[num]);
|
int friendnum = tox_addfriend_norequest(m, pending_frnd_requests[num]);
|
||||||
|
|
||||||
if (num == -1)
|
if (friendnum == -1)
|
||||||
wprintw(self->window, "Failed to add friend.\n");
|
wprintw(self->window, "Failed to add friend.\n");
|
||||||
else {
|
else {
|
||||||
wprintw(self->window, "Friend accepted as: %d.\n", num);
|
wprintw(self->window, "Friend request accepted.\n");
|
||||||
on_friendadded(m, num);
|
on_friendadded(m, friendnum);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -164,13 +172,6 @@ void cmd_add(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
uint8_t id_bin[TOX_FRIEND_ADDRESS_SIZE];
|
|
||||||
char xx[3];
|
|
||||||
uint32_t x;
|
|
||||||
uint8_t *msg;
|
|
||||||
size_t i;
|
|
||||||
int num;
|
|
||||||
|
|
||||||
char *id = argv[1];
|
char *id = argv[1];
|
||||||
|
|
||||||
if (id == NULL) {
|
if (id == NULL) {
|
||||||
@@ -178,6 +179,8 @@ void cmd_add(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
uint8_t *msg;
|
||||||
|
|
||||||
if (argc == 2) {
|
if (argc == 2) {
|
||||||
msg = argv[2];
|
msg = argv[2];
|
||||||
|
|
||||||
@@ -201,6 +204,11 @@ void cmd_add(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
size_t i;
|
||||||
|
char xx[3];
|
||||||
|
uint32_t x;
|
||||||
|
uint8_t id_bin[TOX_FRIEND_ADDRESS_SIZE];
|
||||||
|
|
||||||
for (i = 0; i < TOX_FRIEND_ADDRESS_SIZE; ++i) {
|
for (i = 0; i < TOX_FRIEND_ADDRESS_SIZE; ++i) {
|
||||||
xx[0] = id[2 * i];
|
xx[0] = id[2 * i];
|
||||||
xx[1] = id[2 * i + 1];
|
xx[1] = id[2 * i + 1];
|
||||||
@@ -218,7 +226,7 @@ void cmd_add(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
id[i] = toupper(id[i]);
|
id[i] = toupper(id[i]);
|
||||||
}
|
}
|
||||||
|
|
||||||
num = tox_addfriend(m, id_bin, msg, strlen(msg) + 1);
|
int num = tox_addfriend(m, id_bin, msg, strlen(msg) + 1);
|
||||||
|
|
||||||
switch (num) {
|
switch (num) {
|
||||||
case TOX_FAERR_TOOLONG:
|
case TOX_FAERR_TOOLONG:
|
||||||
@@ -275,7 +283,7 @@ void cmd_connect(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
char *port = argv[2];
|
char *port = argv[2];
|
||||||
char *key = argv[3];
|
char *key = argv[3];
|
||||||
|
|
||||||
if (!ip || !port || !key) {
|
if (ip == NULL || port == NULL || key == NULL) {
|
||||||
wprintw(self->window, "Invalid syntax.\n");
|
wprintw(self->window, "Invalid syntax.\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -296,6 +304,31 @@ void cmd_quit(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
exit_toxic(m);
|
exit_toxic(m);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void cmd_groupchat(ToxWindow *self, Tox *m, int argc, char **argv)
|
||||||
|
{
|
||||||
|
int ngc = get_num_groupchats();
|
||||||
|
|
||||||
|
if (ngc < 0 || ngc > MAX_GROUPCHAT_NUM) {
|
||||||
|
wprintw(self->window, "\nMaximum number of group chats has been reached.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int groupnum = tox_add_groupchat(m);
|
||||||
|
|
||||||
|
if (groupnum == -1) {
|
||||||
|
wprintw(self->window, "Group chat failed to initialize.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (init_groupchat_win(self, m, groupnum) == -1) {
|
||||||
|
wprintw(self->window, "Group chat failed to initialize.\n");
|
||||||
|
tox_del_groupchat(m, groupnum);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
wprintw(self->window, "Group chat created as %d.\n", groupnum);
|
||||||
|
}
|
||||||
|
|
||||||
void cmd_help(ToxWindow *self, Tox *m, int argc, char **argv)
|
void cmd_help(ToxWindow *self, Tox *m, int argc, char **argv)
|
||||||
{
|
{
|
||||||
wclear(self->window);
|
wclear(self->window);
|
||||||
@@ -305,10 +338,13 @@ void cmd_help(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
|
|
||||||
wprintw(self->window, " connect <ip> <port> <key> : Connect to DHT server\n");
|
wprintw(self->window, " connect <ip> <port> <key> : Connect to DHT server\n");
|
||||||
wprintw(self->window, " add <id> <message> : Add friend with optional message\n");
|
wprintw(self->window, " add <id> <message> : Add friend with optional message\n");
|
||||||
|
wprintw(self->window, " accept <n> : Accept friend request\n");
|
||||||
wprintw(self->window, " status <type> <message> : Set your status with optional note\n");
|
wprintw(self->window, " status <type> <message> : Set your status with optional note\n");
|
||||||
wprintw(self->window, " note <message> : Set a personal note\n");
|
wprintw(self->window, " note <message> : Set a personal note\n");
|
||||||
wprintw(self->window, " nick <nickname> : Set your nickname\n");
|
wprintw(self->window, " nick <nickname> : Set your nickname\n");
|
||||||
wprintw(self->window, " accept <number> : Accept friend request\n");
|
wprintw(self->window, " join <n> : Join a group chat\n");
|
||||||
|
wprintw(self->window, " invite <nickname> <n> : Invite friend to a groupchat\n");
|
||||||
|
wprintw(self->window, " groupchat : Create a group chat\n");
|
||||||
wprintw(self->window, " myid : Print your ID\n");
|
wprintw(self->window, " myid : Print your ID\n");
|
||||||
wprintw(self->window, " quit/exit : Exit Toxic\n");
|
wprintw(self->window, " quit/exit : Exit Toxic\n");
|
||||||
wprintw(self->window, " help : Print this message again\n");
|
wprintw(self->window, " help : Print this message again\n");
|
||||||
@@ -322,6 +358,72 @@ void cmd_help(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
wattroff(self->window, COLOR_PAIR(CYAN));
|
wattroff(self->window, COLOR_PAIR(CYAN));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void cmd_invite(ToxWindow *self, Tox *m, int argc, char **argv)
|
||||||
|
{
|
||||||
|
if (argc != 2) {
|
||||||
|
wprintw(self->window, "Invalid syntax.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (argv[1] == NULL || argv[2] == NULL) {
|
||||||
|
wprintw(self->window, "Invalid syntax.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
uint8_t *friendname = argv[1];
|
||||||
|
int groupnum = atoi(argv[2]);
|
||||||
|
|
||||||
|
if (friendname[0] == '\"')
|
||||||
|
friendname[strlen(++friendname)-1] = L'\0';
|
||||||
|
|
||||||
|
int friendnum = get_friendnum(friendname);
|
||||||
|
|
||||||
|
if (friendnum == -1) {
|
||||||
|
wprintw(self->window, "Friend '%s' not found.\n", friendname);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (tox_invite_friend(m, friendnum, groupnum) == -1) {
|
||||||
|
wprintw(self->window, "Failed to invite friend.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
wprintw(self->window, "Invited friend %s to group chat %d.\n", friendname, groupnum);
|
||||||
|
}
|
||||||
|
|
||||||
|
void cmd_join(ToxWindow *self, Tox *m, int argc, char **argv)
|
||||||
|
{
|
||||||
|
if (argc != 1) {
|
||||||
|
wprintw(self->window, "Invalid syntax.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (argv[1] == NULL) {
|
||||||
|
wprintw(self->window, "Invalid syntax.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int num = atoi(argv[1]);
|
||||||
|
|
||||||
|
if (num < 0 || num >= num_grp_requests) {
|
||||||
|
wprintw(self->window, "No pending group chat invites with that number.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int groupnum = tox_join_groupchat(m, num, pending_grp_requests[num]);
|
||||||
|
|
||||||
|
if (groupnum == -1) {
|
||||||
|
wprintw(self->window, "Group chat failed to initialize.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (init_groupchat_win(self, m, groupnum) == -1) {
|
||||||
|
wprintw(self->window, "Group chat failed to initialize.\n");
|
||||||
|
tox_del_groupchat(m, groupnum);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void cmd_msg(ToxWindow *self, Tox *m, int argc, char **argv)
|
void cmd_msg(ToxWindow *self, Tox *m, int argc, char **argv)
|
||||||
{
|
{
|
||||||
/* check arguments */
|
/* check arguments */
|
||||||
@@ -392,18 +494,13 @@ void cmd_nick(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
tox_setname(m, nick, len+1);
|
tox_setname(m, nick, len+1);
|
||||||
prompt_update_nick(self, nick);
|
prompt_update_nick(self, nick, len+1);
|
||||||
|
|
||||||
store_data(m, DATA_FILE);
|
store_data(m, DATA_FILE);
|
||||||
}
|
}
|
||||||
|
|
||||||
void cmd_status(ToxWindow *self, Tox *m, int argc, char **argv)
|
void cmd_status(ToxWindow *self, Tox *m, int argc, char **argv)
|
||||||
{
|
{
|
||||||
if (argc < 1 || argc > 2) {
|
|
||||||
wprintw(self->window, "Wrong number of arguments.\n");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
uint8_t *msg = NULL;
|
uint8_t *msg = NULL;
|
||||||
|
|
||||||
if (argc == 2) {
|
if (argc == 2) {
|
||||||
@@ -419,6 +516,9 @@ void cmd_status(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
wprintw(self->window, "Messages must be enclosed in quotes.\n");
|
wprintw(self->window, "Messages must be enclosed in quotes.\n");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
} else if (argc != 1) {
|
||||||
|
wprintw(self->window, "Wrong number of arguments.\n");
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *status = argv[1];
|
char *status = argv[1];
|
||||||
@@ -447,8 +547,9 @@ void cmd_status(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
|
|
||||||
if (msg != NULL) {
|
if (msg != NULL) {
|
||||||
msg[strlen(++msg)-1] = L'\0'; /* remove opening and closing quotes */
|
msg[strlen(++msg)-1] = L'\0'; /* remove opening and closing quotes */
|
||||||
tox_set_statusmessage(m, msg, strlen(msg) + 1);
|
uint16_t len = strlen(msg) + 1;
|
||||||
prompt_update_statusmessage(self, msg);
|
tox_set_statusmessage(m, msg, len);
|
||||||
|
prompt_update_statusmessage(self, msg, len);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -472,9 +573,9 @@ void cmd_note(ToxWindow *self, Tox *m, int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
msg[strlen(++msg)-1] = L'\0';
|
msg[strlen(++msg)-1] = L'\0';
|
||||||
|
uint16_t len = strlen(msg) + 1;
|
||||||
tox_set_statusmessage(m, msg, strlen(msg) + 1);
|
tox_set_statusmessage(m, msg, len);
|
||||||
prompt_update_statusmessage(self, msg);
|
prompt_update_statusmessage(self, msg, len);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void execute(ToxWindow *self, Tox *m, char *u_cmd)
|
static void execute(ToxWindow *self, Tox *m, char *u_cmd)
|
||||||
@@ -565,14 +666,14 @@ static void prompt_onKey(ToxWindow *self, Tox *m, wint_t key)
|
|||||||
/* Add printable characters to line */
|
/* Add printable characters to line */
|
||||||
if (isprint(key)) {
|
if (isprint(key)) {
|
||||||
if (prompt_buf_pos == (sizeof(prompt_buf) - 1)) {
|
if (prompt_buf_pos == (sizeof(prompt_buf) - 1)) {
|
||||||
wprintw(self->window, "\nToo Long.\n");
|
return;
|
||||||
prompt_buf_pos = 0;
|
|
||||||
prompt_buf[0] = 0;
|
|
||||||
} else if (!(prompt_buf_pos == 0) && (prompt_buf_pos < COLS)
|
} else if (!(prompt_buf_pos == 0) && (prompt_buf_pos < COLS)
|
||||||
&& (prompt_buf_pos % (COLS - 3) == 0)) {
|
&& (prompt_buf_pos % (COLS - 3) == 0)) {
|
||||||
|
wprintw(self->window, "\n");
|
||||||
prompt_buf[prompt_buf_pos++] = '\n';
|
prompt_buf[prompt_buf_pos++] = '\n';
|
||||||
} else if (!(prompt_buf_pos == 0) && (prompt_buf_pos > COLS)
|
} else if (!(prompt_buf_pos == 0) && (prompt_buf_pos > COLS)
|
||||||
&& ((prompt_buf_pos - (COLS - 3)) % (COLS) == 0)) {
|
&& ((prompt_buf_pos - (COLS - 3)) % (COLS) == 0)) {
|
||||||
|
wprintw(self->window, "\n");
|
||||||
prompt_buf[prompt_buf_pos++] = '\n';
|
prompt_buf[prompt_buf_pos++] = '\n';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -665,6 +766,63 @@ static void prompt_onInit(ToxWindow *self, Tox *m)
|
|||||||
wclrtoeol(self->window);
|
wclrtoeol(self->window);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void prompt_onFriendRequest(ToxWindow *self, uint8_t *key, uint8_t *data, uint16_t length)
|
||||||
|
{
|
||||||
|
int n = add_friend_req(key);
|
||||||
|
|
||||||
|
if (n == -1) {
|
||||||
|
wprintw(self->window, "Friend request queue is full. Discarding request.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
wprintw(self->window, "\nFriend request from:\n");
|
||||||
|
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < KEY_SIZE_BYTES; ++i) {
|
||||||
|
wprintw(self->window, "%02x", key[i] & 0xff);
|
||||||
|
}
|
||||||
|
|
||||||
|
wprintw(self->window, "\n\nWith the message: %s\n\n", data);
|
||||||
|
wprintw(self->window, "Type \"accept %d\" to accept it.\n", n);
|
||||||
|
|
||||||
|
self->blink = true;
|
||||||
|
beep();
|
||||||
|
}
|
||||||
|
|
||||||
|
void prompt_onGroupInvite(ToxWindow *self, Tox *m, int friendnumber, uint8_t *group_pub_key)
|
||||||
|
{
|
||||||
|
if (friendnumber < 0)
|
||||||
|
return;
|
||||||
|
|
||||||
|
uint8_t name[TOX_MAX_NAME_LENGTH] = {'\0'};
|
||||||
|
|
||||||
|
if (tox_getname(m, friendnumber, name) == -1)
|
||||||
|
return;
|
||||||
|
|
||||||
|
name[TOXIC_MAX_NAME_LENGTH] = '\0'; /* enforce client max name length */
|
||||||
|
wprintw(self->window, "\nGroup chat invite from %s.\n", name);
|
||||||
|
|
||||||
|
int ngc = get_num_groupchats();
|
||||||
|
|
||||||
|
if (ngc < 0 || ngc > MAX_GROUPCHAT_NUM) {
|
||||||
|
wprintw(self->window, "\nMaximum number of group chats has been reached. Discarding invite.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int n = add_group_req(group_pub_key);
|
||||||
|
|
||||||
|
if (n == -1) {
|
||||||
|
wprintw(self->window, "\nGroup chat queue is full. Discarding invite.\n");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
wprintw(self->window, "Type \"join %d\" to join the chat.\n", n);
|
||||||
|
|
||||||
|
self->blink = true;
|
||||||
|
beep();
|
||||||
|
}
|
||||||
|
|
||||||
void prompt_init_statusbar(ToxWindow *self, Tox *m)
|
void prompt_init_statusbar(ToxWindow *self, Tox *m)
|
||||||
{
|
{
|
||||||
int x, y;
|
int x, y;
|
||||||
@@ -692,10 +850,13 @@ ToxWindow new_prompt()
|
|||||||
{
|
{
|
||||||
ToxWindow ret;
|
ToxWindow ret;
|
||||||
memset(&ret, 0, sizeof(ret));
|
memset(&ret, 0, sizeof(ret));
|
||||||
|
|
||||||
ret.onKey = &prompt_onKey;
|
ret.onKey = &prompt_onKey;
|
||||||
ret.onDraw = &prompt_onDraw;
|
ret.onDraw = &prompt_onDraw;
|
||||||
ret.onInit = &prompt_onInit;
|
ret.onInit = &prompt_onInit;
|
||||||
ret.onFriendRequest = &prompt_onFriendRequest;
|
ret.onFriendRequest = &prompt_onFriendRequest;
|
||||||
|
ret.onGroupInvite = &prompt_onGroupInvite;
|
||||||
|
|
||||||
strcpy(ret.name, "prompt");
|
strcpy(ret.name, "prompt");
|
||||||
|
|
||||||
StatusBar *stb = calloc(1, sizeof(StatusBar));
|
StatusBar *stb = calloc(1, sizeof(StatusBar));
|
||||||
|
@@ -7,5 +7,9 @@ ToxWindow new_prompt();
|
|||||||
int add_req(uint8_t *public_key);
|
int add_req(uint8_t *public_key);
|
||||||
unsigned char *hex_string_to_bin(char hex_string[]);
|
unsigned char *hex_string_to_bin(char hex_string[]);
|
||||||
void prompt_init_statusbar(ToxWindow *self, Tox *m);
|
void prompt_init_statusbar(ToxWindow *self, Tox *m);
|
||||||
|
void prompt_update_nick(ToxWindow *prompt, uint8_t *nick, uint16_t len);
|
||||||
|
void prompt_update_statusmessage(ToxWindow *prompt, uint8_t *statusmsg, uint16_t len);
|
||||||
|
void prompt_update_status(ToxWindow *prompt, TOX_USERSTATUS status);
|
||||||
|
void prompt_update_connectionstatus(ToxWindow *prompt, bool is_connected);
|
||||||
|
|
||||||
#endif /* end of include guard: PROMPT_H_UZYGWFFL */
|
#endif /* end of include guard: PROMPT_H_UZYGWFFL */
|
||||||
|
@@ -1,9 +1,14 @@
|
|||||||
/*
|
/*
|
||||||
* Toxic -- Tox Curses Client
|
* Toxic -- Tox Curses Client
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef _windows_h
|
#ifndef _windows_h
|
||||||
#define _windows_h
|
#define _windows_h
|
||||||
|
|
||||||
|
#ifndef TOXICVER
|
||||||
|
#define TOXICVER "NOVER" /* Use the -D flag to set this */
|
||||||
|
#endif
|
||||||
|
|
||||||
#include <curses.h>
|
#include <curses.h>
|
||||||
#include <stdint.h>
|
#include <stdint.h>
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
@@ -14,17 +19,15 @@
|
|||||||
|
|
||||||
#define MAX_WINDOWS_NUM 32
|
#define MAX_WINDOWS_NUM 32
|
||||||
#define MAX_FRIENDS_NUM 100
|
#define MAX_FRIENDS_NUM 100
|
||||||
|
#define MAX_GROUPCHAT_NUM MAX_WINDOWS_NUM - N_DEFAULT_WINS
|
||||||
#define MAX_STR_SIZE 256
|
#define MAX_STR_SIZE 256
|
||||||
#define KEY_SIZE_BYTES 32
|
#define KEY_SIZE_BYTES 32
|
||||||
#define TOXIC_MAX_NAME_LENGTH 30 /* Not to be confused with TOX_MAX_NAME_LENGTH */
|
#define TOXIC_MAX_NAME_LENGTH 30 /* Must be <= TOX_MAX_NAME_LENGTH */
|
||||||
#define N_DEFAULT_WINS 3 /* number of permanent default windows */
|
#define N_DEFAULT_WINS 2 /* number of permanent default windows */
|
||||||
#define UNKNOWN_NAME "Unknown"
|
#define UNKNOWN_NAME "Unknown"
|
||||||
#define EXIT_SUCCESS 0
|
#define EXIT_SUCCESS 0
|
||||||
#define EXIT_FAILURE 1
|
#define EXIT_FAILURE 1
|
||||||
|
#define CURS_Y_OFFSET 3 /* y-axis cursor offset for chat contexts */
|
||||||
#ifndef TOXICVER
|
|
||||||
#define TOXICVER "NOVER" //Use the -D flag to set this
|
|
||||||
#endif
|
|
||||||
|
|
||||||
/* Curses foreground colours (background is black) */
|
/* Curses foreground colours (background is black) */
|
||||||
#define WHITE 0
|
#define WHITE 0
|
||||||
@@ -49,9 +52,11 @@ struct ToxWindow_ {
|
|||||||
void(*onStatusChange)(ToxWindow *, Tox *, int, TOX_USERSTATUS);
|
void(*onStatusChange)(ToxWindow *, Tox *, int, TOX_USERSTATUS);
|
||||||
void(*onStatusMessageChange)(ToxWindow *, int, uint8_t *, uint16_t);
|
void(*onStatusMessageChange)(ToxWindow *, int, uint8_t *, uint16_t);
|
||||||
void(*onAction)(ToxWindow *, Tox *, int, uint8_t *, uint16_t);
|
void(*onAction)(ToxWindow *, Tox *, int, uint8_t *, uint16_t);
|
||||||
|
void(*onGroupMessage)(ToxWindow *, Tox *, int, int, uint8_t *, uint16_t);
|
||||||
|
void(*onGroupInvite)(ToxWindow *, Tox *, int, uint8_t *);
|
||||||
|
|
||||||
char name[TOX_MAX_NAME_LENGTH];
|
char name[TOX_MAX_NAME_LENGTH];
|
||||||
int friendnum;
|
int num;
|
||||||
int x;
|
int x;
|
||||||
|
|
||||||
void *chatwin;
|
void *chatwin;
|
||||||
@@ -68,10 +73,23 @@ typedef struct {
|
|||||||
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH];
|
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH];
|
||||||
uint16_t statusmsg_len;
|
uint16_t statusmsg_len;
|
||||||
uint8_t nick[TOX_MAX_NAME_LENGTH];
|
uint8_t nick[TOX_MAX_NAME_LENGTH];
|
||||||
|
uint16_t nick_len;
|
||||||
TOX_USERSTATUS status;
|
TOX_USERSTATUS status;
|
||||||
bool is_online;
|
bool is_online;
|
||||||
} StatusBar;
|
} StatusBar;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
wchar_t line[MAX_STR_SIZE];
|
||||||
|
size_t pos;
|
||||||
|
WINDOW *history;
|
||||||
|
WINDOW *linewin;
|
||||||
|
} ChatContext;
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
int chatwin;
|
||||||
|
bool active;
|
||||||
|
} GroupChat;
|
||||||
|
|
||||||
void on_request(uint8_t *public_key, uint8_t *data, uint16_t length, void *userdata);
|
void on_request(uint8_t *public_key, uint8_t *data, uint16_t length, void *userdata);
|
||||||
void on_connectionchange(Tox *m, int friendnumber, uint8_t status, void *userdata);
|
void on_connectionchange(Tox *m, int friendnumber, uint8_t status, void *userdata);
|
||||||
void on_message(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata);
|
void on_message(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata);
|
||||||
@@ -80,6 +98,8 @@ void on_nickchange(Tox *m, int friendnumber, uint8_t *string, uint16_t length, v
|
|||||||
void on_statuschange(Tox *m, int friendnumber, TOX_USERSTATUS status, void *userdata);
|
void on_statuschange(Tox *m, int friendnumber, TOX_USERSTATUS status, void *userdata);
|
||||||
void on_statusmessagechange(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata);
|
void on_statusmessagechange(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata);
|
||||||
void on_friendadded(Tox *m, int friendnumber);
|
void on_friendadded(Tox *m, int friendnumber);
|
||||||
|
void on_groupmessage(Tox *m, int groupnumber, int peernumber, uint8_t *message, uint16_t length, void *userdata);
|
||||||
|
void on_groupinvite(Tox *m, int friendnumber, uint8_t *group_pub_key, void *userdata);
|
||||||
ToxWindow *init_windows();
|
ToxWindow *init_windows();
|
||||||
void draw_active_window(Tox *m);
|
void draw_active_window(Tox *m);
|
||||||
int add_window(Tox *m, ToxWindow w);
|
int add_window(Tox *m, ToxWindow w);
|
||||||
|
@@ -124,6 +124,26 @@ void on_friendadded(Tox *m, int friendnumber)
|
|||||||
if (store_data(m, DATA_FILE))
|
if (store_data(m, DATA_FILE))
|
||||||
wprintw(prompt->window, "\nCould not store Tox data\n");
|
wprintw(prompt->window, "\nCould not store Tox data\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void on_groupmessage(Tox *m, int groupnumber, int peernumber, uint8_t *message, uint16_t length, void *userdata)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
||||||
|
if (windows[i].onGroupMessage != NULL)
|
||||||
|
windows[i].onGroupMessage(&windows[i], m, groupnumber, peernumber, message, length);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void on_groupinvite(Tox *m, int friendnumber, uint8_t *group_pub_key, void *userdata)
|
||||||
|
{
|
||||||
|
int i;
|
||||||
|
|
||||||
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
||||||
|
if (windows[i].onGroupInvite != NULL)
|
||||||
|
windows[i].onGroupInvite(&windows[i], m, friendnumber, group_pub_key);
|
||||||
|
}
|
||||||
|
}
|
||||||
/* CALLBACKS END */
|
/* CALLBACKS END */
|
||||||
|
|
||||||
int add_window(Tox *m, ToxWindow w)
|
int add_window(Tox *m, ToxWindow w)
|
||||||
@@ -262,7 +282,6 @@ void prepare_window(WINDOW *w)
|
|||||||
|
|
||||||
void draw_active_window(Tox *m)
|
void draw_active_window(Tox *m)
|
||||||
{
|
{
|
||||||
|
|
||||||
ToxWindow *a = active_window;
|
ToxWindow *a = active_window;
|
||||||
wint_t ch = 0;
|
wint_t ch = 0;
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user