mirror of
https://github.com/transmission/transmission
synced 2025-01-02 13:05:08 +00:00
ffda5bb68a
* refactor: use fmt in gtk/SystemIconTray * refactor: use fmt in gtk/MakeDialog * refactor: remove unused tr_strip_positional_args() Co-authored-by: Mike Gelfand <mikedld@mikedld.com>
533 lines
17 KiB
C++
533 lines
17 KiB
C++
// This file Copyright © 2007-2022 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 <iostream>
|
|
#include <memory>
|
|
#include <string>
|
|
|
|
#include <glibmm.h>
|
|
#include <glibmm/i18n.h>
|
|
|
|
#include <fmt/core.h>
|
|
|
|
#include <libtransmission/transmission.h>
|
|
#include <libtransmission/makemeta.h>
|
|
#include <libtransmission/utils.h> /* tr_formatter_mem_B() */
|
|
|
|
#include "HigWorkarea.h"
|
|
#include "MakeDialog.h"
|
|
#include "PrefsDialog.h"
|
|
#include "Session.h"
|
|
#include "Utils.h"
|
|
|
|
namespace
|
|
{
|
|
|
|
auto const FileChosenKey = Glib::Quark("file-is-chosen");
|
|
|
|
class MakeProgressDialog : public Gtk::Dialog
|
|
{
|
|
public:
|
|
MakeProgressDialog(
|
|
Gtk::Window& parent,
|
|
tr_metainfo_builder& builder,
|
|
std::string const& target,
|
|
Glib::RefPtr<Session> const& core);
|
|
~MakeProgressDialog() override;
|
|
|
|
TR_DISABLE_COPY_MOVE(MakeProgressDialog)
|
|
|
|
private:
|
|
bool onProgressDialogRefresh();
|
|
void onProgressDialogResponse(int response);
|
|
|
|
void addTorrent();
|
|
|
|
private:
|
|
tr_metainfo_builder& builder_;
|
|
std::string const target_;
|
|
Glib::RefPtr<Session> const core_;
|
|
|
|
sigc::connection progress_tag_;
|
|
Gtk::Label* progress_label_ = nullptr;
|
|
Gtk::ProgressBar* progress_bar_ = nullptr;
|
|
};
|
|
|
|
} // namespace
|
|
|
|
class MakeDialog::Impl
|
|
{
|
|
public:
|
|
Impl(MakeDialog& dialog, Glib::RefPtr<Session> const& core);
|
|
|
|
TR_DISABLE_COPY_MOVE(Impl)
|
|
|
|
private:
|
|
void onSourceToggled2(Gtk::ToggleButton* tb, Gtk::FileChooserButton* chooser);
|
|
void onChooserChosen(Gtk::FileChooserButton* chooser);
|
|
void onResponse(int response);
|
|
|
|
void on_drag_data_received(
|
|
Glib::RefPtr<Gdk::DragContext> const& drag_context,
|
|
int x,
|
|
int y,
|
|
Gtk::SelectionData const& selection_data,
|
|
guint info,
|
|
guint time_);
|
|
|
|
void updatePiecesLabel();
|
|
|
|
void setFilename(std::string const& filename);
|
|
|
|
void makeProgressDialog(std::string const& target);
|
|
|
|
private:
|
|
MakeDialog& dialog_;
|
|
Glib::RefPtr<Session> const core_;
|
|
|
|
Gtk::RadioButton* file_radio_ = nullptr;
|
|
Gtk::FileChooserButton* file_chooser_ = nullptr;
|
|
Gtk::RadioButton* folder_radio_ = nullptr;
|
|
Gtk::FileChooserButton* folder_chooser_ = nullptr;
|
|
Gtk::Label* pieces_lb_ = nullptr;
|
|
Gtk::FileChooserButton* destination_chooser_ = nullptr;
|
|
Gtk::CheckButton* comment_check_ = nullptr;
|
|
Gtk::Entry* comment_entry_ = nullptr;
|
|
Gtk::CheckButton* private_check_ = nullptr;
|
|
Gtk::CheckButton* source_check_ = nullptr;
|
|
Gtk::Entry* source_entry_ = nullptr;
|
|
std::unique_ptr<MakeProgressDialog> progress_dialog_;
|
|
Glib::RefPtr<Gtk::TextBuffer> announce_text_buffer_;
|
|
std::unique_ptr<tr_metainfo_builder, void (*)(tr_metainfo_builder*)> builder_ = { nullptr, nullptr };
|
|
};
|
|
|
|
bool MakeProgressDialog::onProgressDialogRefresh()
|
|
{
|
|
Glib::ustring str;
|
|
double const fraction = builder_.pieceCount != 0 ? (double)builder_.pieceIndex / builder_.pieceCount : 0;
|
|
auto const base = Glib::path_get_basename(builder_.top);
|
|
|
|
/* progress label */
|
|
if (!builder_.isDone)
|
|
{
|
|
str = fmt::format(_("Creating '{path}'"), fmt::arg("path", base));
|
|
}
|
|
else if (builder_.result == TrMakemetaResult::OK)
|
|
{
|
|
str = fmt::format(_("Created '{path}'"), fmt::arg("path", base));
|
|
}
|
|
else if (builder_.result == TrMakemetaResult::CANCELLED)
|
|
{
|
|
str = _("Cancelled");
|
|
}
|
|
else if (builder_.result == TrMakemetaResult::ERR_URL)
|
|
{
|
|
str = fmt::format(_("Unsupported URL: '{url}'"), fmt::arg("url", builder_.errfile));
|
|
}
|
|
else if (builder_.result == TrMakemetaResult::ERR_IO_READ)
|
|
{
|
|
str = fmt::format(
|
|
_("Couldn't read '{path}': {error} ({error_code})"),
|
|
fmt::arg("path", builder_.errfile),
|
|
fmt::arg("error", Glib::strerror(builder_.my_errno)),
|
|
fmt::arg("error_code", builder_.my_errno));
|
|
}
|
|
else if (builder_.result == TrMakemetaResult::ERR_IO_WRITE)
|
|
{
|
|
str = fmt::format(
|
|
_("Couldn't save '{path}': {error} ({error_code})"),
|
|
fmt::arg("path", builder_.errfile),
|
|
fmt::arg("error", Glib::strerror(builder_.my_errno)),
|
|
fmt::arg("error_code", builder_.my_errno));
|
|
}
|
|
else
|
|
{
|
|
g_assert_not_reached();
|
|
}
|
|
|
|
gtr_label_set_text(*progress_label_, str);
|
|
|
|
/* progress bar */
|
|
if (builder_.pieceIndex == 0)
|
|
{
|
|
str.clear();
|
|
}
|
|
else
|
|
{
|
|
/* how much data we've scanned through to generate checksums */
|
|
str = fmt::format(
|
|
_("Scanned {file_size}"),
|
|
fmt::arg("file_size", tr_strlsize((uint64_t)builder_.pieceIndex * (uint64_t)builder_.pieceSize)));
|
|
}
|
|
|
|
progress_bar_->set_fraction(fraction);
|
|
progress_bar_->set_text(str);
|
|
|
|
/* buttons */
|
|
set_response_sensitive(Gtk::RESPONSE_CANCEL, !builder_.isDone);
|
|
set_response_sensitive(Gtk::RESPONSE_CLOSE, builder_.isDone);
|
|
set_response_sensitive(Gtk::RESPONSE_ACCEPT, builder_.isDone && builder_.result == TrMakemetaResult::OK);
|
|
|
|
return true;
|
|
}
|
|
|
|
MakeProgressDialog::~MakeProgressDialog()
|
|
{
|
|
progress_tag_.disconnect();
|
|
}
|
|
|
|
void MakeProgressDialog::addTorrent()
|
|
{
|
|
tr_ctor* ctor = tr_ctorNew(core_->get_session());
|
|
tr_ctorSetMetainfoFromFile(ctor, target_.c_str(), nullptr);
|
|
tr_ctorSetDownloadDir(ctor, TR_FORCE, Glib::path_get_dirname(builder_.top).c_str());
|
|
core_->add_ctor(ctor);
|
|
}
|
|
|
|
void MakeProgressDialog::onProgressDialogResponse(int response)
|
|
{
|
|
switch (response)
|
|
{
|
|
case Gtk::RESPONSE_CANCEL:
|
|
builder_.abortFlag = true;
|
|
hide();
|
|
break;
|
|
|
|
case Gtk::RESPONSE_ACCEPT:
|
|
addTorrent();
|
|
[[fallthrough]];
|
|
|
|
case Gtk::RESPONSE_CLOSE:
|
|
hide();
|
|
break;
|
|
|
|
default:
|
|
g_assert(0 && "unhandled response");
|
|
}
|
|
}
|
|
|
|
MakeProgressDialog::MakeProgressDialog(
|
|
Gtk::Window& parent,
|
|
tr_metainfo_builder& builder,
|
|
std::string const& target,
|
|
Glib::RefPtr<Session> const& core)
|
|
: Gtk::Dialog(_("New Torrent"), parent, true)
|
|
, builder_(builder)
|
|
, target_(target)
|
|
, core_(core)
|
|
{
|
|
add_button(_("_Cancel"), Gtk::RESPONSE_CANCEL);
|
|
add_button(_("_Close"), Gtk::RESPONSE_CLOSE);
|
|
add_button(_("_Add"), Gtk::RESPONSE_ACCEPT);
|
|
signal_response().connect(sigc::mem_fun(*this, &MakeProgressDialog::onProgressDialogResponse));
|
|
|
|
auto* fr = Gtk::make_managed<Gtk::Frame>();
|
|
fr->set_border_width(GUI_PAD_BIG);
|
|
fr->set_shadow_type(Gtk::SHADOW_NONE);
|
|
auto* v = Gtk::make_managed<Gtk::Box>(Gtk::ORIENTATION_VERTICAL, GUI_PAD);
|
|
fr->add(*v);
|
|
|
|
progress_label_ = Gtk::make_managed<Gtk::Label>(_("Creating torrent…"));
|
|
progress_label_->set_halign(Gtk::ALIGN_START);
|
|
progress_label_->set_valign(Gtk::ALIGN_CENTER);
|
|
progress_label_->set_justify(Gtk::JUSTIFY_LEFT);
|
|
v->pack_start(*progress_label_, false, false, 0);
|
|
|
|
progress_bar_ = Gtk::make_managed<Gtk::ProgressBar>();
|
|
v->pack_start(*progress_bar_, false, false, 0);
|
|
|
|
progress_tag_ = Glib::signal_timeout().connect_seconds(
|
|
sigc::mem_fun(*this, &MakeProgressDialog::onProgressDialogRefresh),
|
|
SECONDARY_WINDOW_REFRESH_INTERVAL_SECONDS);
|
|
onProgressDialogRefresh();
|
|
|
|
gtr_dialog_set_content(*this, *fr);
|
|
}
|
|
|
|
void MakeDialog::Impl::makeProgressDialog(std::string const& target)
|
|
{
|
|
progress_dialog_ = std::make_unique<MakeProgressDialog>(dialog_, *builder_, target, core_);
|
|
progress_dialog_->signal_hide().connect(
|
|
[this]()
|
|
{
|
|
progress_dialog_.reset();
|
|
|
|
if (builder_->result == TrMakemetaResult::OK)
|
|
{
|
|
dialog_.hide();
|
|
}
|
|
});
|
|
progress_dialog_->show();
|
|
}
|
|
|
|
void MakeDialog::Impl::onResponse(int response)
|
|
{
|
|
if (response == Gtk::RESPONSE_ACCEPT)
|
|
{
|
|
if (builder_ != nullptr)
|
|
{
|
|
auto const comment = comment_entry_->get_text();
|
|
bool const isPrivate = private_check_->get_active();
|
|
bool const useComment = comment_check_->get_active();
|
|
bool const useSource = source_check_->get_active();
|
|
auto const source = source_entry_->get_text();
|
|
|
|
/* destination file */
|
|
auto const dir = destination_chooser_->get_filename();
|
|
auto const base = Glib::path_get_basename(builder_->top);
|
|
auto const target = gtr_sprintf("%s/%s.torrent", dir, base);
|
|
|
|
/* build the array of trackers */
|
|
auto const tracker_text = announce_text_buffer_->get_text(false);
|
|
std::istringstream tracker_strings(tracker_text);
|
|
|
|
std::vector<tr_tracker_info> trackers;
|
|
std::list<std::string> announce_urls;
|
|
int tier = 0;
|
|
|
|
std::string str;
|
|
while (std::getline(tracker_strings, str))
|
|
{
|
|
if (str.empty())
|
|
{
|
|
++tier;
|
|
}
|
|
else
|
|
{
|
|
announce_urls.push_front(str);
|
|
trackers.push_back(tr_tracker_info{ tier, announce_urls.front().data() });
|
|
}
|
|
}
|
|
|
|
/* build the .torrent */
|
|
makeProgressDialog(target);
|
|
tr_makeMetaInfo(
|
|
builder_.get(),
|
|
target.c_str(),
|
|
trackers.data(),
|
|
trackers.size(),
|
|
nullptr,
|
|
0,
|
|
useComment ? comment.c_str() : nullptr,
|
|
isPrivate,
|
|
useSource ? source.c_str() : nullptr);
|
|
}
|
|
}
|
|
else if (response == Gtk::RESPONSE_CLOSE)
|
|
{
|
|
dialog_.hide();
|
|
}
|
|
}
|
|
|
|
/***
|
|
****
|
|
***/
|
|
|
|
namespace
|
|
{
|
|
|
|
void onSourceToggled(Gtk::ToggleButton* tb, Gtk::Widget* widget)
|
|
{
|
|
widget->set_sensitive(tb->get_active());
|
|
}
|
|
|
|
} // namespace
|
|
|
|
void MakeDialog::Impl::updatePiecesLabel()
|
|
{
|
|
char const* filename = builder_ != nullptr ? builder_->top : nullptr;
|
|
|
|
auto gstr = Glib::ustring{ "<i>" };
|
|
|
|
if (filename == nullptr)
|
|
{
|
|
gstr += _("No source selected");
|
|
}
|
|
else
|
|
{
|
|
gstr += fmt::format(
|
|
ngettext("{total_size} in {file_count:L} file", "{total_size} in {file_count:L} files", builder_->fileCount),
|
|
fmt::arg("total_size", tr_strlsize(builder_->totalSize)),
|
|
fmt::arg("file_count", builder_->fileCount));
|
|
gstr += ' ';
|
|
gstr += fmt::format(
|
|
ngettext(
|
|
"({piece_count} BitTorrent piece @ {piece_size})",
|
|
"({piece_count} BitTorrent pieces @ {piece_size})",
|
|
builder_->pieceCount),
|
|
fmt::arg("piece_count", builder_->pieceCount),
|
|
fmt::arg("piece_size", tr_formatter_mem_B(builder_->pieceSize)));
|
|
}
|
|
|
|
gstr += "</i>";
|
|
pieces_lb_->set_markup(gstr);
|
|
}
|
|
|
|
void MakeDialog::Impl::setFilename(std::string const& filename)
|
|
{
|
|
builder_.reset();
|
|
|
|
if (!filename.empty())
|
|
{
|
|
builder_ = { tr_metaInfoBuilderCreate(filename.c_str()), &tr_metaInfoBuilderFree };
|
|
}
|
|
|
|
updatePiecesLabel();
|
|
}
|
|
|
|
void MakeDialog::Impl::onChooserChosen(Gtk::FileChooserButton* chooser)
|
|
{
|
|
chooser->set_data(FileChosenKey, GINT_TO_POINTER(true));
|
|
setFilename(chooser->get_filename());
|
|
}
|
|
|
|
void MakeDialog::Impl::onSourceToggled2(Gtk::ToggleButton* tb, Gtk::FileChooserButton* chooser)
|
|
{
|
|
if (tb->get_active())
|
|
{
|
|
if (chooser->get_data(FileChosenKey) != nullptr)
|
|
{
|
|
onChooserChosen(chooser);
|
|
}
|
|
else
|
|
{
|
|
setFilename({});
|
|
}
|
|
}
|
|
}
|
|
|
|
void MakeDialog::Impl::on_drag_data_received(
|
|
Glib::RefPtr<Gdk::DragContext> const& drag_context,
|
|
int /*x*/,
|
|
int /*y*/,
|
|
Gtk::SelectionData const& selection_data,
|
|
guint /*info*/,
|
|
guint time_)
|
|
{
|
|
bool success = false;
|
|
|
|
if (auto const uris = selection_data.get_uris(); !uris.empty())
|
|
{
|
|
auto const& uri = uris.front();
|
|
auto const filename = Glib::filename_from_uri(uri);
|
|
|
|
if (Glib::file_test(filename, Glib::FILE_TEST_IS_DIR))
|
|
{
|
|
/* a directory was dragged onto the dialog... */
|
|
folder_radio_->set_active(true);
|
|
folder_chooser_->set_current_folder(filename);
|
|
success = true;
|
|
}
|
|
else if (Glib::file_test(filename, Glib::FILE_TEST_IS_REGULAR))
|
|
{
|
|
/* a file was dragged on to the dialog... */
|
|
file_radio_->set_active(true);
|
|
file_chooser_->set_filename(filename);
|
|
success = true;
|
|
}
|
|
}
|
|
|
|
drag_context->drag_finish(success, false, time_);
|
|
}
|
|
|
|
std::unique_ptr<MakeDialog> MakeDialog::create(Gtk::Window& parent, Glib::RefPtr<Session> const& core)
|
|
{
|
|
return std::unique_ptr<MakeDialog>(new MakeDialog(parent, core));
|
|
}
|
|
|
|
MakeDialog::MakeDialog(Gtk::Window& parent, Glib::RefPtr<Session> const& core)
|
|
: Gtk::Dialog(_("New Torrent"), parent)
|
|
, impl_(std::make_unique<Impl>(*this, core))
|
|
{
|
|
}
|
|
|
|
MakeDialog::~MakeDialog() = default;
|
|
|
|
MakeDialog::Impl::Impl(MakeDialog& dialog, Glib::RefPtr<Session> const& core)
|
|
: dialog_(dialog)
|
|
, core_(core)
|
|
{
|
|
guint row = 0;
|
|
|
|
dialog_.add_button(_("_Close"), Gtk::RESPONSE_CLOSE);
|
|
dialog_.add_button(_("_New"), Gtk::RESPONSE_ACCEPT);
|
|
dialog_.signal_response().connect(sigc::mem_fun(*this, &Impl::onResponse));
|
|
|
|
auto* t = Gtk::make_managed<HigWorkarea>();
|
|
|
|
t->add_section_title(row, _("Files"));
|
|
|
|
destination_chooser_ = Gtk::make_managed<Gtk::FileChooserButton>(Gtk::FILE_CHOOSER_ACTION_SELECT_FOLDER);
|
|
destination_chooser_->set_current_folder(Glib::get_user_special_dir(Glib::USER_DIRECTORY_DESKTOP));
|
|
t->add_row(row, _("Sa_ve to:"), *destination_chooser_);
|
|
|
|
Gtk::RadioButton::Group slist;
|
|
|
|
folder_radio_ = Gtk::make_managed<Gtk::RadioButton>(slist, _("Source F_older:"), true);
|
|
folder_radio_->set_active(false);
|
|
folder_chooser_ = Gtk::make_managed<Gtk::FileChooserButton>(Gtk::FILE_CHOOSER_ACTION_SELECT_FOLDER);
|
|
folder_radio_->signal_toggled().connect([this]() { onSourceToggled2(folder_radio_, folder_chooser_); });
|
|
folder_radio_->signal_toggled().connect([this]() { onSourceToggled(folder_radio_, folder_chooser_); });
|
|
folder_chooser_->signal_selection_changed().connect([this]() { onChooserChosen(folder_chooser_); });
|
|
folder_chooser_->set_sensitive(false);
|
|
t->add_row_w(row, *folder_radio_, *folder_chooser_);
|
|
|
|
file_radio_ = Gtk::make_managed<Gtk::RadioButton>(slist, _("Source _File:"), true);
|
|
file_radio_->set_active(true);
|
|
file_chooser_ = Gtk::make_managed<Gtk::FileChooserButton>(Gtk::FILE_CHOOSER_ACTION_OPEN);
|
|
file_radio_->signal_toggled().connect([this]() { onSourceToggled2(file_radio_, file_chooser_); });
|
|
file_radio_->signal_toggled().connect([this]() { onSourceToggled(file_radio_, file_chooser_); });
|
|
file_chooser_->signal_selection_changed().connect([this]() { onChooserChosen(file_chooser_); });
|
|
t->add_row_w(row, *file_radio_, *file_chooser_);
|
|
|
|
pieces_lb_ = Gtk::make_managed<Gtk::Label>();
|
|
pieces_lb_->set_markup(fmt::format(FMT_STRING("<i>{:s}</i>"), _("No source selected")));
|
|
t->add_row(row, {}, *pieces_lb_);
|
|
|
|
t->add_section_divider(row);
|
|
t->add_section_title(row, _("Properties"));
|
|
|
|
auto* v = Gtk::make_managed<Gtk::Box>(Gtk::ORIENTATION_VERTICAL, GUI_PAD_SMALL);
|
|
announce_text_buffer_ = Gtk::TextBuffer::create();
|
|
auto* w = Gtk::make_managed<Gtk::TextView>(announce_text_buffer_);
|
|
w->set_size_request(-1, 80);
|
|
auto* sw = Gtk::make_managed<Gtk::ScrolledWindow>();
|
|
sw->set_policy(Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
|
|
sw->add(*w);
|
|
auto* fr = Gtk::make_managed<Gtk::Frame>();
|
|
fr->set_shadow_type(Gtk::SHADOW_IN);
|
|
fr->add(*sw);
|
|
v->pack_start(*fr, true, true, 0);
|
|
auto* l = Gtk::make_managed<Gtk::Label>();
|
|
l->set_markup(_(
|
|
"To add a backup URL, add it on the next line after a primary URL.\nTo add a new primary URL, add it after a blank line."));
|
|
l->set_justify(Gtk::JUSTIFY_LEFT);
|
|
l->set_halign(Gtk::ALIGN_START);
|
|
l->set_valign(Gtk::ALIGN_CENTER);
|
|
v->pack_start(*l, false, false, 0);
|
|
t->add_tall_row(row, _("_Trackers:"), *v);
|
|
|
|
comment_check_ = Gtk::make_managed<Gtk::CheckButton>(_("Co_mment:"), true);
|
|
comment_check_->set_active(false);
|
|
comment_entry_ = Gtk::make_managed<Gtk::Entry>();
|
|
comment_entry_->set_sensitive(false);
|
|
comment_check_->signal_toggled().connect([this]() { onSourceToggled(comment_check_, comment_entry_); });
|
|
t->add_row_w(row, *comment_check_, *comment_entry_);
|
|
|
|
source_check_ = Gtk::make_managed<Gtk::CheckButton>(_("_Source:"), true);
|
|
source_check_->set_active(false);
|
|
source_entry_ = Gtk::make_managed<Gtk::Entry>();
|
|
source_entry_->set_sensitive(false);
|
|
source_check_->signal_toggled().connect([this]() { onSourceToggled(source_check_, source_entry_); });
|
|
t->add_row_w(row, *source_check_, *source_entry_);
|
|
|
|
private_check_ = t->add_wide_checkbutton(row, _("_Private torrent"), false);
|
|
|
|
gtr_dialog_set_content(dialog_, *t);
|
|
|
|
dialog_.drag_dest_set(Gtk::DEST_DEFAULT_ALL, Gdk::ACTION_COPY);
|
|
dialog_.drag_dest_add_uri_targets();
|
|
dialog_.signal_drag_data_received().connect(sigc::mem_fun(*this, &Impl::on_drag_data_received));
|
|
}
|