2015-03-28 07:56:54 +01:00
|
|
|
/* file_transfers.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 <stdlib.h>
|
|
|
|
#include <time.h>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include "toxic.h"
|
|
|
|
#include "windows.h"
|
|
|
|
#include "friendlist.h"
|
|
|
|
#include "file_transfers.h"
|
|
|
|
#include "line_info.h"
|
|
|
|
#include "misc_tools.h"
|
|
|
|
#include "notify.h"
|
|
|
|
|
|
|
|
extern FriendsList Friends;
|
|
|
|
|
|
|
|
#define NUM_PROG_MARKS 50 /* number of "#"'s in file transfer progress bar. Keep well below MAX_STR_SIZE */
|
|
|
|
|
|
|
|
/* creates initial progress line that will be updated during file transfer.
|
|
|
|
Assumes progline is of size MAX_STR_SIZE */
|
|
|
|
void prep_prog_line(char *progline)
|
|
|
|
{
|
|
|
|
strcpy(progline, "0.0 B/s [");
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < NUM_PROG_MARKS; ++i)
|
|
|
|
strcat(progline, "-");
|
|
|
|
|
|
|
|
strcat(progline, "] 0%");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* prints a progress bar for file transfers.
|
|
|
|
if friendnum is -1 we're sending the file, otherwise we're receiving. */
|
|
|
|
void print_progress_bar(ToxWindow *self, double bps, double pct_done, uint32_t line_id)
|
|
|
|
{
|
2015-03-30 00:33:51 +02:00
|
|
|
if (bps < 0 || pct_done < 0 || pct_done > 100)
|
|
|
|
return;
|
|
|
|
|
2015-03-28 07:56:54 +01:00
|
|
|
char msg[MAX_STR_SIZE];
|
|
|
|
bytes_convert_str(msg, sizeof(msg), bps);
|
|
|
|
strcat(msg, "/s [");
|
|
|
|
|
|
|
|
int n = pct_done / (100 / NUM_PROG_MARKS);
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
for (i = 0; i < n; ++i)
|
|
|
|
strcat(msg, "#");
|
|
|
|
|
|
|
|
for (j = i; j < NUM_PROG_MARKS; ++j)
|
|
|
|
strcat(msg, "-");
|
|
|
|
|
|
|
|
strcat(msg, "] ");
|
|
|
|
|
|
|
|
char pctstr[16];
|
|
|
|
const char *frmt = pct_done == 100 ? "%.f%%" : "%.1f%%";
|
|
|
|
snprintf(pctstr, sizeof(pctstr), frmt, pct_done);
|
|
|
|
strcat(msg, pctstr);
|
|
|
|
|
|
|
|
line_info_set(self, line_id, msg);
|
|
|
|
}
|
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
static void refresh_progress_helper(ToxWindow *self, struct FileTransfer *ft, uint64_t curtime)
|
|
|
|
{
|
|
|
|
if (ft->state == FILE_TRANSFER_INACTIVE)
|
|
|
|
return;
|
|
|
|
|
|
|
|
/* Timeout must be set to 1 second to show correct bytes per second */
|
|
|
|
if (!timed_out(ft->last_progress, curtime, 1))
|
|
|
|
return;
|
2015-03-28 07:56:54 +01:00
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
double remain = ft->file_size - ft->position;
|
|
|
|
double pct_done = remain > 0 ? (1 - (remain / ft->file_size)) * 100 : 100;
|
|
|
|
print_progress_bar(self, ft->bps, pct_done, ft->line_id);
|
|
|
|
|
|
|
|
ft->bps = 0;
|
|
|
|
ft->last_progress = curtime;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* refreshes active file receiver status bars for friendnum */
|
|
|
|
void refresh_file_transfer_progress(ToxWindow *self, Tox *m, uint32_t friendnum)
|
2015-03-28 07:56:54 +01:00
|
|
|
{
|
2015-03-30 00:33:51 +02:00
|
|
|
uint64_t curtime = get_unix_time();
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_FILES; ++i) {
|
|
|
|
refresh_progress_helper(self, &Friends.list[friendnum].file_receiver[i], curtime);
|
|
|
|
refresh_progress_helper(self, &Friends.list[friendnum].file_sender[i], curtime);
|
|
|
|
}
|
2015-03-28 07:56:54 +01:00
|
|
|
}
|
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
/* Returns a pointer to friendnum's FileTransfer struct associated with filenum.
|
|
|
|
* Returns NULL if filenum is invalid.
|
|
|
|
*/
|
|
|
|
struct FileTransfer *get_file_transfer_struct(uint32_t friendnum, uint32_t filenum)
|
2015-03-28 07:56:54 +01:00
|
|
|
{
|
2015-03-30 00:33:51 +02:00
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_FILES; ++i) {
|
|
|
|
struct FileTransfer *ft_send = &Friends.list[friendnum].file_sender[i];
|
|
|
|
|
|
|
|
if (ft_send->state != FILE_TRANSFER_INACTIVE && ft_send->filenum == filenum)
|
|
|
|
return ft_send;
|
|
|
|
|
|
|
|
struct FileTransfer *ft_recv = &Friends.list[friendnum].file_receiver[i];
|
|
|
|
|
|
|
|
if (ft_recv->state != FILE_TRANSFER_INACTIVE && ft_recv->filenum == filenum)
|
|
|
|
return ft_recv;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2015-03-28 07:56:54 +01:00
|
|
|
}
|
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
/* Returns a pointer to friendnum's file receiver associated with index with the direction specified.
|
|
|
|
* Returns NULL on failure.
|
|
|
|
*/
|
|
|
|
struct FileTransfer *get_file_transfer_struct_index(uint32_t friendnum, uint32_t index,
|
|
|
|
FILE_TRANSFER_DIRECTION direction)
|
2015-03-28 07:56:54 +01:00
|
|
|
{
|
2015-03-30 00:33:51 +02:00
|
|
|
if (direction != FILE_TRANSFER_RECV && direction != FILE_TRANSFER_SEND)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_FILES; ++i) {
|
|
|
|
struct FileTransfer *ft = direction == FILE_TRANSFER_SEND ?
|
|
|
|
&Friends.list[friendnum].file_sender[i] :
|
|
|
|
&Friends.list[friendnum].file_receiver[i];
|
|
|
|
if (ft->index == index)
|
|
|
|
return ft;
|
|
|
|
}
|
|
|
|
|
|
|
|
return NULL;
|
2015-03-28 07:56:54 +01:00
|
|
|
}
|
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
/* Returns a pointer to an unused file sender.
|
|
|
|
* Returns NULL if all file senders are in use.
|
|
|
|
*/
|
|
|
|
struct FileTransfer *get_new_file_sender(uint32_t friendnum)
|
2015-03-28 07:56:54 +01:00
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_FILES; ++i) {
|
2015-03-30 00:33:51 +02:00
|
|
|
struct FileTransfer *ft = &Friends.list[friendnum].file_sender[i];
|
|
|
|
|
|
|
|
if (ft->state == FILE_TRANSFER_INACTIVE) {
|
|
|
|
ft->index = i;
|
|
|
|
return ft;
|
2015-03-28 07:56:54 +01:00
|
|
|
}
|
2015-03-30 00:33:51 +02:00
|
|
|
}
|
2015-03-28 07:56:54 +01:00
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
return NULL;
|
|
|
|
}
|
2015-03-28 07:56:54 +01:00
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
/* Returns a pointer to an unused file receiver.
|
|
|
|
* Returns NULL if all file receivers are in use.
|
|
|
|
*/
|
|
|
|
struct FileTransfer *get_new_file_receiver(uint32_t friendnum)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_FILES; ++i) {
|
|
|
|
struct FileTransfer *ft = &Friends.list[friendnum].file_receiver[i];
|
2015-03-28 07:56:54 +01:00
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
if (ft->state == FILE_TRANSFER_INACTIVE) {
|
|
|
|
ft->index = i;
|
|
|
|
return ft;
|
2015-03-28 07:56:54 +01:00
|
|
|
}
|
|
|
|
}
|
2015-03-30 00:33:51 +02:00
|
|
|
|
|
|
|
return NULL;
|
2015-03-28 07:56:54 +01:00
|
|
|
}
|
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
/* Closes file transfer ft.
|
|
|
|
*
|
2015-03-28 07:56:54 +01:00
|
|
|
* Set CTRL to -1 if we don't want to send a control signal.
|
|
|
|
* Set message or self to NULL if we don't want to display a message.
|
|
|
|
*/
|
2015-03-30 00:33:51 +02:00
|
|
|
void close_file_transfer(ToxWindow *self, Tox *m, struct FileTransfer *ft, int CTRL, const char *message,
|
|
|
|
Notification sound_type)
|
2015-03-28 07:56:54 +01:00
|
|
|
{
|
2015-03-30 00:33:51 +02:00
|
|
|
if (!ft)
|
|
|
|
return;
|
2015-03-28 07:56:54 +01:00
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
if (ft->state == FILE_TRANSFER_INACTIVE)
|
|
|
|
return;
|
2015-03-28 07:56:54 +01:00
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
if (ft->file)
|
|
|
|
fclose(ft->file);
|
2015-03-28 07:56:54 +01:00
|
|
|
|
2015-03-30 00:33:51 +02:00
|
|
|
memset(ft, 0, sizeof(struct FileTransfer));
|
2015-03-28 07:56:54 +01:00
|
|
|
|
|
|
|
if (CTRL >= 0)
|
2015-03-30 00:33:51 +02:00
|
|
|
tox_file_control(m, ft->friendnum, ft->filenum, (TOX_FILE_CONTROL) CTRL, NULL);
|
2015-03-28 07:56:54 +01:00
|
|
|
|
|
|
|
if (message && self) {
|
|
|
|
if (self->active_box != -1)
|
|
|
|
box_notify2(self, sound_type, NT_NOFOCUS | NT_WNDALERT_2, self->active_box, "%s", message);
|
|
|
|
else
|
|
|
|
box_notify(self, sound_type, NT_NOFOCUS | NT_WNDALERT_2, &self->active_box, self->name, "%s", message);
|
|
|
|
|
|
|
|
line_info_add(self, NULL, NULL, NULL, SYS_MSG, 0, 0, "%s", message);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Kills all active file transfers for friendnum */
|
|
|
|
void kill_all_file_transfers_friend(Tox *m, uint32_t friendnum)
|
|
|
|
{
|
|
|
|
size_t i;
|
|
|
|
|
|
|
|
for (i = 0; i < MAX_FILES; ++i) {
|
2015-03-30 00:33:51 +02:00
|
|
|
close_file_transfer(NULL, m, &Friends.list[friendnum].file_sender[i], -1, NULL, silent);
|
|
|
|
close_file_transfer(NULL, m, &Friends.list[friendnum].file_receiver[i], -1, NULL, silent);
|
2015-03-28 07:56:54 +01:00
|
|
|
}
|
|
|
|
}
|