/**************************************************************** * RetroShare is distributed under the following license: * * Copyright (C) 2006,2007 crypton * * This program 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 2 * 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 * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. ****************************************************************/ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "TransfersDialog.h" #include #include "DetailsDialog.h" #include "DLListDelegate.h" #include "ULListDelegate.h" #include "FileTransferInfoWidget.h" #include #include #include "xprogressbar.h" #include #include "util/misc.h" #include #include "TransferUserNotify.h" #include "util/QtVersion.h" #include "util/RsFile.h" #include #include #include #include #include /* Images for context menu icons */ #define IMAGE_INFO ":/images/fileinfo.png" #define IMAGE_CANCEL ":/images/delete.png" #define IMAGE_CLEARCOMPLETED ":/images/deleteall.png" #define IMAGE_PLAY ":/images/player_play.png" #define IMAGE_COPYLINK ":/images/copyrslink.png" #define IMAGE_PASTELINK ":/images/pasterslink.png" #define IMAGE_PAUSE ":/images/pause.png" #define IMAGE_RESUME ":/images/resume.png" #define IMAGE_OPENFOLDER ":/images/folderopen.png" #define IMAGE_OPENFILE ":/images/fileopen.png" #define IMAGE_STOP ":/images/stop.png" #define IMAGE_PREVIEW ":/images/preview.png" #define IMAGE_PRIORITY ":/images/filepriority.png" #define IMAGE_PRIORITYLOW ":/images/prioritylow.png" #define IMAGE_PRIORITYNORMAL ":/images/prioritynormal.png" #define IMAGE_PRIORITYHIGH ":/images/priorityhigh.png" #define IMAGE_PRIORITYAUTO ":/images/priorityauto.png" #define IMAGE_SEARCH ":/icons/svg/magnifying-glass.svg" #define IMAGE_EXPAND ":/images/edit_add24.png" #define IMAGE_COLLAPSE ":/images/edit_remove24.png" #define IMAGE_LIBRARY ":/images/library.png" #define IMAGE_COLLCREATE ":/images/library_add.png" #define IMAGE_COLLMODIF ":/images/library_edit.png" #define IMAGE_COLLVIEW ":/images/library_view.png" #define IMAGE_COLLOPEN ":/images/library.png" #define IMAGE_FRIENDSFILES ":/icons/svg/folders.svg" #define IMAGE_MYFILES ":icons/svg/folders1.svg" #define IMAGE_RENAMEFILE ":images/filecomments.png" #define IMAGE_STREAMING ":images/streaming.png" #define IMAGE_TUNNEL_ANON_E2E ":/images/blue_lock.png" #define IMAGE_TUNNEL_ANON ":/images/blue_lock_open.png" #define IMAGE_TUNNEL_FRIEND ":/icons/avatar_128.png" Q_DECLARE_METATYPE(FileProgressInfo) class SortByNameItem : public QStandardItem { public: SortByNameItem(QHeaderView *header) : QStandardItem() { this->header = header; } virtual bool operator<(const QStandardItem &other) const { QStandardItemModel *m = model(); if (m == NULL) { return QStandardItem::operator<(other); } QStandardItem *myParent = parent(); QStandardItem *otherParent = other.parent(); if (myParent == NULL || otherParent == NULL) { return QStandardItem::operator<(other); } QStandardItem *myName = myParent->child(index().row(), COLUMN_NAME); QStandardItem *otherName = otherParent->child(other.index().row(), COLUMN_NAME); if (header == NULL || header->sortIndicatorOrder() == Qt::AscendingOrder) { /* Ascending */ return *myName < *otherName; } /* Descending, sort peers in ascending order */ return !(*myName < *otherName); } private: QHeaderView *header; }; class ProgressItem : public SortByNameItem { public: ProgressItem(QHeaderView *header) : SortByNameItem(header) {} virtual bool operator<(const QStandardItem &other) const { const int role = model() ? model()->sortRole() : Qt::DisplayRole; FileProgressInfo l = data(role).value(); FileProgressInfo r = other.data(role).value(); if (l < r) { return true; } if (l > r) { return false; } return SortByNameItem::operator<(other); } }; class PriorityItem : public SortByNameItem { public: PriorityItem(QHeaderView *header) : SortByNameItem(header) {} virtual bool operator<(const QStandardItem &other) const { const int role = model() ? model()->sortRole() : Qt::DisplayRole; QString l = data(role).value(); QString r = other.data(role).value(); bool bl,br ; int nl = l.toInt(&bl) ; int nr = r.toInt(&br) ; if(bl && br) return nl < nr ; if(bl ^ br) return br ; return SortByNameItem::operator<(other); } }; /** Constructor */ TransfersDialog::TransfersDialog(QWidget *parent) : RsAutoUpdatePage(1000,parent) { /* Invoke the Qt Designer generated object setup routine */ ui.setupUi(this); m_bProcessSettings = false; connect( ui.downloadList, SIGNAL( customContextMenuRequested( QPoint ) ), this, SLOT( downloadListCustomPopupMenu( QPoint ) ) ); // Set Download list model DLListModel = new QStandardItemModel(0,COLUMN_COUNT); DLListModel->setHeaderData(COLUMN_NAME, Qt::Horizontal, tr("Name", "i.e: file name")); DLListModel->setHeaderData(COLUMN_SIZE, Qt::Horizontal, tr("Size", "i.e: file size")); DLListModel->setHeaderData(COLUMN_COMPLETED, Qt::Horizontal, tr("Completed", "")); DLListModel->setHeaderData(COLUMN_DLSPEED, Qt::Horizontal, tr("Speed", "i.e: Download speed")); DLListModel->setHeaderData(COLUMN_PROGRESS, Qt::Horizontal, tr("Progress / Availability", "i.e: % downloaded")); DLListModel->setHeaderData(COLUMN_SOURCES, Qt::Horizontal, tr("Sources", "i.e: Sources")); DLListModel->setHeaderData(COLUMN_STATUS, Qt::Horizontal, tr("Status")); DLListModel->setHeaderData(COLUMN_PRIORITY, Qt::Horizontal, tr("Speed / Queue position")); DLListModel->setHeaderData(COLUMN_REMAINING, Qt::Horizontal, tr("Remaining")); DLListModel->setHeaderData(COLUMN_DOWNLOADTIME, Qt::Horizontal, tr("Download time", "i.e: Estimated Time of Arrival / Time left")); DLListModel->setHeaderData(COLUMN_ID, Qt::Horizontal, tr("Hash")); DLListModel->setHeaderData(COLUMN_LASTDL, Qt::Horizontal, tr("Last Time Seen", "i.e: Last Time Receiced Data")); DLListModel->setHeaderData(COLUMN_PATH, Qt::Horizontal, tr("Path", "i.e: Where file is saved")); DLLFilterModel = new QSortFilterProxyModel(this); DLLFilterModel->setSourceModel( DLListModel); DLLFilterModel->setFilterCaseSensitivity(Qt::CaseInsensitive); ui.downloadList->setModel(DLLFilterModel); DLDelegate = new DLListDelegate(); ui.downloadList->setItemDelegate(DLDelegate); QHeaderView *qhvDLList = ui.downloadList->header(); qhvDLList->setContextMenuPolicy(Qt::CustomContextMenu); connect(qhvDLList, SIGNAL(customContextMenuRequested(QPoint)), this, SLOT(downloadListHeaderCustomPopupMenu(QPoint))); // Why disable autoscroll ? // With disabled autoscroll, the treeview doesn't scroll with cursor move // ui.downloadList->setAutoScroll(false) ; // workaround for Qt bug, should be solved in next Qt release 4.7.0 // http://bugreports.qt.nokia.com/browse/QTBUG-8270 mShortcut = new QShortcut(QKeySequence (Qt::Key_Delete), ui.downloadList, 0, 0, Qt::WidgetShortcut); connect(mShortcut, SIGNAL(activated()), this, SLOT( cancel ())); //Selection Setup selection = ui.downloadList->selectionModel(); ui.downloadList->setSelectionMode(QAbstractItemView::ExtendedSelection); ui.downloadList->setRootIsDecorated(true); /* Set header resize modes and initial section sizes Downloads TreeView*/ QHeaderView * dlheader = ui.downloadList->header () ; QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_NAME, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_SIZE, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_COMPLETED, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_DLSPEED, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_PROGRESS, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_SOURCES, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_STATUS, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_PRIORITY, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_REMAINING, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_DOWNLOADTIME, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_ID, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_LASTDL, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(dlheader, COLUMN_PATH, QHeaderView::Interactive); dlheader->resizeSection ( COLUMN_NAME, 170 ); dlheader->resizeSection ( COLUMN_SIZE, 70 ); dlheader->resizeSection ( COLUMN_COMPLETED, 75 ); dlheader->resizeSection ( COLUMN_DLSPEED, 75 ); dlheader->resizeSection ( COLUMN_PROGRESS, 170 ); dlheader->resizeSection ( COLUMN_SOURCES, 90 ); dlheader->resizeSection ( COLUMN_STATUS, 100 ); dlheader->resizeSection ( COLUMN_PRIORITY, 100 ); dlheader->resizeSection ( COLUMN_REMAINING, 100 ); dlheader->resizeSection ( COLUMN_DOWNLOADTIME, 100 ); dlheader->resizeSection ( COLUMN_ID, 100 ); dlheader->resizeSection ( COLUMN_LASTDL, 100 ); dlheader->resizeSection ( COLUMN_PATH, 100 ); // set default column and sort order for download ui.downloadList->sortByColumn(COLUMN_NAME, Qt::AscendingOrder); connect(ui.filterLineEdit, SIGNAL(textChanged(QString)), this, SLOT(filterChanged(QString))); /* Add filter actions */ QString headerName = DLListModel->headerData(COLUMN_NAME, Qt::Horizontal).toString(); ui.filterLineEdit->addFilter(QIcon(), headerName, COLUMN_NAME , QString("%1 %2").arg(tr("Search"), headerName)); QString headerID = DLListModel->headerData(COLUMN_ID, Qt::Horizontal).toString(); ui.filterLineEdit->addFilter(QIcon(), headerID, COLUMN_ID , QString("%1 %2").arg(tr("Search"), headerID)); connect( ui.uploadsList, SIGNAL( customContextMenuRequested( QPoint ) ), this, SLOT( uploadsListCustomPopupMenu( QPoint ) ) ); // Set Upload list model ULListModel = new QStandardItemModel(0,COLUMN_UCOUNT); ULListModel->setHeaderData(COLUMN_UNAME, Qt::Horizontal, tr("Name", "i.e: file name")); ULListModel->setHeaderData(COLUMN_UPEER, Qt::Horizontal, tr("Peer", "i.e: user name / tunnel id")); ULListModel->setHeaderData(COLUMN_USIZE, Qt::Horizontal, tr("Size", "i.e: file size")); ULListModel->setHeaderData(COLUMN_UTRANSFERRED, Qt::Horizontal, tr("Transferred", "")); ULListModel->setHeaderData(COLUMN_ULSPEED, Qt::Horizontal, tr("Speed", "i.e: upload speed")); ULListModel->setHeaderData(COLUMN_UPROGRESS, Qt::Horizontal, tr("Progress", "i.e: % uploaded")); ULListModel->setHeaderData(COLUMN_UHASH, Qt::Horizontal, tr("Hash", "")); ui.uploadsList->setModel(ULListModel); ULDelegate = new ULListDelegate(); ui.uploadsList->setItemDelegate(ULDelegate); // Why disable autoscroll ? // With disabled autoscroll, the treeview doesn't scroll with cursor move // ui.uploadsList->setAutoScroll(false) ; //Selection Setup selectionUp = ui.uploadsList->selectionModel(); ui.uploadsList->setSelectionMode(QAbstractItemView::ExtendedSelection); ui.uploadsList->setRootIsDecorated(true); /* Set header resize modes and initial section sizes Uploads TreeView*/ QHeaderView * upheader = ui.uploadsList->header () ; QHeaderView_setSectionResizeModeColumn(upheader, COLUMN_UNAME, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(upheader, COLUMN_UPEER, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(upheader, COLUMN_USIZE, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(upheader, COLUMN_UTRANSFERRED, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(upheader, COLUMN_ULSPEED, QHeaderView::Interactive); QHeaderView_setSectionResizeModeColumn(upheader, COLUMN_UPROGRESS, QHeaderView::Interactive); upheader->resizeSection ( COLUMN_UNAME, 260 ); upheader->resizeSection ( COLUMN_UPEER, 120 ); upheader->resizeSection ( COLUMN_USIZE, 70 ); upheader->resizeSection ( COLUMN_UTRANSFERRED, 75 ); upheader->resizeSection ( COLUMN_ULSPEED, 75 ); upheader->resizeSection ( COLUMN_UPROGRESS, 170 ); // set default column and sort order for upload ui.uploadsList->sortByColumn(COLUMN_UNAME, Qt::AscendingOrder); QObject::connect(ui.downloadList->selectionModel(),SIGNAL(selectionChanged (const QItemSelection&, const QItemSelection&)),this,SLOT(showFileDetails())) ; ui.tabWidget->insertTab(2,searchDialog = new SearchDialog(), QIcon(IMAGE_SEARCH), tr("Search")) ; ui.tabWidget->insertTab(3,remoteSharedFiles = new RemoteSharedFilesDialog(), QIcon(IMAGE_FRIENDSFILES), tr("Friends files")) ; ui.tabWidget->addTab(localSharedFiles = new LocalSharedFilesDialog(), QIcon(IMAGE_MYFILES), tr("My files")) ; //ui.tabWidget->addTab( new TurtleRouterStatistics(), tr("Router Statistics")) ; //ui.tabWidget->addTab( new TurtleRouterDialog(), tr("Router Requests")) ; for(int i=0;inbPlugins();++i) if(rsPlugins->plugin(i) != NULL && rsPlugins->plugin(i)->qt_transfers_tab() != NULL) ui.tabWidget->addTab( rsPlugins->plugin(i)->qt_transfers_tab(),QString::fromUtf8(rsPlugins->plugin(i)->qt_transfers_tab_name().c_str()) ) ; ui.tabWidget->setCurrentWidget(ui.uploadsTab); // TurtleRouterDialog *trdl = new TurtleRouterDialog(); // ui.tunnelInfoWidget->setWidget(trdl); // ui.tunnelInfoWidget->setWidgetResizable(true); // ui.tunnelInfoWidget->setHorizontalScrollBarPolicy(Qt::ScrollBarAlwaysOff); // ui.tunnelInfoWidget->setVerticalScrollBarPolicy(Qt::ScrollBarAlwaysOn); // ui.tunnelInfoWidget->viewport()->setBackgroundRole(QPalette::NoRole); // ui.tunnelInfoWidget->setFrameStyle(QFrame::NoFrame); // ui.tunnelInfoWidget->setFocusPolicy(Qt::NoFocus); /** Setup the actions for the context menu */ // Actions. Only need to be defined once. pauseAct = new QAction(QIcon(IMAGE_PAUSE), tr("Pause"), this); connect(pauseAct, SIGNAL(triggered()), this, SLOT(pauseFileTransfer())); resumeAct = new QAction(QIcon(IMAGE_RESUME), tr("Resume"), this); connect(resumeAct, SIGNAL(triggered()), this, SLOT(resumeFileTransfer())); //#ifdef USE_NEW_CHUNK_CHECKING_CODE // *********WARNING********** // csoler: this has been suspended because it needs the file transfer to consider a file as complete only if all chunks are // verified by hash. As users are goign to slowly switch to new checking code, this will not be readily available. // forceCheckAct = new QAction(QIcon(IMAGE_CANCEL), tr( "Force Check" ), this ); connect( forceCheckAct , SIGNAL( triggered() ), this, SLOT( forceCheck() ) ); //#endif cancelAct = new QAction(QIcon(IMAGE_CANCEL), tr( "Cancel" ), this ); connect( cancelAct , SIGNAL( triggered() ), this, SLOT( cancel() ) ); openFolderAct = new QAction(QIcon(IMAGE_OPENFOLDER), tr("Open Folder"), this); connect(openFolderAct, SIGNAL(triggered()), this, SLOT(dlOpenFolder())); openFileAct = new QAction(QIcon(IMAGE_OPENFILE), tr("Open File"), this); connect(openFileAct, SIGNAL(triggered()), this, SLOT(dlOpenFile())); previewFileAct = new QAction(QIcon(IMAGE_PREVIEW), tr("Preview File"), this); connect(previewFileAct, SIGNAL(triggered()), this, SLOT(dlPreviewFile())); detailsFileAct = new QAction(QIcon(IMAGE_INFO), tr("Details..."), this); connect(detailsFileAct, SIGNAL(triggered()), this, SLOT(showDetailsDialog())); clearCompletedAct = new QAction(QIcon(IMAGE_CLEARCOMPLETED), tr( "Clear Completed" ), this ); connect( clearCompletedAct , SIGNAL( triggered() ), this, SLOT( clearcompleted() ) ); copyLinkAct = new QAction(QIcon(IMAGE_COPYLINK), tr( "Copy RetroShare Link" ), this ); connect( copyLinkAct , SIGNAL( triggered() ), this, SLOT( dlCopyLink() ) ); pasteLinkAct = new QAction(QIcon(IMAGE_PASTELINK), tr( "Paste RetroShare Link" ), this ); connect( pasteLinkAct , SIGNAL( triggered() ), this, SLOT( pasteLink() ) ); queueDownAct = new QAction(QIcon(":/images/go-down.png"), tr("Down"), this); connect(queueDownAct, SIGNAL(triggered()), this, SLOT(priorityQueueDown())); queueUpAct = new QAction(QIcon(":/images/go-up.png"), tr("Up"), this); connect(queueUpAct, SIGNAL(triggered()), this, SLOT(priorityQueueUp())); queueTopAct = new QAction(QIcon(":/images/go-top.png"), tr("Top"), this); connect(queueTopAct, SIGNAL(triggered()), this, SLOT(priorityQueueTop())); queueBottomAct = new QAction(QIcon(":/images/go-bottom.png"), tr("Bottom"), this); connect(queueBottomAct, SIGNAL(triggered()), this, SLOT(priorityQueueBottom())); chunkStreamingAct = new QAction(QIcon(IMAGE_STREAMING), tr("Streaming"), this); connect(chunkStreamingAct, SIGNAL(triggered()), this, SLOT(chunkStreaming())); prioritySlowAct = new QAction(QIcon(IMAGE_PRIORITYLOW), tr("Slower"), this); connect(prioritySlowAct, SIGNAL(triggered()), this, SLOT(speedSlow())); priorityMediumAct = new QAction(QIcon(IMAGE_PRIORITYNORMAL), tr("Average"), this); connect(priorityMediumAct, SIGNAL(triggered()), this, SLOT(speedAverage())); priorityFastAct = new QAction(QIcon(IMAGE_PRIORITYHIGH), tr("Faster"), this); connect(priorityFastAct, SIGNAL(triggered()), this, SLOT(speedFast())); chunkRandomAct = new QAction(QIcon(IMAGE_PRIORITYAUTO), tr("Random"), this); connect(chunkRandomAct, SIGNAL(triggered()), this, SLOT(chunkRandom())); chunkProgressiveAct = new QAction(QIcon(IMAGE_PRIORITYAUTO), tr("Progressive"), this); connect(chunkProgressiveAct, SIGNAL(triggered()), this, SLOT(chunkProgressive())); playAct = new QAction(QIcon(IMAGE_PLAY), tr( "Play" ), this ); connect( playAct , SIGNAL( triggered() ), this, SLOT( dlOpenFile() ) ); renameFileAct = new QAction(QIcon(IMAGE_RENAMEFILE), tr("Rename file..."), this); connect(renameFileAct, SIGNAL(triggered()), this, SLOT(renameFile())); specifyDestinationDirectoryAct = new QAction(QIcon(IMAGE_SEARCH),tr("Specify..."),this) ; connect(specifyDestinationDirectoryAct,SIGNAL(triggered()),this,SLOT(chooseDestinationDirectory())); expandAllDLAct= new QAction(QIcon(IMAGE_EXPAND),tr("Expand all"),this); connect(expandAllDLAct,SIGNAL(triggered()),this,SLOT(expandAllDL())); collapseAllDLAct= new QAction(QIcon(IMAGE_COLLAPSE),tr("Collapse all"),this); connect(collapseAllDLAct,SIGNAL(triggered()),this,SLOT(collapseAllDL())); expandAllULAct= new QAction(QIcon(IMAGE_EXPAND),tr("Expand all"),this); connect(expandAllULAct,SIGNAL(triggered()),this,SLOT(expandAllUL())); collapseAllULAct= new QAction(QIcon(IMAGE_COLLAPSE),tr("Collapse all"),this); connect(collapseAllULAct,SIGNAL(triggered()),this,SLOT(collapseAllUL())); collCreateAct= new QAction(QIcon(IMAGE_COLLCREATE), tr("Create Collection..."), this); connect(collCreateAct,SIGNAL(triggered()),this,SLOT(collCreate())); collModifAct= new QAction(QIcon(IMAGE_COLLMODIF), tr("Modify Collection..."), this); connect(collModifAct,SIGNAL(triggered()),this,SLOT(collModif())); collViewAct= new QAction(QIcon(IMAGE_COLLVIEW), tr("View Collection..."), this); connect(collViewAct,SIGNAL(triggered()),this,SLOT(collView())); collOpenAct = new QAction(QIcon(IMAGE_COLLOPEN), tr( "Download from collection file..." ), this ); connect(collOpenAct, SIGNAL(triggered()), this, SLOT(collOpen())); /** Setup the actions for the header context menu */ showDLSizeAct= new QAction(tr("Size"),this); showDLSizeAct->setCheckable(true); showDLSizeAct->setToolTip(tr("Show Size Column")); connect(showDLSizeAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLSizeColumn(bool))) ; showDLCompleteAct= new QAction(tr("Completed"),this); showDLCompleteAct->setCheckable(true); showDLCompleteAct->setToolTip(tr("Show Completed Column")); connect(showDLCompleteAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLCompleteColumn(bool))) ; showDLDLSpeedAct= new QAction(tr("Speed"),this); showDLDLSpeedAct->setCheckable(true); showDLDLSpeedAct->setToolTip(tr("Show Speed Column")); connect(showDLDLSpeedAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLDLSpeedColumn(bool))) ; showDLProgressAct= new QAction(tr("Progress / Availability"),this); showDLProgressAct->setCheckable(true); showDLProgressAct->setToolTip(tr("Show Progress / Availability Column")); connect(showDLProgressAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLProgressColumn(bool))) ; showDLSourcesAct= new QAction(tr("Sources"),this); showDLSourcesAct->setCheckable(true); showDLSourcesAct->setToolTip(tr("Show Sources Column")); connect(showDLSourcesAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLSourcesColumn(bool))) ; showDLStatusAct= new QAction(tr("Status"),this); showDLStatusAct->setCheckable(true); showDLStatusAct->setToolTip(tr("Show Status Column")); connect(showDLStatusAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLStatusColumn(bool))) ; showDLPriorityAct= new QAction(tr("Speed / Queue position"),this); showDLPriorityAct->setCheckable(true); showDLPriorityAct->setToolTip(tr("Show Speed / Queue position Column")); connect(showDLPriorityAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLPriorityColumn(bool))) ; showDLRemainingAct= new QAction(tr("Remaining"),this); showDLRemainingAct->setCheckable(true); showDLRemainingAct->setToolTip(tr("Show Remaining Column")); connect(showDLRemainingAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLRemainingColumn(bool))) ; showDLDownloadTimeAct= new QAction(tr("Download time"),this); showDLDownloadTimeAct->setCheckable(true); showDLDownloadTimeAct->setToolTip(tr("Show Download time Column")); connect(showDLDownloadTimeAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLDownloadTimeColumn(bool))) ; showDLIDAct= new QAction(tr("Hash"),this); showDLIDAct->setCheckable(true); showDLIDAct->setToolTip(tr("Show Hash Column")); connect(showDLIDAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLIDColumn(bool))) ; showDLLastDLAct= new QAction(tr("Last Time Seen"),this); showDLLastDLAct->setCheckable(true); showDLLastDLAct->setToolTip(tr("Show Last Time Seen Column")); connect(showDLLastDLAct,SIGNAL(triggered(bool)),this,SLOT(setShowDLLastDLColumn(bool))) ; showDLPath= new QAction(tr("Path"),this); showDLPath->setCheckable(true); showDLPath->setToolTip(tr("Show Path Column")); connect(showDLPath,SIGNAL(triggered(bool)),this,SLOT(setShowDLPath(bool))) ; /** Setup the actions for the upload context menu */ ulOpenFolderAct = new QAction(QIcon(IMAGE_OPENFOLDER), tr("Open Folder"), this); connect(ulOpenFolderAct, SIGNAL(triggered()), this, SLOT(ulOpenFolder())); ulCopyLinkAct = new QAction(QIcon(IMAGE_COPYLINK), tr( "Copy RetroShare Link" ), this ); connect( ulCopyLinkAct , SIGNAL( triggered() ), this, SLOT( ulCopyLink() ) ); // load settings processSettings(true); int S = QFontMetricsF(font()).height(); QString help_str = tr( "

  File Transfer

