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/>.
|
|
|
|
*
|
2013-07-30 21:47:40 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
#include <stdint.h>
|
2013-08-28 20:47:47 +02:00
|
|
|
#include <stdlib.h>
|
2014-03-14 22:48:52 +01:00
|
|
|
#include <time.h>
|
2014-07-31 21:02:19 +02:00
|
|
|
#include <arpa/inet.h>
|
2013-07-30 21:47:40 +02:00
|
|
|
|
2013-08-24 04:15:15 +02:00
|
|
|
#include <tox/tox.h>
|
2013-07-30 21:47:40 +02:00
|
|
|
|
2014-06-22 02:18:23 +02:00
|
|
|
#include "toxic.h"
|
|
|
|
#include "windows.h"
|
2013-11-19 21:32:35 +01:00
|
|
|
#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"
|
2014-07-21 03:25:21 +02:00
|
|
|
#include "notify.h"
|
2014-07-31 19:49:15 +02:00
|
|
|
#include "help.h"
|
2014-03-16 20:28:46 +01:00
|
|
|
|
2014-07-21 03:25:21 +02:00
|
|
|
#ifdef _AUDIO
|
2014-03-07 03:14:04 +01:00
|
|
|
#include "audio_call.h"
|
2014-03-16 20:28:46 +01:00
|
|
|
#endif
|
2013-07-30 21:47:40 +02:00
|
|
|
|
2014-07-21 03:25:21 +02:00
|
|
|
|
2013-08-28 11:46:09 +02:00
|
|
|
extern char *DATA_FILE;
|
2014-07-31 18:48:49 +02:00
|
|
|
extern char *BLOCK_FILE;
|
2013-09-07 01:59:45 +02:00
|
|
|
extern ToxWindow *prompt;
|
2014-07-31 18:48:49 +02:00
|
|
|
extern struct _Winthread Winthread;
|
|
|
|
extern struct user_settings *user_settings_;
|
|
|
|
extern struct arg_opts arg_opts;
|
2013-09-07 01:59:45 +02:00
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
static uint8_t blocklist_view = 0; /* 0 if we're in friendlist view, 1 if we're in blocklist view */
|
2013-07-31 19:20:03 +02:00
|
|
|
static int num_selected = 0;
|
2014-07-31 18:48:49 +02:00
|
|
|
static int max_friends_index = 0; /* 1 + the index of the last friend in friends array */
|
2014-02-08 05:28:17 +01:00
|
|
|
static int num_friends = 0;
|
2013-07-30 21:47:40 +02:00
|
|
|
|
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};
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
static struct _Blocked_Contacts {
|
|
|
|
int num_selected;
|
|
|
|
int max_index;
|
|
|
|
int num_blocked;
|
|
|
|
BlockedFriend list[MAX_FRIENDS_NUM];
|
|
|
|
int index[MAX_FRIENDS_NUM];
|
|
|
|
} Blocked_Contacts;
|
|
|
|
|
2014-06-22 02:18:23 +02:00
|
|
|
static struct _pendingDel {
|
2014-03-15 22:18:23 +01:00
|
|
|
int num;
|
|
|
|
bool active;
|
2014-06-24 00:54:23 +02:00
|
|
|
WINDOW *popup;
|
2014-03-15 22:18:23 +01:00
|
|
|
} pendingdelete;
|
2014-03-09 05:57:21 +01:00
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
static int save_blocklist(char *path)
|
|
|
|
{
|
|
|
|
if (arg_opts.ignore_data_file)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (path == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
int len = sizeof(BlockedFriend) * Blocked_Contacts.num_blocked;
|
|
|
|
char *data = malloc(len);
|
|
|
|
|
|
|
|
if (data == NULL)
|
|
|
|
exit_toxic_err("Failed in save_blocklist", FATALERR_MEMORY);
|
|
|
|
|
|
|
|
int i;
|
|
|
|
int count = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < Blocked_Contacts.max_index; ++i) {
|
|
|
|
if (count > Blocked_Contacts.num_blocked)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if (Blocked_Contacts.list[i].active) {
|
|
|
|
BlockedFriend tmp;
|
|
|
|
memset(&tmp, 0, sizeof(BlockedFriend));
|
2014-07-31 21:02:19 +02:00
|
|
|
tmp.namelength = htons(Blocked_Contacts.list[i].namelength);
|
2014-07-31 18:48:49 +02:00
|
|
|
memcpy(tmp.name, Blocked_Contacts.list[i].name, Blocked_Contacts.list[i].namelength + 1);
|
|
|
|
memcpy(tmp.pub_key, Blocked_Contacts.list[i].pub_key, TOX_CLIENT_ID_SIZE);
|
2014-07-31 20:53:02 +02:00
|
|
|
|
|
|
|
uint8_t lastonline[sizeof(uint64_t)];
|
|
|
|
memcpy(lastonline, &Blocked_Contacts.list[i].last_on, sizeof(uint64_t));
|
|
|
|
host_to_net(lastonline, sizeof(uint64_t));
|
|
|
|
memcpy(&tmp.last_on, lastonline, sizeof(uint64_t));
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
memcpy(data + count * sizeof(BlockedFriend), &tmp, sizeof(BlockedFriend));
|
|
|
|
++count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
FILE *fp = fopen(path, "wb");
|
|
|
|
|
|
|
|
if (fp == NULL) {
|
|
|
|
free(data);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int ret = 0;
|
|
|
|
|
|
|
|
if (fwrite(data, len, 1, fp) != 1)
|
|
|
|
ret = -1;
|
|
|
|
|
|
|
|
fclose(fp);
|
|
|
|
free(data);
|
|
|
|
return ret;
|
|
|
|
}
|
2013-12-14 02:57:32 +01:00
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
static void sort_blocklist_index(void);
|
|
|
|
|
|
|
|
int load_blocklist(char *path)
|
|
|
|
{
|
|
|
|
if (path == NULL)
|
|
|
|
return -1;
|
2013-12-14 02:57:32 +01:00
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
FILE *fp = fopen(path, "rb");
|
|
|
|
|
|
|
|
if (fp == NULL)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
fseek(fp, 0, SEEK_END);
|
|
|
|
int len = ftell(fp);
|
|
|
|
fseek(fp, 0, SEEK_SET);
|
|
|
|
|
|
|
|
char *data = malloc(len);
|
|
|
|
|
|
|
|
if (data == NULL) {
|
|
|
|
fclose(fp);
|
|
|
|
exit_toxic_err("Failed in load_blocklist", FATALERR_MEMORY);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fread(data, len, 1, fp) != 1) {
|
|
|
|
fclose(fp);
|
|
|
|
free(data);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (len % sizeof(BlockedFriend) != 0) {
|
|
|
|
fclose(fp);
|
|
|
|
free(data);
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int num = len / sizeof(BlockedFriend);
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < num; ++i) {
|
|
|
|
BlockedFriend tmp;
|
|
|
|
memcpy(&tmp, data + i * sizeof(BlockedFriend), sizeof(BlockedFriend));
|
|
|
|
Blocked_Contacts.list[i].active = true;
|
|
|
|
Blocked_Contacts.list[i].num = i;
|
2014-07-31 21:02:19 +02:00
|
|
|
Blocked_Contacts.list[i].namelength = ntohs(tmp.namelength);
|
|
|
|
memcpy(Blocked_Contacts.list[i].name, tmp.name, Blocked_Contacts.list[i].namelength + 1);
|
2014-07-31 18:48:49 +02:00
|
|
|
memcpy(Blocked_Contacts.list[i].pub_key, tmp.pub_key, TOX_CLIENT_ID_SIZE);
|
2014-07-31 20:53:02 +02:00
|
|
|
|
|
|
|
uint8_t lastonline[sizeof(uint64_t)];
|
|
|
|
memcpy(lastonline, &tmp.last_on, sizeof(uint64_t));
|
|
|
|
net_to_host(lastonline, sizeof(uint64_t));
|
|
|
|
memcpy(&Blocked_Contacts.list[i].last_on, lastonline, sizeof(uint64_t));
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
++Blocked_Contacts.num_blocked;
|
|
|
|
}
|
|
|
|
|
|
|
|
Blocked_Contacts.max_index = i + 1;
|
|
|
|
|
|
|
|
free(data);
|
|
|
|
fclose(fp);
|
|
|
|
sort_blocklist_index();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
#define S_WEIGHT 100000
|
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-09 08:50:32 +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)
|
2013-11-09 08:50:32 +01:00
|
|
|
{
|
|
|
|
int i;
|
2013-11-25 04:31:58 +01:00
|
|
|
int n = 0;
|
2013-11-09 08:50:32 +01:00
|
|
|
|
|
|
|
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;
|
2013-11-09 08:50:32 +01:00
|
|
|
}
|
|
|
|
|
2014-02-08 05:28:17 +01:00
|
|
|
qsort(friendlist_index, num_friends, sizeof(int), index_name_cmp);
|
2013-11-09 08:50:32 +01:00
|
|
|
}
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
static int index_name_cmp_block(const void *n1, const void *n2)
|
|
|
|
{
|
|
|
|
return qsort_strcasecmp_hlpr(Blocked_Contacts.list[*(int *) n1].name, Blocked_Contacts.list[*(int *) n2].name);
|
|
|
|
}
|
|
|
|
|
|
|
|
static void sort_blocklist_index(void)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
int n = 0;
|
|
|
|
|
|
|
|
for (i = 0; i < Blocked_Contacts.max_index; ++i) {
|
|
|
|
if (Blocked_Contacts.list[i].active)
|
|
|
|
Blocked_Contacts.index[n++] = Blocked_Contacts.list[i].num;
|
|
|
|
}
|
|
|
|
|
|
|
|
qsort(Blocked_Contacts.index, Blocked_Contacts.num_blocked, sizeof(int), index_name_cmp_block);
|
|
|
|
}
|
|
|
|
|
2014-03-19 02:48:26 +01:00
|
|
|
static void update_friend_last_online(int32_t num, uint64_t timestamp)
|
2014-03-15 22:18:23 +01:00
|
|
|
{
|
|
|
|
friends[num].last_online.last_on = timestamp;
|
2014-06-23 10:57:27 +02:00
|
|
|
friends[num].last_online.tm = *localtime((const time_t*)×tamp);
|
2014-03-19 08:14:08 +01:00
|
|
|
|
|
|
|
/* if the format changes make sure TIME_STR_SIZE is the correct size */
|
2014-07-21 01:12:13 +02:00
|
|
|
const char *t = user_settings_->time == TIME_12 ? "%I:%M %p" : "%H:%M";
|
2014-04-19 23:58:13 +02:00
|
|
|
strftime(friends[num].last_online.hour_min_str, TIME_STR_SIZE, t,
|
|
|
|
&friends[num].last_online.tm);
|
2014-03-15 22:18:23 +01:00
|
|
|
}
|
|
|
|
|
2014-07-07 04:15:35 +02:00
|
|
|
static void friendlist_onMessage(ToxWindow *self, Tox *m, int32_t num, const char *str, uint16_t len)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-11-30 11:35:25 +01:00
|
|
|
if (num >= max_friends_index)
|
2013-08-16 19:11:09 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
if (friends[num].chatwin == -1) {
|
2014-03-07 01:39:57 +01:00
|
|
|
if (get_num_active_windows() < MAX_WINDOWS_NUM) {
|
2014-08-02 00:37:02 +02:00
|
|
|
friends[num].chatwin = add_window(m, new_chat(m, friends[num].num));
|
2013-11-30 11:35:25 +01:00
|
|
|
} else {
|
2014-07-07 04:15:35 +02:00
|
|
|
char nick[TOX_MAX_NAME_LENGTH];
|
2014-07-08 20:31:59 +02:00
|
|
|
get_nick_truncate(m, nick, num);
|
2013-11-30 11:35:25 +01:00
|
|
|
|
2014-07-07 04:15:35 +02:00
|
|
|
char timefrmt[TIME_STR_SIZE];
|
2014-06-24 00:54:23 +02:00
|
|
|
get_time_str(timefrmt, sizeof(timefrmt));
|
2013-11-30 11:35:25 +01:00
|
|
|
|
2014-07-25 04:43:32 +02:00
|
|
|
line_info_add(prompt, timefrmt, nick, NULL, IN_MSG, 0, 0, str);
|
2014-03-25 13:21:50 +01:00
|
|
|
|
2014-07-25 04:43:32 +02:00
|
|
|
const char *msg = "* Warning: Too many windows are open.";
|
|
|
|
line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, msg);
|
2014-08-02 00:37:02 +02:00
|
|
|
sound_notify(prompt, error, NT_WNDALERT_1, NULL);
|
2013-11-30 11:35:25 +01:00
|
|
|
}
|
|
|
|
}
|
2013-07-31 20:20:16 +02:00
|
|
|
}
|
|
|
|
|
2014-03-19 02:48:26 +01:00
|
|
|
static void friendlist_onConnectionChange(ToxWindow *self, Tox *m, int32_t num, uint8_t status)
|
2013-09-05 03:25:59 +02:00
|
|
|
{
|
2013-11-30 11:35:25 +01:00
|
|
|
if (num >= max_friends_index)
|
2013-09-05 03:25:59 +02:00
|
|
|
return;
|
|
|
|
|
2014-03-25 13:21:50 +01:00
|
|
|
friends[num].online = status;
|
2014-03-15 22:18:23 +01:00
|
|
|
update_friend_last_online(num, get_unix_time());
|
2014-03-14 22:48:52 +01:00
|
|
|
store_data(m, DATA_FILE);
|
2014-03-14 04:30:44 +01:00
|
|
|
sort_friendlist_index();
|
2013-09-05 03:25:59 +02:00
|
|
|
}
|
|
|
|
|
2014-07-07 04:15:35 +02:00
|
|
|
static void friendlist_onNickChange(ToxWindow *self, Tox *m, int32_t num, const char *nick, uint16_t len)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-11-30 11:35:25 +01:00
|
|
|
if (len > TOX_MAX_NAME_LENGTH || num >= max_friends_index)
|
2013-08-16 19:11:09 +02:00
|
|
|
return;
|
2013-07-30 21:47:40 +02:00
|
|
|
|
2014-07-08 01:08:33 +02:00
|
|
|
char tempname[TOX_MAX_NAME_LENGTH];
|
|
|
|
strcpy(tempname, nick);
|
|
|
|
len = MIN(len, TOXIC_MAX_NAME_LENGTH - 1);
|
|
|
|
tempname[len] = '\0';
|
|
|
|
snprintf(friends[num].name, sizeof(friends[num].name), "%s", tempname);
|
|
|
|
friends[num].namelength = len;
|
2014-03-14 04:30:44 +01:00
|
|
|
sort_friendlist_index();
|
2013-07-30 21:47:40 +02:00
|
|
|
}
|
|
|
|
|
2014-03-19 02:48:26 +01:00
|
|
|
static void friendlist_onStatusChange(ToxWindow *self, Tox *m, int32_t num, uint8_t status)
|
2013-09-05 03:25:59 +02:00
|
|
|
{
|
2013-11-30 11:35:25 +01:00
|
|
|
if (num >= max_friends_index)
|
2013-09-05 03:25:59 +02:00
|
|
|
return;
|
|
|
|
|
|
|
|
friends[num].status = status;
|
|
|
|
}
|
|
|
|
|
2014-07-07 04:15:35 +02:00
|
|
|
static void friendlist_onStatusMessageChange(ToxWindow *self, int32_t num, const char *status, uint16_t len)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2013-11-30 11:35:25 +01:00
|
|
|
if (len > TOX_MAX_STATUSMESSAGE_LENGTH || num >= max_friends_index)
|
2013-08-16 19:11:09 +02:00
|
|
|
return;
|
2013-07-30 21:47:40 +02:00
|
|
|
|
2014-07-02 23:30:31 +02:00
|
|
|
snprintf(friends[num].statusmsg, sizeof(friends[num].statusmsg), "%s", status);
|
|
|
|
friends[num].statusmsg_len = strlen(friends[num].statusmsg);
|
2013-07-30 21:47:40 +02:00
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
2013-11-09 08:50:32 +01: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;
|
|
|
|
|
2013-11-09 08:50:32 +01:00
|
|
|
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;
|
2013-09-05 03:25:59 +02:00
|
|
|
friends[i].online = false;
|
2013-09-05 07:34:23 +02:00
|
|
|
friends[i].status = TOX_USERSTATUS_NONE;
|
2014-07-21 01:12:13 +02:00
|
|
|
friends[i].logging_on = (bool) user_settings_->autolog == AUTOLOG_ON;
|
2014-07-07 04:15:35 +02:00
|
|
|
tox_get_client_id(m, num, (uint8_t *) friends[i].pub_key);
|
2014-03-15 22:18:23 +01:00
|
|
|
update_friend_last_online(i, tox_get_last_online(m, i));
|
2013-09-04 03:31:50 +02:00
|
|
|
|
2014-07-08 01:08:33 +02:00
|
|
|
char tempname[TOX_MAX_NAME_LENGTH] = {0};
|
2014-07-08 20:31:59 +02:00
|
|
|
int len = get_nick_truncate(m, tempname, num);
|
2014-07-08 01:08:33 +02:00
|
|
|
|
|
|
|
if (len == -1 || tempname[0] == '\0') {
|
2014-07-07 04:15:35 +02:00
|
|
|
strcpy(friends[i].name, UNKNOWN_NAME);
|
2014-04-01 04:00:17 +02:00
|
|
|
friends[i].namelength = strlen(UNKNOWN_NAME);
|
2013-10-19 05:08:37 +02:00
|
|
|
} else { /* Enforce toxic's maximum name length */
|
2014-07-08 01:08:33 +02:00
|
|
|
friends[i].namelength = len;
|
|
|
|
snprintf(friends[i].name, sizeof(friends[i].name), "%s", tempname);
|
2013-09-18 01:11:23 +02:00
|
|
|
}
|
2013-09-04 03:31:50 +02:00
|
|
|
|
2014-02-08 05:28:17 +01:00
|
|
|
num_friends = tox_count_friendlist(m);
|
|
|
|
|
2013-11-09 08:50:32 +01:00
|
|
|
if (i == max_friends_index)
|
|
|
|
++max_friends_index;
|
|
|
|
|
2013-12-04 07:08:26 +01:00
|
|
|
if (sort)
|
2014-03-14 04:30:44 +01:00
|
|
|
sort_friendlist_index();
|
2013-12-04 07:08:26 +01:00
|
|
|
|
2013-11-15 20:59:49 +01:00
|
|
|
return;
|
2013-08-28 11:46:09 +02:00
|
|
|
}
|
|
|
|
}
|
2013-07-30 21:47:40 +02:00
|
|
|
}
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
/* puts blocked friend back in friendlist. fnum is new friend number, bnum is blocked number */
|
|
|
|
static void friendlist_add_blocked(Tox *m, int32_t fnum, int32_t bnum)
|
|
|
|
{
|
|
|
|
if (max_friends_index >= MAX_FRIENDS_NUM)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i <= max_friends_index; ++i) {
|
|
|
|
if (friends[i].active)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
friends[i].num = fnum;
|
|
|
|
friends[i].active = true;
|
|
|
|
friends[i].chatwin = -1;
|
|
|
|
friends[i].status = TOX_USERSTATUS_NONE;
|
|
|
|
friends[i].logging_on = (bool) user_settings_->autolog == AUTOLOG_ON;
|
|
|
|
friends[i].namelength = Blocked_Contacts.list[bnum].namelength;
|
2014-07-31 20:53:02 +02:00
|
|
|
update_friend_last_online(i, Blocked_Contacts.list[bnum].last_on);
|
2014-07-31 18:48:49 +02:00
|
|
|
memcpy(friends[i].name, Blocked_Contacts.list[bnum].name, friends[i].namelength + 1);
|
|
|
|
memcpy(friends[i].pub_key, Blocked_Contacts.list[bnum].pub_key, TOX_CLIENT_ID_SIZE);
|
|
|
|
|
|
|
|
num_friends = tox_count_friendlist(m);
|
|
|
|
|
|
|
|
if (i == max_friends_index)
|
|
|
|
++max_friends_index;
|
|
|
|
|
|
|
|
sort_blocklist_index();
|
|
|
|
sort_friendlist_index();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-04-19 23:58:13 +02:00
|
|
|
static void friendlist_onFileSendRequest(ToxWindow *self, Tox *m, int32_t num, uint8_t filenum,
|
2014-07-07 04:15:35 +02:00
|
|
|
uint64_t filesize, const char *filename, uint16_t filename_len)
|
2013-10-11 06:23:39 +02:00
|
|
|
{
|
2013-11-30 11:35:25 +01:00
|
|
|
if (num >= max_friends_index)
|
2013-10-11 06:23:39 +02:00
|
|
|
return;
|
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
if (friends[num].chatwin == -1) {
|
2014-03-07 01:39:57 +01:00
|
|
|
if (get_num_active_windows() < MAX_WINDOWS_NUM) {
|
2013-11-30 11:35:25 +01:00
|
|
|
friends[num].chatwin = add_window(m, new_chat(m, friends[num].num));
|
2014-08-02 00:37:02 +02:00
|
|
|
|
|
|
|
if (self->active_box != -1)
|
|
|
|
box_notify2(self, transfer_pending, NT_NOFOCUS, self->active_box,
|
|
|
|
"Incoming file reaquest: %s", filename);
|
|
|
|
else
|
|
|
|
box_notify(self, transfer_pending, NT_NOFOCUS, &self->active_box, self->name,
|
|
|
|
"Incoming file reaquest: %s", filename);
|
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
} else {
|
|
|
|
tox_file_send_control(m, num, 1, filenum, TOX_FILECONTROL_KILL, 0, 0);
|
|
|
|
|
2014-07-07 04:15:35 +02:00
|
|
|
char nick[TOX_MAX_NAME_LENGTH];
|
2014-07-08 20:31:59 +02:00
|
|
|
get_nick_truncate(m, nick, num);
|
2013-11-30 11:35:25 +01:00
|
|
|
|
2014-07-25 04:43:32 +02:00
|
|
|
const char *msg = "* File transfer from %s failed: too many windows are open.";
|
|
|
|
line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, msg, nick);
|
2014-07-21 01:12:13 +02:00
|
|
|
|
2014-08-02 00:37:02 +02:00
|
|
|
sound_notify(prompt, error, NT_WNDALERT_1, NULL);
|
2013-11-30 11:35:25 +01:00
|
|
|
}
|
|
|
|
}
|
2013-10-11 06:23:39 +02:00
|
|
|
}
|
|
|
|
|
2014-07-07 04:15:35 +02:00
|
|
|
static void friendlist_onGroupInvite(ToxWindow *self, Tox *m, int32_t num, const char *group_pub_key)
|
2013-11-10 03:43:56 +01:00
|
|
|
{
|
2013-11-30 11:35:25 +01:00
|
|
|
if (num >= max_friends_index)
|
2013-11-10 03:43:56 +01:00
|
|
|
return;
|
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
if (friends[num].chatwin == -1) {
|
2014-03-07 01:39:57 +01:00
|
|
|
if (get_num_active_windows() < MAX_WINDOWS_NUM) {
|
2013-11-30 11:35:25 +01:00
|
|
|
friends[num].chatwin = add_window(m, new_chat(m, friends[num].num));
|
2014-08-02 00:37:02 +02:00
|
|
|
|
|
|
|
if (self->active_box != -1)
|
|
|
|
box_notify2(self, generic_message, NT_WNDALERT_0 | NT_NOFOCUS, self->active_box,
|
|
|
|
"You are invited to join group" );
|
|
|
|
else
|
|
|
|
box_notify(self, generic_message, NT_WNDALERT_0 | NT_NOFOCUS, &self->active_box, self->name,
|
|
|
|
"You are invited to join group" );
|
2014-07-21 01:12:13 +02:00
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
} else {
|
2014-07-07 04:15:35 +02:00
|
|
|
char nick[TOX_MAX_NAME_LENGTH];
|
2014-07-08 20:31:59 +02:00
|
|
|
get_nick_truncate(m, nick, num);
|
2013-11-30 11:35:25 +01:00
|
|
|
|
2014-07-25 04:43:32 +02:00
|
|
|
const char *msg = "* Group chat invite from %s failed: too many windows are open.";
|
|
|
|
line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, msg, nick);
|
2014-07-21 01:12:13 +02:00
|
|
|
|
2014-08-02 00:37:02 +02:00
|
|
|
sound_notify(prompt, error, NT_WNDALERT_1, NULL);
|
2013-11-30 11:35:25 +01:00
|
|
|
}
|
|
|
|
}
|
2013-11-10 03:43:56 +01:00
|
|
|
}
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
/* move friendlist/blocklist cursor up and down */
|
2014-07-31 19:49:15 +02:00
|
|
|
static void select_friend(ToxWindow *self, wint_t key, int *selected, int num)
|
2013-08-07 00:27:51 +02:00
|
|
|
{
|
2014-07-31 19:49:15 +02:00
|
|
|
if (num <= 0)
|
2014-07-31 18:48:49 +02:00
|
|
|
return;
|
|
|
|
|
2013-08-16 19:11:09 +02:00
|
|
|
if (key == KEY_UP) {
|
2014-07-31 18:48:49 +02:00
|
|
|
if (--(*selected) < 0)
|
2014-07-31 19:49:15 +02:00
|
|
|
*selected = num - 1;
|
2013-08-16 19:11:09 +02:00
|
|
|
} else if (key == KEY_DOWN) {
|
2014-07-31 19:49:15 +02:00
|
|
|
*selected = (*selected + 1) % num;
|
2013-11-09 08:50:32 +01:00
|
|
|
}
|
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
|
|
|
{
|
2014-08-01 03:59:53 +02:00
|
|
|
if (friends[f_num].chatwin >= 0) {
|
|
|
|
ToxWindow *toxwin = get_window_ptr(friends[f_num].chatwin);
|
|
|
|
|
|
|
|
if (toxwin != NULL) {
|
|
|
|
kill_chat_window(toxwin);
|
|
|
|
set_active_window(1); /* keep friendlist focused */
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-29 16:14:59 +01:00
|
|
|
tox_del_friend(m, f_num);
|
2013-11-29 23:48:08 +01:00
|
|
|
memset(&friends[f_num], 0, sizeof(ToxicFriend));
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2013-08-28 11:46:09 +02:00
|
|
|
int i;
|
|
|
|
|
2013-11-09 08:50:32 +01:00
|
|
|
for (i = max_friends_index; i > 0; --i) {
|
2014-04-19 23:58:13 +02:00
|
|
|
if (friends[i - 1].active)
|
2013-08-28 11:46:09 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-11-09 08:50:32 +01:00
|
|
|
max_friends_index = i;
|
2014-02-08 05:28:17 +01:00
|
|
|
num_friends = tox_count_friendlist(m);
|
2013-09-05 07:34:23 +02:00
|
|
|
|
2013-11-18 05:14:27 +01:00
|
|
|
/* 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;
|
|
|
|
|
2013-09-05 07:34:23 +02:00
|
|
|
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
|
|
|
{
|
2014-07-08 01:08:33 +02:00
|
|
|
pendingdelete.popup = newwin(3, 22 + TOXIC_MAX_NAME_LENGTH - 1, 8, 8);
|
2014-03-09 05:57:21 +01:00
|
|
|
pendingdelete.active = true;
|
|
|
|
pendingdelete.num = f_num;
|
|
|
|
}
|
|
|
|
|
2014-07-31 19:49:15 +02:00
|
|
|
static void delete_blocked_friend(int32_t bnum);
|
2014-07-31 18:48:49 +02:00
|
|
|
|
2014-03-09 05:57:21 +01:00
|
|
|
/* deactivates delete friend popup and deletes friend if instructed */
|
|
|
|
static void del_friend_deactivate(ToxWindow *self, Tox *m, wint_t key)
|
|
|
|
{
|
2014-07-31 18:48:49 +02:00
|
|
|
if (key == 'y') {
|
|
|
|
if (blocklist_view == 0) {
|
|
|
|
delete_friend(m, pendingdelete.num);
|
|
|
|
sort_friendlist_index();
|
|
|
|
} else {
|
2014-07-31 19:49:15 +02:00
|
|
|
delete_blocked_friend(pendingdelete.num);
|
2014-07-31 18:48:49 +02:00
|
|
|
sort_blocklist_index();
|
|
|
|
}
|
|
|
|
}
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2014-06-24 00:54:23 +02:00
|
|
|
delwin(pendingdelete.popup);
|
2014-03-15 22:18:23 +01:00
|
|
|
memset(&pendingdelete, 0, sizeof(pendingdelete));
|
2014-03-09 05:57:21 +01:00
|
|
|
clear();
|
|
|
|
refresh();
|
|
|
|
}
|
|
|
|
|
2014-07-31 19:49:15 +02:00
|
|
|
static void draw_del_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);
|
2014-07-31 18:48:49 +02:00
|
|
|
|
|
|
|
if (blocklist_view == 0)
|
|
|
|
wprintw(pendingdelete.popup, "%s", friends[pendingdelete.num].name);
|
|
|
|
else
|
|
|
|
wprintw(pendingdelete.popup, "%s", Blocked_Contacts.list[pendingdelete.num].name);
|
|
|
|
|
2014-06-24 00:54:23 +02:00
|
|
|
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-07-31 19:49:15 +02:00
|
|
|
/* deletes contact from blocked list */
|
|
|
|
static void delete_blocked_friend(int32_t bnum)
|
|
|
|
{
|
|
|
|
memset(&Blocked_Contacts.list[bnum], 0, sizeof(BlockedFriend));
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
2014-07-31 20:53:02 +02:00
|
|
|
for (i = Blocked_Contacts.max_index; i > 0; --i) {
|
2014-07-31 19:49:15 +02:00
|
|
|
if (Blocked_Contacts.list[i - 1].active)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
--Blocked_Contacts.num_blocked;
|
|
|
|
Blocked_Contacts.max_index = i;
|
|
|
|
save_blocklist(BLOCK_FILE);
|
|
|
|
|
|
|
|
if (Blocked_Contacts.num_blocked && Blocked_Contacts.num_selected == Blocked_Contacts.num_blocked)
|
|
|
|
--Blocked_Contacts.num_selected;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* deletes contact from friendlist and puts in blocklist */
|
2014-07-31 18:48:49 +02:00
|
|
|
void block_friend(Tox *m, int32_t fnum)
|
|
|
|
{
|
|
|
|
if (Blocked_Contacts.max_index >= MAX_FRIENDS_NUM || num_friends <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i <= Blocked_Contacts.max_index; ++i) {
|
|
|
|
if (Blocked_Contacts.list[i].active)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
Blocked_Contacts.list[i].active = true;
|
|
|
|
Blocked_Contacts.list[i].num = i;
|
|
|
|
Blocked_Contacts.list[i].namelength = friends[fnum].namelength;
|
2014-07-31 20:53:02 +02:00
|
|
|
Blocked_Contacts.list[i].last_on = friends[fnum].last_online.last_on;
|
2014-07-31 18:48:49 +02:00
|
|
|
memcpy(Blocked_Contacts.list[i].pub_key, friends[fnum].pub_key, TOX_CLIENT_ID_SIZE);
|
|
|
|
memcpy(Blocked_Contacts.list[i].name, friends[fnum].name, friends[fnum].namelength + 1);
|
|
|
|
|
|
|
|
++Blocked_Contacts.num_blocked;
|
|
|
|
|
|
|
|
if (i == Blocked_Contacts.max_index)
|
|
|
|
++Blocked_Contacts.max_index;
|
|
|
|
|
|
|
|
delete_friend(m, fnum);
|
|
|
|
save_blocklist(BLOCK_FILE);
|
|
|
|
sort_blocklist_index();
|
|
|
|
sort_friendlist_index();
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* removes friend from blocklist, puts back in friendlist */
|
|
|
|
static void unblock_friend(Tox *m, int32_t bnum)
|
|
|
|
{
|
|
|
|
if (Blocked_Contacts.num_blocked <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int32_t friendnum = tox_add_friend_norequest(m, (uint8_t *) Blocked_Contacts.list[bnum].pub_key);
|
|
|
|
|
|
|
|
if (friendnum == -1) {
|
|
|
|
line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, "Failed to unblock friend");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
friendlist_add_blocked(m, friendnum, bnum);
|
2014-07-31 19:49:15 +02:00
|
|
|
delete_blocked_friend(bnum);
|
2014-07-31 18:48:49 +02:00
|
|
|
sort_blocklist_index();
|
|
|
|
sort_friendlist_index();
|
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
2014-07-31 19:49:15 +02:00
|
|
|
|
|
|
|
if (self->help->active) {
|
|
|
|
help_onKey(self, key);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (key == 'h') {
|
|
|
|
help_init_menu(self);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
if (!blocklist_view && !num_friends && (key != KEY_RIGHT && key != KEY_LEFT))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (blocklist_view && !Blocked_Contacts.num_blocked && (key != KEY_RIGHT && key != KEY_LEFT))
|
2013-11-18 04:28:22 +01:00
|
|
|
return;
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
int f = blocklist_view == 1 ? Blocked_Contacts.index[Blocked_Contacts.num_selected]
|
|
|
|
: friendlist_index[num_selected];
|
2013-11-09 08:50:32 +01:00
|
|
|
|
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-07-31 18:48:49 +02:00
|
|
|
if (key == ltr)
|
|
|
|
return;
|
|
|
|
|
|
|
|
switch (key) {
|
|
|
|
case '\n':
|
|
|
|
if (blocklist_view)
|
|
|
|
break;
|
|
|
|
|
2014-03-30 22:40:13 +02:00
|
|
|
/* 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 {
|
2014-07-25 04:43:32 +02:00
|
|
|
const char *msg = "* Warning: Too many windows are open.";
|
|
|
|
line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, msg);
|
2014-08-02 00:37:02 +02:00
|
|
|
sound_notify(prompt, error, NT_WNDALERT_1, NULL);
|
2014-03-30 22:40:13 +02:00
|
|
|
}
|
2014-07-31 18:48:49 +02:00
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case KEY_DC:
|
2014-03-30 22:40:13 +02:00
|
|
|
del_friend_activate(self, m, f);
|
2014-07-31 18:48:49 +02:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 'b':
|
|
|
|
if (!blocklist_view)
|
|
|
|
block_friend(m, f);
|
|
|
|
else
|
|
|
|
unblock_friend(m, f);
|
|
|
|
break;
|
|
|
|
|
|
|
|
case KEY_RIGHT:
|
|
|
|
case KEY_LEFT:
|
|
|
|
blocklist_view ^= 1;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
if (blocklist_view == 0)
|
|
|
|
select_friend(self, key, &num_selected, num_friends);
|
|
|
|
else
|
|
|
|
select_friend(self, key, &Blocked_Contacts.num_selected, Blocked_Contacts.num_blocked);
|
|
|
|
break;
|
2013-11-18 04:28:22 +01:00
|
|
|
}
|
2013-07-30 21:47:40 +02:00
|
|
|
}
|
|
|
|
|
2014-04-25 05:25:38 +02:00
|
|
|
#define FLIST_OFST 6 /* Accounts for space at top and bottom */
|
2013-12-03 00:23:04 +01:00
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
static void blocklist_onDraw(ToxWindow *self, Tox *m, int y2, int x2)
|
|
|
|
{
|
|
|
|
wattron(self->window, A_BOLD);
|
|
|
|
wprintw(self->window, " Blocked: ");
|
|
|
|
wattroff(self->window, A_BOLD);
|
|
|
|
wprintw(self->window, "%d\n\n", Blocked_Contacts.num_blocked);
|
|
|
|
|
|
|
|
if ((y2 - FLIST_OFST) <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int selected_num = 0;
|
|
|
|
|
|
|
|
/* Determine which portion of friendlist to draw based on current position */
|
|
|
|
int page = Blocked_Contacts.num_selected / (y2 - FLIST_OFST);
|
|
|
|
int start = (y2 - FLIST_OFST) * page;
|
|
|
|
int end = y2 - FLIST_OFST + start;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = start; i < Blocked_Contacts.num_blocked && i < end; ++i) {
|
|
|
|
int f = Blocked_Contacts.index[i];
|
|
|
|
bool f_selected = false;
|
|
|
|
|
|
|
|
if (i == Blocked_Contacts.num_selected) {
|
|
|
|
wattron(self->window, A_BOLD);
|
|
|
|
wprintw(self->window, " > ");
|
|
|
|
wattroff(self->window, A_BOLD);
|
|
|
|
selected_num = f;
|
|
|
|
f_selected = true;
|
|
|
|
} else {
|
|
|
|
wprintw(self->window, " ");
|
|
|
|
}
|
|
|
|
|
|
|
|
wattron(self->window, COLOR_PAIR(RED));
|
|
|
|
wprintw(self->window, "x");
|
|
|
|
wattroff(self->window, COLOR_PAIR(RED));
|
|
|
|
|
|
|
|
if (f_selected)
|
|
|
|
wattron(self->window, COLOR_PAIR(BLUE));
|
|
|
|
|
|
|
|
wattron(self->window, A_BOLD);
|
|
|
|
wprintw(self->window, " %s\n", Blocked_Contacts.list[f].name);
|
|
|
|
wattroff(self->window, A_BOLD);
|
|
|
|
|
|
|
|
if (f_selected)
|
|
|
|
wattroff(self->window, COLOR_PAIR(BLUE));
|
|
|
|
}
|
|
|
|
|
|
|
|
wprintw(self->window, "\n");
|
|
|
|
self->x = x2;
|
|
|
|
|
|
|
|
if (Blocked_Contacts.num_blocked) {
|
|
|
|
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", Blocked_Contacts.list[selected_num].pub_key[i] & 0xff);
|
|
|
|
}
|
|
|
|
|
|
|
|
wrefresh(self->window);
|
2014-07-31 19:49:15 +02:00
|
|
|
draw_del_popup();
|
|
|
|
|
|
|
|
if (self->help->active)
|
|
|
|
help_onDraw(self);
|
2014-07-31 18:48:49 +02:00
|
|
|
}
|
|
|
|
|
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);
|
2013-12-03 00:23:04 +01:00
|
|
|
int x2, y2;
|
|
|
|
getmaxyx(self->window, y2, x2);
|
2013-11-15 23:03:24 +01:00
|
|
|
|
2014-07-04 09:24:29 +02:00
|
|
|
bool fix_statuses = x2 != self->x; /* true if window max x value has changed */
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2013-12-15 01:29:45 +01:00
|
|
|
wattron(self->window, COLOR_PAIR(CYAN));
|
2014-07-31 19:49:15 +02:00
|
|
|
wprintw(self->window, " Press the");
|
2013-12-15 01:29:45 +01:00
|
|
|
wattron(self->window, A_BOLD);
|
2014-07-31 19:49:15 +02:00
|
|
|
wprintw(self->window, " H ");
|
2013-12-15 01:29:45 +01:00
|
|
|
wattroff(self->window, A_BOLD);
|
2014-07-31 19:49:15 +02:00
|
|
|
wprintw(self->window, "key for help\n\n");
|
2013-12-15 01:29:45 +01:00
|
|
|
wattroff(self->window, COLOR_PAIR(CYAN));
|
|
|
|
|
2014-07-31 18:48:49 +02:00
|
|
|
if (blocklist_view == 1) {
|
|
|
|
blocklist_onDraw(self, m, y2, x2);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint64_t cur_time = get_unix_time();
|
|
|
|
struct tm cur_loc_tm = *localtime((const time_t *) &cur_time);
|
|
|
|
|
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);
|
|
|
|
|
2013-12-15 01:29:45 +01:00
|
|
|
wattron(self->window, A_BOLD);
|
2014-03-15 09:14:46 +01:00
|
|
|
wprintw(self->window, " Online: ");
|
2013-12-15 01:29:45 +01:00
|
|
|
wattroff(self->window, A_BOLD);
|
2014-03-15 09:14:46 +01:00
|
|
|
wprintw(self->window, "%d/%d \n\n", nf, num_friends);
|
2013-08-16 19:11:09 +02:00
|
|
|
|
2014-07-04 09:24:29 +02:00
|
|
|
if ((y2 - FLIST_OFST) <= 0)
|
2013-12-03 00:34:14 +01:00
|
|
|
return;
|
|
|
|
|
2014-04-25 05:25:38 +02:00
|
|
|
int selected_num = 0;
|
|
|
|
|
2013-12-03 00:23:04 +01:00
|
|
|
/* 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;
|
|
|
|
|
2013-12-03 00:23:04 +01:00
|
|
|
for (i = start; i < num_friends && i < end; ++i) {
|
2013-11-09 08:50:32 +01:00
|
|
|
int f = friendlist_index[i];
|
2013-11-18 04:28:22 +01:00
|
|
|
bool f_selected = false;
|
2013-11-09 08:50:32 +01:00
|
|
|
|
|
|
|
if (friends[f].active) {
|
2013-11-18 01:45:53 +01:00
|
|
|
if (i == num_selected) {
|
|
|
|
wattron(self->window, A_BOLD);
|
2013-09-02 04:11:47 +02:00
|
|
|
wprintw(self->window, " > ");
|
2013-11-18 01:45:53 +01:00
|
|
|
wattroff(self->window, A_BOLD);
|
2014-04-25 05:25:38 +02:00
|
|
|
selected_num = f;
|
2013-11-18 01:45:53 +01:00
|
|
|
f_selected = true;
|
|
|
|
} else {
|
2013-09-02 04:11:47 +02:00
|
|
|
wprintw(self->window, " ");
|
2013-11-18 01:45:53 +01:00
|
|
|
}
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2013-11-09 08:50:32 +01:00
|
|
|
if (friends[f].online) {
|
2014-03-19 02:48:26 +01:00
|
|
|
uint8_t status = friends[f].status;
|
2013-09-05 06:47:33 +02:00
|
|
|
int colour = WHITE;
|
2013-09-02 04:11:47 +02:00
|
|
|
|
2013-11-12 07:50:04 +01:00
|
|
|
switch (status) {
|
2014-04-19 23:58:13 +02:00
|
|
|
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
|
|
|
}
|
|
|
|
|
2013-09-05 07:34:23 +02:00
|
|
|
wattron(self->window, COLOR_PAIR(colour) | A_BOLD);
|
2014-07-27 01:16:07 +02:00
|
|
|
wprintw(self->window, "%s ", ONLINE_CHAR);
|
2013-09-05 07:34:23 +02:00
|
|
|
wattroff(self->window, COLOR_PAIR(colour) | A_BOLD);
|
2013-11-18 01:45:53 +01:00
|
|
|
|
|
|
|
if (f_selected)
|
2014-03-15 09:14:46 +01:00
|
|
|
wattron(self->window, COLOR_PAIR(BLUE));
|
2013-11-18 01:45:53 +01:00
|
|
|
|
2014-03-15 09:14:46 +01:00
|
|
|
wattron(self->window, A_BOLD);
|
2013-11-18 01:45:53 +01:00
|
|
|
wprintw(self->window, "%s", friends[f].name);
|
2014-03-15 09:14:46 +01:00
|
|
|
wattroff(self->window, A_BOLD);
|
2013-11-18 01:45:53 +01:00
|
|
|
|
|
|
|
if (f_selected)
|
2014-03-15 09:14:46 +01:00
|
|
|
wattroff(self->window, COLOR_PAIR(BLUE));
|
2013-09-10 10:04:13 +02:00
|
|
|
|
2013-11-09 08:50:32 +01:00
|
|
|
/* Reset friends[f].statusmsg on window resize */
|
2013-11-15 23:03:24 +01:00
|
|
|
if (fix_statuses) {
|
2014-07-07 04:15:35 +02:00
|
|
|
char statusmsg[TOX_MAX_STATUSMESSAGE_LENGTH];
|
2014-03-14 04:30:44 +01:00
|
|
|
|
2014-03-13 11:06:53 +01:00
|
|
|
pthread_mutex_lock(&Winthread.lock);
|
2014-07-07 04:15:35 +02:00
|
|
|
tox_get_status_message(m, friends[f].num, (uint8_t *) 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
|
|
|
|
2014-03-13 11:06:53 +01:00
|
|
|
snprintf(friends[f].statusmsg, sizeof(friends[f].statusmsg), "%s", statusmsg);
|
2014-07-02 23:30:31 +02:00
|
|
|
friends[f].statusmsg_len = strlen(friends[f].statusmsg);
|
2013-09-13 08:02:49 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Truncate note if it doesn't fit on one line */
|
2014-03-15 09:14:46 +01:00
|
|
|
uint16_t maxlen = x2 - getcurx(self->window) - 2;
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2013-11-09 08:50:32 +01:00
|
|
|
if (friends[f].statusmsg_len > maxlen) {
|
2014-04-19 23:58:13 +02:00
|
|
|
friends[f].statusmsg[maxlen - 3] = '\0';
|
2013-11-15 23:03:24 +01:00
|
|
|
strcat(friends[f].statusmsg, "...");
|
2013-11-09 08:50:32 +01:00
|
|
|
friends[f].statusmsg[maxlen] = '\0';
|
|
|
|
friends[f].statusmsg_len = maxlen;
|
2013-09-10 10:04:13 +02:00
|
|
|
}
|
|
|
|
|
2014-03-14 22:48:52 +01:00
|
|
|
if (friends[f].statusmsg[0])
|
2014-03-15 09:14:46 +01:00
|
|
|
wprintw(self->window, " %s", friends[f].statusmsg);
|
2014-03-14 22:48:52 +01:00
|
|
|
|
|
|
|
wprintw(self->window, "\n");
|
2013-09-02 04:11:47 +02:00
|
|
|
} else {
|
2014-07-27 01:16:07 +02:00
|
|
|
wprintw(self->window, "%s ", OFFLINE_CHAR);
|
2013-11-18 01:45:53 +01:00
|
|
|
|
|
|
|
if (f_selected)
|
2014-03-15 09:14:46 +01:00
|
|
|
wattron(self->window, COLOR_PAIR(BLUE));
|
2013-11-18 01:45:53 +01:00
|
|
|
|
2014-03-15 09:14:46 +01:00
|
|
|
wattron(self->window, A_BOLD);
|
2014-03-14 04:56:46 +01:00
|
|
|
wprintw(self->window, "%s", friends[f].name);
|
2014-03-15 09:14:46 +01:00
|
|
|
wattroff(self->window, A_BOLD);
|
2013-11-18 01:45:53 +01:00
|
|
|
|
|
|
|
if (f_selected)
|
2014-05-01 10:00:19 +02:00
|
|
|
wattroff(self->window, COLOR_PAIR(BLUE));
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2014-03-15 22:18:23 +01:00
|
|
|
uint64_t last_seen = friends[f].last_online.last_on;
|
2014-03-14 22:48:52 +01:00
|
|
|
|
|
|
|
if (last_seen != 0) {
|
2014-03-15 22:18:23 +01:00
|
|
|
int day_dist = (cur_loc_tm.tm_yday - friends[f].last_online.tm.tm_yday) % 365;
|
2014-07-07 04:15:35 +02:00
|
|
|
const char *hourmin = friends[f].last_online.hour_min_str;
|
2014-03-14 22:48:52 +01:00
|
|
|
|
|
|
|
switch (day_dist) {
|
2014-04-19 23:58:13 +02:00
|
|
|
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;
|
|
|
|
}
|
2014-03-14 22:48:52 +01:00
|
|
|
} else {
|
2014-03-15 09:14:46 +01:00
|
|
|
wprintw(self->window, " Last seen: Never\n");
|
2014-03-14 22:48:52 +01:00
|
|
|
}
|
2013-09-02 04:11:47 +02:00
|
|
|
}
|
2013-11-10 03:43:56 +01:00
|
|
|
}
|
2013-08-16 19:11:09 +02:00
|
|
|
}
|
2013-09-05 03:25:59 +02:00
|
|
|
|
2013-12-03 00:23:04 +01:00
|
|
|
self->x = x2;
|
2014-04-25 05:25:38 +02:00
|
|
|
|
|
|
|
if (num_friends) {
|
|
|
|
wmove(self->window, y2 - 1, 1);
|
|
|
|
|
|
|
|
wattron(self->window, A_BOLD);
|
2014-04-26 10:09:51 +02:00
|
|
|
wprintw(self->window, "ID: ");
|
2014-04-25 05:25:38 +02:00
|
|
|
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-07-05 20:46:16 +02:00
|
|
|
|
|
|
|
wrefresh(self->window);
|
2014-07-31 19:49:15 +02:00
|
|
|
draw_del_popup();
|
|
|
|
|
|
|
|
if (self->help->active)
|
|
|
|
help_onDraw(self);
|
2013-07-30 21:47:40 +02:00
|
|
|
}
|
|
|
|
|
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;
|
2013-08-05 07:57:29 +02:00
|
|
|
}
|
|
|
|
|
2014-07-21 03:25:21 +02:00
|
|
|
#ifdef _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);
|
2014-04-19 23:58:13 +02:00
|
|
|
|
2014-03-11 23:22:27 +01:00
|
|
|
/*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;
|
2014-04-19 23:58:13 +02:00
|
|
|
|
|
|
|
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-07-21 01:12:13 +02:00
|
|
|
if (toxav_get_call_state(av, call_index) == av_CallStarting) { /* Only open windows when call is incoming */
|
2014-07-04 18:04:03 +02:00
|
|
|
friends[id].chatwin = add_window(m, new_chat(m, friends[id].num));
|
2014-07-21 01:12:13 +02:00
|
|
|
}
|
2014-03-07 03:14:04 +01:00
|
|
|
} else {
|
2014-07-07 04:15:35 +02:00
|
|
|
char nick[TOX_MAX_NAME_LENGTH];
|
2014-07-08 20:31:59 +02:00
|
|
|
get_nick_truncate(m, nick, friends[id].num);
|
2014-03-25 08:17:22 +01:00
|
|
|
|
2014-07-25 04:43:32 +02:00
|
|
|
const char *msg = "Audio action from: %s!";
|
|
|
|
line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, 0, msg, nick);
|
2014-03-25 13:21:50 +01:00
|
|
|
|
2014-07-25 04:43:32 +02:00
|
|
|
const char *errmsg = "* Warning: Too many windows are open.";
|
|
|
|
line_info_add(prompt, NULL, NULL, NULL, SYS_MSG, 0, RED, errmsg);
|
2014-07-21 01:12:13 +02:00
|
|
|
|
2014-08-02 00:37:02 +02:00
|
|
|
sound_notify(prompt, error, NT_WNDALERT_1, NULL);
|
2014-03-07 03:14:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2014-07-21 03:25:21 +02:00
|
|
|
#endif /* _AUDIO */
|
2014-03-07 03:14:04 +01:00
|
|
|
|
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));
|
|
|
|
|
2013-11-30 11:35:25 +01:00
|
|
|
ret.active = true;
|
2014-06-29 00:40:22 +02:00
|
|
|
ret.is_friendlist = true;
|
2013-11-30 11:35:25 +01:00
|
|
|
|
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;
|
2013-09-05 03:25:59 +02:00
|
|
|
ret.onConnectionChange = &friendlist_onConnectionChange;
|
2014-03-11 23:22:27 +01:00
|
|
|
ret.onAction = &friendlist_onMessage; /* Action has identical behaviour to message */
|
2013-08-16 19:11:09 +02:00
|
|
|
ret.onNickChange = &friendlist_onNickChange;
|
2013-09-05 03:25:59 +02:00
|
|
|
ret.onStatusChange = &friendlist_onStatusChange;
|
2013-09-03 05:27:34 +02:00
|
|
|
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-04-19 23:58:13 +02:00
|
|
|
|
2014-07-21 03:25:21 +02:00
|
|
|
#ifdef _AUDIO
|
2014-03-07 03:14:04 +01:00
|
|
|
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-07-21 03:25:21 +02:00
|
|
|
#endif /* _AUDIO */
|
2014-08-02 00:37:02 +02:00
|
|
|
|
|
|
|
ret.active_box = -1;
|
2014-07-31 19:49:15 +02:00
|
|
|
|
|
|
|
Help *help = calloc(1, sizeof(Help));
|
|
|
|
|
|
|
|
if (help == NULL)
|
|
|
|
exit_toxic_err("failed in new_friendlist", FATALERR_MEMORY);
|
2014-07-21 01:12:13 +02:00
|
|
|
|
2014-07-31 19:49:15 +02:00
|
|
|
ret.help = help;
|
2014-07-13 21:34:09 +02:00
|
|
|
strcpy(ret.name, "contacts");
|
2013-08-16 19:11:09 +02:00
|
|
|
return ret;
|
2013-07-30 21:47:40 +02:00
|
|
|
}
|