/* Copyright (C) 2014 by Project Tox 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 "filetransferwidget.h" #include "ui_filetransferwidget.h" #include "src/core.h" #include "src/misc/style.h" #include "src/widget/widget.h" #include #include #include #include #include #include #include #include #include #include FileTransferWidget::FileTransferWidget(QWidget *parent, ToxFile file) : QWidget(parent) , ui(new Ui::FileTransferWidget) , fileInfo(file) , lastTick(QTime::currentTime()) , backgroundColor(Style::getColor(Style::LightGrey)) , buttonColor(Style::getColor(Style::Yellow)) { ui->setupUi(this); // hide the QWidget background (background-color: transparent doesn't seem to work) setAttribute(Qt::WA_TranslucentBackground, true); ui->previewLabel->hide(); ui->filenameLabel->setText(file.fileName); ui->progressBar->setValue(0); ui->fileSizeLabel->setText(getHumanReadableSize(file.filesize)); ui->etaLabel->setText(""); backgroundColorAnimation = new QVariantAnimation(this); backgroundColorAnimation->setDuration(500); backgroundColorAnimation->setEasingCurve(QEasingCurve::OutCubic); connect(backgroundColorAnimation, &QVariantAnimation::valueChanged, this, [this](const QVariant& val) { backgroundColor = val.value(); update(); }); buttonColorAnimation = new QVariantAnimation(this); buttonColorAnimation->setDuration(500); buttonColorAnimation->setEasingCurve(QEasingCurve::OutCubic); connect(buttonColorAnimation, &QVariantAnimation::valueChanged, this, [this](const QVariant& val) { buttonColor = val.value(); update(); }); setBackgroundColor(Style::getColor(Style::LightGrey), false); connect(Core::getInstance(), &Core::fileTransferInfo, this, &FileTransferWidget::onFileTransferInfo); connect(Core::getInstance(), &Core::fileTransferAccepted, this, &FileTransferWidget::onFileTransferAccepted); connect(Core::getInstance(), &Core::fileTransferCancelled, this, &FileTransferWidget::onFileTransferCancelled); connect(Core::getInstance(), &Core::fileTransferPaused, this, &FileTransferWidget::onFileTransferPaused); connect(Core::getInstance(), &Core::fileTransferFinished, this, &FileTransferWidget::onFileTransferFinished); setupButtons(); //preview if(fileInfo.direction == ToxFile::SENDING) { showPreview(fileInfo.filePath); ui->progressLabel->setText(tr("Waiting to send...", "file transfer widget")); } else ui->progressLabel->setText(tr("Accept to receive this file", "file transfer widget")); setFixedHeight(78); } FileTransferWidget::~FileTransferWidget() { delete ui; } void FileTransferWidget::autoAcceptTransfer(const QString &path) { QString filepath; int number = 0; QString suffix = QFileInfo(fileInfo.fileName).completeSuffix(); QString base = QFileInfo(fileInfo.fileName).baseName(); do { filepath = QString("%1/%2%3.%4").arg(path, base, number > 0 ? QString(" (%1)").arg(QString::number(number)) : QString(), suffix); number++; } while(QFileInfo(filepath).exists()); //Do not automatically accept the file-transfer if the path is not writable. //The user can still accept it manually. if(isFilePathWritable(filepath)) Core::getInstance()->acceptFileRecvRequest(fileInfo.friendId, fileInfo.fileNum, filepath); else qDebug() << "Warning: Cannot write to " << filepath; } void FileTransferWidget::acceptTransfer(const QString &filepath) { if(filepath.isEmpty()) return; //test if writable if(!isFilePathWritable(filepath)) { 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")); return; } //everything ok! Core::getInstance()->acceptFileRecvRequest(fileInfo.friendId, fileInfo.fileNum, filepath); } void FileTransferWidget::setBackgroundColor(const QColor &c, bool whiteFont) { if(c != backgroundColor) { backgroundColorAnimation->setStartValue(backgroundColor); backgroundColorAnimation->setEndValue(c); backgroundColorAnimation->start(); } setProperty("fontColor", whiteFont ? "white" : "black"); setStyleSheet(Style::getStylesheet(":/ui/fileTransferInstance/filetransferWidget.css")); Style::repolish(this); update(); } void FileTransferWidget::setButtonColor(const QColor &c) { if(c != buttonColor) { buttonColorAnimation->setStartValue(buttonColor); buttonColorAnimation->setEndValue(c); buttonColorAnimation->start(); } } bool FileTransferWidget::isFilePathWritable(const QString &filepath) const { QFile tmp(filepath); bool writable = tmp.open(QIODevice::WriteOnly); tmp.remove(); return writable; } bool FileTransferWidget::drawButtonAreaNeeded() const { return (ui->bottomButton->isVisible() || ui->topButton->isVisible()) && !(ui->topButton->isVisible() && ui->topButton->objectName() == "ok"); } void FileTransferWidget::paintEvent(QPaintEvent *) { // required by Hi-DPI support as border-image doesn't work. QPainter painter(this); painter.setRenderHint(QPainter::Antialiasing); painter.setPen(Qt::NoPen); qreal ratio = static_cast(geometry().height()) / static_cast(geometry().width()); const int r = 24; const int buttonFieldWidth = 34; const int lineWidth = 1; // draw background if(drawButtonAreaNeeded()) painter.setClipRect(QRect(0,0,width()-buttonFieldWidth,height())); painter.setBrush(QBrush(backgroundColor)); painter.drawRoundRect(geometry(), r * ratio, r); if(drawButtonAreaNeeded()) { // draw button background (top) painter.setBrush(QBrush(buttonColor)); painter.setClipRect(QRect(width()-buttonFieldWidth+lineWidth,0,buttonFieldWidth,height()/2-ceil(lineWidth/2.0))); painter.drawRoundRect(geometry(), r * ratio, r); // draw button background (bottom) painter.setBrush(QBrush(buttonColor)); painter.setClipRect(QRect(width()-buttonFieldWidth+lineWidth,height()/2+lineWidth/2,buttonFieldWidth,height()/2)); painter.drawRoundRect(geometry(), r * ratio, r); } } void FileTransferWidget::onFileTransferInfo(ToxFile file) { QTime now = QTime::currentTime(); qint64 dt = lastTick.msecsTo(now); //ms if(fileInfo != file || dt < 1000) return; fileInfo = file; if(fileInfo.status == ToxFile::TRANSMITTING) { // update progress qreal progress = static_cast(file.bytesSent) / static_cast(file.filesize); ui->progressBar->setValue(static_cast(progress * 100.0)); // ETA, speed qreal deltaSecs = dt / 1000.0; qint64 deltaBytes = qMax(file.bytesSent - lastBytesSent, qint64(0)); qreal bytesPerSec = static_cast(static_cast(deltaBytes) / deltaSecs); // calculate mean meanIndex = meanIndex % TRANSFER_ROLLING_AVG_COUNT; meanData[meanIndex++] = bytesPerSec; qreal meanBytesPerSec = 0.0; for(size_t i = 0; i < TRANSFER_ROLLING_AVG_COUNT; ++i) meanBytesPerSec += meanData[i]; meanBytesPerSec /= static_cast(TRANSFER_ROLLING_AVG_COUNT); // update UI if(meanBytesPerSec > 0) { // ETA QTime toGo = QTime(0,0).addSecs((file.filesize - file.bytesSent) / meanBytesPerSec); QString format = toGo.hour() > 0 ? "hh:mm:ss" : "mm:ss"; ui->etaLabel->setText(toGo.toString(format)); } else { ui->etaLabel->setText(""); } ui->progressLabel->setText(getHumanReadableSize(meanBytesPerSec) + "/s"); lastBytesSent = file.bytesSent; } lastTick = now; // trigger repaint update(); } void FileTransferWidget::onFileTransferAccepted(ToxFile file) { if(fileInfo != file) return; fileInfo = file; setBackgroundColor(Style::getColor(Style::LightGrey), false); setupButtons(); } void FileTransferWidget::onFileTransferCancelled(ToxFile file) { if(fileInfo != file) return; fileInfo = file; setBackgroundColor(Style::getColor(Style::Red), true); setupButtons(); hideWidgets(); disconnect(Core::getInstance(), 0, this, 0); } void FileTransferWidget::onFileTransferPaused(ToxFile file) { if(fileInfo != file) return; fileInfo = file; ui->etaLabel->setText(""); ui->progressLabel->setText(tr("paused", "file transfer widget")); // reset mean meanIndex = 0; for(size_t i=0; itopButton->setIcon(QIcon(":/ui/fileTransferInstance/yes.svg")); ui->topButton->setObjectName("ok"); ui->topButton->show(); ui->bottomButton->setIcon(QIcon(":/ui/fileTransferInstance/dir.svg")); ui->bottomButton->setObjectName("dir"); ui->bottomButton->show(); // preview if(fileInfo.direction == ToxFile::RECEIVING) showPreview(fileInfo.filePath); disconnect(Core::getInstance(), 0, this, 0); } QString FileTransferWidget::getHumanReadableSize(qint64 size) { static const char* suffix[] = {"B","kiB","MiB","GiB","TiB"}; int exp = 0; if (size > 0) exp = std::min( (int) (log(size) / log(1024)), (int) (sizeof(suffix) / sizeof(suffix[0]) - 1)); return QString().setNum(size / pow(1024, exp),'f', exp > 1 ? 2 : 0).append(suffix[exp]); } void FileTransferWidget::hideWidgets() { ui->topButton->hide(); ui->bottomButton->hide(); ui->progressBar->hide(); ui->progressLabel->hide(); ui->etaLabel->hide(); } void FileTransferWidget::setupButtons() { switch(fileInfo.status) { case ToxFile::TRANSMITTING: ui->topButton->setIcon(QIcon(":/ui/fileTransferInstance/no.svg")); ui->topButton->setObjectName("cancel"); ui->bottomButton->setIcon(QIcon(":/ui/fileTransferInstance/pause.svg")); ui->bottomButton->setObjectName("pause"); setButtonColor(Style::getColor(Style::Green)); break; case ToxFile::PAUSED: ui->topButton->setIcon(QIcon(":/ui/fileTransferInstance/no.svg")); ui->topButton->setObjectName("cancel"); ui->bottomButton->setIcon(QIcon(":/ui/fileTransferInstance/arrow_white.svg")); ui->bottomButton->setObjectName("resume"); setButtonColor(Style::getColor(Style::LightGrey)); break; case ToxFile::STOPPED: case ToxFile::BROKEN: //TODO: ? ui->topButton->setIcon(QIcon(":/ui/fileTransferInstance/no.svg")); ui->topButton->setObjectName("cancel"); if(fileInfo.direction == ToxFile::SENDING) { ui->bottomButton->setIcon(QIcon(":/ui/fileTransferInstance/pause.svg")); ui->bottomButton->setObjectName("pause"); } else { ui->bottomButton->setIcon(QIcon(":/ui/fileTransferInstance/yes.svg")); ui->bottomButton->setObjectName("accept"); } break; } } void FileTransferWidget::handleButton(QPushButton *btn) { if(fileInfo.direction == ToxFile::SENDING) { if(btn->objectName() == "cancel") Core::getInstance()->cancelFileSend(fileInfo.friendId, fileInfo.fileNum); else if(btn->objectName() == "pause") Core::getInstance()->pauseResumeFileSend(fileInfo.friendId, fileInfo.fileNum); else if(btn->objectName() == "resume") Core::getInstance()->pauseResumeFileSend(fileInfo.friendId, fileInfo.fileNum); } else { if(btn->objectName() == "cancel") Core::getInstance()->cancelFileRecv(fileInfo.friendId, fileInfo.fileNum); else if(btn->objectName() == "pause") Core::getInstance()->pauseResumeFileRecv(fileInfo.friendId, fileInfo.fileNum); else if(btn->objectName() == "resume") Core::getInstance()->pauseResumeFileRecv(fileInfo.friendId, fileInfo.fileNum); else if(btn->objectName() == "accept") { QString path = QFileDialog::getSaveFileName(0, tr("Save a file","Title of the file saving dialog"), QDir::home().filePath(fileInfo.fileName)); acceptTransfer(path); } } if(btn->objectName() == "ok") { Widget::confirmExecutableOpen(QFileInfo(fileInfo.filePath)); } else if (btn->objectName() == "dir") { QString dirPath = QFileInfo(fileInfo.filePath).dir().path(); QDesktopServices::openUrl(QUrl("file://" + dirPath, QUrl::TolerantMode)); } } void FileTransferWidget::showPreview(const QString &filename) { static const QStringList previewExtensions = { "png", "jpeg", "jpg", "gif" }; if(previewExtensions.contains(QFileInfo(filename).suffix())) { const int size = qMax(ui->previewLabel->width(), ui->previewLabel->height()); QPixmap pmap = QPixmap(filename).scaled(QSize(size, size), Qt::KeepAspectRatioByExpanding, Qt::SmoothTransformation); ui->previewLabel->setPixmap(pmap); ui->previewLabel->show(); // Show preview, but make sure it's not larger than 50% of the screen width/height QRect maxSize = QApplication::desktop()->screenGeometry(); maxSize.setWidth(0.5*maxSize.width()); maxSize.setHeight(0.5*maxSize.height()); QImage image = QImage(filename); QSize imageSize(image.width(), image.height()); if (imageSize.width() > maxSize.width() || imageSize.height() > maxSize.height()) { imageSize.scale(maxSize.width(), maxSize.height(), Qt::KeepAspectRatio); ui->previewLabel->setToolTip(""); } else { ui->previewLabel->setToolTip(""); } } } void FileTransferWidget::on_topButton_clicked() { handleButton(ui->topButton); } void FileTransferWidget::on_bottomButton_clicked() { handleButton(ui->bottomButton); }