mirror of
https://github.com/qTox/qTox.git
synced 2024-03-22 14:00:36 +08:00
504 lines
16 KiB
C++
504 lines
16 KiB
C++
/*
|
|
Copyright (C) 2014 by Project Tox <https://tox.im>
|
|
|
|
This file is part of qTox, a Qt-based graphical interface for Tox.
|
|
|
|
This program is libre 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.
|
|
This program 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 COPYING file for more details.
|
|
*/
|
|
|
|
#include "filetransferinstance.h"
|
|
#include "core.h"
|
|
#include "misc/settings.h"
|
|
#include "misc/style.h"
|
|
#include <math.h>
|
|
#include <QFileDialog>
|
|
#include <QMessageBox>
|
|
#include <QBuffer>
|
|
#include <QDebug>
|
|
#include <QPainter>
|
|
|
|
#define MAX_CONTENT_WIDTH 250
|
|
#define MAX_PREVIEW_SIZE 25*1024*1024
|
|
|
|
uint FileTransferInstance::Idconter = 0;
|
|
|
|
FileTransferInstance::FileTransferInstance(ToxFile File)
|
|
: lastBytesSent{0},
|
|
fileNum{File.fileNum}, friendId{File.friendId}, direction{File.direction}
|
|
{
|
|
id = Idconter++;
|
|
state = tsPending;
|
|
remotePaused = false;
|
|
lastUpdateTime = QDateTime::currentDateTime();
|
|
|
|
filename = File.fileName;
|
|
|
|
// update this whenever you change the font in innerStyle.css
|
|
QFontMetrics fm(Style::getFont(Style::Small));
|
|
|
|
filenameElided = fm.elidedText(filename, Qt::ElideRight, MAX_CONTENT_WIDTH);
|
|
size = getHumanReadableSize(File.filesize);
|
|
contentPrefWidth = std::max(fm.boundingRect(filenameElided).width(), fm.width(size));
|
|
|
|
speed = "0B/s";
|
|
eta = "00:00";
|
|
|
|
if (File.direction == ToxFile::SENDING)
|
|
{
|
|
if (File.file->size() <= MAX_PREVIEW_SIZE)
|
|
{
|
|
QImage preview;
|
|
File.file->seek(0);
|
|
if (preview.loadFromData(File.file->readAll()))
|
|
{
|
|
pic = preview.scaled(100, 50, Qt::KeepAspectRatio, Qt::SmoothTransformation);
|
|
}
|
|
}
|
|
File.file->seek(0);
|
|
}
|
|
}
|
|
|
|
QString FileTransferInstance::getHumanReadableSize(unsigned long long size)
|
|
{
|
|
static const char* suffix[] = {"B","kiB","MiB","GiB","TiB"};
|
|
int exp = 0;
|
|
if (size)
|
|
exp = std::min( (int) (log(size) / log(1024)), (int) (sizeof(suffix) / sizeof(suffix[0]) - 1));
|
|
return QString().setNum(size / pow(1024, exp),'f',2).append(suffix[exp]);
|
|
}
|
|
|
|
void FileTransferInstance::onFileTransferInfo(int FriendId, int FileNum, int64_t Filesize, int64_t BytesSent, ToxFile::FileDirection Direction)
|
|
{
|
|
if (FileNum != fileNum || FriendId != friendId || Direction != direction)
|
|
return;
|
|
|
|
// state = tsProcessing;
|
|
QDateTime now = QDateTime::currentDateTime();
|
|
long recenttimediff = lastUpdateTime.msecsTo(now);
|
|
if (recenttimediff < 1000) //update every 1s
|
|
return;
|
|
|
|
long timediff = effStartTime.msecsTo(now);
|
|
if (timediff <= 0)
|
|
return;
|
|
|
|
long avgspeed = (BytesSent - previousBytesSent) / timediff * 1000;
|
|
long recentspeed = (BytesSent - lastBytesSent) / recenttimediff * 1000;
|
|
|
|
speed = getHumanReadableSize(recentspeed)+"/s";
|
|
size = getHumanReadableSize(Filesize);
|
|
totalBytes = Filesize;
|
|
|
|
if (!avgspeed)
|
|
return;
|
|
int etaSecs = (Filesize - BytesSent) / avgspeed;
|
|
QTime etaTime(0,0);
|
|
etaTime = etaTime.addSecs(etaSecs);
|
|
eta = etaTime.toString("mm:ss");
|
|
|
|
lastBytesSent = BytesSent;
|
|
lastUpdateTime = now;
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::onFileTransferCancelled(int FriendId, int FileNum, ToxFile::FileDirection Direction)
|
|
{
|
|
if (FileNum != fileNum || FriendId != friendId || Direction != direction)
|
|
return;
|
|
disconnect(Core::getInstance(),0,this,0);
|
|
state = tsCanceled;
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::onFileTransferFinished(ToxFile File)
|
|
{
|
|
if (File.fileNum != fileNum || File.friendId != friendId || File.direction != direction)
|
|
return;
|
|
disconnect(Core::getInstance(),0,this,0);
|
|
|
|
if (File.direction == ToxFile::RECEIVING)
|
|
{
|
|
QImage preview;
|
|
QFile previewFile(File.filePath);
|
|
if (previewFile.open(QIODevice::ReadOnly) && previewFile.size() <= MAX_PREVIEW_SIZE) // Don't preview big (>25MiB) images
|
|
{
|
|
if (preview.loadFromData(previewFile.readAll()))
|
|
{
|
|
pic = preview.scaled(100, 50, Qt::KeepAspectRatio, Qt::SmoothTransformation);
|
|
}
|
|
previewFile.close();
|
|
}
|
|
}
|
|
|
|
state = tsFinished;
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::onFileTransferAccepted(ToxFile File)
|
|
{
|
|
if (File.fileNum != fileNum || File.friendId != friendId || File.direction != direction)
|
|
return;
|
|
|
|
remotePaused = false;
|
|
state = tsProcessing;
|
|
effStartTime = QDateTime::currentDateTime();
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::onFileTransferRemotePausedUnpaused(ToxFile File, bool paused)
|
|
{
|
|
if (File.fileNum != fileNum || File.friendId != friendId || File.direction != direction)
|
|
return;
|
|
|
|
remotePaused = paused;
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::onFileTransferPaused(int FriendId, int FileNum, ToxFile::FileDirection Direction)
|
|
{
|
|
if (FileNum != fileNum || FriendId != friendId || Direction != direction)
|
|
return;
|
|
|
|
state = tsPaused;
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::cancelTransfer()
|
|
{
|
|
Core::getInstance()->cancelFileSend(friendId, fileNum);
|
|
state = tsCanceled;
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::rejectRecvRequest()
|
|
{
|
|
Core::getInstance()->rejectFileRecvRequest(friendId, fileNum);
|
|
onFileTransferCancelled(friendId, fileNum, direction);
|
|
state = tsCanceled;
|
|
emit stateUpdated();
|
|
}
|
|
|
|
// for whatever the fuck reason, QFileInfo::isWritable() always fails for files that don't exist
|
|
// which makes it useless for our case
|
|
// since QDir doesn't have an isWritable(), the only option I can think of is to make/delete the file
|
|
// surely this is a common problem that has a qt-implemented solution?
|
|
bool isFileWritable(QString& path)
|
|
{
|
|
QFile file(path);
|
|
bool exists = file.exists();
|
|
bool out = file.open(QIODevice::WriteOnly);
|
|
file.close();
|
|
if (!exists)
|
|
file.remove();
|
|
return out;
|
|
}
|
|
|
|
void FileTransferInstance::acceptRecvRequest()
|
|
{
|
|
QString path = Settings::getInstance().getAutoAcceptDir(Core::getInstance()->getFriendAddress(friendId));
|
|
if (path.isEmpty()) path = Settings::getInstance().getGlobalAutoAcceptDir();
|
|
if (!path.isEmpty())
|
|
{
|
|
QDir dir(path);
|
|
path = dir.filePath(filename);
|
|
QFileInfo info(path);
|
|
if (info.exists()) // emulate chrome
|
|
{
|
|
QString name = info.baseName(), ext = info.completeSuffix();
|
|
int i = 0;
|
|
do
|
|
{
|
|
path = dir.filePath(name + QString(" (%1)").arg(++i) + "." + ext);
|
|
}
|
|
while (QFileInfo(path).exists());
|
|
}
|
|
qDebug() << "File: auto saving to" << path;
|
|
}
|
|
else
|
|
{
|
|
while (true)
|
|
{
|
|
path = QFileDialog::getSaveFileName(0, tr("Save a file","Title of the file saving dialog"), QDir::home().filePath(filename));
|
|
if (path.isEmpty())
|
|
return;
|
|
else
|
|
{
|
|
if (isFileWritable(path))
|
|
break;
|
|
else
|
|
QMessageBox::warning(0, tr("Location not writable","Title of permissions popup"), tr("You do not have permission to write that location. Choose another, or cancel the save dialog.", "text of permissions popup"));
|
|
}
|
|
}
|
|
}
|
|
|
|
savePath = path;
|
|
|
|
Core::getInstance()->acceptFileRecvRequest(friendId, fileNum, path);
|
|
state = tsProcessing;
|
|
|
|
effStartTime = QDateTime::currentDateTime();
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::pauseResumeRecv()
|
|
{
|
|
if (!(state == tsProcessing || state == tsPaused))
|
|
return;
|
|
|
|
if (remotePaused)
|
|
return;
|
|
|
|
Core::getInstance()->pauseResumeFileRecv(friendId, fileNum);
|
|
// if (state == tsProcessing)
|
|
// state = tsPaused;
|
|
// else state = tsProcessing;
|
|
if (state == tsPaused)
|
|
{
|
|
effStartTime = QDateTime::currentDateTime();
|
|
previousBytesSent = lastBytesSent;
|
|
}
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
void FileTransferInstance::pauseResumeSend()
|
|
{
|
|
if (!(state == tsProcessing || state == tsPaused))
|
|
return;
|
|
|
|
if (remotePaused)
|
|
return;
|
|
|
|
Core::getInstance()->pauseResumeFileSend(friendId, fileNum);
|
|
// if (state == tsProcessing)
|
|
// state = tsPaused;
|
|
// else state = tsProcessing;
|
|
if (state == tsPaused)
|
|
{
|
|
effStartTime = QDateTime::currentDateTime();
|
|
previousBytesSent = lastBytesSent;
|
|
}
|
|
|
|
emit stateUpdated();
|
|
}
|
|
|
|
QString FileTransferInstance::QImage2base64(const QImage &img)
|
|
{
|
|
QByteArray ba;
|
|
QBuffer buffer(&ba);
|
|
buffer.open(QIODevice::WriteOnly);
|
|
img.save(&buffer, "PNG"); // writes image into ba in PNG format
|
|
return ba.toBase64();
|
|
}
|
|
|
|
QString FileTransferInstance::getHtmlImage()
|
|
{
|
|
//qDebug() << "QString FileTransferInstance::getHtmlImage() " << state;
|
|
|
|
QString res;
|
|
if (state == tsPending || state == tsProcessing || state == tsPaused)
|
|
{
|
|
QImage leftBtnImg(":/ui/fileTransferInstance/stopFileButton.png");
|
|
QImage rightBtnImg;
|
|
if (state == tsProcessing)
|
|
rightBtnImg = QImage(":/ui/fileTransferInstance/pauseFileButton.png");
|
|
else if (state == tsPaused)
|
|
rightBtnImg = QImage(":/ui/fileTransferInstance/resumeFileButton.png");
|
|
else
|
|
{
|
|
if (direction == ToxFile::SENDING)
|
|
rightBtnImg = QImage(":/ui/fileTransferInstance/pauseGreyFileButton.png");
|
|
else
|
|
rightBtnImg = QImage(":/ui/fileTransferInstance/acceptFileButton.png");
|
|
}
|
|
|
|
if (remotePaused)
|
|
rightBtnImg = QImage(":/ui/fileTransferInstance/pauseGreyFileButton.png");
|
|
|
|
res = draw2ButtonsForm("silver", leftBtnImg, rightBtnImg);
|
|
} else if (state == tsBroken)
|
|
{
|
|
QImage leftBtnImg(":/ui/fileTransferInstance/stopFileButton.png");
|
|
QImage rightBtnImg(":/ui/fileTransferInstance/pauseGreyFileButton.png");
|
|
|
|
res = draw2ButtonsForm("red", leftBtnImg, rightBtnImg);
|
|
} else if (state == tsCanceled)
|
|
{
|
|
res = drawButtonlessForm("red");
|
|
} else if (state == tsFinished)
|
|
{
|
|
res = drawButtonlessForm("green");
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
void FileTransferInstance::pressFromHtml(QString code)
|
|
{
|
|
if (state == tsFinished || state == tsCanceled)
|
|
return;
|
|
|
|
if (direction == ToxFile::SENDING)
|
|
{
|
|
if (code == "btnA")
|
|
cancelTransfer();
|
|
else if (code == "btnB")
|
|
pauseResumeSend();
|
|
} else {
|
|
if (code == "btnA")
|
|
rejectRecvRequest();
|
|
else if (code == "btnB")
|
|
{
|
|
if (state == tsPending)
|
|
acceptRecvRequest();
|
|
else
|
|
pauseResumeRecv();
|
|
}
|
|
}
|
|
}
|
|
|
|
QString FileTransferInstance::drawButtonlessForm(const QString &type)
|
|
{
|
|
QString imgAStr;
|
|
QString imgBStr;
|
|
|
|
if (type == "red")
|
|
{
|
|
imgAStr = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyLRedFileButton.png")) + "\">";
|
|
imgBStr = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyRRedFileButton.png")) + "\">";
|
|
} else {
|
|
imgAStr = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyLGreenFileButton.png")) + "\">";
|
|
imgBStr = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyRGreenFileButton.png")) + "\">";
|
|
}
|
|
|
|
QString content = "<p>" + filenameElided + "</p><p>" + size + "</p>";
|
|
|
|
return wrapIntoForm(content, type, imgAStr, imgBStr);
|
|
}
|
|
|
|
QString FileTransferInstance::insertMiniature(const QString &type)
|
|
{
|
|
if (pic == QImage())
|
|
return QString();
|
|
|
|
QString widgetId = QString::number(getId());
|
|
|
|
QString res;
|
|
res = "<td><div class=" + type + ">\n";
|
|
res += "<img src=\"data:mini." + widgetId + "/png;base64," + QImage2base64(pic) + "\">";
|
|
res += "</div></td>\n";
|
|
return res;
|
|
}
|
|
|
|
QString FileTransferInstance::draw2ButtonsForm(const QString &type, const QImage &imgA, const QImage &imgB)
|
|
{
|
|
QString widgetId = QString::number(getId());
|
|
QString imgAstr = "<img src=\"data:ftrans." + widgetId + ".btnA/png;base64," + QImage2base64(imgA) + "\">";
|
|
QString imgBstr = "<img src=\"data:ftrans." + widgetId + ".btnB/png;base64," + QImage2base64(imgB) + "\">";
|
|
|
|
QString content;
|
|
QString progrBar = "<img src=\"data:progressbar." + widgetId + "/png;base64," +
|
|
QImage2base64(drawProgressBarImg(double(lastBytesSent)/totalBytes, MAX_CONTENT_WIDTH, 9)) + "\">";
|
|
|
|
content = "<p>" + filenameElided + "</p>";
|
|
content += "<table cellspacing=\"0\"><tr>";
|
|
content += "<td>" + size + "</td>";
|
|
content += "<td align=center>" + speed + "</td>";
|
|
content += "<td align=right>ETA: " + eta + "</td>";
|
|
content += "</tr><tr><td colspan=3>";
|
|
content += progrBar;
|
|
content += "</td></tr></table>";
|
|
|
|
return wrapIntoForm(content, type, imgAstr, imgBstr);
|
|
}
|
|
|
|
QString FileTransferInstance::wrapIntoForm(const QString& content, const QString &type, const QString &imgAstr, const QString &imgBstr)
|
|
{
|
|
QString w = QString::number(QImage(":/ui/fileTransferInstance/emptyLRedFileButton.png").size().width());
|
|
QString imgLeftA, imgLeftB;
|
|
|
|
if (type == "green")
|
|
{
|
|
imgLeftA = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyLGreenFileButton.png").mirrored(true,false)) + "\">";
|
|
imgLeftB = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyLGreenFileButton.png").mirrored(true,true)) + "\">";
|
|
}
|
|
|
|
if (type == "silver")
|
|
{
|
|
imgLeftA = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/sliverRTEdge.png").mirrored(true,false)) + "\">";
|
|
imgLeftB = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/sliverRTEdge.png").mirrored(true,true)) + "\">";
|
|
}
|
|
|
|
if (type == "red")
|
|
{
|
|
imgLeftA = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyLRedFileButton.png").mirrored(true,false)) + "\">";
|
|
imgLeftB = "<img src=\"data:placeholder/png;base64," + QImage2base64(QImage(":/ui/fileTransferInstance/emptyLRedFileButton.png").mirrored(true,true)) + "\">";
|
|
}
|
|
|
|
QString res;
|
|
res = "<table cellspacing=\"0\">\n";
|
|
res += "<tr valign=middle>\n";
|
|
res += "<td width=" + w + ">\n";
|
|
res += "<div class=button>" + imgLeftA + "<br>" + imgLeftB + "</div>\n";
|
|
res += "</td>\n";
|
|
res += insertMiniature(type);
|
|
res += "<td width=" + QString::number(contentPrefWidth) + ">\n";
|
|
res += "<div class=" + type + ">";
|
|
res += content;
|
|
res += "</div>\n";
|
|
res += "</td>\n";
|
|
res += "<td width=3>\n";
|
|
res += "<div class=" + type + "></div>\n";
|
|
res += "</td>\n";
|
|
res += "<td>\n";
|
|
res += "<div class=button>" + imgAstr + "<br>" + imgBstr + "</div>\n";
|
|
res += "</td>\n";
|
|
res += "</tr>\n";
|
|
res += "</table>\n";
|
|
|
|
return res;
|
|
}
|
|
|
|
QImage FileTransferInstance::drawProgressBarImg(const double &part, int w, int h)
|
|
{
|
|
QImage progressBar(w, h, QImage::Format_Mono);
|
|
|
|
QPainter qPainter(&progressBar);
|
|
qPainter.setBrush(Qt::NoBrush);
|
|
qPainter.setPen(Qt::black);
|
|
qPainter.drawRect(0, 0, w - 1, h - 1);
|
|
|
|
qPainter.setBrush(Qt::SolidPattern);
|
|
qPainter.setPen(Qt::black);
|
|
qPainter.drawRect(1, 0, (w - 2) * (part), h - 1);
|
|
|
|
return progressBar;
|
|
}
|
|
|
|
void FileTransferInstance::onFileTransferBrokenUnbroken(ToxFile File, bool broken)
|
|
{
|
|
if (File.fileNum != fileNum || File.friendId != friendId || File.direction != direction)
|
|
return;
|
|
|
|
if (broken)
|
|
state = tsBroken;
|
|
else
|
|
state = tsProcessing;
|
|
|
|
emit stateUpdated();
|
|
}
|