1
0
mirror of https://github.com/Tha14/toxic.git synced 2024-11-15 02:23:04 +01:00
toxic/src/windows.c

428 lines
11 KiB
C
Raw Normal View History

2014-02-25 08:28:24 +01:00
/* windows.c
*
*
* Copyright (C) 2014 Toxic All Rights Reserved.
*
* This file is part of Toxic.
*
* Toxic is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* Toxic is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with Toxic. If not, see <http://www.gnu.org/licenses/>.
*
*/
#ifdef HAVE_CONFIG_H
#include "config.h"
#endif
2013-08-23 23:03:44 +02:00
#include <stdlib.h>
#include <string.h>
2014-03-13 11:06:53 +01:00
#include <pthread.h>
2013-08-23 23:03:44 +02:00
2013-08-13 02:21:03 +02:00
#include "friendlist.h"
#include "prompt.h"
#include "toxic_windows.h"
2014-02-26 11:23:11 +01:00
#include "groupchat.h"
2013-08-13 02:21:03 +02:00
2013-08-15 12:11:48 +02:00
extern char *DATA_FILE;
2014-03-13 11:06:53 +01:00
extern struct _Winthread Winthread;
static ToxWindow windows[MAX_WINDOWS_NUM];
static ToxWindow *active_window;
extern ToxWindow *prompt;
2013-08-13 02:21:03 +02:00
2014-03-07 01:39:57 +01:00
static int num_active_windows;
2013-08-13 02:21:03 +02:00
/* CALLBACKS START */
2014-03-19 02:48:26 +01:00
void on_request(Tox *m, 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;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-08-16 19:11:09 +02:00
if (windows[i].onFriendRequest != NULL)
2014-03-19 02:48:26 +01:00
windows[i].onFriendRequest(&windows[i], m, public_key, data, length);
2013-08-16 19:11:09 +02:00
}
2013-08-13 02:21:03 +02:00
}
2014-03-19 02:48:26 +01:00
void on_connectionchange(Tox *m, int32_t friendnumber, uint8_t status, void *userdata)
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
if (windows[i].onConnectionChange != NULL)
windows[i].onConnectionChange(&windows[i], m, friendnumber, status);
}
}
2014-03-19 02:48:26 +01:00
void on_typing_change(Tox *m, int32_t friendnumber, int is_typing, void *userdata)
2014-02-23 10:28:33 +01:00
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2014-02-23 10:28:33 +01:00
if (windows[i].onTypingChange != NULL)
windows[i].onTypingChange(&windows[i], m, friendnumber, is_typing);
}
}
2014-03-19 02:48:26 +01:00
void on_message(Tox *m, int32_t 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;
2014-03-18 03:18:04 +01: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
}
2014-03-19 02:48:26 +01:00
void on_action(Tox *m, int32_t 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;
2014-03-18 03:18:04 +01: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
}
2014-03-19 02:48:26 +01:00
void on_nickchange(Tox *m, int32_t friendnumber, uint8_t *string, uint16_t length, void *userdata)
2013-08-13 02:21:03 +02:00
{
2013-09-23 02:25:48 +02:00
if (friendnumber < 0 || friendnumber > MAX_FRIENDS_NUM)
return;
2013-08-16 19:11:09 +02:00
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-09-04 08:05:36 +02:00
if (windows[i].onNickChange != NULL)
windows[i].onNickChange(&windows[i], m, friendnumber, string, length);
2013-08-16 19:11:09 +02:00
}
2013-09-09 06:56:47 +02:00
if (store_data(m, DATA_FILE))
wprintw(prompt->window, "\nCould not store Tox data\n");
2013-08-13 02:21:03 +02:00
}
2014-03-19 02:48:26 +01:00
void on_statusmessagechange(Tox *m, int32_t 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;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
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
}
2014-03-19 02:48:26 +01:00
void on_statuschange(Tox *m, int32_t friendnumber, uint8_t status, void *userdata)
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
if (windows[i].onStatusChange != NULL)
windows[i].onStatusChange(&windows[i], m, friendnumber, status);
}
}
2014-03-19 02:48:26 +01:00
void on_friendadded(Tox *m, int32_t friendnumber, bool sort)
2013-08-13 02:21:03 +02:00
{
2013-11-15 20:59:49 +01:00
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-11-15 20:59:49 +01:00
if (windows[i].onFriendAdded != NULL)
windows[i].onFriendAdded(&windows[i], m, friendnumber, sort);
2013-11-15 20:59:49 +01:00
}
2013-08-16 19:11:09 +02:00
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
}
2013-09-15 22:38:38 +02:00
2013-10-11 10:42:30 +02:00
void on_groupmessage(Tox *m, int groupnumber, int peernumber, uint8_t *message, uint16_t length,
void *userdata)
2013-09-15 22:38:38 +02:00
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-09-15 22:38:38 +02:00
if (windows[i].onGroupMessage != NULL)
2013-09-18 02:54:25 +02:00
windows[i].onGroupMessage(&windows[i], m, groupnumber, peernumber, message, length);
2013-09-15 22:38:38 +02:00
}
}
2013-12-14 02:57:32 +01:00
void on_groupaction(Tox *m, int groupnumber, int peernumber, uint8_t *action, uint16_t length,
void *userdata)
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-12-14 02:57:32 +01:00
if (windows[i].onGroupAction != NULL)
windows[i].onGroupAction(&windows[i], m, groupnumber, peernumber, action, length);
}
}
2014-03-19 02:48:26 +01:00
void on_groupinvite(Tox *m, int32_t friendnumber, uint8_t *group_pub_key, void *userdata)
2013-09-15 22:38:38 +02:00
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-09-15 22:38:38 +02:00
if (windows[i].onGroupInvite != NULL)
windows[i].onGroupInvite(&windows[i], m, friendnumber, group_pub_key);
}
}
2013-10-10 10:52:05 +02:00
2013-11-26 00:49:31 +01:00
void on_group_namelistchange(Tox *m, int groupnumber, int peernumber, uint8_t change, void *userdata)
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
if (windows[i].onGroupNamelistChange != NULL)
2013-11-26 00:49:31 +01:00
windows[i].onGroupNamelistChange(&windows[i], m, groupnumber, peernumber, change);
}
}
2014-03-19 02:48:26 +01:00
void on_file_sendrequest(Tox *m, int32_t friendnumber, uint8_t filenumber, uint64_t filesize,
2013-10-10 10:52:05 +02:00
uint8_t *filename, uint16_t filename_length, void *userdata)
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-10-10 10:52:05 +02:00
if (windows[i].onFileSendRequest != NULL)
windows[i].onFileSendRequest(&windows[i], m, friendnumber, filenumber, filesize,
filename, filename_length);
}
}
2014-03-19 02:48:26 +01:00
void on_file_control (Tox *m, int32_t friendnumber, uint8_t receive_send, uint8_t filenumber,
2013-10-10 10:52:05 +02:00
uint8_t control_type, uint8_t *data, uint16_t length, void *userdata)
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-10-10 10:52:05 +02:00
if (windows[i].onFileControl != NULL)
windows[i].onFileControl(&windows[i], m, friendnumber, receive_send, filenumber,
control_type, data, length);
}
}
2014-03-19 02:48:26 +01:00
void on_file_data(Tox *m, int32_t friendnumber, uint8_t filenumber, uint8_t *data, uint16_t length,
2013-10-11 10:42:30 +02:00
void *userdata)
2013-10-10 10:52:05 +02:00
{
int i;
2014-03-18 03:18:04 +01:00
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
2013-10-10 10:52:05 +02:00
if (windows[i].onFileData != NULL)
windows[i].onFileData(&windows[i], m, friendnumber, filenumber, data, length);
}
}
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;
int i;
for (i = 0; i < MAX_WINDOWS_NUM; i++) {
if (windows[i].active)
continue;
w.window = newwin(LINES - 2, COLS, 0, 0);
if (w.window == NULL)
return -1;
#ifdef URXVT_FIX
/* Fixes text color problem on some terminals. */
wbkgd(w.window, COLOR_PAIR(6));
#endif
windows[i] = w;
w.onInit(&w, m);
2014-03-07 01:39:57 +01:00
++num_active_windows;
return i;
}
return -1;
2013-08-13 02:21:03 +02:00
}
/* Deletes window w and cleans up */
void del_window(ToxWindow *w)
2013-08-13 02:21:03 +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);
memset(w, 0, sizeof(ToxWindow));
2013-08-16 19:11:09 +02:00
clear();
refresh();
2014-03-07 01:39:57 +01:00
--num_active_windows;
2013-08-13 02:21:03 +02:00
}
/* Shows next window when tab or back-tab is pressed */
2013-12-07 04:07:37 +01:00
void set_next_window(int ch)
2013-08-13 02:21:03 +02:00
{
ToxWindow *end = windows + MAX_WINDOWS_NUM - 1;
ToxWindow *inf = active_window;
while (true) {
if (ch == T_KEY_NEXT) {
if (++active_window > end)
active_window = windows;
} else if (--active_window < windows)
active_window = end;
if (active_window->window)
return;
if (active_window == inf) { /* infinite loop check */
endwin();
fprintf(stderr, "set_next_window() failed. Aborting...\n");
exit(EXIT_FAILURE);
2013-08-16 19:11:09 +02:00
}
2013-08-13 02:21:03 +02:00
}
}
void set_active_window(int index)
2013-08-13 02:21:03 +02:00
{
if (index < 0 || index >= MAX_WINDOWS_NUM)
return;
active_window = windows + index;
2013-08-13 02:21:03 +02:00
}
ToxWindow *init_windows(Tox *m)
2013-08-13 02:21:03 +02:00
{
int n_prompt = add_window(m, new_prompt());
if (n_prompt == -1 || add_window(m, new_friendlist()) == -1) {
2013-08-16 19:11:09 +02:00
endwin();
2013-09-12 07:33:41 +02:00
fprintf(stderr, "add_window() failed. Aborting...\n");
exit(EXIT_FAILURE);
2013-08-16 19:11:09 +02:00
}
prompt = &windows[n_prompt];
active_window = prompt;
2013-08-16 19:11:09 +02:00
return prompt;
2013-08-13 02:21:03 +02:00
}
2013-11-29 01:45:28 +01:00
static void draw_window_tab(ToxWindow toxwin)
{
/* alert0 takes priority */
if (toxwin.alert0)
attron(COLOR_PAIR(GREEN));
else if (toxwin.alert1)
2013-11-29 01:45:28 +01:00
attron(COLOR_PAIR(RED));
else if (toxwin.alert2)
attron(COLOR_PAIR(MAGENTA));
2013-11-29 01:45:28 +01:00
clrtoeol();
printw(" [%s]", toxwin.name);
if (toxwin.alert0)
attroff(COLOR_PAIR(GREEN));
else if (toxwin.alert1)
2013-11-29 01:45:28 +01:00
attroff(COLOR_PAIR(RED));
else if (toxwin.alert2)
attroff(COLOR_PAIR(MAGENTA));
2013-11-29 01:45:28 +01:00
}
2013-11-26 23:39:11 +01:00
static void draw_bar(void)
2013-08-13 02:21:03 +02:00
{
attron(COLOR_PAIR(BLUE));
2013-08-16 19:11:09 +02:00
mvhline(LINES - 2, 0, '_', COLS);
attroff(COLOR_PAIR(BLUE));
2013-08-16 19:11:09 +02:00
move(LINES - 1, 0);
attron(COLOR_PAIR(BLUE) | A_BOLD);
2013-08-31 08:22:07 +02:00
printw(" TOXIC " TOXICVER " |");
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) {
if (windows[i].active) {
if (windows + i == active_window) {
#ifdef URXVT_FIX
attron(A_BOLD | COLOR_PAIR(GREEN));
} else {
#endif
2013-08-16 19:11:09 +02:00
attron(A_BOLD);
}
2013-08-13 02:21:03 +02:00
2013-11-29 01:45:28 +01:00
draw_window_tab(windows[i]);
2013-08-13 02:21:03 +02:00
2013-11-29 01:45:28 +01:00
if (windows + i == active_window) {
#ifdef URXVT_FIX
attroff(A_BOLD | COLOR_PAIR(GREEN));
} else {
#endif
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
}
2013-08-23 23:03:44 +02:00
void draw_active_window(Tox *m)
2013-08-13 02:21:03 +02:00
{
ToxWindow *a = active_window;
a->alert0 = false;
a->alert1 = false;
a->alert2 = false;
wint_t ch = 0;
draw_bar();
touchwin(a->window);
#ifndef WIN32
wresize(a->window, LINES - 2, COLS);
#endif
2014-03-09 07:02:54 +01:00
a->onDraw(a, m);
wrefresh(a->window);
2013-08-16 19:11:09 +02:00
/* Handle input */
#ifdef HAVE_WIDECHAR
if (wget_wch(stdscr, &ch) == ERR)
#else
if ((ch = getch()) == ERR)
#endif
return;
2013-08-16 19:11:09 +02:00
2014-03-13 11:06:53 +01:00
if (ch == T_KEY_NEXT || ch == T_KEY_PREV) {
2013-12-07 04:07:37 +01:00
set_next_window((int) ch);
} else {
2014-03-13 11:06:53 +01:00
pthread_mutex_lock(&Winthread.lock);
2013-08-16 19:11:09 +02:00
a->onKey(a, m, ch);
2014-03-13 11:06:53 +01:00
pthread_mutex_unlock(&Winthread.lock);
}
2013-08-13 02:21:03 +02:00
}
2014-03-07 01:39:57 +01:00
int get_num_active_windows(void)
{
2014-03-07 01:39:57 +01:00
return num_active_windows;
2013-11-30 22:09:45 +01:00
}
2014-02-26 11:23:11 +01:00
/* destroys all chat and groupchat windows (should only be called on shutdown) */
void kill_all_windows(void)
{
int i;
for (i = 0; i < MAX_WINDOWS_NUM; ++i) {
if (windows[i].is_chat)
kill_chat_window(&windows[i]);
else if (windows[i].is_groupchat)
kill_groupchat_window(&windows[i]);
}
}