\

Retroshare brings two ways of transferring files: direct transfers from your friends, and \ distant anonymous tunnelled transfers. In addition, file transfer is multi-source and allows swarming \ (you can be a source while downloading)

\

You can share files using the icon from the left side bar. \ These files will be listed in the My Files tab. You can decide for each friend group whether they can or not see these files \ in their Friends Files tab

\

The search tab reports files from your friends' file lists, and distant files that can be reached \ anonymously using the multi-hop tunnelling system.

\ ").arg(QString::number(2*S)).arg(QString::number(S)) ; registerHelpButton(ui.helpButton,help_str,"TransfersDialog") ; } TransfersDialog::~TransfersDialog() { // save settings processSettings(false); } void TransfersDialog::activatePage(TransfersDialog::Page page) { switch(page) { case TransfersDialog::SearchTab: ui.tabWidget->setCurrentWidget(searchDialog) ; break ; case TransfersDialog::LocalSharedFilesTab: ui.tabWidget->setCurrentWidget(localSharedFiles) ; break ; case TransfersDialog::RemoteSharedFilesTab: ui.tabWidget->setCurrentWidget(remoteSharedFiles) ; break ; } } UserNotify *TransfersDialog::getUserNotify(QObject *parent) { return new TransferUserNotify(parent); } void TransfersDialog::processSettings(bool bLoad) { m_bProcessSettings = true; QHeaderView *DLHeader = ui.downloadList->header () ; QHeaderView *ULHeader = ui.uploadsList->header () ; Settings->beginGroup(QString("TransfersDialog")); if (bLoad) { // load settings // state of the lists DLHeader->restoreState(Settings->value("downloadList").toByteArray()); ULHeader->restoreState(Settings->value("uploadList").toByteArray()); // state of splitter ui.splitter->restoreState(Settings->value("Splitter").toByteArray()); setShowDLSizeColumn(Settings->value("showDLSizeColumn", !ui.downloadList->isColumnHidden(COLUMN_SIZE)).toBool()); setShowDLCompleteColumn(Settings->value("showDLCompleteColumn", !ui.downloadList->isColumnHidden(COLUMN_COMPLETED)).toBool()); setShowDLDLSpeedColumn(Settings->value("showDLDLSpeedColumn", !ui.downloadList->isColumnHidden(COLUMN_DLSPEED)).toBool()); setShowDLProgressColumn(Settings->value("showDLProgressColumn", !ui.downloadList->isColumnHidden(COLUMN_PROGRESS)).toBool()); setShowDLSourcesColumn(Settings->value("showDLSourcesColumn", !ui.downloadList->isColumnHidden(COLUMN_SOURCES)).toBool()); setShowDLStatusColumn(Settings->value("showDLStatusColumn", !ui.downloadList->isColumnHidden(COLUMN_STATUS)).toBool()); setShowDLPriorityColumn(Settings->value("showDLPriorityColumn", !ui.downloadList->isColumnHidden(COLUMN_PRIORITY)).toBool()); setShowDLRemainingColumn(Settings->value("showDLRemainingColumn", !ui.downloadList->isColumnHidden(COLUMN_REMAINING)).toBool()); setShowDLDownloadTimeColumn(Settings->value("showDLDownloadTimeColumn", !ui.downloadList->isColumnHidden(COLUMN_DOWNLOADTIME)).toBool()); setShowDLIDColumn(Settings->value("showDLIDColumn", !ui.downloadList->isColumnHidden(COLUMN_ID)).toBool()); setShowDLLastDLColumn(Settings->value("showDLLastDLColumn", !ui.downloadList->isColumnHidden(COLUMN_LASTDL)).toBool()); setShowDLPath(Settings->value("showDLPath", !ui.downloadList->isColumnHidden(COLUMN_PATH)).toBool()); // selected tab ui.tabWidget->setCurrentIndex(Settings->value("selectedTab").toInt()); } else { // save settings // state of the lists Settings->setValue("downloadList", DLHeader->saveState()); Settings->setValue("uploadList", ULHeader->saveState()); // state of splitter Settings->setValue("Splitter", ui.splitter->saveState()); Settings->setValue("showDLSizeColumn", !ui.downloadList->isColumnHidden(COLUMN_SIZE)); Settings->setValue("showDLCompleteColumn", !ui.downloadList->isColumnHidden(COLUMN_COMPLETED)); Settings->setValue("showDLDLSpeedColumn", !ui.downloadList->isColumnHidden(COLUMN_DLSPEED)); Settings->setValue("showDLProgressColumn", !ui.downloadList->isColumnHidden(COLUMN_PROGRESS)); Settings->setValue("showDLSourcesColumn", !ui.downloadList->isColumnHidden(COLUMN_SOURCES)); Settings->setValue("showDLStatusColumn", !ui.downloadList->isColumnHidden(COLUMN_STATUS)); Settings->setValue("showDLPriorityColumn", !ui.downloadList->isColumnHidden(COLUMN_PRIORITY)); Settings->setValue("showDLRemainingColumn", !ui.downloadList->isColumnHidden(COLUMN_REMAINING)); Settings->setValue("showDLDownloadTimeColumn", !ui.downloadList->isColumnHidden(COLUMN_DOWNLOADTIME)); Settings->setValue("showDLIDColumn", !ui.downloadList->isColumnHidden(COLUMN_ID)); Settings->setValue("showDLLastDLColumn", !ui.downloadList->isColumnHidden(COLUMN_LASTDL)); Settings->setValue("showDLPath", !ui.downloadList->isColumnHidden(COLUMN_PATH)); // selected tab Settings->setValue("selectedTab", ui.tabWidget->currentIndex()); } Settings->endGroup(); m_bProcessSettings = false; } // replaced by shortcut //void TransfersDialog::keyPressEvent(QKeyEvent *e) //{ // if(e->key() == Qt::Key_Delete) // { // cancel() ; // e->accept() ; // } // else // RsAutoUpdatePage::keyPressEvent(e) ; //} void TransfersDialog::downloadListCustomPopupMenu( QPoint /*point*/ ) { std::set items ; getDLSelectedItems(&items, NULL) ; bool single = (items.size() == 1) ; bool atLeastOne_Waiting = false ; bool atLeastOne_Downloading = false ; bool atLeastOne_Complete = false ; bool atLeastOne_Queued = false ; bool atLeastOne_Paused = false ; bool add_PlayOption = false ; bool add_PreviewOption=false ; bool add_OpenFileOption = false ; bool add_CopyLink = false ; bool add_PasteLink = false ; bool add_CollActions = false ; FileInfo info; QMenu priorityQueueMenu(tr("Move in Queue..."), this); priorityQueueMenu.setIcon(QIcon(IMAGE_PRIORITY)); priorityQueueMenu.addAction(queueTopAct); priorityQueueMenu.addAction(queueUpAct); priorityQueueMenu.addAction(queueDownAct); priorityQueueMenu.addAction(queueBottomAct); QMenu prioritySpeedMenu(tr("Priority (Speed)..."), this); prioritySpeedMenu.setIcon(QIcon(IMAGE_PRIORITY)); prioritySpeedMenu.addAction(prioritySlowAct); prioritySpeedMenu.addAction(priorityMediumAct); prioritySpeedMenu.addAction(priorityFastAct); QMenu chunkMenu(tr("Chunk strategy"), this); chunkMenu.setIcon(QIcon(IMAGE_PRIORITY)); chunkMenu.addAction(chunkStreamingAct); chunkMenu.addAction(chunkProgressiveAct); chunkMenu.addAction(chunkRandomAct); QMenu collectionMenu(tr("Collection"), this); collectionMenu.setIcon(QIcon(IMAGE_LIBRARY)); collectionMenu.addAction(collCreateAct); collectionMenu.addAction(collModifAct); collectionMenu.addAction(collViewAct); collectionMenu.addAction(collOpenAct); QMenu contextMnu( this ); if(!RSLinkClipboard::empty(RetroShareLink::TYPE_FILE)) add_PasteLink=true; if(!items.empty()) { add_CopyLink = true ; //Look for all selected items std::set::const_iterator it = items.begin(); std::set::const_iterator end = items.end(); for (; it != end ; ++it) { RsFileHash fileHash = *it; //Look only for first column == File List //Get Info for current item if (rsFiles->FileDetails(fileHash , RS_FILE_HINTS_DOWNLOAD, info)) { /*const uint32_t FT_STATE_FAILED = 0x0000; *const uint32_t FT_STATE_OKAY = 0x0001; *const uint32_t FT_STATE_WAITING = 0x0002; *const uint32_t FT_STATE_DOWNLOADING = 0x0003; *const uint32_t FT_STATE_COMPLETE = 0x0004; *const uint32_t FT_STATE_QUEUED = 0x0005; *const uint32_t FT_STATE_PAUSED = 0x0006; *const uint32_t FT_STATE_CHECKING_HASH = 0x0007; */ if (info.downloadStatus == FT_STATE_WAITING) { atLeastOne_Waiting = true ; } if (info.downloadStatus == FT_STATE_DOWNLOADING) { atLeastOne_Downloading=true ; } if (info.downloadStatus == FT_STATE_COMPLETE) { atLeastOne_Complete = true ; add_OpenFileOption = single ; } if (info.downloadStatus == FT_STATE_QUEUED) { atLeastOne_Queued = true ; } if (info.downloadStatus == FT_STATE_PAUSED) { atLeastOne_Paused = true ; } size_t pos = info.fname.find_last_of('.') ; if (pos != std::string::npos) { // Check if the file is a media file if (misc::isPreviewable(info.fname.substr(pos + 1).c_str())) { add_PreviewOption = (info.downloadStatus != FT_STATE_COMPLETE) ; add_PlayOption = !add_PreviewOption ; } // Check if the file is a collection if (RsCollection::ExtensionString == info.fname.substr(pos + 1).c_str()) { add_CollActions = (info.downloadStatus == FT_STATE_COMPLETE); } } } } } if (atLeastOne_Waiting || atLeastOne_Downloading || atLeastOne_Queued || atLeastOne_Paused) { contextMnu.addMenu( &prioritySpeedMenu) ; } if (atLeastOne_Queued) { contextMnu.addMenu( &priorityQueueMenu) ; } if ( (!items.empty()) && (atLeastOne_Downloading || atLeastOne_Queued || atLeastOne_Waiting || atLeastOne_Paused)) { contextMnu.addMenu(&chunkMenu) ; if (single) { contextMnu.addAction( renameFileAct) ; } QMenu *directoryMenu = contextMnu.addMenu(QIcon(IMAGE_OPENFOLDER), tr("Set destination directory")) ; directoryMenu->addAction(specifyDestinationDirectoryAct) ; // Now get the list of existing directories. std::list< SharedDirInfo> dirs ; rsFiles->getSharedDirectories( dirs) ; for (std::list::const_iterator it(dirs.begin());it!=dirs.end();++it){ // Check for existence of directory name QFile directory( QString::fromUtf8((*it).filename.c_str())) ; if (!directory.exists()) continue ; if (!(directory.permissions() & QFile::WriteOwner)) continue ; QAction *act = new QAction(QString::fromUtf8((*it).virtualname.c_str()), directoryMenu) ; act->setData(QString::fromUtf8( (*it).filename.c_str() ) ) ; connect(act, SIGNAL(triggered()), this, SLOT(setDestinationDirectory())) ; directoryMenu->addAction( act) ; } } if (atLeastOne_Paused) { contextMnu.addAction(resumeAct) ; } if (atLeastOne_Downloading || atLeastOne_Queued || atLeastOne_Waiting) { contextMnu.addAction(pauseAct) ; } if (!atLeastOne_Complete && !items.empty()) { contextMnu.addAction(forceCheckAct) ; contextMnu.addAction(cancelAct) ; } if (add_PlayOption) { contextMnu.addAction(playAct) ; } if (atLeastOne_Paused || atLeastOne_Downloading || atLeastOne_Complete || add_PlayOption) { contextMnu.addSeparator() ;//------------------------------------------------ } if (single) { if (add_OpenFileOption) contextMnu.addAction( openFileAct) ; if (add_PreviewOption) contextMnu.addAction( previewFileAct) ; contextMnu.addAction( openFolderAct) ; contextMnu.addAction( detailsFileAct) ; contextMnu.addSeparator() ;//-------------------------------------------- } contextMnu.addAction( clearCompletedAct) ; contextMnu.addSeparator() ;//------------------------------------------------ if (add_CopyLink) { contextMnu.addAction( copyLinkAct) ; } if (add_PasteLink) { contextMnu.addAction( pasteLinkAct) ; } if (add_CopyLink || add_PasteLink) { contextMnu.addSeparator() ;//-------------------------------------------- } if (DLLFilterModel->rowCount()>0 ) { contextMnu.addAction( expandAllDLAct ) ; contextMnu.addAction( collapseAllDLAct ) ; } contextMnu.addSeparator() ;//----------------------------------------------- collCreateAct->setEnabled(true) ; collModifAct->setEnabled(single && add_CollActions) ; collViewAct->setEnabled(single && add_CollActions) ; collOpenAct->setEnabled(true) ; contextMnu.addMenu(&collectionMenu) ; contextMnu.exec(QCursor::pos()) ; } void TransfersDialog::downloadListHeaderCustomPopupMenu( QPoint /*point*/ ) { std::cerr << "TransfersDialog::downloadListHeaderCustomPopupMenu()" << std::endl; QMenu contextMnu( this ); showDLSizeAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_SIZE)); showDLCompleteAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_COMPLETED)); showDLDLSpeedAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_DLSPEED)); showDLProgressAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_PROGRESS)); showDLSourcesAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_SOURCES)); showDLStatusAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_STATUS)); showDLPriorityAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_PRIORITY)); showDLRemainingAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_REMAINING)); showDLDownloadTimeAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_DOWNLOADTIME)); showDLIDAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_ID)); showDLLastDLAct->setChecked(!ui.downloadList->isColumnHidden(COLUMN_LASTDL)); showDLPath->setChecked(!ui.downloadList->isColumnHidden(COLUMN_PATH)); QMenu *menu = contextMnu.addMenu(tr("Columns")); menu->addAction(showDLSizeAct); menu->addAction(showDLCompleteAct); menu->addAction(showDLDLSpeedAct); menu->addAction(showDLProgressAct); menu->addAction(showDLSourcesAct); menu->addAction(showDLStatusAct); menu->addAction(showDLPriorityAct); menu->addAction(showDLRemainingAct); menu->addAction(showDLDownloadTimeAct); menu->addAction(showDLIDAct); menu->addAction(showDLLastDLAct); menu->addAction(showDLPath); contextMnu.exec(QCursor::pos()); } void TransfersDialog::uploadsListCustomPopupMenu( QPoint /*point*/ ) { std::set items; getULSelectedItems(&items, NULL); bool single = (items.size() == 1); bool add_CopyLink = !items.empty(); QMenu contextMnu( this ); if(single) contextMnu.addAction( ulOpenFolderAct); if (add_CopyLink) contextMnu.addAction( ulCopyLinkAct); if (ULListModel->rowCount()>0 ) { if(single || add_CopyLink) contextMnu.addSeparator() ;//----------------------------------------------- contextMnu.addAction( expandAllULAct ) ; contextMnu.addAction( collapseAllULAct ) ; } contextMnu.exec(QCursor::pos()); } void TransfersDialog::chooseDestinationDirectory() { QString dest_dir = QFileDialog::getExistingDirectory(this,tr("Choose directory")) ; if(dest_dir.isNull()) return ; std::set items ; getDLSelectedItems(&items, NULL); for(std::set::const_iterator it(items.begin());it!=items.end();++it) { std::cerr << "Setting new directory " << dest_dir.toUtf8().data() << " to file " << *it << std::endl; rsFiles->setDestinationDirectory(*it,dest_dir.toUtf8().data() ) ; } } void TransfersDialog::setDestinationDirectory() { std::string dest_dir(qobject_cast(sender())->data().toString().toUtf8().data()) ; std::set items ; getDLSelectedItems(&items, NULL); for(std::set::const_iterator it(items.begin());it!=items.end();++it) { std::cerr << "Setting new directory " << dest_dir << " to file " << *it << std::endl; rsFiles->setDestinationDirectory(*it,dest_dir) ; } } int TransfersDialog::addDLItem(int row, const FileInfo &fileInfo) { QString fileHash = QString::fromStdString(fileInfo.hash.toStdString()); double fileDlspeed = (fileInfo.downloadStatus == FT_STATE_DOWNLOADING) ? (fileInfo.tfRate * 1024.0) : 0.0; QString status; switch (fileInfo.downloadStatus) { case FT_STATE_FAILED: status = tr("Failed"); break; case FT_STATE_OKAY: status = tr("Okay"); break; case FT_STATE_WAITING: status = tr("Waiting"); break; case FT_STATE_DOWNLOADING: status = tr("Downloading"); break; case FT_STATE_COMPLETE: status = tr("Complete"); break; case FT_STATE_QUEUED: status = tr("Queued"); break; case FT_STATE_PAUSED: status = tr("Paused"); break; case FT_STATE_CHECKING_HASH:status = tr("Checking..."); break; default: status = tr("Unknown"); break; } QString priority; if (fileInfo.downloadStatus == FT_STATE_QUEUED) { priority = QString::number(fileInfo.queue_position); } else { switch (fileInfo.priority) { case SPEED_LOW: priority = tr("Slower");break; case SPEED_NORMAL: priority = tr("Average");break; case SPEED_HIGH: priority = tr("Faster");break; default: priority = tr("Average");break; } } qlonglong completed = fileInfo.transfered; qlonglong remaining = fileInfo.size - fileInfo.transfered; qlonglong downloadtime = (fileInfo.tfRate > 0)?( (fileInfo.size - fileInfo.transfered) / (fileInfo.tfRate * 1024.0) ) : 0 ; qint64 qi64LastDL = fileInfo.lastTS ; //std::numeric_limits::max(); if (qi64LastDL == 0) // file is complete, or any raison why the time has not been set properly { QFileInfo file; if (fileInfo.downloadStatus == FT_STATE_COMPLETE) file = QFileInfo(QString::fromUtf8(fileInfo.path.c_str()), QString::fromUtf8(fileInfo.fname.c_str())); else file = QFileInfo(QString::fromUtf8(rsFiles->getPartialsDirectory().c_str()), QString::fromUtf8(fileInfo.hash.toStdString().c_str())); /*Get Last Access on File */ if (file.exists()) qi64LastDL = file.lastModified().toTime_t(); } QString strPath = QString::fromUtf8(fileInfo.path.c_str()); QString strPathAfterDL = strPath; strPathAfterDL.replace(QString::fromUtf8(rsFiles->getDownloadDirectory().c_str()),""); FileChunksInfo fcinfo; if (!rsFiles->FileDownloadChunksDetails(fileInfo.hash, fcinfo)) { return -1; } FileProgressInfo pinfo; pinfo.cmap = fcinfo.chunks; pinfo.type = FileProgressInfo::DOWNLOAD_LINE; pinfo.progress = (fileInfo.size == 0) ? 0 : (completed * 100.0 / fileInfo.size); pinfo.nb_chunks = pinfo.cmap._map.empty() ? 0 : fcinfo.chunks.size(); for (uint32_t i = 0; i < fcinfo.chunks.size(); ++i) switch(fcinfo.chunks[i]) { case FileChunksInfo::CHUNK_CHECKING: pinfo.chunks_in_checking.push_back(i); break ; case FileChunksInfo::CHUNK_ACTIVE: pinfo.chunks_in_progress.push_back(i); break ; case FileChunksInfo::CHUNK_DONE: case FileChunksInfo::CHUNK_OUTSTANDING: break ; } QString tooltip; if (fileInfo.downloadStatus == FT_STATE_CHECKING_HASH) { tooltip = tr("If the hash of the downloaded data does\nnot correspond to the hash announced\nby the file source. The data is likely \nto be corrupted.\n\nRetroShare will ask the source a detailed \nmap of the data; it will compare and invalidate\nbad blocks, and download them again\n\nTry to be patient!") ; } if (row < 0) { row = DLListModel->rowCount(); DLListModel->insertRow(row); // change progress column to own class for sorting DLListModel->setItem(row, COLUMN_PROGRESS, new ProgressItem(NULL)); DLListModel->setItem(row, COLUMN_PRIORITY, new PriorityItem(NULL)); DLListModel->setData(DLListModel->index(row, COLUMN_SIZE), QVariant((qlonglong) fileInfo.size)); DLListModel->setData(DLListModel->index(row, COLUMN_ID), fileHash, Qt::DisplayRole); DLListModel->setData(DLListModel->index(row, COLUMN_ID), fileHash, Qt::UserRole); } QString fileName = QString::fromUtf8(fileInfo.fname.c_str()); DLListModel->setData(DLListModel->index(row, COLUMN_NAME), fileName); DLListModel->setData(DLListModel->index(row, COLUMN_NAME), FilesDefs::getIconFromFilename(fileName), Qt::DecorationRole); DLListModel->setData(DLListModel->index(row, COLUMN_COMPLETED), QVariant((qlonglong)completed)); DLListModel->setData(DLListModel->index(row, COLUMN_DLSPEED), QVariant((double)fileDlspeed)); DLListModel->setData(DLListModel->index(row, COLUMN_PROGRESS), QVariant((float)pinfo.progress)); DLListModel->setData(DLListModel->index(row, COLUMN_PROGRESS), QVariant::fromValue(pinfo), Qt::UserRole); DLListModel->setData(DLListModel->index(row, COLUMN_STATUS), QVariant(status)); DLListModel->setData(DLListModel->index(row, COLUMN_PRIORITY), QVariant(priority)); DLListModel->setData(DLListModel->index(row, COLUMN_REMAINING), QVariant((qlonglong)remaining)); DLListModel->setData(DLListModel->index(row, COLUMN_DOWNLOADTIME), QVariant((qlonglong)downloadtime)); DLListModel->setData(DLListModel->index(row, COLUMN_LASTDL), QVariant(qi64LastDL)); DLListModel->setData(DLListModel->index(row, COLUMN_PATH), QVariant(strPathAfterDL)); DLListModel->item(row,COLUMN_PATH)->setToolTip(strPath); DLListModel->item(row,COLUMN_STATUS)->setToolTip(tooltip); QStandardItem *dlItem = DLListModel->item(row); std::set used_rows ; int active = 0; if (fileInfo.downloadStatus != FT_STATE_COMPLETE) { for (std::list::const_iterator pit = fileInfo.peers.begin() ; pit != fileInfo.peers.end(); ++pit) { const TransferInfo &transferInfo = *pit; //unique combination: fileHash + peerId, variant: hash + peerName (too long) QString hashFileAndPeerId = fileHash + QString::fromStdString(transferInfo.peerId.toStdString()); double peerDlspeed = 0; if ((uint32_t)transferInfo.status == FT_STATE_DOWNLOADING && fileInfo.downloadStatus != FT_STATE_PAUSED && fileInfo.downloadStatus != FT_STATE_COMPLETE) peerDlspeed = transferInfo.tfRate * 1024.0; FileProgressInfo peerpinfo; peerpinfo.cmap = fcinfo.compressed_peer_availability_maps[transferInfo.peerId]; peerpinfo.type = FileProgressInfo::DOWNLOAD_SOURCE ; peerpinfo.progress = 0.0; // we don't display completion for sources. peerpinfo.nb_chunks = peerpinfo.cmap._map.empty() ? 0 : fcinfo.chunks.size(); int row_id = addPeerToDLItem(dlItem, transferInfo.peerId, hashFileAndPeerId, peerDlspeed, transferInfo.status, peerpinfo); used_rows.insert(row_id); /* get the sources (number of online peers) */ if (transferInfo.tfRate > 0 && fileInfo.downloadStatus == FT_STATE_DOWNLOADING) ++active; } } float fltSources = active + (float)fileInfo.peers.size()/1000; DLListModel->setData(DLListModel->index(row, COLUMN_SOURCES), fltSources); // This is not optimal, but we deal with a small number of elements. The reverse order is really important, // because rows after the deleted rows change positions ! // for (int r = dlItem->rowCount() - 1; r >= 0; --r) { if (used_rows.find(r) == used_rows.end()) { dlItem->removeRow(r); } } return row; } int TransfersDialog::addPeerToDLItem(QStandardItem *dlItem, const RsPeerId& peer_ID, const QString& coreID, double dlspeed, uint32_t status, const FileProgressInfo& peerInfo) { // try to find the item int childRow = -1; QStandardItem *childId = NULL; for (int count = 0; (childId = dlItem->child(count, COLUMN_ID)) != NULL; ++count) { if (childId->data(Qt::UserRole).toString() == coreID) { childRow = count; break; } } QStandardItem *siName = NULL; QStandardItem *siStatus = NULL; if (childRow == -1) { // set this false if you want to expand on double click dlItem->setEditable(false); QHeaderView *header = ui.downloadList->header(); QStandardItem *iName = new QStandardItem(); //COLUMN_NAME QStandardItem *iSize = new SortByNameItem(header); //COLUMN_SIZE QStandardItem *iCompleted = new SortByNameItem(header); //COLUMN_COMPLETED QStandardItem *iDlSpeed = new SortByNameItem(header); //COLUMN_DLSPEED QStandardItem *iProgress = new ProgressItem(header); //COLUMN_PROGRESS QStandardItem *iSource = new SortByNameItem(header); //COLUMN_SOURCES QStandardItem *iStatus = new SortByNameItem(header); //COLUMN_STATUS QStandardItem *iPriority = new SortByNameItem(header); //COLUMN_PRIORITY QStandardItem *iRemaining = new SortByNameItem(header); //COLUMN_REMAINING QStandardItem *iDownloadTime = new SortByNameItem(header); //COLUMN_DOWNLOADTIME QStandardItem *iID = new SortByNameItem(header); //COLUMN_ID QStandardItem *iLastDL = new SortByNameItem(header); //COLUMN_LASTDL QStandardItem *iPath = new SortByNameItem(header); //COLUMN_PATH siName = iName; siStatus = iStatus; QList items; QString iconName; QString tooltip; iName->setData(QVariant(getPeerName(peer_ID, iconName, tooltip)), Qt::DisplayRole); iName->setData(QIcon(iconName), Qt::DecorationRole); iName->setData(QVariant(tooltip), Qt::ToolTipRole); iSize->setData(QVariant(QString()), Qt::DisplayRole); iCompleted->setData(QVariant(QString()), Qt::DisplayRole); iDlSpeed->setData(QVariant((double)dlspeed), Qt::DisplayRole); iProgress->setData(QVariant((float)peerInfo.progress), Qt::DisplayRole); iProgress->setData(QVariant::fromValue(peerInfo), Qt::UserRole); iSource->setData(QVariant(QString()), Qt::DisplayRole); iPriority->setData(QVariant(QString()), Qt::DisplayRole); // blank field for priority iRemaining->setData(QVariant(QString()), Qt::DisplayRole); iDownloadTime->setData(QVariant(QString()), Qt::DisplayRole); iID->setData(QVariant() , Qt::DisplayRole); iID->setData(QVariant(coreID), Qt::UserRole); iLastDL->setData(QVariant(QString()), Qt::DisplayRole); iPath->setData(QVariant(QString()), Qt::DisplayRole); items.append(iName); items.append(iSize); items.append(iCompleted); items.append(iDlSpeed); items.append(iProgress); items.append(iSource); items.append(iStatus); items.append(iPriority); items.append(iRemaining); items.append(iDownloadTime); items.append(iID); items.append(iLastDL); items.append(iPath); dlItem->appendRow(items); childRow = dlItem->rowCount() - 1; } else { // just update the child (peer) dlItem->child(childRow, COLUMN_DLSPEED)->setData(QVariant((double)dlspeed), Qt::DisplayRole); dlItem->child(childRow, COLUMN_PROGRESS)->setData(QVariant((float)peerInfo.progress), Qt::DisplayRole); dlItem->child(childRow, COLUMN_PROGRESS)->setData(QVariant::fromValue(peerInfo), Qt::UserRole); siName = dlItem->child(childRow,COLUMN_NAME); siStatus = dlItem->child(childRow, COLUMN_STATUS); } switch (status) { case FT_STATE_FAILED: siStatus->setData(QVariant(tr("Failed"))) ; siName->setData(QIcon(":/images/Client1.png"), Qt::StatusTipRole); break ; case FT_STATE_OKAY: siStatus->setData(QVariant(tr("Okay"))); siName->setData(QIcon(":/images/Client2.png"), Qt::StatusTipRole); break ; case FT_STATE_WAITING: siStatus->setData(QVariant(tr(""))); siName->setData(QIcon(":/images/Client3.png"), Qt::StatusTipRole); break ; case FT_STATE_DOWNLOADING: siStatus->setData(QVariant(tr("Transferring"))); siName->setData(QIcon(":/images/Client0.png"), Qt::StatusTipRole); break ; case FT_STATE_COMPLETE: siStatus->setData(QVariant(tr("Complete"))); siName->setData(QIcon(":/images/Client0.png"), Qt::StatusTipRole); break ; default: siStatus->setData(QVariant(tr(""))); siName->setData(QIcon(":/images/Client4.png"), Qt::StatusTipRole); } return childRow; } int TransfersDialog::addULItem(int row, const FileInfo &fileInfo) { if (fileInfo.peers.empty()) return -1; //No Peers, nothing to do. QString fileHash = QString::fromStdString(fileInfo.hash.toStdString()); RsPeerId ownId = rsPeers->getOwnId(); QString fileName = QString::fromUtf8(fileInfo.fname.c_str()); qlonglong fileSize = fileInfo.size; if(row < 0 ) { row = ULListModel->rowCount(); ULListModel->insertRow(row); // change progress column to own class for sorting //ULListModel->setItem(row, COLUMN_UPROGRESS, new ProgressItem(NULL)); ULListModel->setData(ULListModel->index(row, COLUMN_UNAME), fileName); ULListModel->setData(ULListModel->index(row, COLUMN_UNAME), FilesDefs::getIconFromFilename(fileName), Qt::DecorationRole); ULListModel->setData(ULListModel->index(row, COLUMN_UHASH), fileHash); ULListModel->setData(ULListModel->index(row, COLUMN_UHASH), fileHash, Qt::UserRole); } ULListModel->setData(ULListModel->index(row, COLUMN_USIZE), QVariant((qlonglong)fileSize)); //Reset Parent info if child present ULListModel->setData(ULListModel->index(row, COLUMN_UPEER), QVariant(QString(tr("%1 tunnels").arg(fileInfo.peers.size()))) ); ULListModel->setData(ULListModel->index(row, COLUMN_UPEER), QIcon(), Qt::DecorationRole); ULListModel->setData(ULListModel->index(row, COLUMN_UPEER), QVariant(), Qt::ToolTipRole); ULListModel->setData(ULListModel->index(row, COLUMN_UTRANSFERRED), QVariant()); ULListModel->setData(ULListModel->index(row, COLUMN_UPROGRESS), QVariant()); QStandardItem *ulItem = ULListModel->item(row); std::set used_rows ; double peerULSpeedTotal = 0; bool bOnlyOne = ( fileInfo.peers.size() == 1 ); for(std::list::const_iterator pit = fileInfo.peers.begin() ; pit != fileInfo.peers.end(); ++pit) { const TransferInfo &transferInfo = *pit; if (transferInfo.peerId == ownId) //don't display transfer to ourselves continue ; //unique combination: fileHash + peerId, variant: hash + peerName (too long) QString hashFileAndPeerId = fileHash + QString::fromStdString(transferInfo.peerId.toStdString()); qlonglong completed = transferInfo.transfered; double peerULSpeed = transferInfo.tfRate * 1024.0; FileProgressInfo peerpinfo ; if(!rsFiles->FileUploadChunksDetails(fileInfo.hash, transferInfo.peerId, peerpinfo.cmap) ) continue ; // Estimate the completion. We need something more accurate, meaning that we need to // transmit the completion info. // uint32_t chunk_size = 1024*1024 ; uint32_t nb_chunks = (uint32_t)((fileInfo.size + (uint64_t)chunk_size - 1) / (uint64_t)(chunk_size)) ; uint32_t filled_chunks = peerpinfo.cmap.filledChunks(nb_chunks) ; peerpinfo.type = FileProgressInfo::UPLOAD_LINE ; peerpinfo.nb_chunks = peerpinfo.cmap._map.empty()?0:nb_chunks ; if(filled_chunks > 0 && nb_chunks > 0) { completed = peerpinfo.cmap.computeProgress(fileInfo.size,chunk_size) ; peerpinfo.progress = completed / (float)fileInfo.size * 100.0f ; } else { completed = transferInfo.transfered % chunk_size ; // use the position with respect to last request. peerpinfo.progress = (fileInfo.size>0)?((transferInfo.transfered % chunk_size)*100.0/fileInfo.size):0 ; } if (bOnlyOne) { //Only one peer so update parent QString iconName; QString tooltip; ULListModel->setData(ULListModel->index(row, COLUMN_UPEER), QVariant(getPeerName(transferInfo.peerId, iconName, tooltip))); ULListModel->setData(ULListModel->index(row, COLUMN_UPEER), QIcon(iconName), Qt::DecorationRole); ULListModel->setData(ULListModel->index(row, COLUMN_UPEER), QVariant(tooltip), Qt::ToolTipRole); ULListModel->setData(ULListModel->index(row, COLUMN_UTRANSFERRED), QVariant(completed)); ULListModel->setData(ULListModel->index(row, COLUMN_UPROGRESS), QVariant::fromValue(peerpinfo)); } else { int row_id = addPeerToULItem(ulItem, transferInfo.peerId, hashFileAndPeerId, completed, peerULSpeed, peerpinfo); used_rows.insert(row_id); } peerULSpeedTotal += peerULSpeed; } // Update Parent UpLoad Speed ULListModel->setData(ULListModel->index(row, COLUMN_ULSPEED), QVariant((double)peerULSpeedTotal)); // This is not optimal, but we deal with a small number of elements. The reverse order is really important, // because rows after the deleted rows change positions ! // for (int r = ulItem->rowCount() - 1; r >= 0; --r) { if (used_rows.find(r) == used_rows.end()) { ulItem->removeRow(r); } } return row; } int TransfersDialog::addPeerToULItem(QStandardItem *ulItem, const RsPeerId& peer_ID, const QString& coreID, qlonglong completed, double ulspeed, const FileProgressInfo& peerInfo) { // try to find the item int childRow = -1; QStandardItem *childId = NULL; for (int count = 0; (childId = ulItem->child(count, COLUMN_UHASH)) != NULL; ++count) { if (childId->data(Qt::UserRole).toString() == coreID) { childRow = count; break; } } if (childRow == -1) { // set this false if you want to expand on double click ulItem->setEditable(false); QHeaderView *header = ui.uploadsList->header(); QStandardItem *iName = new QStandardItem(); //COLUMN_UNAME QStandardItem *iPeer = new QStandardItem(); //COLUMN_UPEER QStandardItem *iSize = new SortByNameItem(header); //COLUMN_USIZE QStandardItem *iTransferred = new SortByNameItem(header); //COLUMN_UTRANSFERRED QStandardItem *iULSpeed = new SortByNameItem(header); //COLUMN_ULSPEED QStandardItem *iProgress = new ProgressItem(header); //COLUMN_UPROGRESS QStandardItem *iHash = new SortByNameItem(header); //COLUMN_UHASH QList items; iName->setData( QVariant(QString()), Qt::DisplayRole); QString iconName; QString tooltip; iPeer->setData( QVariant(getPeerName(peer_ID, iconName, tooltip)), Qt::DisplayRole); iPeer->setData( QIcon(iconName), Qt::DecorationRole); iPeer->setData( QVariant(tooltip), Qt::ToolTipRole); iSize->setData( QVariant(QString()), Qt::DisplayRole); iTransferred->setData(QVariant((qlonglong)completed), Qt::DisplayRole); iULSpeed->setData( QVariant((double)ulspeed), Qt::DisplayRole); iProgress->setData( QVariant::fromValue(peerInfo), Qt::DisplayRole); iHash->setData( QVariant(), Qt::DisplayRole); iHash->setData( QVariant(coreID), Qt::UserRole); items.append(iName); items.append(iPeer); items.append(iSize); items.append(iTransferred); items.append(iULSpeed); items.append(iProgress); items.append(iHash); ulItem->appendRow(items); childRow = ulItem->rowCount() - 1; } else { // just update the child (peer) ulItem->child(childRow, COLUMN_ULSPEED)->setData(QVariant((double)ulspeed), Qt::DisplayRole); ulItem->child(childRow, COLUMN_UTRANSFERRED)->setData(QVariant((qlonglong)completed), Qt::DisplayRole); ulItem->child(childRow, COLUMN_UPROGRESS)->setData(QVariant::fromValue(peerInfo), Qt::DisplayRole); } return childRow; } /* get the list of Transfers from the RsIface. **/ void TransfersDialog::updateDisplay() { insertTransfers(); updateDetailsDialog (); } void TransfersDialog::insertTransfers() { /* disable for performance issues, enable after insert all transfers */ ui.downloadList->setSortingEnabled(false); /* get the download lists */ std::list downHashes; rsFiles->FileDownloads(downHashes); /* build set for quick search */ std::set hashs; for (std::list::iterator it = downHashes.begin(); it != downHashes.end(); ++it) { hashs.insert(*it); } /* add downloads, first iterate all rows in list */ int rowCount = DLListModel->rowCount(); for (int row = 0; row < rowCount; ) { RsFileHash hash ( DLListModel->item(row, COLUMN_ID)->data(Qt::UserRole).toString().toStdString()); std::set::iterator hashIt = hashs.find(hash); if (hashIt == hashs.end()) { // remove not existing downloads DLListModel->removeRow(row); rowCount = DLListModel->rowCount(); continue; } FileInfo fileInfo; if (!rsFiles->FileDetails(hash, RS_FILE_HINTS_DOWNLOAD, fileInfo)) { DLListModel->removeRow(row); rowCount = DLListModel->rowCount(); continue; } hashs.erase(hashIt); if (addDLItem(row, fileInfo) < 0) { DLListModel->removeRow(row); rowCount = DLListModel->rowCount(); continue; } ++row; } /* then add new downloads to the list */ for (std::set::iterator hashIt = hashs.begin() ; hashIt != hashs.end(); ++hashIt) { FileInfo fileInfo; if (!rsFiles->FileDetails(*hashIt, RS_FILE_HINTS_DOWNLOAD, fileInfo)) { continue; } addDLItem(-1, fileInfo); } ui.downloadList->setSortingEnabled(true); // Now show upload hashes // /* disable for performance issues, enable after insert all transfers */ ui.uploadsList->setSortingEnabled(false); /* get the upload lists */ std::list upHashes; rsFiles->FileUploads(upHashes); /* build set for quick search */ hashs.clear(); for(std::list::iterator it = upHashes.begin(); it != upHashes.end(); ++it) { hashs.insert(*it); } /* add uploads, first iterate all rows in list */ rowCount = ULListModel->rowCount(); for (int row = 0; row < rowCount; ) { RsFileHash hash ( ULListModel->item(row, COLUMN_UHASH)->data(Qt::UserRole).toString().toStdString()); std::set::iterator hashIt = hashs.find(hash); if (hashIt == hashs.end()) { // remove not existing uploads ULListModel->removeRow(row); rowCount = ULListModel->rowCount(); continue; } FileInfo fileInfo; if (!rsFiles->FileDetails(hash, RS_FILE_HINTS_UPLOAD, fileInfo)) { ULListModel->removeRow(row); rowCount = ULListModel->rowCount(); continue; } hashs.erase(hashIt); if (addULItem(row, fileInfo) < 0) { ULListModel->removeRow(row); rowCount = ULListModel->rowCount(); continue; } ++row; } /* then add new uploads to the list */ for (std::set::iterator hashIt = hashs.begin() ; hashIt != hashs.end(); ++hashIt) { FileInfo fileInfo; if (!rsFiles->FileDetails(*hashIt, RS_FILE_HINTS_UPLOAD, fileInfo)) { continue; } addULItem(-1, fileInfo); } ui.uploadsList->setSortingEnabled(true); downloads = tr("Downloads") + " (" + QString::number(DLListModel->rowCount()) + ")"; uploads = tr("Uploads") + " (" + QString::number(ULListModel->rowCount()) + ")" ; ui.tabWidget->setTabText(0, downloads); ui.tabWidget_UL->setTabText(0, uploads); } QString TransfersDialog::getPeerName(const RsPeerId& id, QString &iconName, QString &tooltip) const { QString res = QString::fromUtf8(rsPeers->getPeerName(id).c_str()) ; // csoler 2009-06-03: This is because turtle tunnels have no name (I didn't want to bother with // connect mgr). In such a case their id can suitably hold for a name. // if(res == "") { res = QString::fromUtf8(rsTurtle->getPeerNameForVirtualPeerId(id).c_str()); if(rsFiles->isEncryptedSource(id)) { iconName = IMAGE_TUNNEL_ANON_E2E; tooltip = tr("Anonymous end-to-end encrypted tunnel 0x")+QString::fromStdString(id.toStdString()).left(8); return tr("Tunnel") + " via " + res ; } iconName = IMAGE_TUNNEL_ANON; tooltip = tr("Anonymous tunnel 0x")+QString::fromStdString(id.toStdString()).left(8); return tr("Tunnel") + " via " + res ; } iconName = IMAGE_TUNNEL_FRIEND; tooltip = res; return res ; } void TransfersDialog::forceCheck() { if (!controlTransferFile(RS_FILE_CTRL_FORCE_CHECK)) std::cerr << "resumeFileTransfer(): can't force check file transfer" << std::endl; } void TransfersDialog::cancel() { bool first = true; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { if (first) { first = false; QString queryWrn2; queryWrn2.clear(); queryWrn2.append(tr("Are you sure that you want to cancel and delete these files?")); if ((QMessageBox::question(this, tr("RetroShare"),queryWrn2,QMessageBox::Yes|QMessageBox::No, QMessageBox::Yes)) == QMessageBox::No) { break; } } rsFiles->FileCancel(*it); } } //void TransfersDialog::handleDownloadRequest(const QString& url) //{ // RetroShareLink link(url); // // if (!link.valid ()) // { // QMessageBox::critical(NULL,"Link error","This link could not be parsed. This is a bug. Please contact the developers.") ; // return; // } // // QVector linkList; // analyzer.getFileInformation (linkList); // // std::list srcIds; // // for (int i = 0, n = linkList.size (); i < n; ++i) // { // const RetroShareLinkData& linkData = linkList[i]; // // rsFiles->FileRequest (linkData.getName ().toStdString (), linkData.getHash ().toStdString (), // linkData.getSize ().toInt (), "", 0, srcIds); // } //} void TransfersDialog::dlCopyLink () { QList links ; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { FileInfo info; if (!rsFiles->FileDetails(*it, RS_FILE_HINTS_DOWNLOAD, info)) { continue; } RetroShareLink link = RetroShareLink::createFile(QString::fromUtf8(info.fname.c_str()), info.size, QString::fromStdString(info.hash.toStdString())); if (link.valid()) { links.push_back(link) ; } } RSLinkClipboard::copyLinks(links) ; } void TransfersDialog::ulCopyLink () { QList links ; std::set items; std::set::iterator it; getULSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { FileInfo info; if (!rsFiles->FileDetails(*it, RS_FILE_HINTS_UPLOAD, info)) { continue; } RetroShareLink link = RetroShareLink::createFile(QString::fromUtf8(info.fname.c_str()), info.size, QString::fromStdString(info.hash.toStdString())); if (link.valid()) { links.push_back(link) ; } } RSLinkClipboard::copyLinks(links) ; } DetailsDialog *TransfersDialog::detailsDialog() { static DetailsDialog *detailsdlg = new DetailsDialog ; return detailsdlg ; } void TransfersDialog::showDetailsDialog() { updateDetailsDialog (); detailsDialog()->show(); } void TransfersDialog::updateDetailsDialog() { std::set items; getDLSelectedItems(&items, NULL); if (!items.empty()) detailsDialog()->setFileHash(*items.begin()); } void TransfersDialog::pasteLink() { RSLinkClipboard::process(RetroShareLink::TYPE_FILE); } void TransfersDialog::getDLSelectedItems(std::set *ids, std::set *rows) { if (ids == NULL && rows == NULL) { return; } if (ids) ids->clear(); if (rows) rows->clear(); QModelIndexList selectedRows = selection->selectedRows(COLUMN_ID); int i, imax = selectedRows.count(); for (i = 0; i < imax; ++i) { QModelIndex index = selectedRows.at(i); if (index.parent().isValid()) index = index.model()->index(index.parent().row(), COLUMN_ID); if (ids) { ids->insert(RsFileHash(index.data(Qt::DisplayRole).toString().toStdString())); ids->insert(RsFileHash(index.data(Qt::UserRole ).toString().toStdString())); } if (rows) { rows->insert(index.row()); } } } void TransfersDialog::getULSelectedItems(std::set *ids, std::set *rows) { if (ids == NULL && rows == NULL) { return; } if (ids) ids->clear(); if (rows) rows->clear(); QModelIndexList indexes = selectionUp->selectedIndexes(); QModelIndex index; foreach(index, indexes) { if (ids) { QStandardItem *id = ULListModel->item(index.row(), COLUMN_UHASH); ids->insert(RsFileHash(id->data(Qt::DisplayRole).toString().toStdString())); } if (rows) { rows->insert(index.row()); } } } bool TransfersDialog::controlTransferFile(uint32_t flags) { bool result = true; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { result &= rsFiles->FileControl(*it, flags); } return result; } void TransfersDialog::pauseFileTransfer() { if (!controlTransferFile(RS_FILE_CTRL_PAUSE)) { std::cerr << "pauseFileTransfer(): can't pause file transfer" << std::endl; } } void TransfersDialog::resumeFileTransfer() { if (!controlTransferFile(RS_FILE_CTRL_START)) { std::cerr << "resumeFileTransfer(): can't resume file transfer" << std::endl; } } void TransfersDialog::dlOpenFolder() { FileInfo info; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { if (!rsFiles->FileDetails(*it, RS_FILE_HINTS_DOWNLOAD, info)) continue; break; } /* make path for downloaded or downloading files */ QFileInfo qinfo; std::string path; if (info.downloadStatus == FT_STATE_COMPLETE) { path = info.path; } else { path = rsFiles->getPartialsDirectory(); } /* open folder with a suitable application */ qinfo.setFile(QString::fromUtf8(path.c_str())); if (qinfo.exists() && qinfo.isDir()) { if (!RsUrlHandler::openUrl(QUrl::fromLocalFile(qinfo.absoluteFilePath()))) { std::cerr << "dlOpenFolder(): can't open folder " << path << std::endl; } } } void TransfersDialog::ulOpenFolder() { FileInfo info; std::set items; std::set::iterator it; getULSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { if (!rsFiles->FileDetails(*it, RS_FILE_HINTS_UPLOAD, info)) continue; break; } /* make path for uploading files */ QFileInfo qinfo; std::string path; path = info.path.substr(0,info.path.length()-info.fname.length()); /* open folder with a suitable application */ qinfo.setFile(QString::fromUtf8(path.c_str())); if (qinfo.exists() && qinfo.isDir()) { if (!RsUrlHandler::openUrl(QUrl::fromLocalFile(qinfo.absoluteFilePath()))) { std::cerr << "ulOpenFolder(): can't open folder " << path << std::endl; } } } void TransfersDialog::dlPreviewFile() { FileInfo info; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { if (!rsFiles->FileDetails(*it, RS_FILE_HINTS_DOWNLOAD, info)) continue; break; } QFileInfo fileNameInfo(QString::fromUtf8(info.fname.c_str())); /* check if the file is a media file */ if (!misc::isPreviewable(fileNameInfo.suffix())) return; /* make path for downloaded or downloading files */ QFileInfo fileInfo; if (info.downloadStatus == FT_STATE_COMPLETE) { fileInfo = QFileInfo(QString::fromUtf8(info.path.c_str()), QString::fromUtf8(info.fname.c_str())); } else { fileInfo = QFileInfo(QString::fromUtf8(rsFiles->getPartialsDirectory().c_str()), QString::fromUtf8(info.hash.toStdString().c_str())); QDir temp; #ifdef WINDOWS_SYS /* the symbolic link must be created on the same drive like the real file, use partial directory */ temp = fileInfo.absoluteDir(); #else temp = QDir::temp(); #endif QString linkName = QFileInfo(temp, fileNameInfo.fileName()).absoluteFilePath(); if (RsFile::CreateLink(fileInfo.absoluteFilePath(), linkName)) { fileInfo.setFile(linkName); } else { std::cerr << "previewTransfer(): can't create link for file " << fileInfo.absoluteFilePath().toStdString() << std::endl; QMessageBox::warning(this, tr("File preview"), tr("Can't create link for file %1.").arg(fileInfo.absoluteFilePath())); return; } } bool previewStarted = false; /* open or preview them with a suitable application */ if (fileInfo.exists() && RsUrlHandler::openUrl(QUrl::fromLocalFile(fileInfo.absoluteFilePath()))) { previewStarted = true; } else { QMessageBox::warning(this, tr("File preview"), tr("File %1 preview failed.").arg(fileInfo.absoluteFilePath())); std::cerr << "previewTransfer(): can't preview file " << fileInfo.absoluteFilePath().toStdString() << std::endl; } if (info.downloadStatus != FT_STATE_COMPLETE) { if (previewStarted) { /* wait for the file to open then remove the link */ QMessageBox::information(this, tr("File preview"), tr("Click OK when program terminates!")); } /* try to delete the preview file */ forever { if (QFile::remove(fileInfo.absoluteFilePath())) { /* preview file could be removed */ break; } /* ask user to try it again */ if (QMessageBox::question(this, tr("File preview"), QString("%1\n\n%2\n\n%3").arg(tr("Could not delete preview file"), fileInfo.absoluteFilePath(), tr("Try it again?")), QMessageBox::Yes, QMessageBox::No) == QMessageBox::No) { break; } } } } void TransfersDialog::dlOpenFile() { FileInfo info; std::set items ; std::set::iterator it ; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { if (!rsFiles->FileDetails(*it, RS_FILE_HINTS_DOWNLOAD, info)) continue; break; } /* make path for downloaded or downloading files */ std::string path; if (info.downloadStatus == FT_STATE_COMPLETE) { path = info.path + "/" + info.fname; /* open file with a suitable application */ QFileInfo qinfo; qinfo.setFile(QString::fromUtf8(path.c_str())); if (qinfo.exists()) { if (!RsUrlHandler::openUrl(QUrl::fromLocalFile(qinfo.absoluteFilePath()))) { std::cerr << "openTransfer(): can't open file " << path << std::endl; } } } else { /* rise a message box for incompleted download file */ QMessageBox::information(this, tr("Open Transfer"), tr("File %1 is not completed. If it is a media file, try to preview it.").arg(QString::fromUtf8(info.fname.c_str()))); } } /* clear download or all queue - for pending dwls */ //void TransfersDialog::clearQueuedDwl() //{ // std::set items; // std::set::iterator it; // getSelectedItems(&items, NULL); // // for (it = items.begin(); it != items.end(); ++it) { // std::string hash = (*it)->data(Qt::DisplayRole).toString().toStdString(); // rsFiles->clearDownload(hash); // } //} //void TransfersDialog::clearQueue() //{ // rsFiles->clearQueue(); //} void TransfersDialog::chunkStreaming() { setChunkStrategy(FileChunksInfo::CHUNK_STRATEGY_STREAMING) ; } void TransfersDialog::chunkRandom() { setChunkStrategy(FileChunksInfo::CHUNK_STRATEGY_RANDOM) ; } void TransfersDialog::chunkProgressive() { setChunkStrategy(FileChunksInfo::CHUNK_STRATEGY_PROGRESSIVE) ; } void TransfersDialog::setChunkStrategy(FileChunksInfo::ChunkStrategy s) { std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { rsFiles->setChunkStrategy(*it, s); } } /* modify download priority actions */ void TransfersDialog::speedSlow() { changeSpeed(0); } void TransfersDialog::speedAverage() { changeSpeed(1); } void TransfersDialog::speedFast() { changeSpeed(2); } void TransfersDialog::priorityQueueUp() { changeQueuePosition(QUEUE_UP); } void TransfersDialog::priorityQueueDown() { changeQueuePosition(QUEUE_DOWN); } void TransfersDialog::priorityQueueTop() { changeQueuePosition(QUEUE_TOP); } void TransfersDialog::priorityQueueBottom() { changeQueuePosition(QUEUE_BOTTOM); } void TransfersDialog::changeSpeed(int speed) { std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { rsFiles->changeDownloadSpeed(*it, speed); } } static bool checkFileName(const QString& name) { if(name.contains('/')) return false ; if(name.contains('\\')) return false ; if(name.contains('|')) return false ; if(name.contains(':')) return false ; if(name.contains('?')) return false ; if(name.contains('>')) return false ; if(name.contains('<')) return false ; if(name.contains('*')) return false ; if(name.length() == 0) return false ; if(name.length() > 255) return false ; return true ; } void TransfersDialog::renameFile() { std::set items; getDLSelectedItems(&items, NULL); if(items.size() != 1) { std::cerr << "Can't rename more than one file. This should not be called." << std::endl; return ; } RsFileHash hash = *(items.begin()) ; FileInfo info ; if (!rsFiles->FileDetails(hash, RS_FILE_HINTS_DOWNLOAD, info)) return ; bool ok = true ; bool first = true ; QString new_name ; do { new_name = QInputDialog::getText(NULL,tr("Change file name"),first?tr("Please enter a new file name"):tr("Please enter a new--and valid--filename"),QLineEdit::Normal,QString::fromUtf8(info.fname.c_str()),&ok) ; if(!ok) return ; first = false ; } while(!checkFileName(new_name)) ; rsFiles->setDestinationName(hash, new_name.toUtf8().data()); } void TransfersDialog::changeQueuePosition(QueueMove mv) { // std::cerr << "In changeQueuePosition (gui)"<< std::endl ; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { rsFiles->changeQueuePosition(*it, mv); } } void TransfersDialog::clearcompleted() { // std::cerr << "TransfersDialog::clearcompleted()" << std::endl; rsFiles->FileClearCompleted(); } void TransfersDialog::showFileDetails() { std::set items ; getDLSelectedItems(&items, NULL) ; if(items.size() != 1) detailsDialog()->setFileHash(RsFileHash()); else detailsDialog()->setFileHash(*items.begin()) ; updateDetailsDialog (); } double TransfersDialog::getProgress(int , QStandardItemModel *) { // return model->data(model->index(row, PROGRESS), Qt::DisplayRole).toDouble(); return 0.0 ; } double TransfersDialog::getSpeed(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_DLSPEED), Qt::DisplayRole).toDouble(); } QString TransfersDialog::getFileName(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_NAME), Qt::DisplayRole).toString(); } QString TransfersDialog::getStatus(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_STATUS), Qt::DisplayRole).toString(); } QString TransfersDialog::getID(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_ID), Qt::UserRole).toString().left(40); // gets only the "hash" part of the name } QString TransfersDialog::getID(int row, QSortFilterProxyModel *filter) { QModelIndex index = filter->mapToSource(filter->index(row, COLUMN_ID)); return filter->sourceModel()->data(index, Qt::UserRole).toString().left(40); // gets only the "hash" part of the name } QString TransfersDialog::getPriority(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_PRIORITY), Qt::DisplayRole).toString(); } qlonglong TransfersDialog::getFileSize(int row, QStandardItemModel *model) { bool ok = false; return model->data(model->index(row, COLUMN_SIZE), Qt::DisplayRole).toULongLong(&ok); } qlonglong TransfersDialog::getTransfered(int row, QStandardItemModel *model) { bool ok = false; return model->data(model->index(row, COLUMN_COMPLETED), Qt::DisplayRole).toULongLong(&ok); } qlonglong TransfersDialog::getRemainingTime(int row, QStandardItemModel *model) { bool ok = false; return model->data(model->index(row, COLUMN_REMAINING), Qt::DisplayRole).toULongLong(&ok); } qlonglong TransfersDialog::getDownloadTime(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_DOWNLOADTIME), Qt::DisplayRole).toULongLong(); } qlonglong TransfersDialog::getLastDL(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_LASTDL), Qt::DisplayRole).toULongLong(); } qlonglong TransfersDialog::getPath(int row, QStandardItemModel *model) { return model->data(model->index(row, COLUMN_PATH), Qt::DisplayRole).toULongLong(); } QString TransfersDialog::getSources(int row, QStandardItemModel *model) { double dblValue = model->data(model->index(row, COLUMN_SOURCES), Qt::DisplayRole).toDouble(); QString temp = QString("%1 (%2)").arg((int)dblValue).arg((int)((fmod(dblValue,1)*1000)+0.5)); return temp; } void TransfersDialog::collCreate() { std::vector dirVec; std::set items ; std::set::iterator it ; getDLSelectedItems(&items, NULL); for (it = items.begin(); it != items.end(); ++it) { FileInfo info; if (!rsFiles->FileDetails(*it, RS_FILE_HINTS_DOWNLOAD, info)) continue; DirDetails details; details.name = info.fname; details.hash = info.hash; details.count = info.size; details.type = DIR_TYPE_FILE; dirVec.push_back(details); } RsCollection(dirVec).openNewColl(this); } void TransfersDialog::collModif() { FileInfo info; std::set items ; std::set::iterator it ; getDLSelectedItems(&items, NULL); if (items.size() != 1) return; it = items.begin(); RsFileHash hash = *it; if (!rsFiles->FileDetails(hash, RS_FILE_HINTS_DOWNLOAD, info)) return; /* make path for downloaded files */ if (info.downloadStatus == FT_STATE_COMPLETE) { std::string path; path = info.path + "/" + info.fname; /* open collection */ QFileInfo qinfo; qinfo.setFile(QString::fromUtf8(path.c_str())); if (qinfo.exists()) { if (qinfo.absoluteFilePath().endsWith(RsCollection::ExtensionString)) { RsCollection collection; collection.openColl(qinfo.absoluteFilePath()); } } } } void TransfersDialog::collView() { FileInfo info; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); if (items.size() != 1) return; it = items.begin(); RsFileHash hash = *it; if (!rsFiles->FileDetails(hash, RS_FILE_HINTS_DOWNLOAD, info)) return; /* make path for downloaded files */ if (info.downloadStatus == FT_STATE_COMPLETE) { std::string path; path = info.path + "/" + info.fname; /* open collection */ QFileInfo qinfo; qinfo.setFile(QString::fromUtf8(path.c_str())); if (qinfo.exists()) { if (qinfo.absoluteFilePath().endsWith(RsCollection::ExtensionString)) { RsCollection collection; collection.openColl(qinfo.absoluteFilePath(), true); } } } } void TransfersDialog::collOpen() { FileInfo info; std::set items; std::set::iterator it; getDLSelectedItems(&items, NULL); if (items.size() == 1) { it = items.begin(); RsFileHash hash = *it; if (rsFiles->FileDetails(hash, RS_FILE_HINTS_DOWNLOAD, info)) { /* make path for downloaded files */ if (info.downloadStatus == FT_STATE_COMPLETE) { std::string path; path = info.path + "/" + info.fname; /* open file with a suitable application */ QFileInfo qinfo; qinfo.setFile(QString::fromUtf8(path.c_str())); if (qinfo.exists()) { if (qinfo.absoluteFilePath().endsWith(RsCollection::ExtensionString)) { RsCollection collection; if (collection.load(qinfo.absoluteFilePath())) { collection.downloadFiles(); return; } } } } } } RsCollection collection; if (collection.load(this)) { collection.downloadFiles(); } } void TransfersDialog::setShowDLSizeColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_SIZE)) != show) { ui.downloadList->setColumnHidden(COLUMN_SIZE, !show); } } void TransfersDialog::setShowDLCompleteColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_COMPLETED)) != show) { ui.downloadList->setColumnHidden(COLUMN_COMPLETED, !show); } } void TransfersDialog::setShowDLDLSpeedColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_DLSPEED)) != show) { ui.downloadList->setColumnHidden(COLUMN_DLSPEED, !show); } } void TransfersDialog::setShowDLProgressColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_PROGRESS)) != show) { ui.downloadList->setColumnHidden(COLUMN_PROGRESS, !show); } } void TransfersDialog::setShowDLSourcesColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_SOURCES)) != show) { ui.downloadList->setColumnHidden(COLUMN_SOURCES, !show); } } void TransfersDialog::setShowDLStatusColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_STATUS)) != show) { ui.downloadList->setColumnHidden(COLUMN_STATUS, !show); } } void TransfersDialog::setShowDLPriorityColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_PRIORITY)) != show) { ui.downloadList->setColumnHidden(COLUMN_PRIORITY, !show); } } void TransfersDialog::setShowDLRemainingColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_REMAINING)) != show) { ui.downloadList->setColumnHidden(COLUMN_REMAINING, !show); } } void TransfersDialog::setShowDLDownloadTimeColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_DOWNLOADTIME)) != show) { ui.downloadList->setColumnHidden(COLUMN_DOWNLOADTIME, !show); } } void TransfersDialog::setShowDLIDColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_ID)) != show) { ui.downloadList->setColumnHidden(COLUMN_ID, !show); } } void TransfersDialog::setShowDLLastDLColumn(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_LASTDL)) != show) { ui.downloadList->setColumnHidden(COLUMN_LASTDL, !show); } } void TransfersDialog::setShowDLPath(bool show) { if ( (!ui.downloadList->isColumnHidden(COLUMN_PATH)) != show) { ui.downloadList->setColumnHidden(COLUMN_PATH, !show); } } void TransfersDialog::expandAllDL() { ui.downloadList->expandAll(); } void TransfersDialog::collapseAllDL() { ui.downloadList->collapseAll(); } void TransfersDialog::expandAllUL() { ui.uploadsList->expandAll(); } void TransfersDialog::collapseAllUL() { ui.uploadsList->collapseAll(); } void TransfersDialog::filterChanged(const QString& /*text*/) { int filterColumn = ui.filterLineEdit->currentFilter(); QString text = ui.filterLineEdit->text(); DLLFilterModel->setFilterKeyColumn(filterColumn); DLLFilterModel->setFilterRegExp(text); }