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/>.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
2020-04-18 02:00:00 +02:00
|
|
|
#include <ctype.h>
|
|
|
|
#include <pthread.h>
|
2013-08-23 23:03:44 +02:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
2020-04-18 02:00:00 +02:00
|
|
|
#include "avatars.h"
|
2014-06-22 03:41:38 +02:00
|
|
|
#include "chat.h"
|
2020-04-18 02:00:00 +02:00
|
|
|
#include "conference.h"
|
|
|
|
#include "file_transfers.h"
|
|
|
|
#include "friendlist.h"
|
2014-06-22 03:41:38 +02:00
|
|
|
#include "line_info.h"
|
2014-10-03 23:53:50 +02:00
|
|
|
#include "misc_tools.h"
|
2020-04-18 02:00:00 +02:00
|
|
|
#include "prompt.h"
|
2014-07-28 01:06:25 +02:00
|
|
|
#include "settings.h"
|
2020-04-18 02:00:00 +02:00
|
|
|
#include "toxic.h"
|
|
|
|
#include "windows.h"
|
2015-04-04 09:26:38 +02:00
|
|
|
|
2021-01-19 04:37:14 +01:00
|
|
|
#ifdef GAMES
|
|
|
|
#include "game_base.h"
|
|
|
|
#endif
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
ToxWindow *windows[MAX_WINDOWS_NUM];
|
|
|
|
static uint8_t active_window_index;
|
|
|
|
static int num_active_windows;
|
2013-11-30 11:35:25 +01:00
|
|
|
|
2013-08-13 02:21:03 +02:00
|
|
|
/* CALLBACKS START */
|
2018-07-18 17:41:17 +02:00
|
|
|
void on_friend_request(Tox *m, const uint8_t *public_key, const uint8_t *data, size_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2014-10-03 23:53:50 +02:00
|
|
|
char msg[MAX_STR_SIZE + 1];
|
|
|
|
length = copy_tox_str(msg, sizeof(msg), (const char *) data, length);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2018-10-28 12:09:37 +01:00
|
|
|
if (windows[i] != NULL && windows[i]->onFriendRequest != NULL) {
|
2018-10-10 00:59:25 +02:00
|
|
|
windows[i]->onFriendRequest(windows[i], m, (const char *) public_key, msg, length);
|
2014-12-10 00:29:07 +01:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2018-10-10 05:06:57 +02:00
|
|
|
void on_friend_connection_status(Tox *m, uint32_t friendnumber, Tox_Connection connection_status, void *userdata)
|
2013-09-05 03:25:59 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2020-11-14 01:38:33 +01:00
|
|
|
on_avatar_friend_connection_status(m, friendnumber, connection_status);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onConnectionChange != NULL) {
|
|
|
|
windows[i]->onConnectionChange(windows[i], m, friendnumber, connection_status);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-09-05 03:25:59 +02:00
|
|
|
}
|
2021-11-19 23:54:35 +01:00
|
|
|
|
|
|
|
flag_interface_refresh();
|
2013-09-05 03:25:59 +02:00
|
|
|
}
|
|
|
|
|
2018-07-18 17:41:17 +02:00
|
|
|
void on_friend_typing(Tox *m, uint32_t friendnumber, bool is_typing, void *userdata)
|
2014-02-23 10:28:33 +01:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2018-07-18 17:33:16 +02:00
|
|
|
if (user_settings->show_typing_other == SHOW_TYPING_OFF) {
|
2014-07-30 02:10:28 +02:00
|
|
|
return;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-07-30 02:10:28 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onTypingChange != NULL) {
|
|
|
|
windows[i]->onTypingChange(windows[i], m, friendnumber, is_typing);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-02-23 10:28:33 +01:00
|
|
|
}
|
2021-11-19 23:54:35 +01:00
|
|
|
|
|
|
|
flag_interface_refresh();
|
2014-02-23 10:28:33 +01:00
|
|
|
}
|
|
|
|
|
2018-10-10 05:06:57 +02:00
|
|
|
void on_friend_message(Tox *m, uint32_t friendnumber, Tox_Message_Type type, const uint8_t *string, size_t length,
|
2018-07-18 17:41:17 +02:00
|
|
|
void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2014-10-03 23:53:50 +02:00
|
|
|
char msg[MAX_STR_SIZE + 1];
|
|
|
|
length = copy_tox_str(msg, sizeof(msg), (const char *) string, length);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onMessage != NULL) {
|
|
|
|
windows[i]->onMessage(windows[i], m, friendnumber, type, msg, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2018-07-18 17:41:17 +02:00
|
|
|
void on_friend_name(Tox *m, uint32_t friendnumber, const uint8_t *string, size_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2014-10-03 23:53:50 +02:00
|
|
|
char nick[TOXIC_MAX_NAME_LENGTH + 1];
|
|
|
|
length = copy_tox_str(nick, sizeof(nick), (const char *) string, length);
|
2014-10-07 22:18:06 +02:00
|
|
|
filter_str(nick, length);
|
2014-10-03 23:53:50 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onNickChange != NULL) {
|
|
|
|
windows[i]->onNickChange(windows[i], m, friendnumber, nick, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
}
|
2013-09-09 06:56:47 +02:00
|
|
|
|
2021-11-19 23:54:35 +01:00
|
|
|
flag_interface_refresh();
|
|
|
|
|
2014-07-09 06:05:13 +02:00
|
|
|
store_data(m, DATA_FILE);
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2018-07-18 17:41:17 +02:00
|
|
|
void on_friend_status_message(Tox *m, uint32_t friendnumber, const uint8_t *string, size_t length, void *userdata)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
UNUSED_VAR(m);
|
|
|
|
|
2015-03-26 03:56:45 +01:00
|
|
|
char msg[TOX_MAX_STATUS_MESSAGE_LENGTH + 1];
|
2014-10-03 23:53:50 +02:00
|
|
|
length = copy_tox_str(msg, sizeof(msg), (const char *) string, length);
|
2014-10-07 22:18:06 +02:00
|
|
|
filter_str(msg, length);
|
2014-10-03 23:53:50 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onStatusMessageChange != NULL) {
|
|
|
|
windows[i]->onStatusMessageChange(windows[i], friendnumber, msg, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
}
|
2021-11-19 23:54:35 +01:00
|
|
|
|
|
|
|
flag_interface_refresh();
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2018-10-10 05:06:57 +02:00
|
|
|
void on_friend_status(Tox *m, uint32_t friendnumber, Tox_User_Status status, void *userdata)
|
2013-09-05 03:25:59 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onStatusChange != NULL) {
|
|
|
|
windows[i]->onStatusChange(windows[i], m, friendnumber, status);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-09-05 03:25:59 +02:00
|
|
|
}
|
2021-11-19 23:54:35 +01:00
|
|
|
|
|
|
|
flag_interface_refresh();
|
2013-09-05 03:25:59 +02:00
|
|
|
}
|
|
|
|
|
2018-07-18 17:41:17 +02:00
|
|
|
void on_friend_added(Tox *m, uint32_t friendnumber, bool sort)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onFriendAdded != NULL) {
|
|
|
|
windows[i]->onFriendAdded(windows[i], m, friendnumber, sort);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-11-15 20:59:49 +01:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2014-07-09 06:05:13 +02:00
|
|
|
store_data(m, DATA_FILE);
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
2013-09-15 22:38:38 +02:00
|
|
|
|
2020-11-08 16:08:24 +01:00
|
|
|
void on_conference_message(Tox *m, uint32_t conferencenumber, uint32_t peernumber, Tox_Message_Type type,
|
2018-07-18 17:41:17 +02:00
|
|
|
const uint8_t *message, size_t length, void *userdata)
|
2013-09-15 22:38:38 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2014-10-03 23:53:50 +02:00
|
|
|
char msg[MAX_STR_SIZE + 1];
|
|
|
|
length = copy_tox_str(msg, sizeof(msg), (const char *) message, length);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2020-11-08 16:08:24 +01:00
|
|
|
if (windows[i] != NULL && windows[i]->onConferenceMessage != NULL) {
|
|
|
|
windows[i]->onConferenceMessage(windows[i], m, conferencenumber, peernumber, type, msg, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-09-15 22:38:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-08 16:08:24 +01:00
|
|
|
void on_conference_invite(Tox *m, uint32_t friendnumber, Tox_Conference_Type type, const uint8_t *conference_pub_key,
|
2018-07-18 17:41:17 +02:00
|
|
|
size_t length, void *userdata)
|
2013-09-15 22:38:38 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2020-11-08 16:08:24 +01:00
|
|
|
if (windows[i] != NULL && windows[i]->onConferenceInvite != NULL) {
|
2021-12-11 23:18:43 +01:00
|
|
|
windows[i]->onConferenceInvite(windows[i], m, friendnumber, type, (const char *) conference_pub_key, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-09-15 22:38:38 +02:00
|
|
|
}
|
|
|
|
}
|
2013-10-10 10:52:05 +02:00
|
|
|
|
2020-11-08 16:08:24 +01:00
|
|
|
void on_conference_peer_list_changed(Tox *m, uint32_t conferencenumber, void *userdata)
|
2013-11-24 03:19:59 +01:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2020-11-08 16:08:24 +01:00
|
|
|
if (windows[i] != NULL && windows[i]->onConferenceNameListChange != NULL) {
|
|
|
|
windows[i]->onConferenceNameListChange(windows[i], m, conferencenumber);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2018-02-25 06:00:06 +01:00
|
|
|
}
|
2021-11-19 23:54:35 +01:00
|
|
|
|
|
|
|
flag_interface_refresh();
|
2018-02-25 06:00:06 +01:00
|
|
|
}
|
|
|
|
|
2020-11-08 16:08:24 +01:00
|
|
|
void on_conference_peer_name(Tox *m, uint32_t conferencenumber, uint32_t peernumber, const uint8_t *name,
|
2018-02-25 06:00:06 +01:00
|
|
|
size_t length, void *userdata)
|
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2018-02-25 09:43:19 +01:00
|
|
|
char nick[TOXIC_MAX_NAME_LENGTH + 1];
|
|
|
|
length = copy_tox_str(nick, sizeof(nick), (const char *) name, length);
|
|
|
|
filter_str(nick, length);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2020-11-08 16:08:24 +01:00
|
|
|
if (windows[i] != NULL && windows[i]->onConferencePeerNameChange != NULL) {
|
|
|
|
windows[i]->onConferencePeerNameChange(windows[i], m, conferencenumber, peernumber, nick, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-11-24 03:19:59 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-08 16:08:24 +01:00
|
|
|
void on_conference_title(Tox *m, uint32_t conferencenumber, uint32_t peernumber, const uint8_t *title, size_t length,
|
2018-07-18 17:41:17 +02:00
|
|
|
void *userdata)
|
2014-11-15 04:15:59 +01:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2014-11-15 04:15:59 +01:00
|
|
|
char data[MAX_STR_SIZE + 1];
|
|
|
|
length = copy_tox_str(data, sizeof(data), (const char *) title, length);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2020-11-08 16:08:24 +01:00
|
|
|
if (windows[i] != NULL && windows[i]->onConferenceTitleChange != NULL) {
|
|
|
|
windows[i]->onConferenceTitleChange(windows[i], m, conferencenumber, peernumber, data, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-11-15 04:15:59 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-26 03:56:45 +01:00
|
|
|
void on_file_chunk_request(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint64_t position,
|
|
|
|
size_t length, void *userdata)
|
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2015-04-04 09:26:38 +02:00
|
|
|
struct FileTransfer *ft = get_file_transfer_struct(friendnumber, filenumber);
|
|
|
|
|
2018-07-18 17:33:16 +02:00
|
|
|
if (!ft) {
|
2015-04-04 09:26:38 +02:00
|
|
|
return;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2015-04-04 09:26:38 +02:00
|
|
|
|
|
|
|
if (ft->file_type == TOX_FILE_KIND_AVATAR) {
|
|
|
|
on_avatar_chunk_request(m, ft, position, length);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onFileChunkRequest != NULL) {
|
|
|
|
windows[i]->onFileChunkRequest(windows[i], m, friendnumber, filenumber, position, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2015-03-26 03:56:45 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void on_file_recv_chunk(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint64_t position,
|
2020-03-15 19:57:00 +01:00
|
|
|
const uint8_t *data, size_t length, void *userdata)
|
2013-10-10 10:52:05 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2015-04-04 09:26:38 +02:00
|
|
|
struct FileTransfer *ft = get_file_transfer_struct(friendnumber, filenumber);
|
|
|
|
|
2018-07-18 17:33:16 +02:00
|
|
|
if (!ft) {
|
2015-04-04 09:26:38 +02:00
|
|
|
return;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2015-04-04 09:26:38 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onFileRecvChunk != NULL) {
|
2021-12-11 23:18:43 +01:00
|
|
|
windows[i]->onFileRecvChunk(windows[i], m, friendnumber, filenumber, position, (const char *) data, length);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-10-10 10:52:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-10 05:06:57 +02:00
|
|
|
void on_file_recv_control(Tox *m, uint32_t friendnumber, uint32_t filenumber, Tox_File_Control control,
|
2018-07-18 17:41:17 +02:00
|
|
|
void *userdata)
|
2013-10-10 10:52:05 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2015-04-04 09:26:38 +02:00
|
|
|
struct FileTransfer *ft = get_file_transfer_struct(friendnumber, filenumber);
|
|
|
|
|
2018-07-18 17:33:16 +02:00
|
|
|
if (!ft) {
|
2015-04-04 09:26:38 +02:00
|
|
|
return;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2015-04-04 09:26:38 +02:00
|
|
|
|
|
|
|
if (ft->file_type == TOX_FILE_KIND_AVATAR) {
|
|
|
|
on_avatar_file_control(m, ft, control);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onFileControl != NULL) {
|
|
|
|
windows[i]->onFileControl(windows[i], m, friendnumber, filenumber, control);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-10-10 10:52:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-26 03:56:45 +01:00
|
|
|
void on_file_recv(Tox *m, uint32_t friendnumber, uint32_t filenumber, uint32_t kind, uint64_t file_size,
|
|
|
|
const uint8_t *filename, size_t filename_length, void *userdata)
|
2013-10-10 10:52:05 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
/* We don't care about receiving avatars */
|
|
|
|
if (kind != TOX_FILE_KIND_DATA) {
|
|
|
|
tox_file_control(m, friendnumber, filenumber, TOX_FILE_CONTROL_CANCEL, NULL);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onFileRecv != NULL) {
|
2021-12-11 23:18:43 +01:00
|
|
|
windows[i]->onFileRecv(windows[i], m, friendnumber, filenumber, file_size, (const char *) filename,
|
2018-10-10 00:59:25 +02:00
|
|
|
filename_length);
|
2018-07-18 17:56:21 +02:00
|
|
|
}
|
2013-10-10 10:52:05 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-18 17:41:17 +02:00
|
|
|
void on_friend_read_receipt(Tox *m, uint32_t friendnumber, uint32_t receipt, void *userdata)
|
2014-09-07 08:43:53 +02:00
|
|
|
{
|
2020-03-15 19:57:00 +01:00
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL && windows[i]->onReadReceipt != NULL) {
|
|
|
|
windows[i]->onReadReceipt(windows[i], m, friendnumber, receipt);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-09-07 08:43:53 +02:00
|
|
|
}
|
|
|
|
}
|
2021-01-17 19:29:13 +01:00
|
|
|
|
|
|
|
void on_lossless_custom_packet(Tox *m, uint32_t friendnumber, const uint8_t *data, size_t length, void *userdata)
|
|
|
|
{
|
|
|
|
UNUSED_VAR(userdata);
|
|
|
|
|
|
|
|
if (length == 0 || data == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint8_t type = data[0];
|
|
|
|
|
|
|
|
switch (type) {
|
2021-01-19 04:37:14 +01:00
|
|
|
#ifdef GAMES
|
|
|
|
|
2021-01-17 19:29:13 +01:00
|
|
|
case CUSTOM_PACKET_GAME_INVITE: {
|
|
|
|
for (size_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
ToxWindow *window = windows[i];
|
|
|
|
|
|
|
|
if (window != NULL && window->onGameInvite != NULL) {
|
|
|
|
window->onGameInvite(window, m, friendnumber, data + 1, length - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case CUSTOM_PACKET_GAME_DATA: {
|
|
|
|
for (size_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
ToxWindow *window = windows[i];
|
|
|
|
|
|
|
|
if (window != NULL && window->onGameData != NULL) {
|
|
|
|
window->onGameData(window, m, friendnumber, data + 1, length - 1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2021-01-19 04:37:14 +01:00
|
|
|
#endif // GAMES
|
|
|
|
|
2021-01-17 19:29:13 +01:00
|
|
|
default: {
|
|
|
|
fprintf(stderr, "Got unknown custom packet of type: %u\n", type);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-13 02:21:03 +02:00
|
|
|
/* CALLBACKS END */
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
int add_window(Tox *m, ToxWindow *w)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2018-07-18 17:33:16 +02:00
|
|
|
if (LINES < 2) {
|
2013-08-16 19:11:09 +02:00
|
|
|
return -1;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; i++) {
|
|
|
|
if (windows[i] != NULL) {
|
2013-08-17 10:00:19 +02:00
|
|
|
continue;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
w->index = i;
|
2020-11-30 05:26:51 +01:00
|
|
|
w->window = newwin(LINES, COLS, 0, 0);
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
if (w->window == NULL) {
|
2013-08-17 10:00:19 +02:00
|
|
|
return -1;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2013-09-26 15:51:21 +02:00
|
|
|
#ifdef URXVT_FIX
|
|
|
|
/* Fixes text color problem on some terminals. */
|
2018-10-10 00:59:25 +02:00
|
|
|
wbkgd(w->window, COLOR_PAIR(6));
|
2013-09-26 15:51:21 +02:00
|
|
|
#endif
|
2013-08-17 10:00:19 +02:00
|
|
|
windows[i] = w;
|
2014-03-19 08:14:08 +01:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
if (w->onInit) {
|
|
|
|
w->onInit(w, m);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2014-03-07 01:39:57 +01:00
|
|
|
++num_active_windows;
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
void set_active_window_index(uint8_t index)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2018-10-10 00:59:25 +02:00
|
|
|
if (index < MAX_WINDOWS_NUM) {
|
|
|
|
active_window_index = index;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2021-01-25 23:09:23 +01:00
|
|
|
/* Displays the next window if `ch` is equal to the next window key binding.
|
|
|
|
* Otherwise displays the previous window.
|
|
|
|
*/
|
2021-11-22 02:16:02 +01:00
|
|
|
static void set_next_window(int ch)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2021-11-19 23:54:35 +01:00
|
|
|
uint8_t index = 0;
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
if (ch == user_settings->key_next_tab) {
|
|
|
|
for (uint8_t i = active_window_index + 1; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] != NULL) {
|
2021-11-19 23:54:35 +01:00
|
|
|
index = i;
|
|
|
|
break;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
|
|
|
}
|
2018-10-10 00:59:25 +02:00
|
|
|
} else {
|
|
|
|
uint8_t start = active_window_index == 0 ? MAX_WINDOWS_NUM - 1 : active_window_index - 1;
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = start; i > 0; --i) {
|
|
|
|
if (windows[i] != NULL) {
|
2021-11-19 23:54:35 +01:00
|
|
|
index = i;
|
|
|
|
break;
|
2018-10-10 00:59:25 +02:00
|
|
|
}
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
2018-10-10 00:59:25 +02:00
|
|
|
|
2021-11-19 23:54:35 +01:00
|
|
|
set_active_window_index(index);
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2014-07-09 01:24:44 +02:00
|
|
|
/* Deletes window w and cleans up */
|
|
|
|
void del_window(ToxWindow *w)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2018-10-10 00:59:25 +02:00
|
|
|
uint8_t idx = w->index;
|
2020-11-30 05:26:51 +01:00
|
|
|
delwin(w->window_bar);
|
2014-07-09 01:24:44 +02:00
|
|
|
delwin(w->window);
|
2020-11-30 05:26:51 +01:00
|
|
|
free(w);
|
2018-10-10 00:59:25 +02:00
|
|
|
windows[idx] = NULL;
|
2014-07-09 01:24:44 +02:00
|
|
|
|
|
|
|
clear();
|
|
|
|
refresh();
|
2021-01-25 23:09:23 +01:00
|
|
|
|
|
|
|
if (num_active_windows > 0) {
|
2021-06-28 19:54:25 +02:00
|
|
|
if (active_window_index == 2) { // if closing current window would bring us to friend list
|
|
|
|
set_next_window(-1); // skip back to the home window instead. FIXME: magic numbers
|
|
|
|
}
|
|
|
|
|
2021-01-25 23:09:23 +01:00
|
|
|
set_next_window(-1);
|
2021-06-28 19:54:25 +02:00
|
|
|
|
2021-01-25 23:09:23 +01:00
|
|
|
--num_active_windows;
|
|
|
|
}
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
ToxWindow *init_windows(Tox *m)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2020-11-30 05:26:51 +01:00
|
|
|
if (COLS <= CHATBOX_HEIGHT + WINDOW_BAR_HEIGHT) {
|
|
|
|
exit_toxic_err("add_window() for prompt failed in init_windows", FATALERR_WININIT);
|
|
|
|
}
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
prompt = new_prompt();
|
2020-11-30 05:26:51 +01:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
int n_prompt = add_window(m, prompt);
|
|
|
|
|
|
|
|
if (n_prompt < 0) {
|
|
|
|
exit_toxic_err("add_window() for prompt failed in init_windows", FATALERR_WININIT);
|
|
|
|
}
|
2013-08-21 01:37:05 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
if (add_window(m, new_friendlist()) == -1) {
|
|
|
|
exit_toxic_err("add_window() for friendlist failed in init_windows", FATALERR_WININIT);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
set_active_window_index(n_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
|
|
|
}
|
|
|
|
|
2014-07-01 05:56:47 +02:00
|
|
|
void on_window_resize(void)
|
2014-06-13 07:42:20 +02:00
|
|
|
{
|
2014-06-29 00:40:22 +02:00
|
|
|
endwin();
|
2014-06-13 07:42:20 +02:00
|
|
|
refresh();
|
|
|
|
clear();
|
2014-07-01 05:56:47 +02:00
|
|
|
|
2020-12-11 07:46:01 +01:00
|
|
|
int x2;
|
|
|
|
int y2;
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
2020-11-30 05:26:51 +01:00
|
|
|
ToxWindow *w = windows[i];
|
|
|
|
|
|
|
|
if (w == NULL) {
|
2014-07-01 05:56:47 +02:00
|
|
|
continue;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-07-01 05:56:47 +02:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
if (w->type == WINDOW_TYPE_FRIEND_LIST) {
|
|
|
|
delwin(w->window_bar);
|
2014-07-10 04:11:31 +02:00
|
|
|
delwin(w->window);
|
2020-11-30 05:26:51 +01:00
|
|
|
w->window = newwin(LINES, COLS, 0, 0);
|
|
|
|
w->window_bar = subwin(w->window, WINDOW_BAR_HEIGHT, COLS, LINES - 2, 0);
|
2014-07-02 01:39:25 +02:00
|
|
|
continue;
|
2014-07-10 04:11:31 +02:00
|
|
|
}
|
2014-07-02 01:39:25 +02:00
|
|
|
|
2021-01-19 04:37:14 +01:00
|
|
|
#ifdef GAMES
|
|
|
|
|
2020-12-11 07:46:01 +01:00
|
|
|
if (w->type == WINDOW_TYPE_GAME) {
|
|
|
|
delwin(w->window_bar);
|
|
|
|
delwin(w->window);
|
|
|
|
delwin(w->game->window);
|
|
|
|
w->window = newwin(LINES, COLS, 0, 0);
|
|
|
|
|
|
|
|
getmaxyx(w->window, y2, x2);
|
|
|
|
|
2021-06-28 19:54:25 +02:00
|
|
|
if (y2 <= 0 || x2 <= 0) {
|
|
|
|
fprintf(stderr, "Failed to resize game window: max_x: %d, max_y: %d\n", x2, y2);
|
|
|
|
delwin(w->window);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-12-11 07:46:01 +01:00
|
|
|
w->window_bar = subwin(w->window, WINDOW_BAR_HEIGHT, COLS, LINES - 2, 0);
|
|
|
|
w->game->window = subwin(w->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, x2, 0, 0);
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2021-01-19 04:37:14 +01:00
|
|
|
#endif // GAMES
|
|
|
|
|
2018-07-18 17:33:16 +02:00
|
|
|
if (w->help->active) {
|
2014-07-05 20:46:16 +02:00
|
|
|
wclear(w->help->win);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-07-05 20:46:16 +02:00
|
|
|
|
2020-11-09 23:01:22 +01:00
|
|
|
if (w->type == WINDOW_TYPE_CONFERENCE) {
|
2014-07-01 20:55:59 +02:00
|
|
|
delwin(w->chatwin->sidebar);
|
2014-10-08 20:53:09 +02:00
|
|
|
w->chatwin->sidebar = NULL;
|
|
|
|
} else {
|
2014-07-01 20:55:59 +02:00
|
|
|
delwin(w->stb->topline);
|
2014-10-08 20:53:09 +02:00
|
|
|
}
|
2014-07-01 20:55:59 +02:00
|
|
|
|
|
|
|
delwin(w->chatwin->linewin);
|
|
|
|
delwin(w->chatwin->history);
|
2020-11-30 05:26:51 +01:00
|
|
|
delwin(w->window_bar);
|
2014-07-10 04:11:31 +02:00
|
|
|
delwin(w->window);
|
2014-07-01 20:55:59 +02:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
w->window = newwin(LINES, COLS, 0, 0);
|
|
|
|
|
|
|
|
getmaxyx(w->window, y2, x2);
|
|
|
|
|
|
|
|
if (y2 <= 0 || x2 <= 0) {
|
|
|
|
fprintf(stderr, "Failed to resize window: max_x: %d, max_y: %d\n", x2, y2);
|
|
|
|
delwin(w->window);
|
2021-06-28 19:54:25 +02:00
|
|
|
continue;
|
2020-11-30 05:26:51 +01:00
|
|
|
}
|
2014-07-01 05:56:47 +02:00
|
|
|
|
2014-10-08 20:53:09 +02:00
|
|
|
if (w->show_peerlist) {
|
2020-11-30 05:26:51 +01:00
|
|
|
w->chatwin->history = subwin(w->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, x2 - SIDEBAR_WIDTH - 1, 0, 0);
|
|
|
|
w->chatwin->sidebar = subwin(w->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, SIDEBAR_WIDTH, 0, x2 - SIDEBAR_WIDTH);
|
2014-07-01 05:56:47 +02:00
|
|
|
} else {
|
2020-11-30 05:26:51 +01:00
|
|
|
w->chatwin->history = subwin(w->window, y2 - CHATBOX_HEIGHT - WINDOW_BAR_HEIGHT, x2, 0, 0);
|
2014-10-08 20:53:09 +02:00
|
|
|
|
2020-11-09 23:01:22 +01:00
|
|
|
if (w->type != WINDOW_TYPE_CONFERENCE) {
|
2020-11-30 05:26:51 +01:00
|
|
|
w->stb->topline = subwin(w->window, TOP_BAR_HEIGHT, x2, 0, 0);
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-07-01 05:56:47 +02:00
|
|
|
}
|
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
w->window_bar = subwin(w->window, WINDOW_BAR_HEIGHT, x2, y2 - (CHATBOX_HEIGHT + WINDOW_BAR_HEIGHT), 0);
|
|
|
|
w->chatwin->linewin = subwin(w->window, CHATBOX_HEIGHT, x2, y2 - WINDOW_BAR_HEIGHT, 0);
|
|
|
|
|
2014-09-23 03:24:45 +02:00
|
|
|
#ifdef AUDIO
|
2016-09-25 03:07:04 +02:00
|
|
|
|
2014-07-01 20:55:59 +02:00
|
|
|
if (w->chatwin->infobox.active) {
|
|
|
|
delwin(w->chatwin->infobox.win);
|
|
|
|
w->chatwin->infobox.win = newwin(INFOBOX_HEIGHT, INFOBOX_WIDTH + 1, 1, x2 - INFOBOX_WIDTH);
|
2014-07-01 17:17:52 +02:00
|
|
|
}
|
2016-09-25 03:07:04 +02:00
|
|
|
|
2018-10-08 19:39:04 +02:00
|
|
|
#endif /* AUDIO */
|
2014-07-01 17:17:52 +02:00
|
|
|
|
2014-07-01 20:55:59 +02:00
|
|
|
scrollok(w->chatwin->history, 0);
|
2020-11-30 05:26:51 +01:00
|
|
|
wmove(w->window, y2 - CURS_Y_OFFSET, 0);
|
2014-07-01 05:56:47 +02:00
|
|
|
}
|
2014-06-13 07:42:20 +02:00
|
|
|
}
|
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
static void draw_window_tab(WINDOW *win, ToxWindow *toxwin, bool active_window)
|
2013-11-29 01:45:28 +01:00
|
|
|
{
|
2015-08-28 03:29:34 +02:00
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
2016-09-25 03:07:04 +02:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
bool has_alert = toxwin->alert != WINDOW_ALERT_NONE;
|
2020-11-21 17:25:41 +01:00
|
|
|
unsigned int pending_messages = toxwin->pending_messages;
|
|
|
|
|
2015-08-28 03:29:34 +02:00
|
|
|
pthread_mutex_unlock(&Winthread.lock);
|
|
|
|
|
2020-11-21 17:25:41 +01:00
|
|
|
WINDOW_TYPE type = toxwin->type;
|
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
if (active_window) {
|
2020-12-05 21:39:33 +01:00
|
|
|
wattron(win, A_BOLD | COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, " [");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_ACCENT));
|
|
|
|
wattron(win, COLOR_PAIR(BAR_TEXT));
|
2020-11-30 05:26:51 +01:00
|
|
|
} else {
|
|
|
|
if (has_alert) {
|
2020-12-05 21:39:33 +01:00
|
|
|
wattron(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, " [");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wattron(win, A_BOLD | COLOR_PAIR(toxwin->alert));
|
|
|
|
} else {
|
2020-12-05 21:39:33 +01:00
|
|
|
wattron(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, " [");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_ACCENT));
|
|
|
|
wattron(win, COLOR_PAIR(BAR_TEXT));
|
2020-11-30 05:26:51 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-11-21 17:25:41 +01:00
|
|
|
if (active_window || (type == WINDOW_TYPE_PROMPT || type == WINDOW_TYPE_FRIEND_LIST)) {
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, "%s", toxwin->name);
|
2020-11-19 07:30:30 +01:00
|
|
|
} else {
|
2020-11-21 17:25:41 +01:00
|
|
|
if (pending_messages > 0) {
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, "%u", pending_messages);
|
2020-11-21 17:25:41 +01:00
|
|
|
} else {
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, "-");
|
2020-11-21 17:25:41 +01:00
|
|
|
}
|
2020-11-19 07:30:30 +01:00
|
|
|
}
|
2015-08-28 03:29:34 +02:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
if (active_window) {
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_TEXT));
|
|
|
|
wattron(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, "]");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, A_BOLD | COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
} else {
|
|
|
|
if (has_alert) {
|
|
|
|
wattroff(win, A_BOLD | COLOR_PAIR(toxwin->alert));
|
2020-12-05 21:39:33 +01:00
|
|
|
wattron(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, "]");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
} else {
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_TEXT));
|
|
|
|
wattron(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, "]");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_ACCENT));
|
2020-11-30 05:26:51 +01:00
|
|
|
}
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-11-29 01:45:28 +01:00
|
|
|
}
|
2013-11-26 23:39:11 +01:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
void draw_window_bar(ToxWindow *self)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2020-11-30 05:26:51 +01:00
|
|
|
WINDOW *win = self->window_bar;
|
|
|
|
wclear(win);
|
2016-07-27 02:07:31 +02:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
if (self->scroll_pause) {
|
2020-12-05 21:39:33 +01:00
|
|
|
wattron(win, A_BLINK | A_BOLD | COLOR_PAIR(BAR_NOTIFY));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, "^");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, A_BLINK | A_BOLD | COLOR_PAIR(BAR_NOTIFY));
|
2020-11-30 05:26:51 +01:00
|
|
|
} else {
|
2020-12-05 21:39:33 +01:00
|
|
|
wattron(win, COLOR_PAIR(BAR_TEXT));
|
2020-11-30 05:26:51 +01:00
|
|
|
wprintw(win, " ");
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_TEXT));
|
2018-10-10 00:59:25 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
if (windows[i] == NULL) {
|
2014-06-12 00:47:18 +02:00
|
|
|
continue;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-06-12 00:47:18 +02:00
|
|
|
|
2020-11-19 07:30:30 +01:00
|
|
|
bool active_window = i == active_window_index;
|
2020-11-30 05:26:51 +01:00
|
|
|
draw_window_tab(win, windows[i], active_window);
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
int cur_x;
|
|
|
|
int cur_y;
|
|
|
|
UNUSED_VAR(cur_y);
|
2016-07-27 02:07:31 +02:00
|
|
|
|
2020-11-30 05:26:51 +01:00
|
|
|
getyx(win, cur_y, cur_x);
|
|
|
|
|
2020-12-05 21:39:33 +01:00
|
|
|
wattron(win, COLOR_PAIR(BAR_TEXT));
|
2020-11-30 05:26:51 +01:00
|
|
|
mvwhline(win, 0, cur_x, ' ', COLS - cur_x);
|
2020-12-05 21:39:33 +01:00
|
|
|
wattroff(win, COLOR_PAIR(BAR_TEXT));
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
|
|
|
|
2020-10-13 22:12:55 +02:00
|
|
|
/*
|
|
|
|
* Gets current char from stdscr and puts it in ch.
|
|
|
|
*
|
|
|
|
* Return 1 if char is printable.
|
|
|
|
* Return 0 if char is not printable.
|
|
|
|
* Return -1 on error.
|
|
|
|
*/
|
|
|
|
static int get_current_char(wint_t *ch)
|
|
|
|
{
|
|
|
|
wint_t tmpchar = 0;
|
|
|
|
bool is_printable = false;
|
|
|
|
|
|
|
|
#ifdef HAVE_WIDECHAR
|
|
|
|
int status = wget_wch(stdscr, &tmpchar);
|
|
|
|
|
|
|
|
if (status == ERR) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (status == OK) {
|
|
|
|
is_printable = iswprint(tmpchar);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else
|
|
|
|
tmpchar = getch();
|
|
|
|
|
|
|
|
if (tmpchar == ERR) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
is_printable = isprint(tmpchar);
|
|
|
|
#endif /* HAVE_WIDECHAR */
|
|
|
|
|
|
|
|
*ch = tmpchar;
|
|
|
|
|
|
|
|
return (int) is_printable;
|
|
|
|
}
|
|
|
|
|
|
|
|
static struct key_sequence_codes {
|
|
|
|
wchar_t *code;
|
|
|
|
wint_t key;
|
|
|
|
} Keys[] = {
|
2020-10-14 01:52:43 +02:00
|
|
|
{ L"[1;5A", T_KEY_C_UP },
|
|
|
|
{ L"[1;5B", T_KEY_C_DOWN },
|
2020-10-13 22:12:55 +02:00
|
|
|
{ L"[1;5C", T_KEY_C_RIGHT },
|
2020-10-14 01:52:43 +02:00
|
|
|
{ L"[1;5D", T_KEY_C_LEFT },
|
2020-10-13 22:12:55 +02:00
|
|
|
{ NULL, 0 }
|
|
|
|
};
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Return key code corresponding to character sequence queued in stdscr.
|
|
|
|
* Return -1 if sequence is unknown.
|
|
|
|
*/
|
|
|
|
#define MAX_SEQUENCE_SIZE 5
|
|
|
|
static wint_t get_input_sequence_code(void)
|
|
|
|
{
|
|
|
|
wchar_t code[MAX_SEQUENCE_SIZE + 1];
|
|
|
|
|
|
|
|
size_t length = 0;
|
|
|
|
wint_t ch = 0;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < MAX_SEQUENCE_SIZE; ++i) {
|
|
|
|
int res = get_current_char(&ch);
|
|
|
|
|
|
|
|
if (res < 0) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
++length;
|
|
|
|
code[i] = (wchar_t) ch;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (length == 0) {
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
code[length] = L'\0';
|
|
|
|
|
|
|
|
for (size_t i = 0; Keys[i].key != 0; ++i) {
|
|
|
|
if (wcscmp(code, Keys[i].code) == 0) {
|
|
|
|
return Keys[i].key;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-08-23 23:03:44 +02:00
|
|
|
void draw_active_window(Tox *m)
|
2013-08-13 02:21:03 +02:00
|
|
|
{
|
2018-10-10 00:59:25 +02:00
|
|
|
ToxWindow *a = windows[active_window_index];
|
|
|
|
|
|
|
|
if (a == NULL) {
|
|
|
|
return;
|
|
|
|
}
|
2015-08-28 03:29:34 +02:00
|
|
|
|
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
2014-07-21 01:12:13 +02:00
|
|
|
a->alert = WINDOW_ALERT_NONE;
|
2020-11-21 17:25:41 +01:00
|
|
|
a->pending_messages = 0;
|
2021-11-19 23:54:35 +01:00
|
|
|
bool flag_refresh = Winthread.flag_refresh;
|
2015-08-28 03:29:34 +02:00
|
|
|
pthread_mutex_unlock(&Winthread.lock);
|
2013-12-04 07:08:26 +01:00
|
|
|
|
2021-11-19 23:54:35 +01:00
|
|
|
if (flag_refresh) {
|
|
|
|
touchwin(a->window);
|
|
|
|
a->onDraw(a, m);
|
|
|
|
wrefresh(a->window);
|
|
|
|
}
|
2021-11-26 14:51:38 +01:00
|
|
|
|
2021-11-19 23:54:35 +01:00
|
|
|
#ifdef AUDIO
|
|
|
|
else if (a->is_call && timed_out(a->chatwin->infobox.lastupdate, 1)) {
|
|
|
|
touchwin(a->window);
|
|
|
|
a->onDraw(a, m);
|
|
|
|
wrefresh(a->window);
|
|
|
|
}
|
2021-11-26 14:51:38 +01:00
|
|
|
|
2021-11-21 20:52:09 +01:00
|
|
|
#endif // AUDIO
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2021-01-19 04:37:14 +01:00
|
|
|
#ifdef GAMES
|
2021-11-20 04:54:35 +01:00
|
|
|
|
2020-12-11 07:46:01 +01:00
|
|
|
if (a->type == WINDOW_TYPE_GAME) {
|
2021-11-19 23:54:35 +01:00
|
|
|
if (!flag_refresh) { // we always want to be continously refreshing game windows
|
|
|
|
touchwin(a->window);
|
|
|
|
a->onDraw(a, m);
|
|
|
|
wrefresh(a->window);
|
|
|
|
}
|
|
|
|
|
2020-12-11 07:46:01 +01:00
|
|
|
int ch = getch();
|
|
|
|
|
|
|
|
if (ch == ERR) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-11-19 23:54:35 +01:00
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
|
|
|
flag_interface_refresh();
|
|
|
|
pthread_mutex_unlock(&Winthread.lock);
|
|
|
|
|
2020-12-11 07:46:01 +01:00
|
|
|
if (ch == user_settings->key_next_tab || ch == user_settings->key_prev_tab) {
|
|
|
|
set_next_window(ch);
|
|
|
|
}
|
|
|
|
|
|
|
|
a->onKey(a, m, ch, false);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-01-19 04:37:14 +01:00
|
|
|
#endif // GAMES
|
|
|
|
|
2020-10-13 22:12:55 +02:00
|
|
|
wint_t ch = 0;
|
|
|
|
int printable = get_current_char(&ch);
|
2014-03-30 22:40:13 +02:00
|
|
|
|
2020-10-13 22:12:55 +02:00
|
|
|
if (printable < 0) {
|
2014-03-17 01:18:31 +01:00
|
|
|
return;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2021-11-19 23:54:35 +01:00
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
|
|
|
flag_interface_refresh();
|
|
|
|
pthread_mutex_unlock(&Winthread.lock);
|
|
|
|
|
2020-10-13 22:12:55 +02:00
|
|
|
if (printable == 0 && (ch == user_settings->key_next_tab || ch == user_settings->key_prev_tab)) {
|
2013-12-07 04:07:37 +01:00
|
|
|
set_next_window((int) ch);
|
2020-10-13 22:12:55 +02:00
|
|
|
return;
|
2020-11-09 23:01:22 +01:00
|
|
|
} else if ((printable == 0) && (a->type != WINDOW_TYPE_FRIEND_LIST)) {
|
2014-03-13 11:06:53 +01:00
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
2020-10-13 22:12:55 +02:00
|
|
|
bool input_ret = a->onKey(a, m, ch, (bool) printable);
|
2014-03-13 11:06:53 +01:00
|
|
|
pthread_mutex_unlock(&Winthread.lock);
|
2020-10-13 22:12:55 +02:00
|
|
|
|
|
|
|
if (input_ret) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// if an unprintable key code is unrecognized by input handler we attempt to manually decode char sequence
|
|
|
|
wint_t tmp = get_input_sequence_code();
|
|
|
|
|
|
|
|
if (tmp != (wint_t) -1) {
|
|
|
|
ch = tmp;
|
|
|
|
}
|
2014-03-13 11:06:53 +01:00
|
|
|
}
|
2020-10-13 22:12:55 +02:00
|
|
|
|
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
|
|
|
a->onKey(a, m, ch, (bool) printable);
|
|
|
|
pthread_mutex_unlock(&Winthread.lock);
|
2013-08-13 02:21:03 +02:00
|
|
|
}
|
2013-11-30 11:35:25 +01:00
|
|
|
|
2020-10-13 22:12:55 +02:00
|
|
|
/* Refresh inactive windows to prevent scrolling bugs.
|
|
|
|
* Call at least once per second.
|
|
|
|
*/
|
2014-06-01 09:38:20 +02:00
|
|
|
void refresh_inactive_windows(void)
|
|
|
|
{
|
2018-10-10 00:59:25 +02:00
|
|
|
for (uint8_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
ToxWindow *toxwin = windows[i];
|
2014-06-01 09:38:20 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
if (toxwin == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
2014-06-01 09:38:20 +02:00
|
|
|
|
2020-11-09 23:01:22 +01:00
|
|
|
if ((i != active_window_index) && (toxwin->type != WINDOW_TYPE_FRIEND_LIST)) {
|
2015-07-01 04:40:45 +02:00
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
2018-10-10 00:59:25 +02:00
|
|
|
line_info_print(toxwin);
|
2015-07-01 04:40:45 +02:00
|
|
|
pthread_mutex_unlock(&Winthread.lock);
|
|
|
|
}
|
2014-06-01 09:38:20 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
/* Returns a pointer to the ToxWindow in the ith index.
|
|
|
|
* Returns NULL if no ToxWindow exists.
|
|
|
|
*/
|
|
|
|
ToxWindow *get_window_ptr(size_t index)
|
2014-08-01 03:59:53 +02:00
|
|
|
{
|
2018-10-10 00:59:25 +02:00
|
|
|
if (index >= MAX_WINDOWS_NUM) {
|
|
|
|
return NULL;
|
2018-07-18 17:33:16 +02:00
|
|
|
}
|
2014-08-01 03:59:53 +02:00
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
return windows[index];
|
2014-08-01 03:59:53 +02:00
|
|
|
}
|
|
|
|
|
2018-10-10 00:59:25 +02:00
|
|
|
/* Returns a pointer to the currently active ToxWindow. */
|
2017-05-17 02:31:23 +02:00
|
|
|
ToxWindow *get_active_window(void)
|
|
|
|
{
|
2018-10-10 00:59:25 +02:00
|
|
|
return windows[active_window_index];
|
2017-05-17 02:31:23 +02:00
|
|
|
}
|
|
|
|
|
2014-10-01 22:24:36 +02:00
|
|
|
void force_refresh(WINDOW *w)
|
|
|
|
{
|
|
|
|
wclear(w);
|
|
|
|
endwin();
|
|
|
|
refresh();
|
|
|
|
}
|
|
|
|
|
2014-03-07 01:39:57 +01:00
|
|
|
int get_num_active_windows(void)
|
2013-11-30 11:35:25 +01:00
|
|
|
{
|
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
|
|
|
|
2021-11-20 15:49:04 +01:00
|
|
|
/* Returns the number of active windows of given type. */
|
|
|
|
size_t get_num_active_windows_type(WINDOW_TYPE type)
|
|
|
|
{
|
|
|
|
size_t count = 0;
|
|
|
|
|
|
|
|
for (size_t i = 0; i < MAX_WINDOWS_NUM; ++i) {
|
|
|
|
ToxWindow *w = windows[i];
|
|
|
|
|
|
|
|
if (w == NULL) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (w->type == type) {
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return count;
|
|
|
|
}
|
|
|
|
|
2020-11-08 16:08:24 +01:00
|
|
|
/* destroys all chat and conference windows (should only be called on shutdown) */
|
2014-08-05 23:38:33 +02:00
|
|
|
void kill_all_windows(Tox *m)
|
2014-02-26 11:23:11 +01:00
|
|
|
{
|
2021-11-22 02:16:02 +01:00
|
|
|
for (size_t i = 2; i < MAX_WINDOWS_NUM; ++i) {
|
2020-12-11 07:46:01 +01:00
|
|
|
ToxWindow *w = windows[i];
|
|
|
|
|
|
|
|
if (w == NULL) {
|
2018-10-10 00:59:25 +02:00
|
|
|
continue;
|
|
|
|
}
|
2014-02-26 11:23:11 +01:00
|
|
|
|
2021-06-28 19:54:25 +02:00
|
|
|
switch (w->type) {
|
|
|
|
case WINDOW_TYPE_CHAT: {
|
|
|
|
kill_chat_window(w, m);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
case WINDOW_TYPE_CONFERENCE: {
|
|
|
|
free_conference(w, w->num);
|
|
|
|
break;
|
|
|
|
}
|
2021-01-19 04:37:14 +01:00
|
|
|
|
|
|
|
#ifdef GAMES
|
2021-06-28 19:54:25 +02:00
|
|
|
|
|
|
|
case WINDOW_TYPE_GAME: {
|
|
|
|
game_kill(w);
|
|
|
|
break;
|
|
|
|
}
|
2021-01-19 04:37:14 +01:00
|
|
|
|
|
|
|
#endif // GAMES
|
2021-06-28 19:54:25 +02:00
|
|
|
|
|
|
|
default: {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2014-02-26 11:23:11 +01:00
|
|
|
}
|
2014-08-08 01:24:56 +02:00
|
|
|
|
2018-10-09 20:21:33 +02:00
|
|
|
/* TODO: use enum instead of magic indices */
|
2018-10-10 00:59:25 +02:00
|
|
|
kill_friendlist(windows[1]);
|
|
|
|
kill_prompt_window(windows[0]);
|
2014-02-26 11:23:11 +01:00
|
|
|
}
|