2013-09-21 02:35:03 +02:00
|
|
|
/*
|
|
|
|
* Toxic -- Tox Curses Client
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <limits.h>
|
|
|
|
|
|
|
|
#include "toxic_windows.h"
|
2013-12-01 04:12:43 +01:00
|
|
|
#include "misc_tools.h"
|
2013-09-21 02:35:03 +02:00
|
|
|
|
|
|
|
// XXX: FIX
|
|
|
|
unsigned char *hex_string_to_bin(char hex_string[])
|
|
|
|
{
|
|
|
|
size_t len = strlen(hex_string);
|
|
|
|
unsigned char *val = malloc(len);
|
|
|
|
|
|
|
|
if (val == NULL) {
|
|
|
|
endwin();
|
|
|
|
fprintf(stderr, "malloc() failed. Aborting...\n");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
char *pos = hex_string;
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < len; ++i, pos += 2)
|
|
|
|
sscanf(pos, "%2hhx", &val[i]);
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Get the current local time */
|
|
|
|
struct tm *get_time(void)
|
|
|
|
{
|
|
|
|
struct tm *timeinfo;
|
|
|
|
time_t now;
|
|
|
|
time(&now);
|
|
|
|
timeinfo = localtime(&now);
|
|
|
|
return timeinfo;
|
|
|
|
}
|
|
|
|
|
2013-10-11 10:42:30 +02:00
|
|
|
/* Prints the time to given window */
|
|
|
|
void print_time(WINDOW *window)
|
|
|
|
{
|
|
|
|
struct tm *timeinfo = get_time();
|
|
|
|
|
|
|
|
wattron(window, COLOR_PAIR(CYAN));
|
|
|
|
wprintw(window, "[%02d:%02d:%02d] ", timeinfo->tm_hour, timeinfo->tm_min, timeinfo->tm_sec);
|
|
|
|
wattroff(window, COLOR_PAIR(CYAN));
|
|
|
|
}
|
|
|
|
|
2013-10-19 05:58:33 +02:00
|
|
|
/* Returns 1 if the string is empty, 0 otherwise */
|
2013-09-21 02:35:03 +02:00
|
|
|
int string_is_empty(char *string)
|
|
|
|
{
|
2013-10-22 07:59:06 +02:00
|
|
|
return string[0] == '\0';
|
2013-09-21 02:35:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/* convert wide characters to null terminated string */
|
|
|
|
uint8_t *wcs_to_char(wchar_t *string)
|
|
|
|
{
|
2013-10-23 09:24:08 +02:00
|
|
|
uint8_t *ret = NULL;
|
2013-10-25 06:29:40 +02:00
|
|
|
size_t len = wcstombs(NULL, string, 0);
|
2013-09-21 02:35:03 +02:00
|
|
|
|
|
|
|
if (len != (size_t) -1) {
|
2013-10-25 06:33:00 +02:00
|
|
|
ret = malloc(++len);
|
2013-10-25 06:29:40 +02:00
|
|
|
|
2013-09-21 02:35:03 +02:00
|
|
|
if (ret != NULL)
|
|
|
|
wcstombs(ret, string, len);
|
|
|
|
} else {
|
|
|
|
ret = malloc(2);
|
2013-10-25 06:29:40 +02:00
|
|
|
|
2013-09-21 02:35:03 +02:00
|
|
|
if (ret != NULL) {
|
|
|
|
ret[0] = ' ';
|
|
|
|
ret[1] = '\0';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ret == NULL) {
|
|
|
|
endwin();
|
|
|
|
fprintf(stderr, "malloc() failed. Aborting...\n");
|
|
|
|
exit(EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* convert a wide char to null terminated string */
|
|
|
|
char *wc_to_char(wchar_t ch)
|
|
|
|
{
|
|
|
|
static char ret[MB_LEN_MAX + 1];
|
2013-10-25 06:29:40 +02:00
|
|
|
int len = wctomb(ret, ch);
|
2013-09-21 02:35:03 +02:00
|
|
|
|
|
|
|
if (len == -1) {
|
|
|
|
ret[0] = ' ';
|
|
|
|
ret[1] = '\0';
|
|
|
|
} else {
|
|
|
|
ret[len] = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
return ret;
|
|
|
|
}
|
2013-11-15 01:14:54 +01:00
|
|
|
|
|
|
|
/* Returns true if connection has timed out, false otherwise */
|
|
|
|
bool timed_out(uint64_t timestamp, uint64_t curtime, uint64_t timeout)
|
|
|
|
{
|
|
|
|
return timestamp + timeout <= curtime;
|
2013-11-15 01:34:40 +01:00
|
|
|
}
|
2013-11-19 08:31:22 +01:00
|
|
|
|
2013-11-29 01:45:28 +01:00
|
|
|
/* Colours the window tab according to type. Beeps if is_beep is true */
|
|
|
|
void alert_window(ToxWindow *self, int type, bool is_beep)
|
2013-11-19 08:31:22 +01:00
|
|
|
{
|
2013-11-29 01:45:28 +01:00
|
|
|
if (type == WINDOW_ALERT_1)
|
|
|
|
self->alert1 = true;
|
|
|
|
else if(type == WINDOW_ALERT_2)
|
|
|
|
self->alert2 = true;
|
|
|
|
|
|
|
|
if (is_beep)
|
|
|
|
beep();
|
2013-11-19 08:31:22 +01:00
|
|
|
}
|
2013-11-24 08:33:03 +01:00
|
|
|
|
|
|
|
/* case-insensitive string compare function for use with qsort - same return logic as strcmp */
|
|
|
|
int name_compare(const void *nick1, const void *nick2)
|
|
|
|
{
|
2013-11-28 08:53:43 +01:00
|
|
|
char s[TOX_MAX_NAME_LENGTH];
|
|
|
|
char t[TOX_MAX_NAME_LENGTH];
|
2013-11-24 21:18:05 +01:00
|
|
|
strcpy(s, (const char *) nick1);
|
|
|
|
strcpy(t, (const char *) nick2);
|
2013-11-24 08:33:03 +01:00
|
|
|
|
|
|
|
int i;
|
|
|
|
|
2013-11-28 08:53:43 +01:00
|
|
|
for (i = 0; s[i] && t[i]; ++i) {
|
2013-11-24 08:33:03 +01:00
|
|
|
s[i] = tolower(s[i]);
|
|
|
|
t[i] = tolower(t[i]);
|
|
|
|
|
|
|
|
if (s[i] != t[i])
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return s[i] - t[i];
|
2013-11-24 21:18:05 +01:00
|
|
|
}
|
2013-11-28 08:53:43 +01:00
|
|
|
|
|
|
|
/* Returns true if nick is valid. A valid toxic nick:
|
|
|
|
- cannot be empty
|
|
|
|
- cannot start with a space
|
|
|
|
- must not contain contiguous spaces */
|
|
|
|
bool valid_nick(uint8_t *nick)
|
|
|
|
{
|
|
|
|
if (!nick[0] || nick[0] == ' ')
|
|
|
|
return false;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; nick[i]; ++i) {
|
|
|
|
if (nick[i] == ' ' && nick[i+1] == ' ')
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
2013-11-29 00:56:56 +01:00
|
|
|
}
|
2013-12-01 04:12:43 +01:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Buffer helper tools. Assumes buffers are no larger than MAX_STR_SIZE
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Adds char to buffer at pos */
|
|
|
|
void add_char_to_buf(wint_t ch, wchar_t *buf, size_t *pos, size_t *len)
|
|
|
|
{
|
|
|
|
if (*pos < 0 || *len >= MAX_STR_SIZE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* if cursor is at end of line simply insert char at last position and increment */
|
|
|
|
if (buf[*pos] == L'\0') {
|
|
|
|
buf[(*pos)++] = ch;
|
|
|
|
buf[*pos] = L'\0';
|
|
|
|
++(*len);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* move all chars in front of pos one space forward and insert char in pos */
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = *len; i >= *pos && i >= 0; --i)
|
|
|
|
buf[i+1] = buf[i];
|
|
|
|
|
|
|
|
buf[(*pos)++] = ch;
|
|
|
|
++(*len);
|
|
|
|
}
|
|
|
|
|
2013-12-01 08:58:21 +01:00
|
|
|
/* Deletes the character before pos */
|
|
|
|
void del_char_buf_bck(wchar_t *buf, size_t *pos, size_t *len)
|
2013-12-01 04:12:43 +01:00
|
|
|
{
|
|
|
|
if (*pos <= 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* if cursor is at end of line delete previous char */
|
|
|
|
if (buf[*pos] == L'\0') {
|
|
|
|
buf[--(*pos)] = L'\0';
|
|
|
|
--(*len);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
/* similar to add_char_to_buf but deletes a char */
|
|
|
|
for (i = *pos-1; i <= *len; ++i)
|
|
|
|
buf[i] = buf[i+1];
|
|
|
|
|
|
|
|
--(*pos);
|
|
|
|
--(*len);
|
|
|
|
}
|
|
|
|
|
2013-12-01 08:58:21 +01:00
|
|
|
/* Deletes the character at pos */
|
|
|
|
void del_char_buf_frnt(wchar_t *buf, size_t *pos, size_t *len)
|
|
|
|
{
|
|
|
|
if (*pos < 0 || *pos >= *len)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = *pos; i < *len; ++i)
|
|
|
|
buf[i] = buf[i+1];
|
|
|
|
|
|
|
|
--(*len);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* nulls buf and sets pos and len to 0 */
|
2013-12-01 04:12:43 +01:00
|
|
|
void reset_buf(wchar_t *buf, size_t *pos, size_t *len)
|
|
|
|
{
|
|
|
|
buf[0] = L'\0';
|
|
|
|
*pos = 0;
|
|
|
|
*len = 0;
|
|
|
|
}
|