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-10 01:25:09 +01:00
|
|
|
extern ToxWindow *prompt;
|
|
|
|
|
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();
|
|
|
|
|
2013-12-05 09:29:39 +01:00
|
|
|
wattron(window, COLOR_PAIR(BLUE));
|
2013-10-11 10:42:30 +02:00
|
|
|
wprintw(window, "[%02d:%02d:%02d] ", timeinfo->tm_hour, timeinfo->tm_min, timeinfo->tm_sec);
|
2013-12-05 09:29:39 +01:00
|
|
|
wattroff(window,COLOR_PAIR(BLUE));
|
2013-10-11 10:42:30 +02:00
|
|
|
}
|
|
|
|
|
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
|
|
|
}
|
|
|
|
|
2013-12-08 04:10:32 +01:00
|
|
|
/* convert a multibyte string to a wide character string (must provide buffer) */
|
2014-01-23 05:29:28 +01:00
|
|
|
int mbs_to_wcs_buf(wchar_t *buf, const uint8_t *string, size_t n)
|
2013-12-08 04:10:32 +01:00
|
|
|
{
|
|
|
|
size_t len = mbstowcs(NULL, string, 0) + 1;
|
|
|
|
|
|
|
|
if (n < len)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if ((len = mbstowcs(buf, string, n)) == (size_t) -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* converts wide character string into a multibyte string.
|
2013-12-08 10:16:49 +01:00
|
|
|
Same thing as wcs_to_mbs() but caller must provide its own buffer */
|
|
|
|
int wcs_to_mbs_buf(uint8_t *buf, const wchar_t *string, size_t n)
|
2013-12-08 04:10:32 +01:00
|
|
|
{
|
|
|
|
size_t len = wcstombs(NULL, string, 0) + 1;
|
|
|
|
|
|
|
|
if (n < len)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
if ((len = wcstombs(buf, string, n)) == (size_t) -1)
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
return len;
|
|
|
|
}
|
|
|
|
|
2013-12-08 10:16:49 +01:00
|
|
|
/* convert wide characters to multibyte string: string returned must be free'd */
|
|
|
|
uint8_t *wcs_to_mbs(wchar_t *string)
|
2013-09-21 02:35:03 +02:00
|
|
|
{
|
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-12-08 04:10:32 +01:00
|
|
|
if (ret != NULL) {
|
|
|
|
if (wcstombs(ret, string, len) == (size_t) -1)
|
|
|
|
return NULL;
|
|
|
|
}
|
2013-09-21 02:35:03 +02:00
|
|
|
} 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;
|
|
|
|
}
|
|
|
|
|
2013-12-08 04:10:32 +01:00
|
|
|
/* convert a wide char to multibyte string */
|
2013-09-21 02:35:03 +02:00
|
|
|
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-12-07 01:41:53 +01:00
|
|
|
switch (type) {
|
|
|
|
case WINDOW_ALERT_0:
|
|
|
|
self->alert0 = true;
|
|
|
|
break;
|
|
|
|
case WINDOW_ALERT_1:
|
2013-11-29 01:45:28 +01:00
|
|
|
self->alert1 = true;
|
2013-12-07 01:41:53 +01:00
|
|
|
break;
|
|
|
|
case WINDOW_ALERT_2:
|
2013-11-29 01:45:28 +01:00
|
|
|
self->alert2 = true;
|
2013-12-07 01:41:53 +01:00
|
|
|
break;
|
|
|
|
}
|
2013-12-10 01:25:09 +01:00
|
|
|
|
|
|
|
StatusBar *stb = prompt->stb;
|
|
|
|
|
|
|
|
if (is_beep && stb->status != TOX_USERSTATUS_BUSY)
|
2013-11-29 01:45:28 +01:00
|
|
|
beep();
|
2013-11-19 08:31:22 +01:00
|
|
|
}
|
2013-11-24 08:33:03 +01:00
|
|
|
|
2013-12-08 07:18:10 +01:00
|
|
|
/* case-insensitive string compare function for use with qsort */
|
|
|
|
int qsort_strcasecmp_hlpr(const void *nick1, const void *nick2)
|
2013-11-24 08:33:03 +01:00
|
|
|
{
|
2013-12-08 07:18:10 +01:00
|
|
|
return strcasecmp((const char *) nick1, (const char *) nick2);
|
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-11 06:10:09 +01:00
|
|
|
|
|
|
|
/* Moves cursor to the end of the line in given window */
|
|
|
|
void mv_curs_end(WINDOW *w, size_t len, int max_y, int max_x)
|
|
|
|
{
|
|
|
|
int end_y = (len / max_x) + (max_y - CURS_Y_OFFSET);
|
|
|
|
int end_x = len % max_x;
|
|
|
|
wmove(w, end_y, end_x);
|
2013-12-11 09:29:31 +01:00
|
|
|
}
|
2014-02-12 00:52:04 +01:00
|
|
|
|
|
|
|
/* Returns base file name from path or original file name if no path is supplied */
|
|
|
|
uint8_t *get_file_name(uint8_t *pathname)
|
|
|
|
{
|
|
|
|
int idx = strlen(pathname) - 1;
|
|
|
|
|
|
|
|
while (idx >= 0 && pathname[idx] == '/')
|
|
|
|
pathname[idx--] = '\0';
|
|
|
|
|
|
|
|
uint8_t *filename = strrchr(pathname, '/'); // Try unix style paths
|
|
|
|
|
|
|
|
if (filename != NULL) {
|
|
|
|
if (!strlen(++filename))
|
|
|
|
filename = pathname;
|
|
|
|
} else {
|
|
|
|
filename = strrchr(pathname, '\\'); // Try windows style paths
|
|
|
|
|
|
|
|
if (filename == NULL)
|
|
|
|
filename = pathname;
|
|
|
|
}
|
|
|
|
|
|
|
|
return filename;
|
|
|
|
}
|