1
0
mirror of https://github.com/Tha14/toxic.git synced 2024-11-15 06:33:01 +01:00
toxic/src/friendlist.c

633 lines
20 KiB
C
Raw Normal View History

2014-02-25 08:28:24 +01:00
/* friendlist.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/>.
*
*/
#include <string.h>
#include <stdint.h>
2013-08-28 20:47:47 +02:00
#include <stdlib.h>
#include <time.h>
#include <tox/tox.h>
#include "toxic.h"
#include "windows.h"
#include "chat.h"
2013-08-13 00:50:43 +02:00
#include "friendlist.h"
2013-11-24 23:12:24 +01:00
#include "misc_tools.h"
2014-03-25 13:21:50 +01:00
#include "line_info.h"
2014-04-07 10:42:10 +02:00
#include "settings.h"
#ifdef _SUPPORT_AUDIO
2014-03-07 03:14:04 +01:00
#include "audio_call.h"
#endif
2013-08-28 11:46:09 +02:00
extern char *DATA_FILE;
extern ToxWindow *prompt;
static int max_friends_index = 0; /* marks the index of the last friend in friends array */
2013-07-31 19:20:03 +02:00
static int num_selected = 0;
static int num_friends = 0;
2014-03-13 11:06:53 +01:00
extern struct _Winthread Winthread;
2014-04-07 10:42:10 +02:00
extern struct user_settings *user_settings;
2013-11-30 00:52:21 +01:00
ToxicFriend friends[MAX_FRIENDS_NUM];
2013-11-24 23:12:24 +01:00
static int friendlist_index[MAX_FRIENDS_NUM] = {0};
static struct _pendingDel {
int num;
bool active;
2014-06-24 00:54:23 +02:00
WINDOW *popup;
} pendingdelete;
2014-03-09 05:57:21 +01:00
#define S_WEIGHT 100000
2013-12-14 02:57:32 +01:00
2013-11-26 00:49:31 +01:00
static int index_name_cmp(const void *n1, const void *n2)
2013-11-24 23:12:24 +01:00
{
2013-12-08 07:18:10 +01:00
int res = qsort_strcasecmp_hlpr(friends[*(int *) n1].name, friends[*(int *) n2].name);
2013-11-25 00:22:48 +01:00
/* Use weight to make qsort always put online friends before offline */
2013-12-14 02:57:32 +01:00
res = friends[*(int *) n1].online ? (res - S_WEIGHT) : (res + S_WEIGHT);
res = friends[*(int *) n2].online ? (res + S_WEIGHT) : (res - S_WEIGHT);
2013-11-25 00:22:48 +01:00
return res;
2013-11-24 23:12:24 +01:00
}
2013-11-25 00:22:48 +01:00
/* sorts friendlist_index first by connection status then alphabetically */
2014-03-14 04:30:44 +01:00
void sort_friendlist_index(void)
{
int i;
2013-11-25 04:31:58 +01:00
int n = 0;
for (i = 0; i < max_friends_index; ++i) {
2013-11-25 00:22:48 +01:00
if (friends[i].active)
2013-11-25 04:31:58 +01:00
friendlist_index[n++] = friends[i].num;
}
qsort(friendlist_index, num_friends, sizeof(int), index_name_cmp);
}
2014-03-19 02:48:26 +01:00
static void update_friend_last_online(int32_t num, uint64_t timestamp)
{
friends[num].last_online.last_on = timestamp;
2014-06-23 10:57:27 +02:00
friends[num].last_online.tm = *localtime((const time_t*)&timestamp);
2014-03-19 08:14:08 +01:00
/* if the format changes make sure TIME_STR_SIZE is the correct size */
const char *t = user_settings->time == TIME_12 ? "%I:%M %p" : "%H:%M";
strftime(friends[num].last_online.hour_min_str, TIME_STR_SIZE, t,
&friends[num].last_online.tm);
}
2014-03-19 02:48:26 +01:00
static void friendlist_onMessage(ToxWindow *self, Tox *m, int32_t num, uint8_t *str, uint16_t len)
2013-08-07 00:27:51 +02:00
{
if (num >= max_friends_index)
2013-08-16 19:11:09 +02:00
return;
if (friends[num].chatwin == -1) {
2014-03-07 01:39:57 +01:00
if (get_num_active_windows() < MAX_WINDOWS_NUM) {
friends[num].chatwin = add_window(m, new_chat(m, friends[num].num));
} else {
str[len] = '\0';
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
2014-04-01 09:53:12 +02:00
int n_len = tox_get_name(m, num, nick);
n_len = MIN(n_len, TOXIC_MAX_NAME_LENGTH - 1);
nick[n_len] = '\0';
2014-03-25 13:21:50 +01:00
uint8_t timefrmt[TIME_STR_SIZE];
2014-06-24 00:54:23 +02:00
get_time_str(timefrmt, sizeof(timefrmt));
2014-03-25 13:21:50 +01:00
line_info_add(prompt, timefrmt, nick, NULL, str, IN_MSG, 0, 0);
uint8_t *msg = "* Warning: Too many windows are open.";
line_info_add(prompt, NULL, NULL, NULL, msg, SYS_MSG, 0, RED);
alert_window(prompt, WINDOW_ALERT_1, true);
}
}
}
2014-03-19 02:48:26 +01:00
static void friendlist_onConnectionChange(ToxWindow *self, Tox *m, int32_t num, uint8_t status)
{
if (num >= max_friends_index)
return;
2014-03-25 13:21:50 +01:00
friends[num].online = status;
update_friend_last_online(num, get_unix_time());
store_data(m, DATA_FILE);
2014-03-14 04:30:44 +01:00
sort_friendlist_index();
}
2014-03-19 02:48:26 +01:00
static void friendlist_onNickChange(ToxWindow *self, Tox *m, int32_t num, uint8_t *str, uint16_t len)
2013-08-07 00:27:51 +02:00
{
if (len > TOX_MAX_NAME_LENGTH || num >= max_friends_index)
2013-08-16 19:11:09 +02:00
return;
len = MIN(len, TOXIC_MAX_NAME_LENGTH - 1);
str[len] = '\0';
2014-03-28 08:46:00 +01:00
strcpy(friends[num].name, str);
friends[num].namelength = len;
2014-03-14 04:30:44 +01:00
sort_friendlist_index();
}
2014-03-19 02:48:26 +01:00
static void friendlist_onStatusChange(ToxWindow *self, Tox *m, int32_t num, uint8_t status)
{
if (num >= max_friends_index)
return;
friends[num].status = status;
}
2014-03-19 02:48:26 +01:00
static void friendlist_onStatusMessageChange(ToxWindow *self, int32_t num, uint8_t *str, uint16_t len)
2013-08-07 00:27:51 +02:00
{
if (len > TOX_MAX_STATUSMESSAGE_LENGTH || num >= max_friends_index)
2013-08-16 19:11:09 +02:00
return;
str[len] = '\0';
snprintf(friends[num].statusmsg, sizeof(friends[num].statusmsg), "%s", str);
len = strlen(friends[num].statusmsg);
friends[num].statusmsg_len = len;
friends[num].statusmsg[len] = '\0';
}
2014-03-19 02:48:26 +01:00
void friendlist_onFriendAdded(ToxWindow *self, Tox *m, int32_t num, bool sort)
2013-08-07 00:27:51 +02:00
{
if (max_friends_index < 0 || max_friends_index >= MAX_FRIENDS_NUM)
2013-11-15 20:59:49 +01:00
return;
2013-08-16 19:11:09 +02:00
2013-08-28 11:46:09 +02:00
int i;
for (i = 0; i <= max_friends_index; ++i) {
2013-08-28 11:46:09 +02:00
if (!friends[i].active) {
friends[i].num = num;
friends[i].active = true;
friends[i].chatwin = -1;
friends[i].online = false;
friends[i].status = TOX_USERSTATUS_NONE;
friends[i].namelength = tox_get_name(m, num, friends[i].name);
2014-04-07 10:42:10 +02:00
friends[i].logging_on = (bool) user_settings->autolog == AUTOLOG_ON;
2014-02-22 23:58:36 +01:00
tox_get_client_id(m, num, friends[i].pub_key);
update_friend_last_online(i, tox_get_last_online(m, i));
if (friends[i].namelength == -1 || friends[i].name[0] == '\0') {
2013-11-19 00:52:46 +01:00
strcpy(friends[i].name, (uint8_t *) UNKNOWN_NAME);
friends[i].namelength = strlen(UNKNOWN_NAME);
2013-10-19 05:08:37 +02:00
} else { /* Enforce toxic's maximum name length */
friends[i].namelength = MIN(friends[i].namelength, TOXIC_MAX_NAME_LENGTH);
friends[i].name[friends[i].namelength] = '\0';
}
num_friends = tox_count_friendlist(m);
if (i == max_friends_index)
++max_friends_index;
if (sort)
2014-03-14 04:30:44 +01:00
sort_friendlist_index();
2013-11-15 20:59:49 +01:00
return;
2013-08-28 11:46:09 +02:00
}
}
}
static void friendlist_onFileSendRequest(ToxWindow *self, Tox *m, int32_t num, uint8_t filenum,
uint64_t filesize, uint8_t *filename, uint16_t filename_len)
2013-10-11 06:23:39 +02:00
{
if (num >= max_friends_index)
2013-10-11 06:23:39 +02:00
return;
if (friends[num].chatwin == -1) {
2014-03-07 01:39:57 +01:00
if (get_num_active_windows() < MAX_WINDOWS_NUM) {
friends[num].chatwin = add_window(m, new_chat(m, friends[num].num));
} else {
tox_file_send_control(m, num, 1, filenum, TOX_FILECONTROL_KILL, 0, 0);
uint8_t nick[TOX_MAX_NAME_LENGTH];
2014-04-01 09:53:12 +02:00
int n_len = tox_get_name(m, num, nick);
n_len = MIN(n_len, TOXIC_MAX_NAME_LENGTH - 1);
nick[n_len] = '\0';
2014-03-25 13:21:50 +01:00
uint8_t msg[MAX_STR_SIZE];
snprintf(msg, sizeof(msg), "* File transfer from %s failed: too many windows are open.", nick);
line_info_add(prompt, NULL, NULL, NULL, msg, SYS_MSG, 0, RED);
alert_window(prompt, WINDOW_ALERT_1, true);
}
}
2013-10-11 06:23:39 +02:00
}
2014-03-19 02:48:26 +01:00
static void friendlist_onGroupInvite(ToxWindow *self, Tox *m, int32_t num, uint8_t *group_pub_key)
2013-11-10 03:43:56 +01:00
{
if (num >= max_friends_index)
2013-11-10 03:43:56 +01:00
return;
if (friends[num].chatwin == -1) {
2014-03-07 01:39:57 +01:00
if (get_num_active_windows() < MAX_WINDOWS_NUM) {
friends[num].chatwin = add_window(m, new_chat(m, friends[num].num));
} else {
uint8_t nick[TOX_MAX_NAME_LENGTH];
2014-04-01 09:53:12 +02:00
int n_len = tox_get_name(m, num, nick);
n_len = MIN(n_len, TOXIC_MAX_NAME_LENGTH - 1);
nick[n_len] = '\0';
2014-03-25 13:21:50 +01:00
uint8_t msg[MAX_STR_SIZE];
snprintf(msg, sizeof(msg), "* Group chat invite from %s failed: too many windows are open.", nick);
line_info_add(prompt, NULL, NULL, NULL, msg, SYS_MSG, 0, RED);
alert_window(prompt, WINDOW_ALERT_1, true);
}
}
2013-11-10 03:43:56 +01:00
}
2013-11-27 07:54:22 +01:00
static void select_friend(ToxWindow *self, Tox *m, wint_t key)
2013-08-07 00:27:51 +02:00
{
2013-08-16 19:11:09 +02:00
if (key == KEY_UP) {
if (--num_selected < 0)
num_selected = num_friends - 1;
2013-08-16 19:11:09 +02:00
} else if (key == KEY_DOWN) {
num_selected = (num_selected + 1) % num_friends;
}
2013-08-28 11:46:09 +02:00
}
2014-03-19 02:48:26 +01:00
static void delete_friend(Tox *m, int32_t f_num)
2013-08-28 11:46:09 +02:00
{
tox_del_friend(m, f_num);
memset(&friends[f_num], 0, sizeof(ToxicFriend));
2013-08-28 11:46:09 +02:00
int i;
for (i = max_friends_index; i > 0; --i) {
if (friends[i - 1].active)
2013-08-28 11:46:09 +02:00
break;
}
max_friends_index = i;
num_friends = tox_count_friendlist(m);
/* make sure num_selected stays within num_friends range */
if (num_friends && num_selected == num_friends)
2013-11-18 04:28:22 +01:00
--num_selected;
2014-03-14 04:30:44 +01:00
sort_friendlist_index();
store_data(m, DATA_FILE);
2013-08-28 11:46:09 +02:00
}
2014-03-09 05:57:21 +01:00
/* activates delete friend popup */
2014-03-19 02:48:26 +01:00
static void del_friend_activate(ToxWindow *self, Tox *m, int32_t f_num)
2014-03-09 05:57:21 +01:00
{
int x2, y2;
getmaxyx(self->window, y2, x2);
2014-06-24 00:54:23 +02:00
pendingdelete.popup = newwin(3, 22 + TOXIC_MAX_NAME_LENGTH, 8, 8);
2014-03-09 05:57:21 +01:00
pendingdelete.active = true;
pendingdelete.num = f_num;
}
/* deactivates delete friend popup and deletes friend if instructed */
static void del_friend_deactivate(ToxWindow *self, Tox *m, wint_t key)
{
if (key == 'y')
delete_friend(m, pendingdelete.num);
2014-06-24 00:54:23 +02:00
delwin(pendingdelete.popup);
memset(&pendingdelete, 0, sizeof(pendingdelete));
2014-03-09 05:57:21 +01:00
clear();
refresh();
}
2014-06-24 00:54:23 +02:00
static void draw_popup(void)
2014-03-09 05:57:21 +01:00
{
2014-06-24 00:54:23 +02:00
if (!pendingdelete.active)
2014-03-09 05:57:21 +01:00
return;
2014-06-24 00:54:23 +02:00
wattron(pendingdelete.popup, A_BOLD);
box(pendingdelete.popup, ACS_VLINE, ACS_HLINE);
wattroff(pendingdelete.popup, A_BOLD);
2014-03-09 07:02:54 +01:00
2014-06-24 00:54:23 +02:00
wmove(pendingdelete.popup, 1, 1);
wprintw(pendingdelete.popup, "Delete contact ");
wattron(pendingdelete.popup, A_BOLD);
wprintw(pendingdelete.popup, "%s", friends[pendingdelete.num].name);
wattroff(pendingdelete.popup, A_BOLD);
wprintw(pendingdelete.popup, "? y/n");
2014-03-09 07:02:54 +01:00
2014-06-24 00:54:23 +02:00
wrefresh(pendingdelete.popup);
2014-03-09 05:57:21 +01:00
}
2014-03-30 22:40:13 +02:00
static void friendlist_onKey(ToxWindow *self, Tox *m, wint_t key, bool ltr)
2013-08-28 11:46:09 +02:00
{
if (num_friends == 0)
2013-11-18 04:28:22 +01:00
return;
int f = friendlist_index[num_selected];
2014-03-09 05:57:21 +01:00
/* lock screen and force decision on deletion popup */
if (pendingdelete.active) {
if (key == 'y' || key == 'n')
del_friend_deactivate(self, m, key);
return;
}
2014-03-30 22:40:13 +02:00
if (key != ltr) {
if (key == '\n') {
/* Jump to chat window if already open */
if (friends[f].chatwin != -1) {
set_active_window(friends[f].chatwin);
} else if (get_num_active_windows() < MAX_WINDOWS_NUM) {
friends[f].chatwin = add_window(m, new_chat(m, friends[f].num));
set_active_window(friends[f].chatwin);
} else {
uint8_t *msg = "* Warning: Too many windows are open.";
line_info_add(prompt, NULL, NULL, NULL, msg, SYS_MSG, 0, RED);
alert_window(prompt, WINDOW_ALERT_1, true);
}
} else if (key == KEY_DC) {
del_friend_activate(self, m, f);
} else {
2014-03-30 22:40:13 +02:00
select_friend(self, m, key);
}
2013-11-18 04:28:22 +01:00
}
}
#define FLIST_OFST 6 /* Accounts for space at top and bottom */
2013-08-23 23:03:44 +02:00
static void friendlist_onDraw(ToxWindow *self, Tox *m)
2013-08-07 00:27:51 +02:00
{
2013-08-16 19:11:09 +02:00
curs_set(0);
werase(self->window);
int x2, y2;
getmaxyx(self->window, y2, x2);
2013-11-15 23:03:24 +01:00
2014-03-14 04:56:46 +01:00
uint64_t cur_time = get_unix_time();
2014-06-23 10:57:27 +02:00
struct tm cur_loc_tm = *localtime((const time_t*)&cur_time);
2014-03-14 04:56:46 +01:00
bool fix_statuses = x2 != self->x; /* true if window x axis has changed */
2013-08-16 19:11:09 +02:00
wattron(self->window, COLOR_PAIR(CYAN));
wprintw(self->window, " Open a chat window with the");
wattron(self->window, A_BOLD);
wprintw(self->window, " Enter ");
wattroff(self->window, A_BOLD);
wprintw(self->window, "key. Delete a friend with the");
wattron(self->window, A_BOLD);
wprintw(self->window, " Delete ");
wattroff(self->window, A_BOLD);
wprintw(self->window, "key.\n\n");
wattroff(self->window, COLOR_PAIR(CYAN));
2014-03-13 11:43:53 +01:00
pthread_mutex_lock(&Winthread.lock);
int nf = tox_get_num_online_friends(m);
pthread_mutex_unlock(&Winthread.lock);
wattron(self->window, A_BOLD);
wprintw(self->window, " Online: ");
wattroff(self->window, A_BOLD);
wprintw(self->window, "%d/%d \n\n", nf, num_friends);
2013-08-16 19:11:09 +02:00
2013-12-03 09:44:02 +01:00
if ((y2 - FLIST_OFST) <= 0) /* don't allow division by zero */
2013-12-03 00:34:14 +01:00
return;
int selected_num = 0;
/* Determine which portion of friendlist to draw based on current position */
int page = num_selected / (y2 - FLIST_OFST);
int start = (y2 - FLIST_OFST) * page;
int end = y2 - FLIST_OFST + start;
2013-08-16 19:11:09 +02:00
int i;
for (i = start; i < num_friends && i < end; ++i) {
int f = friendlist_index[i];
2013-11-18 04:28:22 +01:00
bool f_selected = false;
if (friends[f].active) {
if (i == num_selected) {
wattron(self->window, A_BOLD);
2013-09-02 04:11:47 +02:00
wprintw(self->window, " > ");
wattroff(self->window, A_BOLD);
selected_num = f;
f_selected = true;
} else {
2013-09-02 04:11:47 +02:00
wprintw(self->window, " ");
}
if (friends[f].online) {
2014-03-19 02:48:26 +01:00
uint8_t status = friends[f].status;
int colour = WHITE;
2013-09-02 04:11:47 +02:00
2013-11-12 07:50:04 +01:00
switch (status) {
case TOX_USERSTATUS_NONE:
colour = GREEN;
break;
case TOX_USERSTATUS_AWAY:
colour = YELLOW;
break;
case TOX_USERSTATUS_BUSY:
colour = RED;
break;
case TOX_USERSTATUS_INVALID:
colour = MAGENTA;
break;
2013-09-02 04:11:47 +02:00
}
wattron(self->window, COLOR_PAIR(colour) | A_BOLD);
wprintw(self->window, "O ");
wattroff(self->window, COLOR_PAIR(colour) | A_BOLD);
if (f_selected)
wattron(self->window, COLOR_PAIR(BLUE));
wattron(self->window, A_BOLD);
wprintw(self->window, "%s", friends[f].name);
wattroff(self->window, A_BOLD);
if (f_selected)
wattroff(self->window, COLOR_PAIR(BLUE));
/* Reset friends[f].statusmsg on window resize */
2013-11-15 23:03:24 +01:00
if (fix_statuses) {
uint8_t statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH] = {'\0'};
2014-03-14 04:30:44 +01:00
2014-03-13 11:06:53 +01:00
pthread_mutex_lock(&Winthread.lock);
uint16_t s_len = tox_get_status_message(m, friends[f].num, statusmsg,
TOX_MAX_STATUSMESSAGE_LENGTH);
2014-03-13 11:06:53 +01:00
pthread_mutex_unlock(&Winthread.lock);
2014-03-14 04:30:44 +01:00
friends[f].statusmsg_len = s_len;
friends[f].statusmsg[s_len] = '\0';
2014-03-13 11:06:53 +01:00
snprintf(friends[f].statusmsg, sizeof(friends[f].statusmsg), "%s", statusmsg);
}
/* Truncate note if it doesn't fit on one line */
uint16_t maxlen = x2 - getcurx(self->window) - 2;
if (friends[f].statusmsg_len > maxlen) {
friends[f].statusmsg[maxlen - 3] = '\0';
2013-11-15 23:03:24 +01:00
strcat(friends[f].statusmsg, "...");
friends[f].statusmsg[maxlen] = '\0';
friends[f].statusmsg_len = maxlen;
}
if (friends[f].statusmsg[0])
wprintw(self->window, " %s", friends[f].statusmsg);
wprintw(self->window, "\n");
2013-09-02 04:11:47 +02:00
} else {
wprintw(self->window, "o ");
if (f_selected)
wattron(self->window, COLOR_PAIR(BLUE));
wattron(self->window, A_BOLD);
2014-03-14 04:56:46 +01:00
wprintw(self->window, "%s", friends[f].name);
wattroff(self->window, A_BOLD);
if (f_selected)
2014-05-01 10:00:19 +02:00
wattroff(self->window, COLOR_PAIR(BLUE));
uint64_t last_seen = friends[f].last_online.last_on;
if (last_seen != 0) {
int day_dist = (cur_loc_tm.tm_yday - friends[f].last_online.tm.tm_yday) % 365;
const uint8_t *hourmin = friends[f].last_online.hour_min_str;
switch (day_dist) {
case 0:
wprintw(self->window, " Last seen: Today %s\n", hourmin);
break;
case 1:
wprintw(self->window, " Last seen: Yesterday %s\n", hourmin);
break;
default:
wprintw(self->window, " Last seen: %d days ago\n", day_dist);
break;
}
} else {
wprintw(self->window, " Last seen: Never\n");
}
2013-09-02 04:11:47 +02:00
}
2013-11-10 03:43:56 +01:00
}
2013-08-16 19:11:09 +02:00
}
self->x = x2;
2013-08-16 19:11:09 +02:00
wrefresh(self->window);
2014-06-24 00:54:23 +02:00
draw_popup();
if (num_friends) {
wmove(self->window, y2 - 1, 1);
wattron(self->window, A_BOLD);
wprintw(self->window, "ID: ");
wattroff(self->window, A_BOLD);
int i;
for (i = 0; i < TOX_CLIENT_ID_SIZE; ++i)
wprintw(self->window, "%02X", friends[selected_num].pub_key[i] & 0xff);
}
}
2014-03-19 02:48:26 +01:00
void disable_chatwin(int32_t f_num)
2013-08-07 00:27:51 +02:00
{
2013-08-16 19:11:09 +02:00
friends[f_num].chatwin = -1;
}
2014-03-07 03:14:04 +01:00
#ifdef _SUPPORT_AUDIO
2014-05-16 20:00:01 +02:00
static void friendlist_onAv(ToxWindow *self, ToxAv *av, int call_index)
2014-03-07 03:14:04 +01:00
{
2014-05-16 20:00:01 +02:00
int id = toxav_get_peer_id(av, call_index, 0);
/*id++;*/
2014-03-12 00:25:13 +01:00
if ( id != ErrorInternal && id >= max_friends_index)
2014-03-07 03:14:04 +01:00
return;
Tox *m = toxav_get_tox(av);
2014-03-07 03:14:04 +01:00
if (friends[id].chatwin == -1) {
2014-03-11 01:04:53 +01:00
if (get_num_active_windows() < MAX_WINDOWS_NUM) {
2014-03-07 03:14:04 +01:00
friends[id].chatwin = add_window(m, new_chat(m, friends[id].num));
} else {
uint8_t nick[TOX_MAX_NAME_LENGTH] = {'\0'};
2014-04-01 09:53:12 +02:00
int n_len = tox_get_name(m, id, nick);
n_len = MIN(n_len, TOXIC_MAX_NAME_LENGTH - 1);
nick[n_len] = '\0';
2014-03-25 08:17:22 +01:00
2014-03-25 13:21:50 +01:00
uint8_t msg[MAX_STR_SIZE];
snprintf(msg, sizeof(msg), "Audio action from: %s!", nick);
line_info_add(prompt, NULL, NULL, NULL, msg, SYS_MSG, 0, 0);
uint8_t *errmsg = "* Warning: Too many windows are open.";
line_info_add(prompt, NULL, NULL, NULL, errmsg, SYS_MSG, 0, RED);
2014-03-07 03:14:04 +01:00
alert_window(prompt, WINDOW_ALERT_0, true);
}
}
}
#endif /* _SUPPORT_AUDIO */
2013-11-03 01:32:35 +01:00
ToxWindow new_friendlist(void)
2013-08-16 19:11:09 +02:00
{
ToxWindow ret;
memset(&ret, 0, sizeof(ret));
ret.active = true;
2013-08-16 19:11:09 +02:00
ret.onKey = &friendlist_onKey;
ret.onDraw = &friendlist_onDraw;
2013-11-15 20:59:49 +01:00
ret.onFriendAdded = &friendlist_onFriendAdded;
2013-08-16 19:11:09 +02:00
ret.onMessage = &friendlist_onMessage;
ret.onConnectionChange = &friendlist_onConnectionChange;
ret.onAction = &friendlist_onMessage; /* Action has identical behaviour to message */
2013-08-16 19:11:09 +02:00
ret.onNickChange = &friendlist_onNickChange;
ret.onStatusChange = &friendlist_onStatusChange;
ret.onStatusMessageChange = &friendlist_onStatusMessageChange;
2013-10-11 06:23:39 +02:00
ret.onFileSendRequest = &friendlist_onFileSendRequest;
2013-11-10 03:43:56 +01:00
ret.onGroupInvite = &friendlist_onGroupInvite;
2014-03-07 03:14:04 +01:00
#ifdef _SUPPORT_AUDIO
ret.onInvite = &friendlist_onAv;
ret.onRinging = &friendlist_onAv;
ret.onStarting = &friendlist_onAv;
ret.onEnding = &friendlist_onAv;
ret.onError = &friendlist_onAv;
ret.onStart = &friendlist_onAv;
ret.onCancel = &friendlist_onAv;
ret.onReject = &friendlist_onAv;
ret.onEnd = &friendlist_onAv;
2014-03-14 23:08:08 +01:00
ret.onRequestTimeout = &friendlist_onAv;
ret.onPeerTimeout = &friendlist_onAv;
2014-05-16 20:00:01 +02:00
2014-06-21 01:58:00 +02:00
ret.call_idx = -1;
ret.device_selection[0] = ret.device_selection[1] = -1;
2014-03-07 03:14:04 +01:00
#endif /* _SUPPORT_AUDIO */
2013-08-16 19:11:09 +02:00
strcpy(ret.name, "friends");
2013-08-16 19:11:09 +02:00
return ret;
}