ardour/gtk2_ardour/plugin_manager_ui.cc

517 lines
17 KiB
C++
Raw Normal View History

2021-05-26 19:02:11 -04:00
/*
* Copyright (C) 2021 Robin Gareus <robin@gareus.org>
*
* 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.
*/
#ifdef WAF_BUILD
#include "gtk2ardour-config.h"
#endif
2021-06-18 15:21:46 -04:00
#include <cassert>
#include <gtkmm/frame.h>
#include "pbd/unwind.h"
2021-05-26 19:02:11 -04:00
#include "ardour/types_convert.h"
#include "gtkmm2ext/gui_thread.h"
2021-06-18 15:21:46 -04:00
#include "ardour_message.h"
2021-05-26 19:02:11 -04:00
#include "plugin_manager_ui.h"
2021-06-18 15:21:46 -04:00
#include "plugin_scan_dialog.h"
2021-05-26 19:02:11 -04:00
#include "pbd/i18n.h"
using namespace ARDOUR;
PluginManagerUI::PluginManagerUI ()
: ArdourWindow (_("Plugin Manager"))
2021-06-18 15:21:46 -04:00
, _btn_rescan_all (_("Re-scan All"))
, _btn_rescan_err (_("Re-scan Faulty"))
2021-05-26 19:02:11 -04:00
, _btn_rescan_sel (_("Re-scan Selected"))
, _btn_clear (_("Clear Stale Scan Log"))
2021-06-18 15:21:46 -04:00
, _in_row_change (false)
2021-05-26 19:02:11 -04:00
{
plugin_model = Gtk::ListStore::create (plugin_columns);
2021-06-18 15:21:46 -04:00
Gtk::CellRendererToggle* cell_blacklist = Gtk::manage (new Gtk::CellRendererToggle ());
Gtk::TreeViewColumn* column_blacklist = Gtk::manage (new Gtk::TreeViewColumn (_("BL"), *cell_blacklist));
cell_blacklist->property_activatable() = true;
cell_blacklist->property_radio() = false;
column_blacklist->add_attribute (cell_blacklist->property_active (), plugin_columns.blacklisted);
column_blacklist->add_attribute (cell_blacklist->property_activatable (), plugin_columns.can_blacklist);
Gtk::CellRendererToggle* cell_fav = Gtk::manage (new Gtk::CellRendererToggle ());
Gtk::TreeViewColumn* column_fav = Gtk::manage (new Gtk::TreeViewColumn (_("Fav"), *cell_fav));
cell_fav->property_activatable() = true;
cell_fav->property_radio() = true;
column_fav->add_attribute (cell_fav->property_active (), plugin_columns.favorite);
column_fav->add_attribute (cell_fav->property_activatable (), plugin_columns.can_fav_hide);
Gtk::CellRendererToggle* cell_hidden = Gtk::manage (new Gtk::CellRendererToggle ());
Gtk::TreeViewColumn* column_hidden = Gtk::manage (new Gtk::TreeViewColumn (_("Hide"), *cell_hidden));
cell_hidden->property_activatable() = true;
cell_hidden->property_radio() = true;
column_hidden->add_attribute (cell_hidden->property_active (), plugin_columns.hidden);
column_hidden->add_attribute (cell_hidden->property_activatable (), plugin_columns.can_fav_hide);
2021-05-26 19:02:11 -04:00
plugin_display.append_column (_("Status"), plugin_columns.status);
2021-06-18 15:21:46 -04:00
plugin_display.append_column (*column_blacklist);
plugin_display.append_column (*column_fav);
plugin_display.append_column (*column_hidden);
2021-05-26 19:02:11 -04:00
plugin_display.append_column (_("Name"), plugin_columns.name);
plugin_display.append_column (_("Creator"), plugin_columns.creator);
plugin_display.append_column (_("Type"), plugin_columns.type);
2021-06-18 15:21:46 -04:00
plugin_display.append_column (_("File/ID"), plugin_columns.path);
2021-05-26 19:02:11 -04:00
plugin_display.set_model (plugin_model);
plugin_display.set_headers_visible (true);
plugin_display.set_headers_clickable (true);
plugin_display.set_reorderable (false);
plugin_display.set_rules_hint (true);
2021-06-18 15:21:46 -04:00
plugin_display.set_enable_search(true);
plugin_display.get_column (4)->set_resizable (true);
plugin_display.get_column (5)->set_resizable (true);
2021-05-26 19:02:11 -04:00
2021-06-18 15:21:46 -04:00
#if 0
if (UIConfiguration::instance().get_use_tooltips()) {
recent_session_display.set_tooltip_column(1); // plugin_columns.tip
}
#endif
for (int i = 0; i < 8; ++i) {
2021-05-26 19:02:11 -04:00
Gtk::TreeView::Column* column = plugin_display.get_column(i);
if (column) {
column->set_sort_column(i);
}
}
plugin_model->set_sort_column (plugin_columns.name.index(), Gtk::SORT_ASCENDING);
plugin_display.set_name("PluginSelectorDisplay");
2021-06-18 15:21:46 -04:00
plugin_display.get_selection()->set_mode (Gtk::SELECTION_SINGLE);
2021-05-26 19:02:11 -04:00
plugin_display.get_selection()->signal_changed().connect (sigc::mem_fun(*this, &PluginManagerUI::selection_changed));
2021-06-18 15:21:46 -04:00
//plugin_display.signal_row_activated().connect_notify (sigc::mem_fun(*this, &PluginManagerUI::row_activated));
2021-05-26 19:02:11 -04:00
_scroller.add (plugin_display);
_scroller.set_policy (Gtk::POLICY_AUTOMATIC, Gtk::POLICY_AUTOMATIC);
_log.set_editable (false);
_log.set_wrap_mode (Gtk::WRAP_WORD);
_log_scroller.set_shadow_type(Gtk::SHADOW_NONE);
_log_scroller.set_border_width(0);
_log_scroller.add (_log);
_log_scroller.set_policy (Gtk::POLICY_NEVER, Gtk::POLICY_AUTOMATIC);
_pane.add (_scroller);
_pane.add (_log_scroller);
_pane.set_divider (0, .85);
2021-06-18 15:21:46 -04:00
Gtk::Label* lbl = Gtk::manage (new Gtk::Label ("")); // spacer
Gtk::Frame* f_info = Gtk::manage (new Gtk::Frame (_("Plugin Count")));
Gtk::Frame* f_actions = Gtk::manage (new Gtk::Frame (_("Scan Actions")));
Gtk::VBox* b_actions = Gtk::manage (new Gtk::VBox ());
f_info->add (_tbl_nfo);
f_actions->add (*b_actions);
_tbl_nfo.set_border_width (4);
b_actions->pack_start (_btn_clear);
b_actions->pack_start (_btn_rescan_sel);
b_actions->pack_start (_btn_rescan_err);
b_actions->pack_start (_btn_rescan_all);
b_actions->set_spacing (4);
b_actions->set_border_width (4);
2021-05-26 19:02:11 -04:00
/* top level packing */
2021-06-18 15:21:46 -04:00
_top.attach (*lbl, 0, 1, 0, 1, Gtk::SHRINK, Gtk::EXPAND | Gtk::FILL, 4, 0);
_top.attach (*f_info, 0, 1, 1, 2, Gtk::FILL | Gtk::SHRINK, Gtk::SHRINK, 4, 4);
_top.attach (*f_actions, 0, 1, 2, 3, Gtk::FILL | Gtk::SHRINK, Gtk::SHRINK, 4, 4);
_top.attach (_pane, 1, 2, 0, 3, Gtk::EXPAND | Gtk::FILL, Gtk::EXPAND | Gtk::FILL, 4, 0);
2021-05-26 19:02:11 -04:00
add (_top);
2021-06-18 15:21:46 -04:00
_top.show_all ();
2021-05-26 19:02:11 -04:00
_log.set_size_request (400, -1);
set_size_request (-1, 600);
/* connect to signals */
2021-06-18 15:21:46 -04:00
PluginManager::instance ().PluginListChanged.connect (_manager_connections, invalidator (*this), boost::bind (&PluginManagerUI::refill, this), gui_context());
PluginManager::instance ().PluginStatusChanged.connect (_manager_connections, invalidator (*this), boost::bind (&PluginManagerUI::plugin_status_changed, this, _1, _2, _3), gui_context());
2021-05-26 19:02:11 -04:00
_btn_rescan_all.signal_clicked.connect (sigc::mem_fun (*this, &PluginManagerUI::rescan_all));
2021-06-18 15:21:46 -04:00
_btn_rescan_err.signal_clicked.connect (sigc::mem_fun (*this, &PluginManagerUI::rescan_faulty));
2021-05-26 19:02:11 -04:00
_btn_rescan_sel.signal_clicked.connect (sigc::mem_fun (*this, &PluginManagerUI::rescan_selected));
_btn_clear.signal_clicked.connect (sigc::mem_fun (*this, &PluginManagerUI::clear_log));
2021-06-18 15:21:46 -04:00
cell_fav->signal_toggled().connect (sigc::mem_fun (*this, &PluginManagerUI::favorite_changed));
cell_hidden->signal_toggled().connect (sigc::mem_fun (*this, &PluginManagerUI::hidden_changed));
cell_blacklist->signal_toggled().connect (sigc::mem_fun (*this, &PluginManagerUI::blacklist_changed));
/* populate */
refill ();
2021-05-26 19:02:11 -04:00
}
PluginManagerUI::~PluginManagerUI ()
{
}
void
PluginManagerUI::on_show ()
{
ArdourWindow::on_show ();
}
static std::string
2021-06-18 15:21:46 -04:00
status_text (PluginScanLogEntry const& psle, PluginManager::PluginStatusType status)
2021-05-26 19:02:11 -04:00
{
if (!psle.recent ()) {
return "Stale";
}
PluginScanLogEntry::PluginScanResult sr = psle.result ();
2021-06-18 15:21:46 -04:00
if (sr == PluginScanLogEntry::OK || sr == PluginScanLogEntry::Blacklisted) {
if (status == PluginManager::Concealed) {
return _("Concealed");
} else {
return _("OK");
}
}
if ((int)sr & PluginScanLogEntry::TimeOut) {
return _("New");
2021-05-26 19:02:11 -04:00
}
if ((int)sr & PluginScanLogEntry::New) {
2021-06-18 15:21:46 -04:00
return _("New");
2021-05-26 19:02:11 -04:00
}
if ((int)sr & PluginScanLogEntry::Updated) {
2021-06-18 15:21:46 -04:00
return _("Updated");
2021-05-26 19:02:11 -04:00
}
if ((int)sr & PluginScanLogEntry::Error) {
2021-06-18 15:21:46 -04:00
return _("Error");
2021-05-26 19:02:11 -04:00
}
if ((int)sr & PluginScanLogEntry::Incompatible) {
2021-06-18 15:21:46 -04:00
return _("Incompatible");
2021-05-26 19:02:11 -04:00
}
2021-06-18 15:21:46 -04:00
assert (0);
return "?";
2021-05-26 19:02:11 -04:00
}
static bool
is_blacklisted (PluginScanLogEntry const& psle)
{
return (int) psle.result () & PluginScanLogEntry::Blacklisted;
}
2021-06-18 15:21:46 -04:00
static bool
can_blacklist (PluginScanLogEntry const& psle)
{
if (psle.type () == LV2 || psle.type () == LADSPA) {
return false;
}
return ((int) psle.result () & ~PluginScanLogEntry::Blacklisted) == PluginScanLogEntry::OK;
}
2021-05-26 19:02:11 -04:00
static std::string
plugin_type (const PluginType t)
{
/* see also PluginManager::to_generic_vst */
switch (t) {
case Windows_VST:
case LXVST:
case MacVST:
return "VST2.x";
default:
return enum_2_string (t);
}
}
void
PluginManagerUI::refill ()
{
2021-06-18 15:21:46 -04:00
/* save selection and sort-column, clear model to speed-up refill */
Gtk::TreeIter iter = plugin_display.get_selection ()->get_selected ();
boost::shared_ptr<PluginScanLogEntry> sel;
if (iter) {
sel = (*iter)[plugin_columns.psle];
}
2021-05-26 19:02:11 -04:00
plugin_display.set_model (Glib::RefPtr<Gtk::TreeStore>(0));
int sort_col;
Gtk::SortType sort_type;
bool sorted = plugin_model->get_sort_column_id (sort_col, sort_type);
plugin_model->set_sort_column (-2, Gtk::SORT_ASCENDING);
2021-06-18 15:21:46 -04:00
plugin_model->clear ();
2021-05-26 19:02:11 -04:00
2021-06-18 15:21:46 -04:00
bool rescan_err = false;
2021-05-26 19:02:11 -04:00
bool have_stale = false;
2021-06-18 15:21:46 -04:00
std::map<PluginType, unsigned int> plugin_count;
std::vector<boost::shared_ptr<PluginScanLogEntry> > psl;
PluginManager& manager (PluginManager::instance ());
manager.scan_log (psl);
2021-05-26 19:02:11 -04:00
for (std::vector<boost::shared_ptr <PluginScanLogEntry> >::const_iterator i = psl.begin(); i != psl.end(); ++i) {
PluginInfoList const& plugs = (*i)->nfo ();
if (!(*i)->recent ()) {
have_stale = true;
} else if ((*i)->result () == PluginScanLogEntry::Blacklisted) {
2021-06-18 15:21:46 -04:00
/* OK, but manually blacklisted */
2021-05-26 19:02:11 -04:00
} else if ((*i)->result () != PluginScanLogEntry::OK) {
2021-06-18 15:21:46 -04:00
if ((*i)->type () != LV2) {
rescan_err = true;
}
2021-05-26 19:02:11 -04:00
}
if (plugs.size () == 0) {
Gtk::TreeModel::Row newrow = *(plugin_model->append());
2021-06-18 15:21:46 -04:00
newrow[plugin_columns.path] = Glib::path_get_basename ((*i)->path ());
2021-05-26 19:02:11 -04:00
newrow[plugin_columns.type] = plugin_type ((*i)->type ());
newrow[plugin_columns.name] = "-";
newrow[plugin_columns.creator] = "-";
2021-06-18 15:21:46 -04:00
newrow[plugin_columns.status] = status_text (**i, PluginManager::Normal); // XXX
2021-05-26 19:02:11 -04:00
newrow[plugin_columns.blacklisted] = is_blacklisted (**i);
newrow[plugin_columns.psle] = *i;
2021-06-18 15:21:46 -04:00
newrow[plugin_columns.plugin] = ARDOUR::PluginInfoPtr ();
newrow[plugin_columns.favorite] = false;
newrow[plugin_columns.hidden] = false;
newrow[plugin_columns.can_blacklist] = can_blacklist (**i);
newrow[plugin_columns.can_fav_hide] = false;
++plugin_count[(*i)->type ()];
2021-05-26 19:02:11 -04:00
} else {
for (PluginInfoList::const_iterator j = plugs.begin(); j != plugs.end(); ++j) {
2021-06-18 15:21:46 -04:00
PluginManager::PluginStatusType status = manager.get_status (*j);
2021-05-26 19:02:11 -04:00
Gtk::TreeModel::Row newrow = *(plugin_model->append());
2021-06-18 15:21:46 -04:00
newrow[plugin_columns.favorite] = status == PluginManager::Favorite;
newrow[plugin_columns.hidden] = status == PluginManager::Hidden;
newrow[plugin_columns.path] = Glib::path_get_basename ((*i)->path ());
2021-05-26 19:02:11 -04:00
newrow[plugin_columns.type] = plugin_type ((*i)->type ());
newrow[plugin_columns.name] = (*j)->name;
newrow[plugin_columns.creator] = (*j)->creator;
2021-06-18 15:21:46 -04:00
newrow[plugin_columns.status] = status_text (**i, status);
2021-05-26 19:02:11 -04:00
newrow[plugin_columns.blacklisted] = is_blacklisted (**i);
newrow[plugin_columns.psle] = *i;
2021-06-18 15:21:46 -04:00
newrow[plugin_columns.plugin] = *j;
newrow[plugin_columns.can_blacklist] = can_blacklist (**i);
newrow[plugin_columns.can_fav_hide] = status != PluginManager::Concealed;
++plugin_count[(*i)->type ()];
2021-05-26 19:02:11 -04:00
}
}
}
2021-06-18 15:21:46 -04:00
2021-05-26 19:02:11 -04:00
plugin_display.set_model (plugin_model);
if (sorted) {
plugin_model->set_sort_column (sort_col, sort_type);
}
2021-06-18 15:21:46 -04:00
if (sel) {
Gtk::TreeModel::Children rows = plugin_model->children ();
for (Gtk::TreeModel::Children::iterator i = rows.begin (); i != rows.end (); ++i) {
boost::shared_ptr<PluginScanLogEntry> const& srow ((*i)[plugin_columns.psle]);
if (*sel == *srow) {
plugin_display.get_selection ()->select (*i);
break;
}
}
}
plugin_display.set_search_column (4); // Name
int row = 0;
std::list<Gtk::Widget*> children = _tbl_nfo.get_children();
for (std::list<Gtk::Widget*>::iterator child = children.begin(); child != children.end(); ++child) {
_tbl_nfo.remove (**child);
delete *child;
}
for (std::map<PluginType, unsigned int>::const_iterator i = plugin_count.begin (); i != plugin_count.end (); ++i, ++row) {
Gtk::Label* lbl_type = new Gtk::Label (plugin_type (i->first), Gtk::ALIGN_LEFT, Gtk::ALIGN_CENTER);
Gtk::Label* lbl_count = new Gtk::Label (string_compose ("%1", i->second), Gtk::ALIGN_RIGHT, Gtk::ALIGN_CENTER);
_tbl_nfo.attach (*lbl_type , 0, 1, row, row + 1, Gtk::EXPAND | Gtk::FILL, Gtk::SHRINK, 2, 2);
_tbl_nfo.attach (*lbl_count, 1, 2, row, row + 1, Gtk::SHRINK | Gtk::FILL, Gtk::SHRINK, 2, 2);
}
_tbl_nfo.show_all ();
2021-05-26 19:02:11 -04:00
_btn_clear.set_sensitive (have_stale);
2021-06-18 15:21:46 -04:00
_btn_rescan_err.set_sensitive (rescan_err);
2021-05-26 19:02:11 -04:00
}
void
PluginManagerUI::selection_changed ()
{
if (plugin_display.get_selection()->count_selected_rows() != 1) {
_log.get_buffer()->set_text ("-");
return;
}
2021-06-18 15:21:46 -04:00
2021-05-26 19:02:11 -04:00
Gtk::TreeIter iter = plugin_display.get_selection ()->get_selected ();
boost::shared_ptr<PluginScanLogEntry> const& psle ((*iter)[plugin_columns.psle]);
2021-06-18 15:21:46 -04:00
2021-05-26 19:02:11 -04:00
_log.get_buffer()->set_text (psle->log ());
PluginScanLogEntry::PluginScanResult sr = psle->result ();
2021-06-18 15:21:46 -04:00
if (sr == PluginScanLogEntry::OK || psle->type () == LV2) {
2021-05-26 19:02:11 -04:00
_btn_rescan_sel.set_sensitive (false);
} else {
_btn_rescan_sel.set_sensitive (true);
}
}
2021-06-18 15:21:46 -04:00
void
PluginManagerUI::blacklist_changed (std::string const& path)
{
Gtk::TreeIter iter;
if ((iter = plugin_model->get_iter (path))) {
boost::shared_ptr<PluginScanLogEntry> const& psle ((*iter)[plugin_columns.psle]);
if ((*iter)[plugin_columns.blacklisted]) {
PluginScanDialog psd (false, true, this);
PluginManager::instance ().rescan_plugin (psle->type (), psle->path ());
} else {
PluginManager::instance ().blacklist (psle->type (), psle->path ());
}
}
}
2021-05-26 19:02:11 -04:00
void
PluginManagerUI::rescan_all ()
{
2021-06-18 15:21:46 -04:00
ArdourMessageDialog msg (_("Are you sure you want to rescan all plugins?"), false, Gtk::MESSAGE_QUESTION, Gtk::BUTTONS_YES_NO, true);
msg.set_title (_("Rescan Plugins"));
msg.set_secondary_text(_("This starts a fresh scan, dropping all cached plugin data and backlists. Depending on the number if plugins installed this can take a long time."));
if (msg.run() != Gtk::RESPONSE_YES) {
return;
}
msg.hide();
ARDOUR::PluginManager& manager (PluginManager::instance ());
if (true) {
manager.clear_au_blacklist ();
manager.clear_vst_blacklist ();
manager.clear_vst3_blacklist ();
}
manager.clear_au_cache ();
manager.clear_vst_cache ();
manager.clear_vst3_cache ();
PluginScanDialog psd (false, true, this);
psd.start ();
}
void
PluginManagerUI::rescan_faulty ()
{
PluginScanDialog psd (false, true, this);
PluginManager::instance ().rescan_faulty ();
2021-05-26 19:02:11 -04:00
}
void
PluginManagerUI::rescan_selected ()
{
2021-06-18 15:21:46 -04:00
if (plugin_display.get_selection()->count_selected_rows() != 1) {
return;
}
Gtk::TreeIter iter = plugin_display.get_selection ()->get_selected ();
boost::shared_ptr<PluginScanLogEntry> const& psle ((*iter)[plugin_columns.psle]);
PluginScanDialog psd (false, true, this);
PluginManager::instance ().rescan_plugin (psle->type (), psle->path ());
2021-05-26 19:02:11 -04:00
}
void
PluginManagerUI::clear_log ()
{
2021-06-18 15:21:46 -04:00
PluginManager::instance ().clear_stale_log ();
refill ();
}
void
PluginManagerUI::plugin_status_changed (ARDOUR::PluginType t, std::string uid, ARDOUR::PluginManager::PluginStatusType stat)
{
Gtk::TreeModel::Children rows = plugin_model->children ();
for (Gtk::TreeModel::Children::iterator i = rows.begin (); i != rows.end (); ++i) {
PluginInfoPtr pp = (*i)[plugin_columns.plugin];
if (!pp || pp->type != t || pp->unique_id != uid) {
continue;
}
(*i)[plugin_columns.favorite] = (stat == PluginManager::Favorite) ? true : false;
(*i)[plugin_columns.hidden] = (stat == PluginManager::Hidden) ? true : false;
break;
}
}
void
PluginManagerUI::favorite_changed (const std::string& path)
{
if (_in_row_change) {
return;
}
PBD::Unwinder<bool> uw (_in_row_change, true);
Gtk::TreeIter iter;
if ((iter = plugin_model->get_iter (path))) {
bool favorite = !(*iter)[plugin_columns.favorite];
PluginManager::PluginStatusType status = (favorite ? PluginManager::Favorite : PluginManager::Normal);
PluginInfoPtr pi = (*iter)[plugin_columns.plugin];
ARDOUR::PluginManager& manager (PluginManager::instance ());
manager.set_status (pi->type, pi->unique_id, status);
manager.save_statuses ();
}
}
void
PluginManagerUI::hidden_changed (const std::string& path)
{
if (_in_row_change) {
return;
}
PBD::Unwinder<bool> uw (_in_row_change, true);
Gtk::TreeIter iter;
if ((iter = plugin_model->get_iter (path))) {
bool hidden = !(*iter)[plugin_columns.hidden];
PluginManager::PluginStatusType status = (hidden ? PluginManager::Hidden : PluginManager::Normal);
PluginInfoPtr pi = (*iter)[plugin_columns.plugin];
ARDOUR::PluginManager& manager (PluginManager::instance ());
manager.set_status (pi->type, pi->unique_id, status);
manager.save_statuses ();
}
2021-05-26 19:02:11 -04:00
}