transmission/qt/PrefsDialog.cc

877 lines
27 KiB
C++

// This file Copyright © Mnemosyne LLC.
// It may be used under GPLv2 (SPDX: GPL-2.0-only), GPLv3 (SPDX: GPL-3.0-only),
// or any future license endorsed by Mnemosyne LLC.
// License text can be found in the licenses/ folder.
#include "PrefsDialog.h"
#include <cassert>
#include <optional>
#include <QCheckBox>
#include <QComboBox>
#include <QCoreApplication>
#include <QDialogButtonBox>
#include <QDoubleSpinBox>
#include <QFileIconProvider>
#include <QFileInfo>
#include <QHBoxLayout>
#include <QIcon>
#include <QLabel>
#include <QLineEdit>
#include <QMessageBox>
#include <QPlainTextEdit>
#include <QPushButton>
#include <QSpinBox>
#include <QStyle>
#include <QTabWidget>
#include <QTime>
#include <QTimeEdit>
#include <QTimer>
#include <QVBoxLayout>
#include "ColumnResizer.h"
#include "Formatter.h"
#include "FreeSpaceLabel.h"
#include "Prefs.h"
#include "Session.h"
#include "Utils.h"
// ---
namespace
{
class PreferenceWidget
{
static char const* const PrefKey;
public:
explicit PreferenceWidget(QObject* object)
: object_{ object }
{
}
template<typename T>
[[nodiscard]] bool is() const
{
return qobject_cast<T*>(object_) != nullptr;
}
template<typename T>
[[nodiscard]] T const* as() const
{
assert(is<T>());
return static_cast<T const*>(object_);
}
template<typename T>
[[nodiscard]] T* as()
{
assert(is<T>());
return static_cast<T*>(object_);
}
void setPrefKey(int key)
{
object_->setProperty(PrefKey, key);
}
[[nodiscard]] int getPrefKey() const
{
return object_->property(PrefKey).toInt();
}
private:
QObject* object_;
};
char const* const PreferenceWidget::PrefKey = "pref-key";
int qtDayToTrDay(int day)
{
switch (day)
{
case Qt::Monday:
return TR_SCHED_MON;
case Qt::Tuesday:
return TR_SCHED_TUES;
case Qt::Wednesday:
return TR_SCHED_WED;
case Qt::Thursday:
return TR_SCHED_THURS;
case Qt::Friday:
return TR_SCHED_FRI;
case Qt::Saturday:
return TR_SCHED_SAT;
case Qt::Sunday:
return TR_SCHED_SUN;
default:
assert(false && "Invalid day of week");
return 0;
}
}
QString qtDayName(int day)
{
switch (day)
{
case Qt::Monday:
return PrefsDialog::tr("Monday");
case Qt::Tuesday:
return PrefsDialog::tr("Tuesday");
case Qt::Wednesday:
return PrefsDialog::tr("Wednesday");
case Qt::Thursday:
return PrefsDialog::tr("Thursday");
case Qt::Friday:
return PrefsDialog::tr("Friday");
case Qt::Saturday:
return PrefsDialog::tr("Saturday");
case Qt::Sunday:
return PrefsDialog::tr("Sunday");
default:
assert(false && "Invalid day of week");
return {};
}
}
[[nodiscard]] bool isDescendantOf(QObject const* descendant, QObject const* ancestor)
{
if (ancestor == nullptr)
{
return false;
}
while (descendant != nullptr)
{
if (descendant == ancestor)
{
return true;
}
descendant = descendant->parent();
}
return false;
}
} // namespace
bool PrefsDialog::updateWidgetValue(QWidget* widget, int pref_key) const
{
auto pref_widget = PreferenceWidget{ widget };
if (pref_widget.is<QCheckBox>())
{
pref_widget.as<QCheckBox>()->setChecked(prefs_.getBool(pref_key));
}
else if (pref_widget.is<QSpinBox>())
{
pref_widget.as<QSpinBox>()->setValue(prefs_.getInt(pref_key));
}
else if (pref_widget.is<QDoubleSpinBox>())
{
pref_widget.as<QDoubleSpinBox>()->setValue(prefs_.getDouble(pref_key));
}
else if (pref_widget.is<QTimeEdit>())
{
pref_widget.as<QTimeEdit>()->setTime(QTime{ 0, 0 }.addSecs(prefs_.getInt(pref_key) * 60));
}
else if (pref_widget.is<QLineEdit>())
{
pref_widget.as<QLineEdit>()->setText(prefs_.getString(pref_key));
}
else if (pref_widget.is<PathButton>())
{
pref_widget.as<PathButton>()->setPath(prefs_.getString(pref_key));
}
else if (pref_widget.is<FreeSpaceLabel>())
{
pref_widget.as<FreeSpaceLabel>()->setPath(prefs_.getString(pref_key));
}
else if (pref_widget.is<QPlainTextEdit>())
{
pref_widget.as<QPlainTextEdit>()->setPlainText(prefs_.getString(pref_key));
}
else
{
return false;
}
return true;
}
void PrefsDialog::linkWidgetToPref(QWidget* widget, int pref_key)
{
auto pref_widget = PreferenceWidget{ widget };
pref_widget.setPrefKey(pref_key);
updateWidgetValue(widget, pref_key);
widgets_.try_emplace(pref_key, widget);
if (auto const* check_box = qobject_cast<QCheckBox*>(widget); check_box != nullptr)
{
connect(check_box, &QAbstractButton::toggled, this, &PrefsDialog::checkBoxToggled);
return;
}
if (auto const* time_edit = qobject_cast<QTimeEdit*>(widget); time_edit != nullptr)
{
connect(time_edit, &QAbstractSpinBox::editingFinished, this, &PrefsDialog::timeEditingFinished);
return;
}
if (auto const* line_edit = qobject_cast<QLineEdit*>(widget); line_edit != nullptr)
{
connect(line_edit, &QLineEdit::editingFinished, this, &PrefsDialog::lineEditingFinished);
return;
}
if (auto const* path_button = qobject_cast<PathButton*>(widget); path_button != nullptr)
{
connect(path_button, &PathButton::pathChanged, this, &PrefsDialog::pathChanged);
return;
}
if (auto const* spin_box = qobject_cast<QAbstractSpinBox*>(widget); spin_box != nullptr)
{
connect(spin_box, &QAbstractSpinBox::editingFinished, this, &PrefsDialog::spinBoxEditingFinished);
return;
}
}
void PrefsDialog::focusChanged(QWidget* old, QWidget* cur)
{
// We don't want to change the preference every time there's a keystroke
// in a QPlainTextEdit, so instead of connecting to the textChanged signal,
// only update the pref when the text changed AND focus was lost.
char constexpr const* const StartValue = "StartValue";
if (auto* const edit = qobject_cast<QPlainTextEdit*>(cur); isDescendantOf(edit, this))
{
edit->setProperty(StartValue, edit->toPlainText());
}
if (auto const* const edit = qobject_cast<QPlainTextEdit*>(old); isDescendantOf(edit, this))
{
if (auto const val = edit->toPlainText(); val != edit->property(StartValue).toString())
{
setPref(PreferenceWidget{ old }.getPrefKey(), val);
}
}
// (TODO: we probably want to do this for single-line text entries too?)
}
void PrefsDialog::checkBoxToggled(bool checked)
{
auto const pref_widget = PreferenceWidget{ sender() };
if (pref_widget.is<QCheckBox>())
{
setPref(pref_widget.getPrefKey(), checked);
}
}
void PrefsDialog::spinBoxEditingFinished()
{
auto const pref_widget = PreferenceWidget{ sender() };
if (pref_widget.is<QDoubleSpinBox>())
{
setPref(pref_widget.getPrefKey(), pref_widget.as<QDoubleSpinBox>()->value());
}
else if (pref_widget.is<QSpinBox>())
{
setPref(pref_widget.getPrefKey(), pref_widget.as<QSpinBox>()->value());
}
}
void PrefsDialog::timeEditingFinished()
{
auto const pref_widget = PreferenceWidget{ sender() };
if (pref_widget.is<QTimeEdit>())
{
setPref(pref_widget.getPrefKey(), QTime{ 0, 0 }.secsTo(pref_widget.as<QTimeEdit>()->time()) / 60);
}
}
void PrefsDialog::lineEditingFinished()
{
auto const pref_widget = PreferenceWidget{ sender() };
if (pref_widget.is<QLineEdit>())
{
auto const* const line_edit = pref_widget.as<QLineEdit>();
if (line_edit->isModified())
{
setPref(pref_widget.getPrefKey(), line_edit->text());
}
}
}
void PrefsDialog::pathChanged(QString const& path)
{
auto const pref_widget = PreferenceWidget{ sender() };
if (pref_widget.is<PathButton>())
{
setPref(pref_widget.getPrefKey(), path);
}
}
// ---
void PrefsDialog::initRemoteTab()
{
linkWidgetToPref(ui_.enableRpcCheck, Prefs::RPC_ENABLED);
linkWidgetToPref(ui_.rpcPortSpin, Prefs::RPC_PORT);
linkWidgetToPref(ui_.requireRpcAuthCheck, Prefs::RPC_AUTH_REQUIRED);
linkWidgetToPref(ui_.rpcUsernameEdit, Prefs::RPC_USERNAME);
linkWidgetToPref(ui_.rpcPasswordEdit, Prefs::RPC_PASSWORD);
linkWidgetToPref(ui_.enableRpcWhitelistCheck, Prefs::RPC_WHITELIST_ENABLED);
linkWidgetToPref(ui_.rpcWhitelistEdit, Prefs::RPC_WHITELIST);
web_widgets_ << ui_.rpcPortLabel << ui_.rpcPortSpin << ui_.requireRpcAuthCheck << ui_.enableRpcWhitelistCheck;
web_auth_widgets_ << ui_.rpcUsernameLabel << ui_.rpcUsernameEdit << ui_.rpcPasswordLabel << ui_.rpcPasswordEdit;
web_whitelist_widgets_ << ui_.rpcWhitelistLabel << ui_.rpcWhitelistEdit;
unsupported_when_remote_ << ui_.enableRpcCheck << web_widgets_ << web_auth_widgets_ << web_whitelist_widgets_;
connect(ui_.openWebClientButton, &QAbstractButton::clicked, &session_, &Session::launchWebInterface);
}
// ---
void PrefsDialog::altSpeedDaysEdited(int i)
{
int const value = qobject_cast<QComboBox*>(sender())->itemData(i).toInt();
setPref(Prefs::ALT_SPEED_LIMIT_TIME_DAY, value);
}
void PrefsDialog::initSpeedTab()
{
auto const suffix = QStringLiteral(" %1").arg(Speed::display_name(Speed::Units::KByps));
auto const locale = QLocale{};
ui_.uploadSpeedLimitSpin->setSuffix(suffix);
ui_.downloadSpeedLimitSpin->setSuffix(suffix);
ui_.altUploadSpeedLimitSpin->setSuffix(suffix);
ui_.altDownloadSpeedLimitSpin->setSuffix(suffix);
ui_.altSpeedLimitDaysCombo->addItem(tr("Every Day"), QVariant{ TR_SCHED_ALL });
ui_.altSpeedLimitDaysCombo->addItem(tr("Weekdays"), QVariant{ TR_SCHED_WEEKDAY });
ui_.altSpeedLimitDaysCombo->addItem(tr("Weekends"), QVariant{ TR_SCHED_WEEKEND });
ui_.altSpeedLimitDaysCombo->insertSeparator(ui_.altSpeedLimitDaysCombo->count());
for (int i = locale.firstDayOfWeek(); i <= Qt::Sunday; ++i)
{
ui_.altSpeedLimitDaysCombo->addItem(qtDayName(i), qtDayToTrDay(i));
}
for (int i = Qt::Monday; i < locale.firstDayOfWeek(); ++i)
{
ui_.altSpeedLimitDaysCombo->addItem(qtDayName(i), qtDayToTrDay(i));
}
ui_.altSpeedLimitDaysCombo->setCurrentIndex(
ui_.altSpeedLimitDaysCombo->findData(prefs_.getInt(Prefs::ALT_SPEED_LIMIT_TIME_DAY)));
linkWidgetToPref(ui_.uploadSpeedLimitCheck, Prefs::USPEED_ENABLED);
linkWidgetToPref(ui_.uploadSpeedLimitSpin, Prefs::USPEED);
linkWidgetToPref(ui_.downloadSpeedLimitCheck, Prefs::DSPEED_ENABLED);
linkWidgetToPref(ui_.downloadSpeedLimitSpin, Prefs::DSPEED);
linkWidgetToPref(ui_.altUploadSpeedLimitSpin, Prefs::ALT_SPEED_LIMIT_UP);
linkWidgetToPref(ui_.altDownloadSpeedLimitSpin, Prefs::ALT_SPEED_LIMIT_DOWN);
linkWidgetToPref(ui_.altSpeedLimitScheduleCheck, Prefs::ALT_SPEED_LIMIT_TIME_ENABLED);
linkWidgetToPref(ui_.altSpeedLimitStartTimeEdit, Prefs::ALT_SPEED_LIMIT_TIME_BEGIN);
linkWidgetToPref(ui_.altSpeedLimitEndTimeEdit, Prefs::ALT_SPEED_LIMIT_TIME_END);
sched_widgets_ << ui_.altSpeedLimitStartTimeEdit << ui_.altSpeedLimitToLabel << ui_.altSpeedLimitEndTimeEdit
<< ui_.altSpeedLimitDaysLabel << ui_.altSpeedLimitDaysCombo;
auto* cr = new ColumnResizer{ this };
cr->addLayout(ui_.speedLimitsSectionLayout);
cr->addLayout(ui_.altSpeedLimitsSectionLayout);
cr->update();
connect(ui_.altSpeedLimitDaysCombo, qOverload<int>(&QComboBox::activated), this, &PrefsDialog::altSpeedDaysEdited);
}
// ---
void PrefsDialog::initDesktopTab()
{
linkWidgetToPref(ui_.showTrayIconCheck, Prefs::SHOW_TRAY_ICON);
linkWidgetToPref(ui_.startMinimizedCheck, Prefs::START_MINIMIZED);
linkWidgetToPref(ui_.notifyOnTorrentAddedCheck, Prefs::SHOW_NOTIFICATION_ON_ADD);
linkWidgetToPref(ui_.notifyOnTorrentCompletedCheck, Prefs::SHOW_NOTIFICATION_ON_COMPLETE);
linkWidgetToPref(ui_.playSoundOnTorrentCompletedCheck, Prefs::COMPLETE_SOUND_ENABLED);
}
// ---
QString PrefsDialog::getPortStatusText(PrefsDialog::PortTestStatus status) noexcept
{
switch (status)
{
case PORT_TEST_UNKNOWN:
return tr("unknown");
case PORT_TEST_CHECKING:
return tr("checking…");
case PORT_TEST_OPEN:
return tr("open");
case PORT_TEST_CLOSED:
return tr("closed");
case PORT_TEST_ERROR:
return tr("error");
default:
return {};
}
}
void PrefsDialog::updatePortStatusLabel()
{
auto const status_ipv4 = getPortStatusText(port_test_status_[Session::PORT_TEST_IPV4]);
auto const status_ipv6 = getPortStatusText(port_test_status_[Session::PORT_TEST_IPV6]);
ui_.peerPortStatusLabel->setText(
port_test_status_[Session::PORT_TEST_IPV4] == port_test_status_[Session::PORT_TEST_IPV6] ?
tr("Status: <b>%1</b>").arg(status_ipv4) :
tr("Status: <b>%1</b> (IPv4), <b>%2</b> (IPv6)").arg(status_ipv4).arg(status_ipv6));
}
void PrefsDialog::portTestSetEnabled()
{
// Depend on the RPC call status instead of the UI status, so that the widgets
// won't be enabled even if the port peer port changed while we have port-test
// RPC call(s) in-flight.
auto const sensitive = !session_.portTestPending(Session::PORT_TEST_IPV4) &&
!session_.portTestPending(Session::PORT_TEST_IPV6);
ui_.testPeerPortButton->setEnabled(sensitive);
widgets_[Prefs::PEER_PORT]->setEnabled(sensitive);
}
void PrefsDialog::onPortTested(std::optional<bool> result, Session::PortTestIpProtocol ip_protocol)
{
constexpr auto StatusFromResult = [](std::optional<bool> const res)
{
if (!res)
{
return PORT_TEST_ERROR;
}
if (!*res)
{
return PORT_TEST_CLOSED;
}
return PORT_TEST_OPEN;
};
// Only update the UI if the current status is "checking", so that
// we won't show the port test results for the old peer port if it
// changed while we have port-test RPC call(s) in-flight.
if (port_test_status_[ip_protocol] == PORT_TEST_CHECKING)
{
port_test_status_[ip_protocol] = StatusFromResult(result);
updatePortStatusLabel();
}
portTestSetEnabled();
}
void PrefsDialog::onPortTest()
{
port_test_status_[Session::PORT_TEST_IPV4] = PORT_TEST_CHECKING;
port_test_status_[Session::PORT_TEST_IPV6] = PORT_TEST_CHECKING;
updatePortStatusLabel();
session_.portTest(Session::PORT_TEST_IPV4);
session_.portTest(Session::PORT_TEST_IPV6);
portTestSetEnabled();
}
void PrefsDialog::initNetworkTab()
{
ui_.torrentPeerLimitSpin->setRange(1, INT_MAX);
ui_.globalPeerLimitSpin->setRange(1, INT_MAX);
linkWidgetToPref(ui_.peerPortSpin, Prefs::PEER_PORT);
linkWidgetToPref(ui_.randomPeerPortCheck, Prefs::PEER_PORT_RANDOM_ON_START);
linkWidgetToPref(ui_.enablePortForwardingCheck, Prefs::PORT_FORWARDING);
linkWidgetToPref(ui_.torrentPeerLimitSpin, Prefs::PEER_LIMIT_TORRENT);
linkWidgetToPref(ui_.globalPeerLimitSpin, Prefs::PEER_LIMIT_GLOBAL);
linkWidgetToPref(ui_.enableUtpCheck, Prefs::UTP_ENABLED);
linkWidgetToPref(ui_.enablePexCheck, Prefs::PEX_ENABLED);
linkWidgetToPref(ui_.enableDhtCheck, Prefs::DHT_ENABLED);
linkWidgetToPref(ui_.enableLpdCheck, Prefs::LPD_ENABLED);
linkWidgetToPref(ui_.defaultTrackersPlainTextEdit, Prefs::DEFAULT_TRACKERS);
auto* cr = new ColumnResizer{ this };
cr->addLayout(ui_.incomingPeersSectionLayout);
cr->addLayout(ui_.peerLimitsSectionLayout);
cr->update();
connect(ui_.testPeerPortButton, &QAbstractButton::clicked, this, &PrefsDialog::onPortTest);
connect(&session_, &Session::portTested, this, &PrefsDialog::onPortTested);
updatePortStatusLabel();
}
// ---
void PrefsDialog::onBlocklistDialogDestroyed(QObject* o)
{
Q_UNUSED(o)
blocklist_dialog_ = nullptr;
}
void PrefsDialog::onUpdateBlocklistCancelled()
{
disconnect(&session_, &Session::blocklistUpdated, this, &PrefsDialog::onBlocklistUpdated);
blocklist_dialog_->deleteLater();
}
void PrefsDialog::onBlocklistUpdated(int n)
{
blocklist_dialog_->setText(tr("<b>Update succeeded!</b><p>Blocklist now has %Ln rule(s).</p>", nullptr, n));
blocklist_dialog_->setTextFormat(Qt::RichText);
}
void PrefsDialog::onUpdateBlocklistClicked()
{
blocklist_dialog_ = new QMessageBox{ QMessageBox::Information,
QString{},
tr("<b>Update Blocklist</b><p>Getting new blocklist…</p>"),
QMessageBox::Close,
this };
connect(blocklist_dialog_, &QDialog::rejected, this, &PrefsDialog::onUpdateBlocklistCancelled);
connect(&session_, &Session::blocklistUpdated, this, &PrefsDialog::onBlocklistUpdated);
blocklist_dialog_->show();
session_.updateBlocklist();
}
void PrefsDialog::encryptionEdited(int i)
{
int const value(qobject_cast<QComboBox*>(sender())->itemData(i).toInt());
setPref(Prefs::ENCRYPTION, value);
}
void PrefsDialog::initPrivacyTab()
{
ui_.encryptionModeCombo->addItem(tr("Allow encryption"), 0);
ui_.encryptionModeCombo->addItem(tr("Prefer encryption"), 1);
ui_.encryptionModeCombo->addItem(tr("Require encryption"), 2);
linkWidgetToPref(ui_.encryptionModeCombo, Prefs::ENCRYPTION);
linkWidgetToPref(ui_.blocklistCheck, Prefs::BLOCKLIST_ENABLED);
linkWidgetToPref(ui_.blocklistEdit, Prefs::BLOCKLIST_URL);
linkWidgetToPref(ui_.autoUpdateBlocklistCheck, Prefs::BLOCKLIST_UPDATES_ENABLED);
block_widgets_ << ui_.blocklistEdit << ui_.blocklistStatusLabel << ui_.updateBlocklistButton
<< ui_.autoUpdateBlocklistCheck;
auto* cr = new ColumnResizer{ this };
cr->addLayout(ui_.encryptionSectionLayout);
cr->addLayout(ui_.blocklistSectionLayout);
cr->update();
connect(ui_.updateBlocklistButton, &QAbstractButton::clicked, this, &PrefsDialog::onUpdateBlocklistClicked);
connect(ui_.encryptionModeCombo, qOverload<int>(&QComboBox::activated), this, &PrefsDialog::encryptionEdited);
updateBlocklistLabel();
}
// ---
void PrefsDialog::onIdleLimitChanged()
{
//: Spin box format, "Stop seeding if idle for: [ 5 minutes ]"
auto const* const units_format = QT_TRANSLATE_N_NOOP("PrefsDialog", "%1 minute(s)");
auto const placeholder = QStringLiteral("%1");
Utils::updateSpinBoxFormat(ui_.idleLimitSpin, "PrefsDialog", units_format, placeholder);
}
void PrefsDialog::initSeedingTab()
{
ui_.doneSeedingScriptButton->setTitle(tr("Select \"Torrent Done Seeding\" Script"));
linkWidgetToPref(ui_.ratioLimitCheck, Prefs::RATIO_ENABLED);
linkWidgetToPref(ui_.ratioLimitSpin, Prefs::RATIO);
linkWidgetToPref(ui_.idleLimitCheck, Prefs::IDLE_LIMIT_ENABLED);
linkWidgetToPref(ui_.idleLimitSpin, Prefs::IDLE_LIMIT);
linkWidgetToPref(ui_.doneSeedingScriptCheck, Prefs::SCRIPT_TORRENT_DONE_SEEDING_ENABLED);
linkWidgetToPref(ui_.doneSeedingScriptButton, Prefs::SCRIPT_TORRENT_DONE_SEEDING_FILENAME);
linkWidgetToPref(ui_.doneSeedingScriptEdit, Prefs::SCRIPT_TORRENT_DONE_SEEDING_FILENAME);
connect(ui_.idleLimitSpin, qOverload<int>(&QSpinBox::valueChanged), this, &PrefsDialog::onIdleLimitChanged);
updateSeedingWidgetsLocality();
onIdleLimitChanged();
}
void PrefsDialog::onQueueStalledMinutesChanged()
{
//: Spin box format, "Download is inactive if data sharing stopped: [ 5 minutes ago ]"
auto const* const units_format = QT_TRANSLATE_N_NOOP("PrefsDialog", "%1 minute(s) ago");
auto const placeholder = QStringLiteral("%1");
Utils::updateSpinBoxFormat(ui_.queueStalledMinutesSpin, "PrefsDialog", units_format, placeholder);
}
void PrefsDialog::initDownloadingTab()
{
ui_.watchDirButton->setMode(PathButton::DirectoryMode);
ui_.downloadDirButton->setMode(PathButton::DirectoryMode);
ui_.incompleteDirButton->setMode(PathButton::DirectoryMode);
ui_.doneDownloadingScriptButton->setMode(PathButton::FileMode);
ui_.doneSeedingScriptButton->setMode(PathButton::FileMode);
ui_.watchDirButton->setTitle(tr("Select Watch Directory"));
ui_.downloadDirButton->setTitle(tr("Select Destination"));
ui_.incompleteDirButton->setTitle(tr("Select Incomplete Directory"));
ui_.doneDownloadingScriptButton->setTitle(tr("Select \"Torrent Done Downloading\" Script"));
ui_.watchDirStack->setMinimumWidth(200);
ui_.downloadDirFreeSpaceLabel->setSession(session_);
ui_.downloadDirFreeSpaceLabel->setPath(prefs_.getString(Prefs::DOWNLOAD_DIR));
linkWidgetToPref(ui_.watchDirCheck, Prefs::DIR_WATCH_ENABLED);
linkWidgetToPref(ui_.watchDirButton, Prefs::DIR_WATCH);
linkWidgetToPref(ui_.watchDirEdit, Prefs::DIR_WATCH);
linkWidgetToPref(ui_.showTorrentOptionsDialogCheck, Prefs::OPTIONS_PROMPT);
linkWidgetToPref(ui_.startAddedTorrentsCheck, Prefs::START);
linkWidgetToPref(ui_.detectTorrentFromClipboard, Prefs::READ_CLIPBOARD);
linkWidgetToPref(ui_.trashTorrentFileCheck, Prefs::TRASH_ORIGINAL);
linkWidgetToPref(ui_.downloadDirButton, Prefs::DOWNLOAD_DIR);
linkWidgetToPref(ui_.downloadDirEdit, Prefs::DOWNLOAD_DIR);
linkWidgetToPref(ui_.downloadDirFreeSpaceLabel, Prefs::DOWNLOAD_DIR);
linkWidgetToPref(ui_.downloadQueueSizeSpin, Prefs::DOWNLOAD_QUEUE_SIZE);
linkWidgetToPref(ui_.queueStalledMinutesSpin, Prefs::QUEUE_STALLED_MINUTES);
linkWidgetToPref(ui_.renamePartialFilesCheck, Prefs::RENAME_PARTIAL_FILES);
linkWidgetToPref(ui_.incompleteDirCheck, Prefs::INCOMPLETE_DIR_ENABLED);
linkWidgetToPref(ui_.incompleteDirButton, Prefs::INCOMPLETE_DIR);
linkWidgetToPref(ui_.incompleteDirEdit, Prefs::INCOMPLETE_DIR);
linkWidgetToPref(ui_.doneDownloadingScriptCheck, Prefs::SCRIPT_TORRENT_DONE_ENABLED);
linkWidgetToPref(ui_.doneDownloadingScriptButton, Prefs::SCRIPT_TORRENT_DONE_FILENAME);
linkWidgetToPref(ui_.doneDownloadingScriptEdit, Prefs::SCRIPT_TORRENT_DONE_FILENAME);
auto* cr = new ColumnResizer{ this };
cr->addLayout(ui_.addingSectionLayout);
cr->addLayout(ui_.downloadQueueSectionLayout);
cr->addLayout(ui_.incompleteSectionLayout);
cr->update();
connect(
ui_.queueStalledMinutesSpin,
qOverload<int>(&QSpinBox::valueChanged),
this,
&PrefsDialog::onQueueStalledMinutesChanged);
updateDownloadingWidgetsLocality();
onQueueStalledMinutesChanged();
}
void PrefsDialog::updateDownloadingWidgetsLocality()
{
ui_.watchDirStack->setCurrentWidget(is_local_ ? static_cast<QWidget*>(ui_.watchDirButton) : ui_.watchDirEdit);
ui_.downloadDirStack->setCurrentWidget(is_local_ ? static_cast<QWidget*>(ui_.downloadDirButton) : ui_.downloadDirEdit);
ui_.incompleteDirStack->setCurrentWidget(
is_local_ ? static_cast<QWidget*>(ui_.incompleteDirButton) : ui_.incompleteDirEdit);
ui_.doneDownloadingScriptStack->setCurrentWidget(
is_local_ ? static_cast<QWidget*>(ui_.doneDownloadingScriptButton) : ui_.doneDownloadingScriptEdit);
ui_.watchDirStack->setFixedHeight(ui_.watchDirStack->currentWidget()->sizeHint().height());
ui_.downloadDirStack->setFixedHeight(ui_.downloadDirStack->currentWidget()->sizeHint().height());
ui_.incompleteDirStack->setFixedHeight(ui_.incompleteDirStack->currentWidget()->sizeHint().height());
ui_.doneDownloadingScriptStack->setFixedHeight(ui_.doneDownloadingScriptStack->currentWidget()->sizeHint().height());
ui_.downloadDirLabel->setBuddy(ui_.downloadDirStack->currentWidget());
}
void PrefsDialog::updateSeedingWidgetsLocality()
{
ui_.doneSeedingScriptStack->setCurrentWidget(
is_local_ ? static_cast<QWidget*>(ui_.doneSeedingScriptButton) : ui_.doneSeedingScriptEdit);
ui_.doneSeedingScriptStack->setFixedHeight(ui_.doneSeedingScriptStack->currentWidget()->sizeHint().height());
}
// ---
PrefsDialog::PrefsDialog(Session& session, Prefs& prefs, QWidget* parent)
: BaseDialog{ parent }
, session_{ session }
, prefs_{ prefs }
, is_server_{ session.isServer() }
, is_local_{ session_.isLocal() }
{
ui_.setupUi(this);
initSpeedTab();
initDownloadingTab();
initSeedingTab();
initPrivacyTab();
initNetworkTab();
initDesktopTab();
initRemoteTab();
connect(&session_, &Session::sessionUpdated, this, &PrefsDialog::sessionUpdated);
static std::array<int, 10> constexpr InitKeys = {
Prefs::ALT_SPEED_LIMIT_ENABLED,
Prefs::ALT_SPEED_LIMIT_TIME_ENABLED,
Prefs::BLOCKLIST_ENABLED,
Prefs::DIR_WATCH,
Prefs::DOWNLOAD_DIR,
Prefs::ENCRYPTION,
Prefs::INCOMPLETE_DIR,
Prefs::INCOMPLETE_DIR_ENABLED,
Prefs::RPC_ENABLED,
Prefs::SCRIPT_TORRENT_DONE_FILENAME,
};
for (auto const key : InitKeys)
{
refreshPref(key);
}
// if it's a remote session, disable the preferences
// that don't work in remote sessions
if (!is_server_)
{
for (QWidget* const w : unsupported_when_remote_)
{
w->setToolTip(tr("Not supported by remote sessions"));
w->setEnabled(false);
}
}
adjustSize();
connect(qApp, &QApplication::focusChanged, this, &PrefsDialog::focusChanged);
}
void PrefsDialog::setPref(int key, QVariant const& v)
{
prefs_.set(key, v);
refreshPref(key);
}
// ---
void PrefsDialog::sessionUpdated()
{
if (bool const is_local = session_.isLocal(); is_local_ != is_local)
{
is_local_ = is_local;
updateDownloadingWidgetsLocality();
updateSeedingWidgetsLocality();
}
updateBlocklistLabel();
}
void PrefsDialog::updateBlocklistLabel()
{
int const n = session_.blocklistSize();
ui_.blocklistStatusLabel->setText(tr("<i>Blocklist contains %Ln rule(s)</i>", nullptr, n));
}
void PrefsDialog::refreshPref(int key)
{
switch (key)
{
case Prefs::RPC_ENABLED:
case Prefs::RPC_WHITELIST_ENABLED:
case Prefs::RPC_AUTH_REQUIRED:
{
bool const enabled(prefs_.getBool(Prefs::RPC_ENABLED));
bool const whitelist(prefs_.getBool(Prefs::RPC_WHITELIST_ENABLED));
bool const auth(prefs_.getBool(Prefs::RPC_AUTH_REQUIRED));
for (QWidget* const w : web_whitelist_widgets_)
{
w->setEnabled(enabled && whitelist);
}
for (QWidget* const w : web_auth_widgets_)
{
w->setEnabled(enabled && auth);
}
for (QWidget* const w : web_widgets_)
{
w->setEnabled(enabled);
}
break;
}
case Prefs::ALT_SPEED_LIMIT_TIME_ENABLED:
{
bool const enabled = prefs_.getBool(key);
for (QWidget* const w : sched_widgets_)
{
w->setEnabled(enabled);
}
break;
}
case Prefs::BLOCKLIST_ENABLED:
{
bool const enabled = prefs_.getBool(key);
for (QWidget* const w : block_widgets_)
{
w->setEnabled(enabled);
}
break;
}
case Prefs::PEER_PORT:
port_test_status_[Session::PORT_TEST_IPV4] = PORT_TEST_UNKNOWN;
port_test_status_[Session::PORT_TEST_IPV6] = PORT_TEST_UNKNOWN;
updatePortStatusLabel();
portTestSetEnabled();
break;
default:
break;
}
if (auto iter = widgets_.find(key); iter != std::end(widgets_))
{
QWidget* const w = iter->second;
w->blockSignals(true);
if (!updateWidgetValue(w, key) && (key == Prefs::ENCRYPTION))
{
auto* combo_box = qobject_cast<QComboBox*>(w);
int const index = combo_box->findData(prefs_.getInt(key));
combo_box->setCurrentIndex(index);
}
w->blockSignals(false);
}
}