1449 lines
44 KiB
C++
1449 lines
44 KiB
C++
/*
|
|
* This file Copyright (C) 2009-2015 Mnemosyne LLC
|
|
*
|
|
* It may be used under the GNU GPL versions 2 or 3
|
|
* or any future license endorsed by Mnemosyne LLC.
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
#include <cassert>
|
|
|
|
#include <QtGui>
|
|
#include <QCheckBox>
|
|
#include <QIcon>
|
|
#include <QProxyStyle>
|
|
#include <QLabel>
|
|
#include <QFileDialog>
|
|
#include <QMessageBox>
|
|
|
|
#include <libtransmission/transmission.h>
|
|
#include <libtransmission/version.h>
|
|
|
|
#include "AboutDialog.h"
|
|
#include "AddData.h"
|
|
#include "Application.h"
|
|
#include "DetailsDialog.h"
|
|
#include "FilterBar.h"
|
|
#include "Filters.h"
|
|
#include "Formatter.h"
|
|
#include "MainWindow.h"
|
|
#include "MakeDialog.h"
|
|
#include "OptionsDialog.h"
|
|
#include "Prefs.h"
|
|
#include "PrefsDialog.h"
|
|
#include "RelocateDialog.h"
|
|
#include "Session.h"
|
|
#include "SessionDialog.h"
|
|
#include "Speed.h"
|
|
#include "StatsDialog.h"
|
|
#include "TorrentDelegate.h"
|
|
#include "TorrentDelegateMin.h"
|
|
#include "TorrentFilter.h"
|
|
#include "TorrentModel.h"
|
|
#include "Utils.h"
|
|
|
|
#define PREF_VARIANTS_KEY "pref-variants-list"
|
|
#define STATS_MODE_KEY "stats-mode"
|
|
#define SORT_MODE_KEY "sort-mode"
|
|
|
|
namespace
|
|
{
|
|
const QLatin1String TotalRatioStatsModeName ("total-ratio");
|
|
const QLatin1String TotalTransferStatsModeName ("total-transfer");
|
|
const QLatin1String SessionRatioStatsModeName ("session-ratio");
|
|
const QLatin1String SessionTransferStatsModeName ("session-transfer");
|
|
}
|
|
|
|
/**
|
|
* This is a proxy-style for that forces it to be always disabled.
|
|
* We use this to make our torrent list view behave consistently on
|
|
* both GTK and Qt implementations.
|
|
*/
|
|
class ListViewProxyStyle: public QProxyStyle
|
|
{
|
|
public:
|
|
|
|
int styleHint (StyleHint hint,
|
|
const QStyleOption * option = 0,
|
|
const QWidget * widget = 0,
|
|
QStyleHintReturn * returnData = 0) const
|
|
{
|
|
if (hint == QStyle::SH_ItemView_ActivateItemOnSingleClick)
|
|
return 0;
|
|
return QProxyStyle::styleHint (hint, option, widget, returnData);
|
|
}
|
|
};
|
|
|
|
QIcon
|
|
MainWindow::getStockIcon (const QString& name, int fallback)
|
|
{
|
|
QIcon icon = QIcon::fromTheme (name);
|
|
|
|
if (icon.isNull () && (fallback >= 0))
|
|
icon = style ()->standardIcon (QStyle::StandardPixmap (fallback), 0, this);
|
|
|
|
return icon;
|
|
}
|
|
|
|
MainWindow::MainWindow (Session& session, Prefs& prefs, TorrentModel& model, bool minimized):
|
|
mySession (session),
|
|
myPrefs (prefs),
|
|
myModel (model),
|
|
myLastFullUpdateTime (0),
|
|
mySessionDialog (),
|
|
myPrefsDialog (),
|
|
myAboutDialog (),
|
|
myStatsDialog (),
|
|
myDetailsDialog (),
|
|
myFilterModel (prefs),
|
|
myTorrentDelegate (new TorrentDelegate (this)),
|
|
myTorrentDelegateMin (new TorrentDelegateMin (this)),
|
|
myLastSendTime (0),
|
|
myLastReadTime (0),
|
|
myNetworkTimer (this),
|
|
myNetworkError (false),
|
|
myRefreshTrayIconTimer (this),
|
|
myRefreshActionSensitivityTimer (this)
|
|
{
|
|
setAcceptDrops (true);
|
|
|
|
QAction * sep = new QAction (this);
|
|
sep->setSeparator (true);
|
|
|
|
ui.setupUi (this);
|
|
|
|
QStyle * style = this->style ();
|
|
|
|
int i = style->pixelMetric (QStyle::PM_SmallIconSize, 0, this);
|
|
const QSize smallIconSize (i, i);
|
|
|
|
ui.listView->setStyle (new ListViewProxyStyle);
|
|
ui.listView->setAttribute (Qt::WA_MacShowFocusRect, false);
|
|
|
|
// icons
|
|
ui.action_OpenFile->setIcon (getStockIcon (QLatin1String ("document-open"), QStyle::SP_DialogOpenButton));
|
|
ui.action_New->setIcon (getStockIcon (QLatin1String ("document-new"), QStyle::SP_DesktopIcon));
|
|
ui.action_Properties->setIcon (getStockIcon (QLatin1String ("document-properties"), QStyle::SP_DesktopIcon));
|
|
ui.action_OpenFolder->setIcon (getStockIcon (QLatin1String ("folder-open"), QStyle::SP_DirOpenIcon));
|
|
ui.action_Start->setIcon (getStockIcon (QLatin1String ("media-playback-start"), QStyle::SP_MediaPlay));
|
|
ui.action_StartNow->setIcon (getStockIcon (QLatin1String ("media-playback-start"), QStyle::SP_MediaPlay));
|
|
ui.action_Announce->setIcon (getStockIcon (QLatin1String ("network-transmit-receive")));
|
|
ui.action_Pause->setIcon (getStockIcon (QLatin1String ("media-playback-pause"), QStyle::SP_MediaPause));
|
|
ui.action_Remove->setIcon (getStockIcon (QLatin1String ("list-remove"), QStyle::SP_TrashIcon));
|
|
ui.action_Delete->setIcon (getStockIcon (QLatin1String ("edit-delete"), QStyle::SP_TrashIcon));
|
|
ui.action_StartAll->setIcon (getStockIcon (QLatin1String ("media-playback-start"), QStyle::SP_MediaPlay));
|
|
ui.action_PauseAll->setIcon (getStockIcon (QLatin1String ("media-playback-pause"), QStyle::SP_MediaPause));
|
|
ui.action_Quit->setIcon (getStockIcon (QLatin1String ("application-exit")));
|
|
ui.action_SelectAll->setIcon (getStockIcon (QLatin1String ("edit-select-all")));
|
|
ui.action_ReverseSortOrder->setIcon (getStockIcon (QLatin1String ("view-sort-ascending"), QStyle::SP_ArrowDown));
|
|
ui.action_Preferences->setIcon (getStockIcon (QLatin1String ("preferences-system")));
|
|
ui.action_Contents->setIcon (getStockIcon (QLatin1String ("help-contents"), QStyle::SP_DialogHelpButton));
|
|
ui.action_About->setIcon (getStockIcon (QLatin1String ("help-about")));
|
|
ui.action_QueueMoveTop->setIcon (getStockIcon (QLatin1String ("go-top")));
|
|
ui.action_QueueMoveUp->setIcon (getStockIcon (QLatin1String ("go-up"), QStyle::SP_ArrowUp));
|
|
ui.action_QueueMoveDown->setIcon (getStockIcon (QLatin1String ("go-down"), QStyle::SP_ArrowDown));
|
|
ui.action_QueueMoveBottom->setIcon (getStockIcon (QLatin1String ("go-bottom")));
|
|
|
|
// ui signals
|
|
connect (ui.action_Toolbar, SIGNAL (toggled (bool)), this, SLOT (setToolbarVisible (bool)));
|
|
connect (ui.action_Filterbar, SIGNAL (toggled (bool)), this, SLOT (setFilterbarVisible (bool)));
|
|
connect (ui.action_Statusbar, SIGNAL (toggled (bool)), this, SLOT (setStatusbarVisible (bool)));
|
|
connect (ui.action_CompactView, SIGNAL (toggled (bool)), this, SLOT (setCompactView (bool)));
|
|
connect (ui.action_ReverseSortOrder, SIGNAL (toggled (bool)), this, SLOT (setSortAscendingPref (bool)));
|
|
connect (ui.action_Start, SIGNAL (triggered ()), this, SLOT (startSelected ()));
|
|
connect (ui.action_QueueMoveTop, SIGNAL (triggered ()), this, SLOT (queueMoveTop ()));
|
|
connect (ui.action_QueueMoveUp, SIGNAL (triggered ()), this, SLOT (queueMoveUp ()));
|
|
connect (ui.action_QueueMoveDown, SIGNAL (triggered ()), this, SLOT (queueMoveDown ()));
|
|
connect (ui.action_QueueMoveBottom, SIGNAL (triggered ()), this, SLOT (queueMoveBottom ()));
|
|
connect (ui.action_StartNow, SIGNAL (triggered ()), this, SLOT (startSelectedNow ()));
|
|
connect (ui.action_Pause, SIGNAL (triggered ()), this, SLOT (pauseSelected ()));
|
|
connect (ui.action_Remove, SIGNAL (triggered ()), this, SLOT (removeSelected ()));
|
|
connect (ui.action_Delete, SIGNAL (triggered ()), this, SLOT (deleteSelected ()));
|
|
connect (ui.action_Verify, SIGNAL (triggered ()), this, SLOT (verifySelected ()));
|
|
connect (ui.action_Announce, SIGNAL (triggered ()), this, SLOT (reannounceSelected ()));
|
|
connect (ui.action_StartAll, SIGNAL (triggered ()), this, SLOT (startAll ()));
|
|
connect (ui.action_PauseAll, SIGNAL (triggered ()), this, SLOT (pauseAll ()));
|
|
connect (ui.action_OpenFile, SIGNAL (triggered ()), this, SLOT (openTorrent ()));
|
|
connect (ui.action_AddURL, SIGNAL (triggered ()), this, SLOT (openURL ()));
|
|
connect (ui.action_New, SIGNAL (triggered ()), this, SLOT (newTorrent ()));
|
|
connect (ui.action_Preferences, SIGNAL (triggered ()), this, SLOT (openPreferences ()));
|
|
connect (ui.action_Statistics, SIGNAL (triggered ()), this, SLOT (openStats ()));
|
|
connect (ui.action_Donate, SIGNAL (triggered ()), this, SLOT (openDonate ()));
|
|
connect (ui.action_About, SIGNAL (triggered ()), this, SLOT (openAbout ()));
|
|
connect (ui.action_Contents, SIGNAL (triggered ()), this, SLOT (openHelp ()));
|
|
connect (ui.action_OpenFolder, SIGNAL (triggered ()), this, SLOT (openFolder ()));
|
|
connect (ui.action_CopyMagnetToClipboard, SIGNAL (triggered ()), this, SLOT (copyMagnetLinkToClipboard ()));
|
|
connect (ui.action_SetLocation, SIGNAL (triggered ()), this, SLOT (setLocation ()));
|
|
connect (ui.action_Properties, SIGNAL (triggered ()), this, SLOT (openProperties ()));
|
|
connect (ui.action_SessionDialog, SIGNAL (triggered ()), this, SLOT (openSession ()));
|
|
|
|
connect (ui.listView, SIGNAL (activated (QModelIndex)), ui.action_Properties, SLOT (trigger ()));
|
|
|
|
// signals
|
|
connect (ui.action_SelectAll, SIGNAL (triggered ()), ui.listView, SLOT (selectAll ()));
|
|
connect (ui.action_DeselectAll, SIGNAL (triggered ()), ui.listView, SLOT (clearSelection ()));
|
|
|
|
connect (&myFilterModel, SIGNAL (rowsInserted (QModelIndex, int, int)), this, SLOT (refreshActionSensitivitySoon ()));
|
|
connect (&myFilterModel, SIGNAL (rowsRemoved (QModelIndex, int, int)), this, SLOT (refreshActionSensitivitySoon ()));
|
|
|
|
connect (ui.action_Quit, SIGNAL (triggered ()), qApp, SLOT (quit ()));
|
|
|
|
// torrent view
|
|
myFilterModel.setSourceModel (&myModel);
|
|
connect (&myModel, SIGNAL (modelReset ()), this, SLOT (onModelReset ()));
|
|
connect (&myModel, SIGNAL (rowsRemoved (QModelIndex, int, int)), this, SLOT (onModelReset ()));
|
|
connect (&myModel, SIGNAL (rowsInserted (QModelIndex, int, int)), this, SLOT (onModelReset ()));
|
|
connect (&myModel, SIGNAL (dataChanged (QModelIndex, QModelIndex)), this, SLOT (refreshTrayIconSoon ()));
|
|
|
|
ui.listView->setModel (&myFilterModel);
|
|
connect (ui.listView->selectionModel (), SIGNAL (selectionChanged (QItemSelection, QItemSelection)), this, SLOT (refreshActionSensitivitySoon ()));
|
|
|
|
const QPair<QAction *, int> sortModes[] =
|
|
{
|
|
qMakePair (ui.action_SortByActivity, static_cast<int> (SortMode::SORT_BY_ACTIVITY)),
|
|
qMakePair (ui.action_SortByAge, static_cast<int> (SortMode::SORT_BY_AGE)),
|
|
qMakePair (ui.action_SortByETA, static_cast<int> (SortMode::SORT_BY_ETA)),
|
|
qMakePair (ui.action_SortByName, static_cast<int> (SortMode::SORT_BY_NAME)),
|
|
qMakePair (ui.action_SortByProgress, static_cast<int> (SortMode::SORT_BY_PROGRESS)),
|
|
qMakePair (ui.action_SortByQueue, static_cast<int> (SortMode::SORT_BY_QUEUE)),
|
|
qMakePair (ui.action_SortByRatio, static_cast<int> (SortMode::SORT_BY_RATIO)),
|
|
qMakePair (ui.action_SortBySize, static_cast<int> (SortMode::SORT_BY_SIZE)),
|
|
qMakePair (ui.action_SortByState, static_cast<int> (SortMode::SORT_BY_STATE))
|
|
};
|
|
|
|
QActionGroup * actionGroup = new QActionGroup (this);
|
|
|
|
for (const auto& mode: sortModes)
|
|
{
|
|
mode.first->setProperty (SORT_MODE_KEY, mode.second);
|
|
actionGroup->addAction (mode.first);
|
|
}
|
|
|
|
connect (actionGroup, SIGNAL (triggered (QAction *)), this, SLOT (onSortModeChanged (QAction *)));
|
|
|
|
myAltSpeedAction = new QAction (tr ("Speed Limits"), this);
|
|
myAltSpeedAction->setIcon (ui.altSpeedButton->icon ());
|
|
myAltSpeedAction->setCheckable (true);
|
|
connect (myAltSpeedAction, SIGNAL (triggered ()), this, SLOT (toggleSpeedMode ()));
|
|
|
|
QMenu * menu = new QMenu (this);
|
|
menu->addAction (ui.action_OpenFile);
|
|
menu->addAction (ui.action_AddURL);
|
|
menu->addSeparator ();
|
|
menu->addAction (ui.action_ShowMainWindow);
|
|
menu->addAction (ui.action_ShowMessageLog);
|
|
menu->addAction (ui.action_About);
|
|
menu->addSeparator ();
|
|
menu->addAction (ui.action_StartAll);
|
|
menu->addAction (ui.action_PauseAll);
|
|
menu->addAction (myAltSpeedAction);
|
|
menu->addSeparator ();
|
|
menu->addAction (ui.action_Quit);
|
|
myTrayIcon.setContextMenu (menu);
|
|
myTrayIcon.setIcon (QIcon::fromTheme (QLatin1String ("transmission-tray-icon"), qApp->windowIcon ()));
|
|
|
|
connect (&myPrefs, SIGNAL (changed (int)), this, SLOT (refreshPref (int)));
|
|
connect (ui.action_ShowMainWindow, SIGNAL (triggered (bool)), this, SLOT (toggleWindows (bool)));
|
|
connect (&myTrayIcon, SIGNAL (activated (QSystemTrayIcon::ActivationReason)),
|
|
this, SLOT (trayActivated (QSystemTrayIcon::ActivationReason)));
|
|
|
|
toggleWindows (!minimized);
|
|
ui.action_TrayIcon->setChecked (minimized || prefs.getBool (Prefs::SHOW_TRAY_ICON));
|
|
|
|
initStatusBar ();
|
|
ui.verticalLayout->insertWidget (0, myFilterBar = new FilterBar (myPrefs, myModel, myFilterModel));
|
|
|
|
QList<int> initKeys;
|
|
initKeys << Prefs::MAIN_WINDOW_X
|
|
<< Prefs::SHOW_TRAY_ICON
|
|
<< Prefs::SORT_REVERSED
|
|
<< Prefs::SORT_MODE
|
|
<< Prefs::FILTERBAR
|
|
<< Prefs::STATUSBAR
|
|
<< Prefs::STATUSBAR_STATS
|
|
<< Prefs::TOOLBAR
|
|
<< Prefs::ALT_SPEED_LIMIT_ENABLED
|
|
<< Prefs::COMPACT_VIEW
|
|
<< Prefs::DSPEED
|
|
<< Prefs::DSPEED_ENABLED
|
|
<< Prefs::USPEED
|
|
<< Prefs::USPEED_ENABLED
|
|
<< Prefs::RATIO
|
|
<< Prefs::RATIO_ENABLED;
|
|
for (const int key: initKeys)
|
|
refreshPref (key);
|
|
|
|
connect (&mySession, SIGNAL (sourceChanged ()), this, SLOT (onSessionSourceChanged ()));
|
|
connect (&mySession, SIGNAL (statsUpdated ()), this, SLOT (refreshStatusBar ()));
|
|
connect (&mySession, SIGNAL (dataReadProgress ()), this, SLOT (dataReadProgress ()));
|
|
connect (&mySession, SIGNAL (dataSendProgress ()), this, SLOT (dataSendProgress ()));
|
|
connect (&mySession, SIGNAL (httpAuthenticationRequired ()), this, SLOT (wrongAuthentication ()));
|
|
connect (&mySession, SIGNAL (error (QNetworkReply::NetworkError)), this, SLOT (onError (QNetworkReply::NetworkError)));
|
|
connect (&mySession, SIGNAL (errorMessage (QString)), this, SLOT (errorMessage(QString)));
|
|
|
|
if (mySession.isServer ())
|
|
{
|
|
ui.networkLabel->hide ();
|
|
}
|
|
else
|
|
{
|
|
connect (&myNetworkTimer, SIGNAL (timeout ()), this, SLOT (onNetworkTimer ()));
|
|
myNetworkTimer.start (1000);
|
|
}
|
|
|
|
connect (&myRefreshTrayIconTimer, SIGNAL (timeout ()), this, SLOT (refreshTrayIcon ()));
|
|
connect (&myRefreshActionSensitivityTimer, SIGNAL (timeout ()), this, SLOT (refreshActionSensitivity ()));
|
|
|
|
|
|
refreshActionSensitivitySoon ();
|
|
refreshTrayIconSoon ();
|
|
refreshStatusBar ();
|
|
refreshTitle ();
|
|
}
|
|
|
|
MainWindow::~MainWindow ()
|
|
{
|
|
}
|
|
|
|
/****
|
|
*****
|
|
****/
|
|
|
|
void
|
|
MainWindow::onSessionSourceChanged ()
|
|
{
|
|
myModel.clear ();
|
|
}
|
|
|
|
void
|
|
MainWindow::onModelReset ()
|
|
{
|
|
refreshTitle ();
|
|
refreshActionSensitivitySoon ();
|
|
refreshStatusBar ();
|
|
refreshTrayIconSoon ();
|
|
}
|
|
|
|
/****
|
|
*****
|
|
****/
|
|
|
|
void
|
|
MainWindow::onSetPrefs ()
|
|
{
|
|
const QVariantList p = sender ()->property (PREF_VARIANTS_KEY).toList ();
|
|
assert ( (p.size () % 2) == 0);
|
|
for (int i=0, n=p.size (); i<n; i+=2)
|
|
myPrefs.set (p[i].toInt (), p[i+1]);
|
|
}
|
|
|
|
void
|
|
MainWindow::onSetPrefs (bool isChecked)
|
|
{
|
|
if (isChecked)
|
|
onSetPrefs ();
|
|
}
|
|
|
|
void
|
|
MainWindow::initStatusBar ()
|
|
{
|
|
ui.optionsButton->setMenu (createOptionsMenu ());
|
|
|
|
const int minimumSpeedWidth = ui.downloadSpeedLabel->fontMetrics ().width (Formatter::uploadSpeedToString (Speed::fromKBps (999.99)));
|
|
ui.downloadSpeedLabel->setMinimumWidth (minimumSpeedWidth);
|
|
ui.uploadSpeedLabel->setMinimumWidth (minimumSpeedWidth);
|
|
|
|
ui.statsModeButton->setMenu (createStatsModeMenu ());
|
|
|
|
connect (ui.altSpeedButton, SIGNAL (clicked ()), this, SLOT (toggleSpeedMode ()));
|
|
}
|
|
|
|
QMenu *
|
|
MainWindow::createOptionsMenu ()
|
|
{
|
|
const auto initSpeedSubMenu = [this] (QMenu * menu, QAction *& offAction, QAction *& onAction,
|
|
int pref, int enabledPref)
|
|
{
|
|
const int stockSpeeds[] = {5, 10, 20, 30, 40, 50, 75, 100, 150, 200, 250, 500, 750};
|
|
const int currentValue = myPrefs.get<int> (pref);
|
|
|
|
QActionGroup * actionGroup = new QActionGroup (this);
|
|
|
|
offAction = menu->addAction (tr ("Unlimited"));
|
|
offAction->setCheckable (true);
|
|
offAction->setProperty (PREF_VARIANTS_KEY, QVariantList () << enabledPref << false);
|
|
actionGroup->addAction (offAction);
|
|
connect (offAction, SIGNAL (triggered (bool)), this, SLOT (onSetPrefs (bool)));
|
|
|
|
onAction = menu->addAction (tr ("Limited at %1").arg (Formatter::speedToString (Speed::fromKBps (currentValue))));
|
|
onAction->setCheckable (true);
|
|
onAction->setProperty (PREF_VARIANTS_KEY, QVariantList () << pref << currentValue << enabledPref << true);
|
|
actionGroup->addAction (onAction);
|
|
connect (onAction, SIGNAL (triggered (bool)), this, SLOT (onSetPrefs (bool)));
|
|
|
|
menu->addSeparator ();
|
|
|
|
for (const int i: stockSpeeds)
|
|
{
|
|
QAction * action = menu->addAction (Formatter::speedToString (Speed::fromKBps (i)));
|
|
action->setProperty (PREF_VARIANTS_KEY, QVariantList () << pref << i << enabledPref << true);
|
|
connect (action, SIGNAL (triggered (bool)), this, SLOT (onSetPrefs ()));
|
|
}
|
|
};
|
|
|
|
const auto initSeedRatioSubMenu = [this] (QMenu * menu, QAction *& offAction, QAction *& onAction,
|
|
int pref, int enabledPref)
|
|
{
|
|
const double stockRatios[] = {0.25, 0.50, 0.75, 1, 1.5, 2, 3};
|
|
const double currentValue = myPrefs.get<double> (pref);
|
|
|
|
QActionGroup * actionGroup = new QActionGroup (this);
|
|
|
|
offAction = menu->addAction (tr ("Seed Forever"));
|
|
offAction->setCheckable (true);
|
|
offAction->setProperty (PREF_VARIANTS_KEY, QVariantList () << enabledPref << false);
|
|
actionGroup->addAction (offAction);
|
|
connect (offAction, SIGNAL (triggered (bool)), this, SLOT (onSetPrefs (bool)));
|
|
|
|
onAction = menu->addAction (tr ("Stop at Ratio (%1)").arg (Formatter::ratioToString (currentValue)));
|
|
onAction->setCheckable (true);
|
|
onAction->setProperty (PREF_VARIANTS_KEY, QVariantList () << pref << currentValue << enabledPref << true);
|
|
actionGroup->addAction (onAction);
|
|
connect (onAction, SIGNAL (triggered (bool)), this, SLOT (onSetPrefs (bool)));
|
|
|
|
menu->addSeparator ();
|
|
|
|
for (const double i: stockRatios)
|
|
{
|
|
QAction * action = menu->addAction (Formatter::ratioToString (i));
|
|
action->setProperty (PREF_VARIANTS_KEY, QVariantList () << pref << i << enabledPref << true);
|
|
connect (action, SIGNAL (triggered (bool)), this, SLOT (onSetPrefs ()));
|
|
}
|
|
};
|
|
|
|
QMenu * menu = new QMenu (this);
|
|
|
|
initSpeedSubMenu (menu->addMenu (tr ("Limit Download Speed")), myDlimitOffAction, myDlimitOnAction,
|
|
Prefs::DSPEED, Prefs::DSPEED_ENABLED);
|
|
initSpeedSubMenu (menu->addMenu (tr ("Limit Upload Speed")), myUlimitOffAction, myUlimitOnAction,
|
|
Prefs::USPEED, Prefs::USPEED_ENABLED);
|
|
|
|
menu->addSeparator ();
|
|
|
|
initSeedRatioSubMenu (menu->addMenu (tr ("Stop Seeding at Ratio")), myRatioOffAction, myRatioOnAction,
|
|
Prefs::RATIO, Prefs::RATIO_ENABLED);
|
|
|
|
return menu;
|
|
}
|
|
|
|
QMenu *
|
|
MainWindow::createStatsModeMenu ()
|
|
{
|
|
const QPair<QAction *, QLatin1String> statsModes[] =
|
|
{
|
|
qMakePair (ui.action_TotalRatio, TotalRatioStatsModeName),
|
|
qMakePair (ui.action_TotalTransfer, TotalTransferStatsModeName),
|
|
qMakePair (ui.action_SessionRatio, SessionRatioStatsModeName),
|
|
qMakePair (ui.action_SessionTransfer, SessionTransferStatsModeName)
|
|
};
|
|
|
|
QActionGroup * actionGroup = new QActionGroup (this);
|
|
QMenu * menu = new QMenu (this);
|
|
|
|
for (const auto& mode: statsModes)
|
|
{
|
|
mode.first->setProperty (STATS_MODE_KEY, QString (mode.second));
|
|
actionGroup->addAction (mode.first);
|
|
menu->addAction (mode.first);
|
|
}
|
|
|
|
connect (actionGroup, SIGNAL (triggered (QAction *)), this, SLOT (onStatsModeChanged (QAction *)));
|
|
|
|
return menu;
|
|
}
|
|
|
|
/****
|
|
*****
|
|
****/
|
|
|
|
void
|
|
MainWindow::onSortModeChanged (QAction * action)
|
|
{
|
|
myPrefs.set (Prefs::SORT_MODE, SortMode (action->property (SORT_MODE_KEY).toInt ()));
|
|
}
|
|
|
|
void
|
|
MainWindow::setSortAscendingPref (bool b)
|
|
{
|
|
myPrefs.set (Prefs::SORT_REVERSED, b);
|
|
}
|
|
|
|
/****
|
|
*****
|
|
****/
|
|
|
|
void
|
|
MainWindow::showEvent (QShowEvent * event)
|
|
{
|
|
Q_UNUSED (event);
|
|
|
|
ui.action_ShowMainWindow->setChecked (true);
|
|
}
|
|
|
|
/****
|
|
*****
|
|
****/
|
|
|
|
void
|
|
MainWindow::hideEvent (QHideEvent * event)
|
|
{
|
|
Q_UNUSED (event);
|
|
|
|
if (!isVisible ())
|
|
ui.action_ShowMainWindow->setChecked (false);
|
|
}
|
|
|
|
/****
|
|
*****
|
|
****/
|
|
|
|
void
|
|
MainWindow::openSession ()
|
|
{
|
|
Utils::openDialog (mySessionDialog, mySession, myPrefs, this);
|
|
}
|
|
|
|
void
|
|
MainWindow::openPreferences ()
|
|
{
|
|
Utils::openDialog (myPrefsDialog, mySession, myPrefs, this);
|
|
}
|
|
|
|
void
|
|
MainWindow::openProperties ()
|
|
{
|
|
Utils::openDialog (myDetailsDialog, mySession, myPrefs, myModel, this);
|
|
myDetailsDialog->setIds (getSelectedTorrents ());
|
|
}
|
|
|
|
void
|
|
MainWindow::setLocation ()
|
|
{
|
|
RelocateDialog * d = new RelocateDialog (mySession, myModel, getSelectedTorrents (), this);
|
|
d->setAttribute (Qt::WA_DeleteOnClose, true);
|
|
d->show ();
|
|
}
|
|
|
|
// Open Folder & select torrent's file or top folder
|
|
#undef HAVE_OPEN_SELECT
|
|
#if defined (Q_OS_WIN)
|
|
# define HAVE_OPEN_SELECT
|
|
static
|
|
void openSelect (const QString& path)
|
|
{
|
|
const QString explorer = QLatin1String ("explorer");
|
|
QString param;
|
|
if (!QFileInfo (path).isDir ())
|
|
param = QLatin1String ("/select,");
|
|
param += QDir::toNativeSeparators (path);
|
|
QProcess::startDetached (explorer, QStringList (param));
|
|
}
|
|
#elif defined (Q_OS_MAC)
|
|
# define HAVE_OPEN_SELECT
|
|
static
|
|
void openSelect (const QString& path)
|
|
{
|
|
QStringList scriptArgs;
|
|
scriptArgs << QLatin1String ("-e")
|
|
<< QString::fromLatin1 ("tell application \"Finder\" to reveal POSIX file \"%1\"").arg (path);
|
|
QProcess::execute (QLatin1String ("/usr/bin/osascript"), scriptArgs);
|
|
|
|
scriptArgs.clear ();
|
|
scriptArgs << QLatin1String ("-e")
|
|
<< QLatin1String ("tell application \"Finder\" to activate");
|
|
QProcess::execute (QLatin1String ("/usr/bin/osascript"), scriptArgs);
|
|
}
|
|
#endif
|
|
|
|
void
|
|
MainWindow::openFolder ()
|
|
{
|
|
const QSet<int> selectedTorrents = getSelectedTorrents ();
|
|
if (selectedTorrents.size () != 1)
|
|
return;
|
|
|
|
const int torrentId (*selectedTorrents.begin ());
|
|
const Torrent * tor (myModel.getTorrentFromId (torrentId));
|
|
if (tor == nullptr)
|
|
return;
|
|
|
|
QString path (tor->getPath ());
|
|
const FileList& files = tor->files ();
|
|
if (files.isEmpty ())
|
|
return;
|
|
|
|
const QString firstfile = files.at (0).filename;
|
|
int slashIndex = firstfile.indexOf (QLatin1Char ('/'));
|
|
if (slashIndex > -1)
|
|
{
|
|
path = path + QLatin1Char ('/') + firstfile.left (slashIndex);
|
|
}
|
|
#ifdef HAVE_OPEN_SELECT
|
|
else
|
|
{
|
|
openSelect (path + QLatin1Char ('/') + firstfile);
|
|
return;
|
|
}
|
|
#endif
|
|
|
|
QDesktopServices::openUrl (QUrl::fromLocalFile (path));
|
|
}
|
|
|
|
void
|
|
MainWindow::copyMagnetLinkToClipboard ()
|
|
{
|
|
const int id (*getSelectedTorrents ().begin ());
|
|
mySession.copyMagnetLinkToClipboard (id);
|
|
}
|
|
|
|
void
|
|
MainWindow::openStats ()
|
|
{
|
|
Utils::openDialog (myStatsDialog, mySession, this);
|
|
}
|
|
|
|
void
|
|
MainWindow::openDonate ()
|
|
{
|
|
QDesktopServices::openUrl (QUrl (QLatin1String ("http://www.transmissionbt.com/donate.php")));
|
|
}
|
|
|
|
void
|
|
MainWindow::openAbout ()
|
|
{
|
|
Utils::openDialog (myAboutDialog, this);
|
|
}
|
|
|
|
void
|
|
MainWindow::openHelp ()
|
|
{
|
|
QDesktopServices::openUrl (QUrl (QString::fromLatin1 ("http://www.transmissionbt.com/help/gtk/%1.%2x").
|
|
arg (MAJOR_VERSION).arg (MINOR_VERSION / 10)));
|
|
}
|
|
|
|
void
|
|
MainWindow::refreshTitle ()
|
|
{
|
|
QString title (QLatin1String ("Transmission"));
|
|
const QUrl url (mySession.getRemoteUrl ());
|
|
if (!url.isEmpty ())
|
|
//: Second (optional) part of main window title "Transmission - host:port" (added when connected to remote session);
|
|
//: notice that leading space (before the dash) is included here
|
|
title += tr (" - %1:%2").arg (url.host ()).arg (url.port ());
|
|
setWindowTitle (title);
|
|
}
|
|
|
|
void
|
|
MainWindow::refreshTrayIconSoon ()
|
|
{
|
|
if (!myRefreshTrayIconTimer.isActive ())
|
|
{
|
|
myRefreshTrayIconTimer.setSingleShot (true);
|
|
myRefreshTrayIconTimer.start (100);
|
|
}
|
|
}
|
|
void
|
|
MainWindow::refreshTrayIcon ()
|
|
{
|
|
Speed upSpeed, downSpeed;
|
|
size_t upCount, downCount;
|
|
QString tip;
|
|
|
|
myModel.getTransferSpeed (upSpeed, upCount, downSpeed, downCount);
|
|
|
|
if (myNetworkError)
|
|
{
|
|
tip = tr ("Network Error");
|
|
}
|
|
else if (!upCount && !downCount)
|
|
{
|
|
tip = tr ("Idle");
|
|
}
|
|
else if (downCount)
|
|
{
|
|
tip = Formatter::downloadSpeedToString(downSpeed) +
|
|
QLatin1String (" ") +
|
|
Formatter::uploadSpeedToString(upSpeed);
|
|
}
|
|
else if (upCount)
|
|
{
|
|
tip = Formatter::uploadSpeedToString(upSpeed);
|
|
}
|
|
|
|
myTrayIcon.setToolTip (tip);
|
|
}
|
|
|
|
void
|
|
MainWindow::refreshStatusBar ()
|
|
{
|
|
Speed upSpeed, downSpeed;
|
|
size_t upCount, downCount;
|
|
myModel.getTransferSpeed (upSpeed, upCount, downSpeed, downCount);
|
|
|
|
ui.uploadSpeedLabel->setText (Formatter::uploadSpeedToString (upSpeed));
|
|
ui.uploadSpeedLabel->setVisible (downCount || upCount);
|
|
ui.downloadSpeedLabel->setText (Formatter::downloadSpeedToString (downSpeed));
|
|
ui.downloadSpeedLabel->setVisible (downCount);
|
|
|
|
ui.networkLabel->setVisible (!mySession.isServer ());
|
|
|
|
const QString mode (myPrefs.getString (Prefs::STATUSBAR_STATS));
|
|
QString str;
|
|
|
|
if (mode == SessionRatioStatsModeName)
|
|
{
|
|
str = tr ("Ratio: %1").arg (Formatter::ratioToString (mySession.getStats ().ratio));
|
|
}
|
|
else if (mode == SessionTransferStatsModeName)
|
|
{
|
|
const tr_session_stats& stats (mySession.getStats ());
|
|
str = tr ("Down: %1, Up: %2").arg (Formatter::sizeToString (stats.downloadedBytes))
|
|
.arg (Formatter::sizeToString (stats.uploadedBytes));
|
|
}
|
|
else if (mode == TotalTransferStatsModeName)
|
|
{
|
|
const tr_session_stats& stats (mySession.getCumulativeStats ());
|
|
str = tr ("Down: %1, Up: %2").arg (Formatter::sizeToString (stats.downloadedBytes))
|
|
.arg (Formatter::sizeToString (stats.uploadedBytes));
|
|
}
|
|
else // default is "total-ratio"
|
|
{
|
|
assert (mode == TotalRatioStatsModeName);
|
|
str = tr ("Ratio: %1").arg (Formatter::ratioToString (mySession.getCumulativeStats ().ratio));
|
|
}
|
|
|
|
ui.statsLabel->setText (str);
|
|
}
|
|
|
|
|
|
|
|
void
|
|
MainWindow::refreshActionSensitivitySoon ()
|
|
{
|
|
if (!myRefreshActionSensitivityTimer.isActive ())
|
|
{
|
|
myRefreshActionSensitivityTimer.setSingleShot (true);
|
|
myRefreshActionSensitivityTimer.start (100);
|
|
}
|
|
}
|
|
void
|
|
MainWindow::refreshActionSensitivity ()
|
|
{
|
|
int selected (0);
|
|
int paused (0);
|
|
int queued (0);
|
|
int selectedAndPaused (0);
|
|
int selectedAndQueued (0);
|
|
int selectedWithMetadata (0);
|
|
int canAnnounce (0);
|
|
const QAbstractItemModel * model (ui.listView->model ());
|
|
const QItemSelectionModel * selectionModel (ui.listView->selectionModel ());
|
|
const int rowCount (model->rowCount ());
|
|
|
|
// count how many torrents are selected, paused, etc
|
|
for (int row=0; row<rowCount; ++row)
|
|
{
|
|
const QModelIndex modelIndex (model->index (row, 0));
|
|
assert (model == modelIndex.model ());
|
|
const Torrent * tor (model->data (modelIndex, TorrentModel::TorrentRole).value<const Torrent*> ());
|
|
if (tor)
|
|
{
|
|
const bool isSelected (selectionModel->isSelected (modelIndex));
|
|
const bool isPaused (tor->isPaused ());
|
|
const bool isQueued (tor->isQueued ());
|
|
if (isSelected) ++selected;
|
|
if (isQueued) ++queued;
|
|
if (isPaused) ++ paused;
|
|
if (isSelected && isPaused) ++selectedAndPaused;
|
|
if (isSelected && isQueued) ++selectedAndQueued;
|
|
if (isSelected && tor->hasMetadata ()) ++selectedWithMetadata;
|
|
if (tor->canManualAnnounce ()) ++canAnnounce;
|
|
}
|
|
}
|
|
|
|
const bool haveSelection (selected > 0);
|
|
const bool haveSelectionWithMetadata = selectedWithMetadata > 0;
|
|
const bool oneSelection (selected == 1);
|
|
|
|
ui.action_Verify->setEnabled (haveSelectionWithMetadata);
|
|
ui.action_Remove->setEnabled (haveSelection);
|
|
ui.action_Delete->setEnabled (haveSelection);
|
|
ui.action_Properties->setEnabled (haveSelection);
|
|
ui.action_DeselectAll->setEnabled (haveSelection);
|
|
ui.action_SetLocation->setEnabled (haveSelection);
|
|
|
|
ui.action_OpenFolder->setEnabled (oneSelection && haveSelectionWithMetadata && mySession.isLocal ());
|
|
ui.action_CopyMagnetToClipboard->setEnabled (oneSelection);
|
|
|
|
ui.action_SelectAll->setEnabled (selected < rowCount);
|
|
ui.action_StartAll->setEnabled (paused > 0);
|
|
ui.action_PauseAll->setEnabled (paused < rowCount);
|
|
ui.action_Start->setEnabled (selectedAndPaused > 0);
|
|
ui.action_StartNow->setEnabled (selectedAndPaused + selectedAndQueued > 0);
|
|
ui.action_Pause->setEnabled (selectedAndPaused < selected);
|
|
ui.action_Announce->setEnabled (selected > 0 && (canAnnounce == selected));
|
|
|
|
ui.action_QueueMoveTop->setEnabled (haveSelection);
|
|
ui.action_QueueMoveUp->setEnabled (haveSelection);
|
|
ui.action_QueueMoveDown->setEnabled (haveSelection);
|
|
ui.action_QueueMoveBottom->setEnabled (haveSelection);
|
|
|
|
if (!myDetailsDialog.isNull ())
|
|
myDetailsDialog->setIds (getSelectedTorrents ());
|
|
}
|
|
|
|
/**
|
|
***
|
|
**/
|
|
|
|
void
|
|
MainWindow::clearSelection ()
|
|
{
|
|
ui.action_DeselectAll->trigger ();
|
|
}
|
|
|
|
QSet<int>
|
|
MainWindow::getSelectedTorrents (bool withMetadataOnly) const
|
|
{
|
|
QSet<int> ids;
|
|
|
|
for (const QModelIndex& index: ui.listView->selectionModel ()->selectedRows ())
|
|
{
|
|
const Torrent * tor (index.data (TorrentModel::TorrentRole).value<const Torrent*> ());
|
|
if (tor != nullptr && (!withMetadataOnly || tor->hasMetadata ()))
|
|
ids.insert (tor->id ());
|
|
}
|
|
|
|
return ids;
|
|
}
|
|
|
|
void
|
|
MainWindow::startSelected ()
|
|
{
|
|
mySession.startTorrents (getSelectedTorrents ());
|
|
}
|
|
void
|
|
MainWindow::startSelectedNow ()
|
|
{
|
|
mySession.startTorrentsNow (getSelectedTorrents ());
|
|
}
|
|
void
|
|
MainWindow::pauseSelected ()
|
|
{
|
|
mySession.pauseTorrents (getSelectedTorrents ());
|
|
}
|
|
void
|
|
MainWindow::queueMoveTop ()
|
|
{
|
|
mySession.queueMoveTop (getSelectedTorrents ());
|
|
}
|
|
void
|
|
MainWindow::queueMoveUp ()
|
|
{
|
|
mySession.queueMoveUp (getSelectedTorrents ());
|
|
}
|
|
void
|
|
MainWindow::queueMoveDown ()
|
|
{
|
|
mySession.queueMoveDown (getSelectedTorrents ());
|
|
}
|
|
void
|
|
MainWindow::queueMoveBottom ()
|
|
{
|
|
mySession.queueMoveBottom (getSelectedTorrents ());
|
|
}
|
|
void
|
|
MainWindow::startAll ()
|
|
{
|
|
mySession.startTorrents ();
|
|
}
|
|
void
|
|
MainWindow::pauseAll ()
|
|
{
|
|
mySession.pauseTorrents ();
|
|
}
|
|
void
|
|
MainWindow::removeSelected ()
|
|
{
|
|
removeTorrents (false);
|
|
}
|
|
void
|
|
MainWindow::deleteSelected ()
|
|
{
|
|
removeTorrents (true);
|
|
}
|
|
void
|
|
MainWindow::verifySelected ()
|
|
{
|
|
mySession.verifyTorrents (getSelectedTorrents (true));
|
|
}
|
|
void
|
|
MainWindow::reannounceSelected ()
|
|
{
|
|
mySession.reannounceTorrents (getSelectedTorrents ());
|
|
}
|
|
|
|
/**
|
|
***
|
|
**/
|
|
|
|
void
|
|
MainWindow::onStatsModeChanged (QAction * action)
|
|
{
|
|
myPrefs.set (Prefs::STATUSBAR_STATS, action->property (STATS_MODE_KEY).toString ());
|
|
}
|
|
|
|
/**
|
|
***
|
|
**/
|
|
|
|
void
|
|
MainWindow::setCompactView (bool visible)
|
|
{
|
|
myPrefs.set (Prefs::COMPACT_VIEW, visible);
|
|
}
|
|
void
|
|
MainWindow::toggleSpeedMode ()
|
|
{
|
|
myPrefs.toggleBool (Prefs::ALT_SPEED_LIMIT_ENABLED);
|
|
const bool mode = myPrefs.get<bool> (Prefs::ALT_SPEED_LIMIT_ENABLED);
|
|
myAltSpeedAction->setChecked (mode);
|
|
}
|
|
void
|
|
MainWindow::setToolbarVisible (bool visible)
|
|
{
|
|
myPrefs.set (Prefs::TOOLBAR, visible);
|
|
}
|
|
void
|
|
MainWindow::setFilterbarVisible (bool visible)
|
|
{
|
|
myPrefs.set (Prefs::FILTERBAR, visible);
|
|
}
|
|
void
|
|
MainWindow::setStatusbarVisible (bool visible)
|
|
{
|
|
myPrefs.set (Prefs::STATUSBAR, visible);
|
|
}
|
|
|
|
/**
|
|
***
|
|
**/
|
|
|
|
void
|
|
MainWindow::toggleWindows (bool doShow)
|
|
{
|
|
if (!doShow)
|
|
{
|
|
hide ();
|
|
}
|
|
else
|
|
{
|
|
if (!isVisible ()) show ();
|
|
if (isMinimized ()) showNormal ();
|
|
//activateWindow ();
|
|
raise ();
|
|
qApp->setActiveWindow (this);
|
|
}
|
|
}
|
|
|
|
void
|
|
MainWindow::trayActivated (QSystemTrayIcon::ActivationReason reason)
|
|
{
|
|
if ((reason == QSystemTrayIcon::Trigger) ||
|
|
(reason == QSystemTrayIcon::DoubleClick))
|
|
{
|
|
if (isMinimized ())
|
|
toggleWindows (true);
|
|
else
|
|
toggleWindows (!isVisible ());
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
MainWindow::refreshPref (int key)
|
|
{
|
|
bool b;
|
|
int i;
|
|
QString str;
|
|
QActionGroup * actionGroup;
|
|
|
|
switch (key)
|
|
{
|
|
case Prefs::STATUSBAR_STATS:
|
|
str = myPrefs.getString (key);
|
|
actionGroup = ui.action_TotalRatio->actionGroup ();
|
|
assert (actionGroup != nullptr);
|
|
for (QAction * action: actionGroup->actions ())
|
|
action->setChecked (str == action->property (STATS_MODE_KEY).toString ());
|
|
refreshStatusBar ();
|
|
break;
|
|
|
|
case Prefs::SORT_REVERSED:
|
|
ui.action_ReverseSortOrder->setChecked (myPrefs.getBool (key));
|
|
break;
|
|
|
|
case Prefs::SORT_MODE:
|
|
i = myPrefs.get<SortMode> (key).mode ();
|
|
actionGroup = ui.action_SortByActivity->actionGroup ();
|
|
assert (actionGroup != nullptr);
|
|
for (QAction * action: actionGroup->actions ())
|
|
action->setChecked (i == action->property (SORT_MODE_KEY).toInt ());
|
|
break;
|
|
|
|
case Prefs::DSPEED_ENABLED:
|
|
(myPrefs.get<bool> (key) ? myDlimitOnAction : myDlimitOffAction)->setChecked (true);
|
|
break;
|
|
|
|
case Prefs::DSPEED:
|
|
myDlimitOnAction->setText (tr ("Limited at %1").arg (Formatter::speedToString (Speed::fromKBps (myPrefs.get<int> (key)))));
|
|
break;
|
|
|
|
case Prefs::USPEED_ENABLED:
|
|
(myPrefs.get<bool> (key) ? myUlimitOnAction : myUlimitOffAction)->setChecked (true);
|
|
break;
|
|
|
|
case Prefs::USPEED:
|
|
myUlimitOnAction->setText (tr ("Limited at %1").arg (Formatter::speedToString (Speed::fromKBps (myPrefs.get<int> (key)))));
|
|
break;
|
|
|
|
case Prefs::RATIO_ENABLED:
|
|
(myPrefs.get<bool> (key) ? myRatioOnAction : myRatioOffAction)->setChecked (true);
|
|
break;
|
|
|
|
case Prefs::RATIO:
|
|
myRatioOnAction->setText (tr ("Stop at Ratio (%1)").arg (Formatter::ratioToString (myPrefs.get<double> (key))));
|
|
break;
|
|
|
|
case Prefs::FILTERBAR:
|
|
b = myPrefs.getBool (key);
|
|
myFilterBar->setVisible (b);
|
|
ui.action_Filterbar->setChecked (b);
|
|
break;
|
|
|
|
case Prefs::STATUSBAR:
|
|
b = myPrefs.getBool (key);
|
|
ui.statusBar->setVisible (b);
|
|
ui.action_Statusbar->setChecked (b);
|
|
break;
|
|
|
|
case Prefs::TOOLBAR:
|
|
b = myPrefs.getBool (key);
|
|
ui.toolBar->setVisible (b);
|
|
ui.action_Toolbar->setChecked (b);
|
|
break;
|
|
|
|
case Prefs::SHOW_TRAY_ICON:
|
|
b = myPrefs.getBool (key);
|
|
ui.action_TrayIcon->setChecked (b);
|
|
myTrayIcon.setVisible (b);
|
|
qApp->setQuitOnLastWindowClosed (!b);
|
|
refreshTrayIconSoon ();
|
|
break;
|
|
|
|
case Prefs::COMPACT_VIEW: {
|
|
#if QT_VERSION < QT_VERSION_CHECK(5, 4, 0) // QTBUG-33537
|
|
QItemSelectionModel * selectionModel (ui.listView->selectionModel ());
|
|
const QItemSelection selection (selectionModel->selection ());
|
|
const QModelIndex currentIndex (selectionModel->currentIndex ());
|
|
#endif
|
|
b = myPrefs.getBool (key);
|
|
ui.action_CompactView->setChecked (b);
|
|
ui.listView->setItemDelegate (b ? myTorrentDelegateMin : myTorrentDelegate);
|
|
#if QT_VERSION < QT_VERSION_CHECK(5, 4, 0) // QTBUG-33537
|
|
selectionModel->clear ();
|
|
ui.listView->reset (); // force the rows to resize
|
|
selectionModel->select (selection, QItemSelectionModel::Select);
|
|
selectionModel->setCurrentIndex (currentIndex, QItemSelectionModel::NoUpdate);
|
|
#endif
|
|
break;
|
|
}
|
|
|
|
case Prefs::MAIN_WINDOW_X:
|
|
case Prefs::MAIN_WINDOW_Y:
|
|
case Prefs::MAIN_WINDOW_WIDTH:
|
|
case Prefs::MAIN_WINDOW_HEIGHT:
|
|
setGeometry (myPrefs.getInt (Prefs::MAIN_WINDOW_X),
|
|
myPrefs.getInt (Prefs::MAIN_WINDOW_Y),
|
|
myPrefs.getInt (Prefs::MAIN_WINDOW_WIDTH),
|
|
myPrefs.getInt (Prefs::MAIN_WINDOW_HEIGHT));
|
|
break;
|
|
|
|
case Prefs::ALT_SPEED_LIMIT_ENABLED:
|
|
case Prefs::ALT_SPEED_LIMIT_UP:
|
|
case Prefs::ALT_SPEED_LIMIT_DOWN:
|
|
{
|
|
b = myPrefs.getBool (Prefs::ALT_SPEED_LIMIT_ENABLED);
|
|
myAltSpeedAction->setChecked (b);
|
|
ui.altSpeedButton->setChecked (b);
|
|
const QString fmt = b ? tr ("Click to disable Temporary Speed Limits\n (%1 down, %2 up)")
|
|
: tr ("Click to enable Temporary Speed Limits\n (%1 down, %2 up)");
|
|
const Speed d = Speed::fromKBps (myPrefs.getInt (Prefs::ALT_SPEED_LIMIT_DOWN));
|
|
const Speed u = Speed::fromKBps (myPrefs.getInt (Prefs::ALT_SPEED_LIMIT_UP));
|
|
ui.altSpeedButton->setToolTip (fmt.arg (Formatter::speedToString (d))
|
|
.arg (Formatter::speedToString (u)));
|
|
break;
|
|
}
|
|
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
namespace
|
|
{
|
|
const QLatin1String SHOW_OPTIONS_CHECKBOX_NAME ("show-options-checkbox");
|
|
}
|
|
|
|
void
|
|
MainWindow::newTorrent ()
|
|
{
|
|
MakeDialog * dialog = new MakeDialog (mySession, this);
|
|
dialog->setAttribute (Qt::WA_DeleteOnClose);
|
|
dialog->show ();
|
|
}
|
|
|
|
void
|
|
MainWindow::openTorrent ()
|
|
{
|
|
QFileDialog * d;
|
|
d = new QFileDialog (this,
|
|
tr ("Open Torrent"),
|
|
myPrefs.getString (Prefs::OPEN_DIALOG_FOLDER),
|
|
tr ("Torrent Files (*.torrent);;All Files (*.*)"));
|
|
d->setFileMode (QFileDialog::ExistingFiles);
|
|
d->setAttribute (Qt::WA_DeleteOnClose);
|
|
|
|
QCheckBox * b = new QCheckBox (tr ("Show &options dialog"));
|
|
b->setChecked (myPrefs.getBool (Prefs::OPTIONS_PROMPT));
|
|
b->setObjectName (SHOW_OPTIONS_CHECKBOX_NAME);
|
|
auto l = qobject_cast<QGridLayout*> (d->layout ());
|
|
if (l == nullptr)
|
|
{
|
|
l = new QGridLayout;
|
|
d->setLayout (l);
|
|
}
|
|
l->addWidget (b, l->rowCount(), 0, 1, -1, Qt::AlignLeft);
|
|
|
|
connect (d, SIGNAL (filesSelected (QStringList)),
|
|
this, SLOT (addTorrents (QStringList)));
|
|
|
|
d->show ();
|
|
}
|
|
|
|
void
|
|
MainWindow::openURL ()
|
|
{
|
|
QString str = qApp->clipboard ()->text (QClipboard::Selection);
|
|
|
|
if (!AddData::isSupported (str))
|
|
str = qApp->clipboard ()->text (QClipboard::Clipboard);
|
|
|
|
if (!AddData::isSupported (str))
|
|
str.clear ();
|
|
|
|
addTorrent (str, true);
|
|
}
|
|
|
|
void
|
|
MainWindow::addTorrents (const QStringList& filenames)
|
|
{
|
|
bool showOptions = myPrefs.getBool (Prefs::OPTIONS_PROMPT);
|
|
|
|
const QFileDialog * const fileDialog = qobject_cast<const QFileDialog*> (sender ());
|
|
if (fileDialog != NULL)
|
|
{
|
|
const QCheckBox * const b = fileDialog->findChild<const QCheckBox*> (SHOW_OPTIONS_CHECKBOX_NAME);
|
|
if (b != NULL)
|
|
showOptions = b->isChecked ();
|
|
}
|
|
|
|
for (const QString& filename: filenames)
|
|
addTorrent (filename, showOptions);
|
|
}
|
|
|
|
void
|
|
MainWindow::addTorrent (const AddData& addMe, bool showOptions)
|
|
{
|
|
if (showOptions)
|
|
{
|
|
OptionsDialog * o = new OptionsDialog (mySession, myPrefs, addMe, this);
|
|
o->show ();
|
|
qApp->alert (o);
|
|
}
|
|
else
|
|
{
|
|
mySession.addTorrent (addMe);
|
|
qApp->alert (this);
|
|
}
|
|
}
|
|
|
|
void
|
|
MainWindow::removeTorrents (const bool deleteFiles)
|
|
{
|
|
QSet<int> ids;
|
|
QMessageBox msgBox (this);
|
|
QString primary_text, secondary_text;
|
|
int incomplete = 0;
|
|
int connected = 0;
|
|
int count;
|
|
|
|
for (const QModelIndex& index: ui.listView->selectionModel ()->selectedRows ())
|
|
{
|
|
const Torrent * tor (index.data (TorrentModel::TorrentRole).value<const Torrent*> ());
|
|
ids.insert (tor->id ());
|
|
|
|
if (tor->connectedPeers ())
|
|
++connected;
|
|
|
|
if (!tor->isDone ())
|
|
++incomplete;
|
|
}
|
|
|
|
if (ids.isEmpty ())
|
|
return;
|
|
|
|
count = ids.size ();
|
|
|
|
if (!deleteFiles)
|
|
{
|
|
primary_text = (count == 1)
|
|
? tr ("Remove torrent?")
|
|
: tr ("Remove %Ln torrent(s)?", 0, count);
|
|
}
|
|
else
|
|
{
|
|
primary_text = (count == 1)
|
|
? tr ("Delete this torrent's downloaded files?")
|
|
: tr ("Delete these %Ln torrent(s)' downloaded files?", 0, count);
|
|
}
|
|
|
|
if (!incomplete && !connected)
|
|
{
|
|
secondary_text = (count == 1)
|
|
? tr ("Once removed, continuing the transfer will require the torrent file or magnet link.")
|
|
: tr ("Once removed, continuing the transfers will require the torrent files or magnet links.");
|
|
}
|
|
else if (count == incomplete)
|
|
{
|
|
secondary_text = (count == 1)
|
|
? tr ("This torrent has not finished downloading.")
|
|
: tr ("These torrents have not finished downloading.");
|
|
}
|
|
else if (count == connected)
|
|
{
|
|
secondary_text = (count == 1)
|
|
? tr ("This torrent is connected to peers.")
|
|
: tr ("These torrents are connected to peers.");
|
|
}
|
|
else
|
|
{
|
|
if (connected)
|
|
{
|
|
secondary_text = (connected == 1)
|
|
? tr ("One of these torrents is connected to peers.")
|
|
: tr ("Some of these torrents are connected to peers.");
|
|
}
|
|
|
|
if (connected && incomplete)
|
|
{
|
|
secondary_text += QLatin1Char ('\n');
|
|
}
|
|
|
|
if (incomplete)
|
|
{
|
|
secondary_text += (incomplete == 1)
|
|
? tr ("One of these torrents has not finished downloading.")
|
|
: tr ("Some of these torrents have not finished downloading.");
|
|
}
|
|
}
|
|
|
|
msgBox.setWindowTitle (QLatin1String (" "));
|
|
msgBox.setText (QString::fromLatin1 ("<big><b>%1</big></b>").arg (primary_text));
|
|
msgBox.setInformativeText (secondary_text);
|
|
msgBox.setStandardButtons (QMessageBox::Ok | QMessageBox::Cancel);
|
|
msgBox.setDefaultButton (QMessageBox::Cancel);
|
|
msgBox.setIcon (QMessageBox::Question);
|
|
// hack needed to keep the dialog from being too narrow
|
|
auto layout = qobject_cast<QGridLayout*> (msgBox.layout ());
|
|
if (layout == nullptr)
|
|
{
|
|
layout = new QGridLayout;
|
|
msgBox.setLayout (layout);
|
|
}
|
|
QSpacerItem* spacer = new QSpacerItem (450, 0, QSizePolicy::Minimum, QSizePolicy::Expanding);
|
|
layout->addItem (spacer, layout->rowCount (), 0, 1, layout->columnCount ());
|
|
|
|
if (msgBox.exec () == QMessageBox::Ok)
|
|
{
|
|
ui.listView->selectionModel ()->clear ();
|
|
mySession.removeTorrents (ids, deleteFiles);
|
|
}
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
void
|
|
MainWindow::updateNetworkIcon ()
|
|
{
|
|
const time_t now = time (NULL);
|
|
const int period = 3;
|
|
const time_t secondsSinceLastSend = now - myLastSendTime;
|
|
const time_t secondsSinceLastRead = now - myLastReadTime;
|
|
const bool isSending = secondsSinceLastSend <= period;
|
|
const bool isReading = secondsSinceLastRead <= period;
|
|
const char * key;
|
|
|
|
if (myNetworkError)
|
|
key = "network-error";
|
|
else if (isSending && isReading)
|
|
key = "network-transmit-receive";
|
|
else if (isSending)
|
|
key = "network-transmit";
|
|
else if (isReading)
|
|
key = "network-receive";
|
|
else
|
|
key = "network-idle";
|
|
const QIcon icon = getStockIcon (QLatin1String (key), QStyle::SP_DriveNetIcon);
|
|
const QPixmap pixmap = icon.pixmap (16, 16);
|
|
|
|
QString tip;
|
|
const QString url = mySession.getRemoteUrl ().host ();
|
|
if (!myLastReadTime)
|
|
tip = tr ("%1 has not responded yet").arg (url);
|
|
else if (myNetworkError)
|
|
tip = tr (myErrorMessage.toLatin1 ().constData ());
|
|
else if (secondsSinceLastRead < 30)
|
|
tip = tr ("%1 is responding").arg (url);
|
|
else if (secondsSinceLastRead < (60*2))
|
|
tip = tr ("%1 last responded %2 ago").arg (url).arg (Formatter::timeToString (secondsSinceLastRead));
|
|
else
|
|
tip = tr ("%1 is not responding").arg (url);
|
|
|
|
ui.networkLabel->setPixmap (pixmap);
|
|
ui.networkLabel->setToolTip (tip);
|
|
}
|
|
|
|
void
|
|
MainWindow::onNetworkTimer ()
|
|
{
|
|
updateNetworkIcon ();
|
|
}
|
|
|
|
void
|
|
MainWindow::dataReadProgress ()
|
|
{
|
|
if (!myNetworkError)
|
|
myLastReadTime = time (NULL);
|
|
}
|
|
|
|
void
|
|
MainWindow::dataSendProgress ()
|
|
{
|
|
myLastSendTime = time (NULL);
|
|
}
|
|
|
|
void
|
|
MainWindow::onError (QNetworkReply::NetworkError code)
|
|
{
|
|
const bool hadError = myNetworkError;
|
|
const bool haveError = (code != QNetworkReply::NoError)
|
|
&& (code != QNetworkReply::UnknownContentError);
|
|
|
|
myNetworkError = haveError;
|
|
refreshTrayIconSoon();
|
|
updateNetworkIcon();
|
|
|
|
// Refresh our model if we've just gotten a clean connection to the session.
|
|
// That way we can rebuild after a restart of transmission-daemon
|
|
if (hadError && !haveError)
|
|
myModel.clear();
|
|
}
|
|
|
|
void
|
|
MainWindow::errorMessage (const QString& msg)
|
|
{
|
|
myErrorMessage = msg;
|
|
}
|
|
|
|
void
|
|
MainWindow::wrongAuthentication ()
|
|
{
|
|
mySession.stop ();
|
|
openSession ();
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
void
|
|
MainWindow::dragEnterEvent (QDragEnterEvent * event)
|
|
{
|
|
const QMimeData * mime = event->mimeData ();
|
|
|
|
if (mime->hasFormat (QLatin1String ("application/x-bittorrent"))
|
|
|| mime->hasUrls()
|
|
|| mime->text ().trimmed ().endsWith (QLatin1String (".torrent"), Qt::CaseInsensitive)
|
|
|| mime->text ().startsWith (QLatin1String ("magnet:"), Qt::CaseInsensitive))
|
|
event->acceptProposedAction ();
|
|
}
|
|
|
|
void
|
|
MainWindow::dropEvent (QDropEvent * event)
|
|
{
|
|
QStringList list;
|
|
|
|
if (event->mimeData()->hasText())
|
|
{
|
|
list = event->mimeData()->text().trimmed().split(QLatin1Char ('\n'));
|
|
}
|
|
else if (event->mimeData()->hasUrls())
|
|
{
|
|
for (const QUrl& url: event->mimeData()->urls())
|
|
list.append(url.toLocalFile());
|
|
}
|
|
|
|
for (const QString& entry: list)
|
|
{
|
|
QString key = entry.trimmed();
|
|
|
|
if (!key.isEmpty())
|
|
{
|
|
const QUrl url (key);
|
|
|
|
if (url.scheme () == QLatin1String ("file"))
|
|
key = QUrl::fromPercentEncoding (url.path().toUtf8());
|
|
|
|
qApp->addTorrent (key);
|
|
}
|
|
}
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
void
|
|
MainWindow::contextMenuEvent (QContextMenuEvent * event)
|
|
{
|
|
ui.menuTorrent->popup (event->globalPos ());
|
|
}
|