2013-08-23 09:50:04 +02:00
|
|
|
#ifdef HAVE_CONFIG_H
|
|
|
|
#include "config.h"
|
|
|
|
#endif
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2013-08-13 02:21:03 +02:00
|
|
|
#include "friendlist.h"
|
|
|
|
#include "prompt.h"
|
2013-08-23 00:18:53 +02:00
|
|
|
#include "toxic_windows.h"
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-15 12:11:48 +02:00
|
|
|
extern char *DATA_FILE;
|
2013-08-23 23:03:44 +02:00
|
|
|
extern int store_data(Tox *m, char *path);
|
2013-08-15 12:11:48 +02:00
|
|
|
|
2013-08-17 15:40:30 +02:00
|
|
|
static ToxWindow windows[MAX_WINDOWS_NUM];
|
2013-08-17 10:00:19 +02:00
|
|
|
static ToxWindow *active_window;
|
2013-08-16 19:11:09 +02:00
|
|
|
static ToxWindow *prompt;
|
2013-08-23 23:03:44 +02:00
|
|
|
static Tox *m;
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-09-05 06:47:33 +02:00
|
|
|
#define UNKNOWN_NAME "Unknown"
|
2013-09-05 03:25:59 +02:00
|
|
|
|
2013-08-13 02:21:03 +02:00
|
|
|
/* CALLBACKS START */
|
2013-08-16 19:11:09 +02:00
|
|
|
void on_request(uint8_t *public_key, uint8_t *data, uint16_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
int i;
|
2013-09-06 08:51:10 +02:00
|
|
|
|
2013-08-17 15:40:30 +02:00
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2013-08-16 19:11:09 +02:00
|
|
|
if (windows[i].onFriendRequest != NULL)
|
|
|
|
windows[i].onFriendRequest(&windows[i], public_key, data, length);
|
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-09-05 03:25:59 +02:00
|
|
|
void on_connectionchange(Tox *m, int friendnumber, uint8_t status, void *userdata)
|
|
|
|
{
|
2013-09-08 09:18:34 +02:00
|
|
|
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
|
2013-09-05 03:25:59 +02:00
|
|
|
tox_getname(m, friendnumber, (uint8_t *) &nick);
|
|
|
|
|
|
|
|
if (!nick[0])
|
2013-09-05 06:47:33 +02:00
|
|
|
snprintf(nick, sizeof(nick), "%s", UNKNOWN_NAME);
|
2013-09-05 03:25:59 +02:00
|
|
|
|
2013-09-05 07:34:23 +02:00
|
|
|
if (status == 1) {
|
2013-09-06 08:51:10 +02:00
|
|
|
wattron(prompt->window, COLOR_PAIR(GREEN));
|
2013-09-05 07:34:23 +02:00
|
|
|
wattron(prompt->window, A_BOLD);
|
|
|
|
wprintw(prompt->window, "\n%s ", nick);
|
|
|
|
wattroff(prompt->window, A_BOLD);
|
|
|
|
wprintw(prompt->window, "has come online\n");
|
2013-09-06 08:51:10 +02:00
|
|
|
wattroff(prompt->window, COLOR_PAIR(GREEN));
|
2013-09-05 07:34:23 +02:00
|
|
|
} else {
|
2013-09-06 08:51:10 +02:00
|
|
|
wattron(prompt->window, COLOR_PAIR(RED));
|
2013-09-05 07:34:23 +02:00
|
|
|
wattron(prompt->window, A_BOLD);
|
|
|
|
wprintw(prompt->window, "\n%s ", nick);
|
|
|
|
wattroff(prompt->window, A_BOLD);
|
|
|
|
wprintw(prompt->window, "has gone offline\n");
|
2013-09-06 08:51:10 +02:00
|
|
|
wattroff(prompt->window, COLOR_PAIR(RED));
|
2013-09-05 07:34:23 +02:00
|
|
|
}
|
2013-09-05 03:25:59 +02:00
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i].onConnectionChange != NULL)
|
|
|
|
windows[i].onConnectionChange(&windows[i], m, friendnumber, status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
void on_message(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
int i;
|
|
|
|
|
2013-08-17 15:40:30 +02:00
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2013-08-16 19:11:09 +02:00
|
|
|
if (windows[i].onMessage != NULL)
|
|
|
|
windows[i].onMessage(&windows[i], m, friendnumber, string, length);
|
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
void on_action(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
int i;
|
|
|
|
|
2013-08-17 15:40:30 +02:00
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2013-08-16 19:11:09 +02:00
|
|
|
if (windows[i].onAction != NULL)
|
|
|
|
windows[i].onAction(&windows[i], m, friendnumber, string, length);
|
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
void on_nickchange(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
int i;
|
|
|
|
|
2013-08-17 15:40:30 +02:00
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2013-09-04 08:05:36 +02:00
|
|
|
if (windows[i].onNickChange != NULL)
|
2013-08-16 19:11:09 +02:00
|
|
|
windows[i].onNickChange(&windows[i], friendnumber, string, length);
|
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-09-03 05:27:34 +02:00
|
|
|
void on_statusmessagechange(Tox *m, int friendnumber, uint8_t *string, uint16_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
int i;
|
|
|
|
|
2013-08-17 15:40:30 +02:00
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2013-09-03 05:27:34 +02:00
|
|
|
if (windows[i].onStatusMessageChange != NULL)
|
|
|
|
windows[i].onStatusMessageChange(&windows[i], friendnumber, string, length);
|
2013-08-16 19:11:09 +02:00
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-09-05 03:25:59 +02:00
|
|
|
void on_statuschange(Tox *m, int friendnumber, TOX_USERSTATUS status, void *userdata)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i].onStatusChange != NULL)
|
|
|
|
windows[i].onStatusChange(&windows[i], m, friendnumber, status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
void on_friendadded(Tox *m, int friendnumber)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
friendlist_onFriendAdded(m, friendnumber);
|
|
|
|
|
2013-09-02 04:11:47 +02:00
|
|
|
if (store_data(m, DATA_FILE))
|
2013-08-23 23:03:44 +02:00
|
|
|
wprintw(prompt->window, "\nCould not store Tox data\n");
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
/* CALLBACKS END */
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
int add_window(Tox *m, ToxWindow w)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
if (LINES < 2)
|
|
|
|
return -1;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
int i;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; i++) {
|
|
|
|
if (windows[i].window)
|
2013-08-17 10:00:19 +02:00
|
|
|
continue;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
w.window = newwin(LINES - 2, COLS, 0, 0);
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
if (w.window == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
windows[i] = w;
|
|
|
|
w.onInit(&w, m);
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
return i;
|
|
|
|
}
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
return -1;
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Deletes window w and cleans up */
|
2013-08-17 10:00:19 +02:00
|
|
|
void del_window(ToxWindow *w)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-17 10:00:19 +02:00
|
|
|
active_window = windows; // Go to prompt screen
|
2013-08-31 08:22:07 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
delwin(w->window);
|
2013-08-17 10:00:19 +02:00
|
|
|
memset(w, 0, sizeof(ToxWindow));
|
2013-08-31 02:13:29 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
clear();
|
|
|
|
refresh();
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Shows next window when tab or back-tab is pressed */
|
2013-08-17 10:00:19 +02:00
|
|
|
void set_next_window(int ch)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-21 01:37:05 +02:00
|
|
|
ToxWindow *end = windows + MAX_WINDOWS_NUM - 1;
|
2013-08-17 10:00:19 +02:00
|
|
|
ToxWindow *inf = active_window;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
|
|
|
while (true) {
|
2013-08-17 10:00:19 +02:00
|
|
|
if (ch == '\t') {
|
|
|
|
if (++active_window > end)
|
|
|
|
active_window = windows;
|
2013-08-21 01:37:05 +02:00
|
|
|
} else if (--active_window < windows)
|
|
|
|
active_window = end;
|
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
if (active_window->window)
|
|
|
|
return;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
if (active_window == inf) { // infinite loop check
|
|
|
|
endwin();
|
|
|
|
exit(2);
|
2013-08-16 19:11:09 +02:00
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
void set_active_window(int index)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-17 15:40:30 +02:00
|
|
|
if (index < 0 || index >= MAX_WINDOWS_NUM)
|
2013-08-17 10:00:19 +02:00
|
|
|
return;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
|
|
|
active_window = windows + index;
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-08-14 22:02:21 +02:00
|
|
|
ToxWindow *init_windows()
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2013-08-17 16:05:32 +02:00
|
|
|
int n_prompt = add_window(m, new_prompt());
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-28 00:30:38 +02:00
|
|
|
if (n_prompt == -1 || add_window(m, new_friendlist()) == -1) {
|
2013-08-16 19:11:09 +02:00
|
|
|
fprintf(stderr, "add_window() failed.\n");
|
|
|
|
endwin();
|
|
|
|
exit(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
prompt = &windows[n_prompt];
|
2013-08-17 10:00:19 +02:00
|
|
|
active_window = prompt;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
return prompt;
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
static void draw_bar()
|
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
static int odd = 0;
|
|
|
|
int blinkrate = 30;
|
|
|
|
|
2013-09-05 06:47:33 +02:00
|
|
|
attron(COLOR_PAIR(BLUE));
|
2013-08-16 19:11:09 +02:00
|
|
|
mvhline(LINES - 2, 0, '_', COLS);
|
2013-09-05 06:47:33 +02:00
|
|
|
attroff(COLOR_PAIR(BLUE));
|
2013-08-16 19:11:09 +02:00
|
|
|
|
|
|
|
move(LINES - 1, 0);
|
|
|
|
|
2013-09-05 06:47:33 +02:00
|
|
|
attron(COLOR_PAIR(BLUE) | A_BOLD);
|
2013-08-31 08:22:07 +02:00
|
|
|
printw(" TOXIC " TOXICVER " |");
|
2013-09-05 06:47:33 +02:00
|
|
|
attroff(COLOR_PAIR(BLUE) | A_BOLD);
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
int i;
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-28 11:46:09 +02:00
|
|
|
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2013-08-17 10:00:19 +02:00
|
|
|
if (windows[i].window) {
|
2013-08-21 01:37:05 +02:00
|
|
|
if (windows + i == active_window)
|
2013-08-16 19:11:09 +02:00
|
|
|
attron(A_BOLD);
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
odd = (odd + 1) % blinkrate;
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
if (windows[i].blink && (odd < (blinkrate / 2)))
|
2013-09-05 06:47:33 +02:00
|
|
|
attron(COLOR_PAIR(RED));
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-21 01:37:05 +02:00
|
|
|
clrtoeol();
|
2013-09-06 00:24:58 +02:00
|
|
|
printw(" [%s]", windows[i].name);
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
if (windows[i].blink && (odd < (blinkrate / 2)))
|
2013-09-05 06:47:33 +02:00
|
|
|
attroff(COLOR_PAIR(RED));
|
2013-08-13 02:21:03 +02:00
|
|
|
|
2013-08-21 01:37:05 +02:00
|
|
|
if (windows + i == active_window) {
|
2013-08-16 19:11:09 +02:00
|
|
|
attroff(A_BOLD);
|
|
|
|
}
|
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
|
|
|
|
refresh();
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void prepare_window(WINDOW *w)
|
|
|
|
{
|
2013-08-16 19:11:09 +02:00
|
|
|
mvwin(w, 0, 0);
|
|
|
|
wresize(w, LINES - 2, COLS);
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
void draw_active_window(Tox *m)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
|
|
|
|
2013-08-17 10:00:19 +02:00
|
|
|
ToxWindow *a = active_window;
|
2013-08-21 23:19:35 +02:00
|
|
|
wint_t ch = 0;
|
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
prepare_window(a->window);
|
|
|
|
a->blink = false;
|
|
|
|
draw_bar();
|
2013-08-20 20:47:32 +02:00
|
|
|
a->onDraw(a, m);
|
2013-08-16 19:11:09 +02:00
|
|
|
|
|
|
|
/* Handle input */
|
2013-08-22 22:57:34 +02:00
|
|
|
#ifdef HAVE_WIDECHAR
|
2013-08-30 04:56:27 +02:00
|
|
|
wget_wch(stdscr, &ch);
|
2013-08-22 22:57:34 +02:00
|
|
|
#else
|
|
|
|
ch = getch();
|
|
|
|
#endif
|
2013-08-16 19:11:09 +02:00
|
|
|
|
|
|
|
if (ch == '\t' || ch == KEY_BTAB)
|
2013-08-21 23:19:35 +02:00
|
|
|
set_next_window((int) ch);
|
2013-08-16 19:11:09 +02:00
|
|
|
else if (ch != ERR)
|
|
|
|
a->onKey(a, m, ch);
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|