annotate transform/TransformFactory.cpp @ 484:45ded09e4710

* minor text change
author Chris Cannam
date Wed, 19 Nov 2008 21:36:33 +0000
parents f54381e01141
children 9eb7ef610d7f
rev   line source
Chris@330 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@330 2
Chris@330 3 /*
Chris@330 4 Sonic Visualiser
Chris@330 5 An audio file viewer and annotation editor.
Chris@330 6 Centre for Digital Music, Queen Mary, University of London.
Chris@330 7 This file copyright 2006 Chris Cannam and QMUL.
Chris@330 8
Chris@330 9 This program is free software; you can redistribute it and/or
Chris@330 10 modify it under the terms of the GNU General Public License as
Chris@330 11 published by the Free Software Foundation; either version 2 of the
Chris@330 12 License, or (at your option) any later version. See the file
Chris@330 13 COPYING included with this distribution for more information.
Chris@330 14 */
Chris@330 15
Chris@330 16 #include "TransformFactory.h"
Chris@330 17
Chris@330 18 #include "plugin/FeatureExtractionPluginFactory.h"
Chris@330 19 #include "plugin/RealTimePluginFactory.h"
Chris@332 20 #include "plugin/RealTimePluginInstance.h"
Chris@330 21 #include "plugin/PluginXml.h"
Chris@330 22
Chris@475 23 #include <vamp-hostsdk/Plugin.h>
Chris@475 24 #include <vamp-hostsdk/PluginHostAdapter.h>
Chris@475 25 #include <vamp-hostsdk/PluginWrapper.h>
Chris@330 26
Chris@457 27 #include "rdf/PluginRDFIndexer.h"
Chris@457 28 #include "rdf/PluginRDFDescription.h"
Chris@457 29
Chris@446 30 #include "base/XmlExportable.h"
Chris@446 31
Chris@330 32 #include <iostream>
Chris@330 33 #include <set>
Chris@330 34
Chris@330 35 #include <QRegExp>
Chris@350 36 #include <QTextStream>
Chris@330 37
Chris@460 38 #include "base/Thread.h"
Chris@460 39
Chris@443 40 using std::cerr;
Chris@443 41 using std::endl;
Chris@443 42
Chris@330 43 TransformFactory *
Chris@330 44 TransformFactory::m_instance = new TransformFactory;
Chris@330 45
Chris@330 46 TransformFactory *
Chris@330 47 TransformFactory::getInstance()
Chris@330 48 {
Chris@330 49 return m_instance;
Chris@330 50 }
Chris@330 51
Chris@457 52 TransformFactory::TransformFactory() :
Chris@457 53 m_transformsPopulated(false),
Chris@477 54 m_uninstalledTransformsPopulated(false),
Chris@477 55 m_thread(0)
Chris@457 56 {
Chris@457 57 }
Chris@457 58
Chris@330 59 TransformFactory::~TransformFactory()
Chris@330 60 {
Chris@330 61 }
Chris@330 62
Chris@477 63 void
Chris@477 64 TransformFactory::startPopulationThread()
Chris@477 65 {
Chris@482 66 m_uninstalledTransformsMutex.lock();
Chris@477 67
Chris@482 68 if (m_thread) {
Chris@482 69 m_uninstalledTransformsMutex.unlock();
Chris@482 70 return;
Chris@482 71 }
Chris@482 72 m_thread = new UninstalledTransformsPopulateThread(this);
Chris@477 73
Chris@482 74 m_uninstalledTransformsMutex.unlock();
Chris@482 75
Chris@477 76 m_thread->start();
Chris@477 77 }
Chris@477 78
Chris@481 79 void
Chris@481 80 TransformFactory::UninstalledTransformsPopulateThread::run()
Chris@481 81 {
Chris@481 82 m_factory->m_populatingSlowly = true;
Chris@481 83 sleep(1);
Chris@481 84 m_factory->populateUninstalledTransforms();
Chris@481 85 }
Chris@481 86
Chris@330 87 TransformList
Chris@350 88 TransformFactory::getAllTransformDescriptions()
Chris@330 89 {
Chris@460 90 populateTransforms();
Chris@330 91
Chris@330 92 std::set<TransformDescription> dset;
Chris@330 93 for (TransformDescriptionMap::const_iterator i = m_transforms.begin();
Chris@330 94 i != m_transforms.end(); ++i) {
Chris@443 95 // cerr << "inserting transform into set: id = " << i->second.identifier.toStdString() << endl;
Chris@330 96 dset.insert(i->second);
Chris@330 97 }
Chris@330 98
Chris@330 99 TransformList list;
Chris@330 100 for (std::set<TransformDescription>::const_iterator i = dset.begin();
Chris@330 101 i != dset.end(); ++i) {
Chris@443 102 // cerr << "inserting transform into list: id = " << i->identifier.toStdString() << endl;
Chris@330 103 list.push_back(*i);
Chris@330 104 }
Chris@330 105
Chris@330 106 return list;
Chris@330 107 }
Chris@330 108
Chris@350 109 TransformDescription
Chris@350 110 TransformFactory::getTransformDescription(TransformId id)
Chris@350 111 {
Chris@460 112 populateTransforms();
Chris@350 113
Chris@350 114 if (m_transforms.find(id) == m_transforms.end()) {
Chris@350 115 return TransformDescription();
Chris@350 116 }
Chris@350 117
Chris@350 118 return m_transforms[id];
Chris@350 119 }
Chris@350 120
Chris@457 121 TransformList
Chris@457 122 TransformFactory::getUninstalledTransformDescriptions()
Chris@457 123 {
Chris@479 124 m_populatingSlowly = false;
Chris@460 125 populateUninstalledTransforms();
Chris@457 126
Chris@457 127 std::set<TransformDescription> dset;
Chris@457 128 for (TransformDescriptionMap::const_iterator i = m_uninstalledTransforms.begin();
Chris@457 129 i != m_uninstalledTransforms.end(); ++i) {
Chris@457 130 // cerr << "inserting transform into set: id = " << i->second.identifier.toStdString() << endl;
Chris@457 131 dset.insert(i->second);
Chris@457 132 }
Chris@457 133
Chris@457 134 TransformList list;
Chris@457 135 for (std::set<TransformDescription>::const_iterator i = dset.begin();
Chris@457 136 i != dset.end(); ++i) {
Chris@460 137 // cerr << "inserting transform into uninstalled list: id = " << i->identifier.toStdString() << endl;
Chris@457 138 list.push_back(*i);
Chris@457 139 }
Chris@457 140
Chris@457 141 return list;
Chris@457 142 }
Chris@457 143
Chris@457 144 TransformDescription
Chris@457 145 TransformFactory::getUninstalledTransformDescription(TransformId id)
Chris@457 146 {
Chris@479 147 m_populatingSlowly = false;
Chris@460 148 populateUninstalledTransforms();
Chris@457 149
Chris@457 150 if (m_uninstalledTransforms.find(id) == m_uninstalledTransforms.end()) {
Chris@457 151 return TransformDescription();
Chris@457 152 }
Chris@457 153
Chris@457 154 return m_uninstalledTransforms[id];
Chris@457 155 }
Chris@457 156
Chris@457 157 TransformFactory::TransformInstallStatus
Chris@457 158 TransformFactory::getTransformInstallStatus(TransformId id)
Chris@457 159 {
Chris@460 160 populateTransforms();
Chris@457 161
Chris@457 162 if (m_transforms.find(id) != m_transforms.end()) {
Chris@457 163 return TransformInstalled;
Chris@457 164 }
Chris@473 165
Chris@473 166 if (!m_uninstalledTransformsMutex.tryLock()) {
Chris@473 167 // uninstalled transforms are being populated; this may take some time,
Chris@473 168 // and they aren't critical
Chris@473 169 return TransformUnknown;
Chris@473 170 }
Chris@473 171
Chris@473 172 if (!m_uninstalledTransformsPopulated) {
Chris@473 173 m_uninstalledTransformsMutex.unlock();
Chris@479 174 m_populatingSlowly = false;
Chris@473 175 populateUninstalledTransforms();
Chris@473 176 m_uninstalledTransformsMutex.lock();
Chris@473 177 }
Chris@473 178
Chris@457 179 if (m_uninstalledTransforms.find(id) != m_uninstalledTransforms.end()) {
Chris@482 180 m_uninstalledTransformsMutex.unlock();
Chris@457 181 return TransformNotInstalled;
Chris@457 182 }
Chris@473 183
Chris@473 184 m_uninstalledTransformsMutex.unlock();
Chris@457 185 return TransformUnknown;
Chris@457 186 }
Chris@457 187
Chris@457 188
Chris@330 189 std::vector<QString>
Chris@330 190 TransformFactory::getAllTransformTypes()
Chris@330 191 {
Chris@460 192 populateTransforms();
Chris@330 193
Chris@330 194 std::set<QString> types;
Chris@330 195 for (TransformDescriptionMap::const_iterator i = m_transforms.begin();
Chris@330 196 i != m_transforms.end(); ++i) {
Chris@330 197 types.insert(i->second.type);
Chris@330 198 }
Chris@330 199
Chris@330 200 std::vector<QString> rv;
Chris@330 201 for (std::set<QString>::iterator i = types.begin(); i != types.end(); ++i) {
Chris@330 202 rv.push_back(*i);
Chris@330 203 }
Chris@330 204
Chris@330 205 return rv;
Chris@330 206 }
Chris@330 207
Chris@330 208 std::vector<QString>
Chris@330 209 TransformFactory::getTransformCategories(QString transformType)
Chris@330 210 {
Chris@460 211 populateTransforms();
Chris@330 212
Chris@330 213 std::set<QString> categories;
Chris@330 214 for (TransformDescriptionMap::const_iterator i = m_transforms.begin();
Chris@330 215 i != m_transforms.end(); ++i) {
Chris@330 216 if (i->second.type == transformType) {
Chris@330 217 categories.insert(i->second.category);
Chris@330 218 }
Chris@330 219 }
Chris@330 220
Chris@330 221 bool haveEmpty = false;
Chris@330 222
Chris@330 223 std::vector<QString> rv;
Chris@330 224 for (std::set<QString>::iterator i = categories.begin();
Chris@330 225 i != categories.end(); ++i) {
Chris@330 226 if (*i != "") rv.push_back(*i);
Chris@330 227 else haveEmpty = true;
Chris@330 228 }
Chris@330 229
Chris@330 230 if (haveEmpty) rv.push_back(""); // make sure empty category sorts last
Chris@330 231
Chris@330 232 return rv;
Chris@330 233 }
Chris@330 234
Chris@330 235 std::vector<QString>
Chris@330 236 TransformFactory::getTransformMakers(QString transformType)
Chris@330 237 {
Chris@460 238 populateTransforms();
Chris@330 239
Chris@330 240 std::set<QString> makers;
Chris@330 241 for (TransformDescriptionMap::const_iterator i = m_transforms.begin();
Chris@330 242 i != m_transforms.end(); ++i) {
Chris@330 243 if (i->second.type == transformType) {
Chris@330 244 makers.insert(i->second.maker);
Chris@330 245 }
Chris@330 246 }
Chris@330 247
Chris@330 248 bool haveEmpty = false;
Chris@330 249
Chris@330 250 std::vector<QString> rv;
Chris@330 251 for (std::set<QString>::iterator i = makers.begin();
Chris@330 252 i != makers.end(); ++i) {
Chris@330 253 if (*i != "") rv.push_back(*i);
Chris@330 254 else haveEmpty = true;
Chris@330 255 }
Chris@330 256
Chris@330 257 if (haveEmpty) rv.push_back(""); // make sure empty category sorts last
Chris@330 258
Chris@330 259 return rv;
Chris@330 260 }
Chris@330 261
Chris@330 262 void
Chris@330 263 TransformFactory::populateTransforms()
Chris@330 264 {
Chris@460 265 MutexLocker locker(&m_transformsMutex,
Chris@460 266 "TransformFactory::populateTransforms");
Chris@460 267 if (m_transformsPopulated) {
Chris@460 268 return;
Chris@460 269 }
Chris@460 270
Chris@330 271 TransformDescriptionMap transforms;
Chris@330 272
Chris@330 273 populateFeatureExtractionPlugins(transforms);
Chris@330 274 populateRealTimePlugins(transforms);
Chris@330 275
Chris@330 276 // disambiguate plugins with similar names
Chris@330 277
Chris@330 278 std::map<QString, int> names;
Chris@330 279 std::map<QString, QString> pluginSources;
Chris@330 280 std::map<QString, QString> pluginMakers;
Chris@330 281
Chris@330 282 for (TransformDescriptionMap::iterator i = transforms.begin();
Chris@330 283 i != transforms.end(); ++i) {
Chris@330 284
Chris@330 285 TransformDescription desc = i->second;
Chris@330 286
Chris@330 287 QString td = desc.name;
Chris@330 288 QString tn = td.section(": ", 0, 0);
Chris@330 289 QString pn = desc.identifier.section(":", 1, 1);
Chris@330 290
Chris@330 291 if (pluginSources.find(tn) != pluginSources.end()) {
Chris@330 292 if (pluginSources[tn] != pn && pluginMakers[tn] != desc.maker) {
Chris@330 293 ++names[tn];
Chris@330 294 }
Chris@330 295 } else {
Chris@330 296 ++names[tn];
Chris@330 297 pluginSources[tn] = pn;
Chris@330 298 pluginMakers[tn] = desc.maker;
Chris@330 299 }
Chris@330 300 }
Chris@330 301
Chris@330 302 std::map<QString, int> counts;
Chris@330 303 m_transforms.clear();
Chris@330 304
Chris@330 305 for (TransformDescriptionMap::iterator i = transforms.begin();
Chris@330 306 i != transforms.end(); ++i) {
Chris@330 307
Chris@330 308 TransformDescription desc = i->second;
Chris@330 309 QString identifier = desc.identifier;
Chris@330 310 QString maker = desc.maker;
Chris@330 311
Chris@330 312 QString td = desc.name;
Chris@330 313 QString tn = td.section(": ", 0, 0);
Chris@330 314 QString to = td.section(": ", 1);
Chris@330 315
Chris@330 316 if (names[tn] > 1) {
Chris@330 317 maker.replace(QRegExp(tr(" [\\(<].*$")), "");
Chris@330 318 tn = QString("%1 [%2]").arg(tn).arg(maker);
Chris@330 319 }
Chris@330 320
Chris@330 321 if (to != "") {
Chris@330 322 desc.name = QString("%1: %2").arg(tn).arg(to);
Chris@330 323 } else {
Chris@330 324 desc.name = tn;
Chris@330 325 }
Chris@330 326
Chris@330 327 m_transforms[identifier] = desc;
Chris@330 328 }
Chris@457 329
Chris@457 330 m_transformsPopulated = true;
Chris@330 331 }
Chris@330 332
Chris@330 333 void
Chris@330 334 TransformFactory::populateFeatureExtractionPlugins(TransformDescriptionMap &transforms)
Chris@330 335 {
Chris@330 336 std::vector<QString> plugs =
Chris@330 337 FeatureExtractionPluginFactory::getAllPluginIdentifiers();
Chris@330 338
Chris@330 339 for (size_t i = 0; i < plugs.size(); ++i) {
Chris@330 340
Chris@330 341 QString pluginId = plugs[i];
Chris@330 342
Chris@330 343 FeatureExtractionPluginFactory *factory =
Chris@330 344 FeatureExtractionPluginFactory::instanceFor(pluginId);
Chris@330 345
Chris@330 346 if (!factory) {
Chris@443 347 cerr << "WARNING: TransformFactory::populateTransforms: No feature extraction plugin factory for instance " << pluginId.toLocal8Bit().data() << endl;
Chris@330 348 continue;
Chris@330 349 }
Chris@330 350
Chris@330 351 Vamp::Plugin *plugin =
Chris@350 352 factory->instantiatePlugin(pluginId, 44100);
Chris@330 353
Chris@330 354 if (!plugin) {
Chris@443 355 cerr << "WARNING: TransformFactory::populateTransforms: Failed to instantiate plugin " << pluginId.toLocal8Bit().data() << endl;
Chris@330 356 continue;
Chris@330 357 }
Chris@330 358
Chris@330 359 QString pluginName = plugin->getName().c_str();
Chris@330 360 QString category = factory->getPluginCategory(pluginId);
Chris@330 361
Chris@330 362 Vamp::Plugin::OutputList outputs =
Chris@330 363 plugin->getOutputDescriptors();
Chris@330 364
Chris@330 365 for (size_t j = 0; j < outputs.size(); ++j) {
Chris@330 366
Chris@330 367 QString transformId = QString("%1:%2")
Chris@330 368 .arg(pluginId).arg(outputs[j].identifier.c_str());
Chris@330 369
Chris@330 370 QString userName;
Chris@330 371 QString friendlyName;
Chris@330 372 QString units = outputs[j].unit.c_str();
Chris@330 373 QString description = plugin->getDescription().c_str();
Chris@330 374 QString maker = plugin->getMaker().c_str();
Chris@330 375 if (maker == "") maker = tr("<unknown maker>");
Chris@330 376
Chris@443 377 QString longDescription = description;
Chris@443 378
Chris@443 379 if (longDescription == "") {
Chris@330 380 if (outputs.size() == 1) {
Chris@443 381 longDescription = tr("Extract features using \"%1\" plugin (from %2)")
Chris@330 382 .arg(pluginName).arg(maker);
Chris@330 383 } else {
Chris@443 384 longDescription = tr("Extract features using \"%1\" output of \"%2\" plugin (from %3)")
Chris@330 385 .arg(outputs[j].name.c_str()).arg(pluginName).arg(maker);
Chris@330 386 }
Chris@330 387 } else {
Chris@330 388 if (outputs.size() == 1) {
Chris@443 389 longDescription = tr("%1 using \"%2\" plugin (from %3)")
Chris@443 390 .arg(longDescription).arg(pluginName).arg(maker);
Chris@330 391 } else {
Chris@443 392 longDescription = tr("%1 using \"%2\" output of \"%3\" plugin (from %4)")
Chris@443 393 .arg(longDescription).arg(outputs[j].name.c_str()).arg(pluginName).arg(maker);
Chris@330 394 }
Chris@330 395 }
Chris@330 396
Chris@330 397 if (outputs.size() == 1) {
Chris@330 398 userName = pluginName;
Chris@330 399 friendlyName = pluginName;
Chris@330 400 } else {
Chris@330 401 userName = QString("%1: %2")
Chris@330 402 .arg(pluginName)
Chris@330 403 .arg(outputs[j].name.c_str());
Chris@330 404 friendlyName = outputs[j].name.c_str();
Chris@330 405 }
Chris@330 406
Chris@330 407 bool configurable = (!plugin->getPrograms().empty() ||
Chris@330 408 !plugin->getParameterDescriptors().empty());
Chris@330 409
Chris@443 410 // cerr << "Feature extraction plugin transform: " << transformId.toStdString() << " friendly name: " << friendlyName.toStdString() << endl;
Chris@330 411
Chris@330 412 transforms[transformId] =
Chris@330 413 TransformDescription(tr("Analysis"),
Chris@332 414 category,
Chris@332 415 transformId,
Chris@332 416 userName,
Chris@332 417 friendlyName,
Chris@332 418 description,
Chris@443 419 longDescription,
Chris@332 420 maker,
Chris@332 421 units,
Chris@332 422 configurable);
Chris@330 423 }
Chris@330 424
Chris@330 425 delete plugin;
Chris@330 426 }
Chris@330 427 }
Chris@330 428
Chris@330 429 void
Chris@330 430 TransformFactory::populateRealTimePlugins(TransformDescriptionMap &transforms)
Chris@330 431 {
Chris@330 432 std::vector<QString> plugs =
Chris@330 433 RealTimePluginFactory::getAllPluginIdentifiers();
Chris@330 434
Chris@330 435 static QRegExp unitRE("[\\[\\(]([A-Za-z0-9/]+)[\\)\\]]$");
Chris@330 436
Chris@330 437 for (size_t i = 0; i < plugs.size(); ++i) {
Chris@330 438
Chris@330 439 QString pluginId = plugs[i];
Chris@330 440
Chris@330 441 RealTimePluginFactory *factory =
Chris@330 442 RealTimePluginFactory::instanceFor(pluginId);
Chris@330 443
Chris@330 444 if (!factory) {
Chris@443 445 cerr << "WARNING: TransformFactory::populateTransforms: No real time plugin factory for instance " << pluginId.toLocal8Bit().data() << endl;
Chris@330 446 continue;
Chris@330 447 }
Chris@330 448
Chris@330 449 const RealTimePluginDescriptor *descriptor =
Chris@330 450 factory->getPluginDescriptor(pluginId);
Chris@330 451
Chris@330 452 if (!descriptor) {
Chris@443 453 cerr << "WARNING: TransformFactory::populateTransforms: Failed to query plugin " << pluginId.toLocal8Bit().data() << endl;
Chris@330 454 continue;
Chris@330 455 }
Chris@330 456
Chris@330 457 //!!! if (descriptor->controlOutputPortCount == 0 ||
Chris@330 458 // descriptor->audioInputPortCount == 0) continue;
Chris@330 459
Chris@443 460 // std::cout << "TransformFactory::populateRealTimePlugins: plugin " << pluginId.toStdString() << " has " << descriptor->controlOutputPortCount << " control output ports, " << descriptor->audioOutputPortCount << " audio outputs, " << descriptor->audioInputPortCount << " audio inputs" << endl;
Chris@330 461
Chris@330 462 QString pluginName = descriptor->name.c_str();
Chris@330 463 QString category = factory->getPluginCategory(pluginId);
Chris@330 464 bool configurable = (descriptor->parameterCount > 0);
Chris@330 465 QString maker = descriptor->maker.c_str();
Chris@330 466 if (maker == "") maker = tr("<unknown maker>");
Chris@330 467
Chris@330 468 if (descriptor->audioInputPortCount > 0) {
Chris@330 469
Chris@330 470 for (size_t j = 0; j < descriptor->controlOutputPortCount; ++j) {
Chris@330 471
Chris@330 472 QString transformId = QString("%1:%2").arg(pluginId).arg(j);
Chris@330 473 QString userName;
Chris@330 474 QString units;
Chris@330 475 QString portName;
Chris@330 476
Chris@330 477 if (j < descriptor->controlOutputPortNames.size() &&
Chris@330 478 descriptor->controlOutputPortNames[j] != "") {
Chris@330 479
Chris@330 480 portName = descriptor->controlOutputPortNames[j].c_str();
Chris@330 481
Chris@330 482 userName = tr("%1: %2")
Chris@330 483 .arg(pluginName)
Chris@330 484 .arg(portName);
Chris@330 485
Chris@330 486 if (unitRE.indexIn(portName) >= 0) {
Chris@330 487 units = unitRE.cap(1);
Chris@330 488 }
Chris@330 489
Chris@330 490 } else if (descriptor->controlOutputPortCount > 1) {
Chris@330 491
Chris@330 492 userName = tr("%1: Output %2")
Chris@330 493 .arg(pluginName)
Chris@330 494 .arg(j + 1);
Chris@330 495
Chris@330 496 } else {
Chris@330 497
Chris@330 498 userName = pluginName;
Chris@330 499 }
Chris@330 500
Chris@330 501 QString description;
Chris@330 502
Chris@330 503 if (portName != "") {
Chris@330 504 description = tr("Extract \"%1\" data output from \"%2\" effect plugin (from %3)")
Chris@330 505 .arg(portName)
Chris@330 506 .arg(pluginName)
Chris@330 507 .arg(maker);
Chris@330 508 } else {
Chris@330 509 description = tr("Extract data output %1 from \"%2\" effect plugin (from %3)")
Chris@330 510 .arg(j + 1)
Chris@330 511 .arg(pluginName)
Chris@330 512 .arg(maker);
Chris@330 513 }
Chris@330 514
Chris@330 515 transforms[transformId] =
Chris@330 516 TransformDescription(tr("Effects Data"),
Chris@332 517 category,
Chris@332 518 transformId,
Chris@332 519 userName,
Chris@332 520 userName,
Chris@443 521 "",
Chris@332 522 description,
Chris@332 523 maker,
Chris@332 524 units,
Chris@332 525 configurable);
Chris@330 526 }
Chris@330 527 }
Chris@330 528
Chris@330 529 if (!descriptor->isSynth || descriptor->audioInputPortCount > 0) {
Chris@330 530
Chris@330 531 if (descriptor->audioOutputPortCount > 0) {
Chris@330 532
Chris@330 533 QString transformId = QString("%1:A").arg(pluginId);
Chris@330 534 QString type = tr("Effects");
Chris@330 535
Chris@330 536 QString description = tr("Transform audio signal with \"%1\" effect plugin (from %2)")
Chris@330 537 .arg(pluginName)
Chris@330 538 .arg(maker);
Chris@330 539
Chris@330 540 if (descriptor->audioInputPortCount == 0) {
Chris@330 541 type = tr("Generators");
Chris@330 542 QString description = tr("Generate audio signal using \"%1\" plugin (from %2)")
Chris@330 543 .arg(pluginName)
Chris@330 544 .arg(maker);
Chris@330 545 }
Chris@330 546
Chris@330 547 transforms[transformId] =
Chris@330 548 TransformDescription(type,
Chris@332 549 category,
Chris@332 550 transformId,
Chris@332 551 pluginName,
Chris@332 552 pluginName,
Chris@443 553 "",
Chris@332 554 description,
Chris@332 555 maker,
Chris@332 556 "",
Chris@332 557 configurable);
Chris@330 558 }
Chris@330 559 }
Chris@330 560 }
Chris@330 561 }
Chris@330 562
Chris@457 563 void
Chris@457 564 TransformFactory::populateUninstalledTransforms()
Chris@457 565 {
Chris@460 566 populateTransforms();
Chris@460 567
Chris@460 568 MutexLocker locker(&m_uninstalledTransformsMutex,
Chris@460 569 "TransformFactory::populateUninstalledTransforms");
Chris@460 570 if (m_uninstalledTransformsPopulated) return;
Chris@460 571
Chris@461 572 PluginRDFIndexer::getInstance()->indexConfiguredURLs();
Chris@457 573
Chris@457 574 //!!! This will be amazingly slow
Chris@457 575
Chris@457 576 QStringList ids = PluginRDFIndexer::getInstance()->getIndexedPluginIds();
Chris@457 577
Chris@457 578 for (QStringList::const_iterator i = ids.begin(); i != ids.end(); ++i) {
Chris@457 579
Chris@457 580 PluginRDFDescription desc(*i);
Chris@457 581
Chris@457 582 QString name = desc.getPluginName();
Chris@457 583 // if (name == "") {
Chris@457 584 // std::cerr << "TransformFactory::populateUninstalledTransforms: "
Chris@457 585 // << "No name available for plugin " << i->toStdString()
Chris@457 586 // << ", skipping" << std::endl;
Chris@457 587 // continue;
Chris@457 588 // }
Chris@457 589
Chris@457 590 QString description = desc.getPluginDescription();
Chris@457 591 QString maker = desc.getPluginMaker();
Chris@462 592 QString infoUrl = desc.getPluginInfoURL();
Chris@457 593
Chris@457 594 QStringList oids = desc.getOutputIds();
Chris@457 595
Chris@457 596 for (QStringList::const_iterator j = oids.begin(); j != oids.end(); ++j) {
Chris@457 597
Chris@457 598 TransformId tid = Transform::getIdentifierForPluginOutput(*i, *j);
Chris@457 599
Chris@457 600 if (m_transforms.find(tid) != m_transforms.end()) {
Chris@457 601 std::cerr << "TransformFactory::populateUninstalledTransforms: "
Chris@468 602 << tid.toStdString() << " is installed; adding info url if appropriate, skipping rest" << std::endl;
Chris@468 603 if (infoUrl != "") {
Chris@468 604 if (m_transforms[tid].infoUrl == "") {
Chris@468 605 m_transforms[tid].infoUrl = infoUrl;
Chris@468 606 }
Chris@468 607 }
Chris@457 608 continue;
Chris@457 609 }
Chris@457 610
Chris@457 611 std::cerr << "TransformFactory::populateUninstalledTransforms: "
Chris@457 612 << "adding " << tid.toStdString() << std::endl;
Chris@457 613
Chris@457 614 QString oname = desc.getOutputName(*j);
Chris@457 615 if (oname == "") oname = *j;
Chris@457 616
Chris@457 617 TransformDescription td;
Chris@457 618 td.type = tr("Analysis"); //!!! should be enum or something
Chris@457 619 td.category = "";
Chris@457 620 td.identifier = tid;
Chris@457 621
Chris@457 622 if (oids.size() == 1) {
Chris@457 623 td.name = name;
Chris@457 624 } else if (name != "") {
Chris@457 625 td.name = tr("%1: %2").arg(name).arg(oname);
Chris@457 626 }
Chris@457 627
Chris@462 628 QString longDescription = description;
Chris@462 629 //!!! basically duplicated from above
Chris@462 630 if (longDescription == "") {
Chris@462 631 if (oids.size() == 1) {
Chris@462 632 longDescription = tr("Extract features using \"%1\" plugin (from %2)")
Chris@462 633 .arg(name).arg(maker);
Chris@462 634 } else {
Chris@462 635 longDescription = tr("Extract features using \"%1\" output of \"%2\" plugin (from %3)")
Chris@462 636 .arg(oname).arg(name).arg(maker);
Chris@462 637 }
Chris@462 638 } else {
Chris@462 639 if (oids.size() == 1) {
Chris@462 640 longDescription = tr("%1 using \"%2\" plugin (from %3)")
Chris@462 641 .arg(longDescription).arg(name).arg(maker);
Chris@462 642 } else {
Chris@462 643 longDescription = tr("%1 using \"%2\" output of \"%3\" plugin (from %4)")
Chris@462 644 .arg(longDescription).arg(oname).arg(name).arg(maker);
Chris@462 645 }
Chris@462 646 }
Chris@462 647
Chris@457 648 td.friendlyName = name; //!!!???
Chris@457 649 td.description = description;
Chris@462 650 td.longDescription = longDescription;
Chris@457 651 td.maker = maker;
Chris@462 652 td.infoUrl = infoUrl;
Chris@457 653 td.units = "";
Chris@457 654 td.configurable = false;
Chris@457 655
Chris@457 656 m_uninstalledTransforms[tid] = td;
Chris@457 657 }
Chris@457 658 }
Chris@457 659
Chris@457 660 m_uninstalledTransformsPopulated = true;
Chris@460 661
Chris@460 662 std::cerr << "populateUninstalledTransforms exiting" << std::endl;
Chris@457 663 }
Chris@350 664
Chris@350 665 Transform
Chris@350 666 TransformFactory::getDefaultTransformFor(TransformId id, size_t rate)
Chris@350 667 {
Chris@350 668 Transform t;
Chris@350 669 t.setIdentifier(id);
Chris@350 670 if (rate != 0) t.setSampleRate(rate);
Chris@350 671
Chris@351 672 Vamp::PluginBase *plugin = instantiateDefaultPluginFor(id, rate);
Chris@350 673
Chris@350 674 if (plugin) {
Chris@366 675 t.setPluginVersion(QString("%1").arg(plugin->getPluginVersion()));
Chris@350 676 setParametersFromPlugin(t, plugin);
Chris@350 677 makeContextConsistentWithPlugin(t, plugin);
Chris@350 678 delete plugin;
Chris@350 679 }
Chris@350 680
Chris@350 681 return t;
Chris@350 682 }
Chris@350 683
Chris@350 684 Vamp::PluginBase *
Chris@351 685 TransformFactory::instantiatePluginFor(const Transform &transform)
Chris@351 686 {
Chris@351 687 Vamp::PluginBase *plugin = instantiateDefaultPluginFor
Chris@351 688 (transform.getIdentifier(), transform.getSampleRate());
Chris@351 689 if (plugin) {
Chris@351 690 setPluginParameters(transform, plugin);
Chris@351 691 }
Chris@351 692 return plugin;
Chris@351 693 }
Chris@351 694
Chris@351 695 Vamp::PluginBase *
Chris@351 696 TransformFactory::instantiateDefaultPluginFor(TransformId identifier, size_t rate)
Chris@350 697 {
Chris@350 698 Transform t;
Chris@350 699 t.setIdentifier(identifier);
Chris@350 700 if (rate == 0) rate = 44100;
Chris@350 701 QString pluginId = t.getPluginIdentifier();
Chris@350 702
Chris@350 703 Vamp::PluginBase *plugin = 0;
Chris@350 704
Chris@350 705 if (t.getType() == Transform::FeatureExtraction) {
Chris@350 706
Chris@350 707 FeatureExtractionPluginFactory *factory =
Chris@350 708 FeatureExtractionPluginFactory::instanceFor(pluginId);
Chris@350 709
Chris@439 710 if (factory) {
Chris@439 711 plugin = factory->instantiatePlugin(pluginId, rate);
Chris@439 712 }
Chris@350 713
Chris@350 714 } else {
Chris@350 715
Chris@350 716 RealTimePluginFactory *factory =
Chris@350 717 RealTimePluginFactory::instanceFor(pluginId);
Chris@439 718
Chris@439 719 if (factory) {
Chris@439 720 plugin = factory->instantiatePlugin(pluginId, 0, 0, rate, 1024, 1);
Chris@439 721 }
Chris@350 722 }
Chris@350 723
Chris@350 724 return plugin;
Chris@350 725 }
Chris@350 726
Chris@350 727 Vamp::Plugin *
Chris@350 728 TransformFactory::downcastVampPlugin(Vamp::PluginBase *plugin)
Chris@350 729 {
Chris@350 730 Vamp::Plugin *vp = dynamic_cast<Vamp::Plugin *>(plugin);
Chris@350 731 if (!vp) {
Chris@443 732 // cerr << "makeConsistentWithPlugin: not a Vamp::Plugin" << endl;
Chris@350 733 vp = dynamic_cast<Vamp::PluginHostAdapter *>(plugin); //!!! why?
Chris@350 734 }
Chris@350 735 if (!vp) {
Chris@443 736 // cerr << "makeConsistentWithPlugin: not a Vamp::PluginHostAdapter" << endl;
Chris@350 737 vp = dynamic_cast<Vamp::HostExt::PluginWrapper *>(plugin); //!!! no, I mean really why?
Chris@350 738 }
Chris@350 739 if (!vp) {
Chris@443 740 // cerr << "makeConsistentWithPlugin: not a Vamp::HostExt::PluginWrapper" << endl;
Chris@350 741 }
Chris@350 742 return vp;
Chris@350 743 }
Chris@350 744
Chris@330 745 bool
Chris@330 746 TransformFactory::haveTransform(TransformId identifier)
Chris@330 747 {
Chris@460 748 populateTransforms();
Chris@330 749 return (m_transforms.find(identifier) != m_transforms.end());
Chris@330 750 }
Chris@330 751
Chris@330 752 QString
Chris@330 753 TransformFactory::getTransformName(TransformId identifier)
Chris@330 754 {
Chris@330 755 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@330 756 return m_transforms[identifier].name;
Chris@330 757 } else return "";
Chris@330 758 }
Chris@330 759
Chris@330 760 QString
Chris@330 761 TransformFactory::getTransformFriendlyName(TransformId identifier)
Chris@330 762 {
Chris@330 763 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@330 764 return m_transforms[identifier].friendlyName;
Chris@330 765 } else return "";
Chris@330 766 }
Chris@330 767
Chris@330 768 QString
Chris@330 769 TransformFactory::getTransformUnits(TransformId identifier)
Chris@330 770 {
Chris@330 771 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@330 772 return m_transforms[identifier].units;
Chris@330 773 } else return "";
Chris@330 774 }
Chris@330 775
Chris@472 776 QString
Chris@472 777 TransformFactory::getTransformInfoUrl(TransformId identifier)
Chris@472 778 {
Chris@472 779 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@472 780 return m_transforms[identifier].infoUrl;
Chris@472 781 } else return "";
Chris@472 782 }
Chris@472 783
Chris@350 784 Vamp::Plugin::InputDomain
Chris@350 785 TransformFactory::getTransformInputDomain(TransformId identifier)
Chris@350 786 {
Chris@350 787 Transform transform;
Chris@350 788 transform.setIdentifier(identifier);
Chris@350 789
Chris@350 790 if (transform.getType() != Transform::FeatureExtraction) {
Chris@350 791 return Vamp::Plugin::TimeDomain;
Chris@350 792 }
Chris@350 793
Chris@350 794 Vamp::Plugin *plugin =
Chris@351 795 downcastVampPlugin(instantiateDefaultPluginFor(identifier, 0));
Chris@350 796
Chris@350 797 if (plugin) {
Chris@350 798 Vamp::Plugin::InputDomain d = plugin->getInputDomain();
Chris@350 799 delete plugin;
Chris@350 800 return d;
Chris@350 801 }
Chris@350 802
Chris@350 803 return Vamp::Plugin::TimeDomain;
Chris@350 804 }
Chris@350 805
Chris@330 806 bool
Chris@330 807 TransformFactory::isTransformConfigurable(TransformId identifier)
Chris@330 808 {
Chris@330 809 if (m_transforms.find(identifier) != m_transforms.end()) {
Chris@330 810 return m_transforms[identifier].configurable;
Chris@330 811 } else return false;
Chris@330 812 }
Chris@330 813
Chris@330 814 bool
Chris@330 815 TransformFactory::getTransformChannelRange(TransformId identifier,
Chris@330 816 int &min, int &max)
Chris@330 817 {
Chris@330 818 QString id = identifier.section(':', 0, 2);
Chris@330 819
Chris@330 820 if (FeatureExtractionPluginFactory::instanceFor(id)) {
Chris@330 821
Chris@330 822 Vamp::Plugin *plugin =
Chris@330 823 FeatureExtractionPluginFactory::instanceFor(id)->
Chris@350 824 instantiatePlugin(id, 44100);
Chris@330 825 if (!plugin) return false;
Chris@330 826
Chris@330 827 min = plugin->getMinChannelCount();
Chris@330 828 max = plugin->getMaxChannelCount();
Chris@330 829 delete plugin;
Chris@330 830
Chris@330 831 return true;
Chris@330 832
Chris@330 833 } else if (RealTimePluginFactory::instanceFor(id)) {
Chris@330 834
Chris@350 835 // don't need to instantiate
Chris@350 836
Chris@330 837 const RealTimePluginDescriptor *descriptor =
Chris@330 838 RealTimePluginFactory::instanceFor(id)->
Chris@330 839 getPluginDescriptor(id);
Chris@330 840 if (!descriptor) return false;
Chris@330 841
Chris@330 842 min = descriptor->audioInputPortCount;
Chris@330 843 max = descriptor->audioInputPortCount;
Chris@330 844
Chris@330 845 return true;
Chris@330 846 }
Chris@330 847
Chris@330 848 return false;
Chris@330 849 }
Chris@332 850
Chris@332 851 void
Chris@332 852 TransformFactory::setParametersFromPlugin(Transform &transform,
Chris@332 853 Vamp::PluginBase *plugin)
Chris@332 854 {
Chris@332 855 Transform::ParameterMap pmap;
Chris@332 856
Chris@350 857 //!!! record plugin & API version
Chris@350 858
Chris@350 859 //!!! check that this is the right plugin!
Chris@350 860
Chris@332 861 Vamp::PluginBase::ParameterList parameters =
Chris@332 862 plugin->getParameterDescriptors();
Chris@332 863
Chris@332 864 for (Vamp::PluginBase::ParameterList::const_iterator i = parameters.begin();
Chris@332 865 i != parameters.end(); ++i) {
Chris@332 866 pmap[i->identifier.c_str()] = plugin->getParameter(i->identifier);
Chris@332 867 }
Chris@332 868
Chris@332 869 transform.setParameters(pmap);
Chris@332 870
Chris@332 871 if (plugin->getPrograms().empty()) {
Chris@332 872 transform.setProgram("");
Chris@332 873 } else {
Chris@332 874 transform.setProgram(plugin->getCurrentProgram().c_str());
Chris@332 875 }
Chris@332 876
Chris@332 877 RealTimePluginInstance *rtpi =
Chris@332 878 dynamic_cast<RealTimePluginInstance *>(plugin);
Chris@332 879
Chris@332 880 Transform::ConfigurationMap cmap;
Chris@332 881
Chris@332 882 if (rtpi) {
Chris@332 883
Chris@332 884 RealTimePluginInstance::ConfigurationPairMap configurePairs =
Chris@332 885 rtpi->getConfigurePairs();
Chris@332 886
Chris@332 887 for (RealTimePluginInstance::ConfigurationPairMap::const_iterator i
Chris@332 888 = configurePairs.begin(); i != configurePairs.end(); ++i) {
Chris@332 889 cmap[i->first.c_str()] = i->second.c_str();
Chris@332 890 }
Chris@332 891 }
Chris@332 892
Chris@332 893 transform.setConfiguration(cmap);
Chris@332 894 }
Chris@332 895
Chris@332 896 void
Chris@350 897 TransformFactory::setPluginParameters(const Transform &transform,
Chris@350 898 Vamp::PluginBase *plugin)
Chris@350 899 {
Chris@350 900 //!!! check plugin & API version (see e.g. PluginXml::setParameters)
Chris@350 901
Chris@350 902 //!!! check that this is the right plugin!
Chris@350 903
Chris@350 904 RealTimePluginInstance *rtpi =
Chris@350 905 dynamic_cast<RealTimePluginInstance *>(plugin);
Chris@350 906
Chris@350 907 if (rtpi) {
Chris@350 908 const Transform::ConfigurationMap &cmap = transform.getConfiguration();
Chris@350 909 for (Transform::ConfigurationMap::const_iterator i = cmap.begin();
Chris@350 910 i != cmap.end(); ++i) {
Chris@350 911 rtpi->configure(i->first.toStdString(), i->second.toStdString());
Chris@350 912 }
Chris@350 913 }
Chris@350 914
Chris@350 915 if (transform.getProgram() != "") {
Chris@350 916 plugin->selectProgram(transform.getProgram().toStdString());
Chris@350 917 }
Chris@350 918
Chris@350 919 const Transform::ParameterMap &pmap = transform.getParameters();
Chris@350 920
Chris@350 921 Vamp::PluginBase::ParameterList parameters =
Chris@350 922 plugin->getParameterDescriptors();
Chris@350 923
Chris@350 924 for (Vamp::PluginBase::ParameterList::const_iterator i = parameters.begin();
Chris@350 925 i != parameters.end(); ++i) {
Chris@350 926 QString key = i->identifier.c_str();
Chris@350 927 Transform::ParameterMap::const_iterator pmi = pmap.find(key);
Chris@350 928 if (pmi != pmap.end()) {
Chris@350 929 plugin->setParameter(i->identifier, pmi->second);
Chris@350 930 }
Chris@350 931 }
Chris@350 932 }
Chris@350 933
Chris@350 934 void
Chris@332 935 TransformFactory::makeContextConsistentWithPlugin(Transform &transform,
Chris@332 936 Vamp::PluginBase *plugin)
Chris@332 937 {
Chris@350 938 const Vamp::Plugin *vp = downcastVampPlugin(plugin);
Chris@332 939
Chris@332 940 if (!vp) {
Chris@332 941 // time domain input for real-time effects plugin
Chris@332 942 if (!transform.getBlockSize()) {
Chris@332 943 if (!transform.getStepSize()) transform.setStepSize(1024);
Chris@332 944 transform.setBlockSize(transform.getStepSize());
Chris@332 945 } else {
Chris@332 946 transform.setStepSize(transform.getBlockSize());
Chris@332 947 }
Chris@332 948 } else {
Chris@332 949 Vamp::Plugin::InputDomain domain = vp->getInputDomain();
Chris@332 950 if (!transform.getStepSize()) {
Chris@332 951 transform.setStepSize(vp->getPreferredStepSize());
Chris@332 952 }
Chris@332 953 if (!transform.getBlockSize()) {
Chris@332 954 transform.setBlockSize(vp->getPreferredBlockSize());
Chris@332 955 }
Chris@332 956 if (!transform.getBlockSize()) {
Chris@332 957 transform.setBlockSize(1024);
Chris@332 958 }
Chris@332 959 if (!transform.getStepSize()) {
Chris@332 960 if (domain == Vamp::Plugin::FrequencyDomain) {
Chris@443 961 // cerr << "frequency domain, step = " << blockSize/2 << endl;
Chris@332 962 transform.setStepSize(transform.getBlockSize()/2);
Chris@332 963 } else {
Chris@443 964 // cerr << "time domain, step = " << blockSize/2 << endl;
Chris@332 965 transform.setStepSize(transform.getBlockSize());
Chris@332 966 }
Chris@332 967 }
Chris@332 968 }
Chris@332 969 }
Chris@332 970
Chris@350 971 QString
Chris@350 972 TransformFactory::getPluginConfigurationXml(const Transform &t)
Chris@332 973 {
Chris@350 974 QString xml;
Chris@350 975
Chris@351 976 Vamp::PluginBase *plugin = instantiateDefaultPluginFor
Chris@351 977 (t.getIdentifier(), 0);
Chris@350 978 if (!plugin) {
Chris@443 979 cerr << "TransformFactory::getPluginConfigurationXml: "
Chris@350 980 << "Unable to instantiate plugin for transform \""
Chris@443 981 << t.getIdentifier().toStdString() << "\"" << endl;
Chris@350 982 return xml;
Chris@332 983 }
Chris@332 984
Chris@351 985 setPluginParameters(t, plugin);
Chris@351 986
Chris@350 987 QTextStream out(&xml);
Chris@350 988 PluginXml(plugin).toXml(out);
Chris@350 989 delete plugin;
Chris@332 990
Chris@350 991 return xml;
Chris@350 992 }
Chris@332 993
Chris@350 994 void
Chris@350 995 TransformFactory::setParametersFromPluginConfigurationXml(Transform &t,
Chris@350 996 QString xml)
Chris@350 997 {
Chris@351 998 Vamp::PluginBase *plugin = instantiateDefaultPluginFor
Chris@351 999 (t.getIdentifier(), 0);
Chris@350 1000 if (!plugin) {
Chris@443 1001 cerr << "TransformFactory::setParametersFromPluginConfigurationXml: "
Chris@350 1002 << "Unable to instantiate plugin for transform \""
Chris@443 1003 << t.getIdentifier().toStdString() << "\"" << endl;
Chris@350 1004 return;
Chris@332 1005 }
Chris@332 1006
Chris@350 1007 PluginXml(plugin).setParametersFromXml(xml);
Chris@350 1008 setParametersFromPlugin(t, plugin);
Chris@350 1009 delete plugin;
Chris@332 1010 }
Chris@332 1011
Chris@443 1012 TransformFactory::SearchResults
Chris@443 1013 TransformFactory::search(QString keyword)
Chris@443 1014 {
Chris@443 1015 QStringList keywords;
Chris@443 1016 keywords << keyword;
Chris@443 1017 return search(keywords);
Chris@443 1018 }
Chris@443 1019
Chris@443 1020 TransformFactory::SearchResults
Chris@443 1021 TransformFactory::search(QStringList keywords)
Chris@443 1022 {
Chris@460 1023 populateTransforms();
Chris@443 1024
Chris@447 1025 if (keywords.size() > 1) {
Chris@447 1026 // Additional score for all keywords in a row
Chris@447 1027 keywords.push_back(keywords.join(" "));
Chris@447 1028 }
Chris@447 1029
Chris@443 1030 SearchResults results;
Chris@457 1031 TextMatcher matcher;
Chris@443 1032
Chris@443 1033 for (TransformDescriptionMap::const_iterator i = m_transforms.begin();
Chris@443 1034 i != m_transforms.end(); ++i) {
Chris@443 1035
Chris@457 1036 TextMatcher::Match match;
Chris@443 1037
Chris@457 1038 match.key = i->first;
Chris@443 1039
Chris@457 1040 matcher.test(match, keywords, i->second.type, tr("Plugin type"), 5);
Chris@457 1041 matcher.test(match, keywords, i->second.category, tr("Category"), 20);
Chris@457 1042 matcher.test(match, keywords, i->second.identifier, tr("System Identifier"), 6);
Chris@457 1043 matcher.test(match, keywords, i->second.name, tr("Name"), 30);
Chris@457 1044 matcher.test(match, keywords, i->second.description, tr("Description"), 20);
Chris@457 1045 matcher.test(match, keywords, i->second.maker, tr("Maker"), 10);
Chris@457 1046 matcher.test(match, keywords, i->second.units, tr("Units"), 10);
Chris@457 1047
Chris@457 1048 if (match.score > 0) results[i->first] = match;
Chris@457 1049 }
Chris@457 1050
Chris@460 1051 if (!m_uninstalledTransformsMutex.tryLock()) {
Chris@460 1052 // uninstalled transforms are being populated; this may take some time,
Chris@484 1053 // and they aren't critical, but we will speed them up if necessary
Chris@460 1054 std::cerr << "TransformFactory::search: Uninstalled transforms mutex is held, skipping" << std::endl;
Chris@484 1055 m_populatingSlowly = false;
Chris@460 1056 return results;
Chris@460 1057 }
Chris@460 1058
Chris@460 1059 if (!m_uninstalledTransformsPopulated) {
Chris@460 1060 std::cerr << "WARNING: TransformFactory::search: Uninstalled transforms are not populated yet" << endl
Chris@460 1061 << "and are not being populated either -- was the thread not started correctly?" << endl;
Chris@460 1062 m_uninstalledTransformsMutex.unlock();
Chris@460 1063 return results;
Chris@460 1064 }
Chris@460 1065
Chris@460 1066 m_uninstalledTransformsMutex.unlock();
Chris@457 1067
Chris@457 1068 for (TransformDescriptionMap::const_iterator i = m_uninstalledTransforms.begin();
Chris@457 1069 i != m_uninstalledTransforms.end(); ++i) {
Chris@457 1070
Chris@457 1071 TextMatcher::Match match;
Chris@457 1072
Chris@457 1073 match.key = i->first;
Chris@457 1074
Chris@457 1075 matcher.test(match, keywords, i->second.type, tr("Plugin type"), 2);
Chris@457 1076 matcher.test(match, keywords, i->second.category, tr("Category"), 10);
Chris@457 1077 matcher.test(match, keywords, i->second.identifier, tr("System Identifier"), 3);
Chris@457 1078 matcher.test(match, keywords, i->second.name, tr("Name"), 15);
Chris@457 1079 matcher.test(match, keywords, i->second.description, tr("Description"), 10);
Chris@457 1080 matcher.test(match, keywords, i->second.maker, tr("Maker"), 5);
Chris@457 1081 matcher.test(match, keywords, i->second.units, tr("Units"), 5);
Chris@443 1082
Chris@443 1083 if (match.score > 0) results[i->first] = match;
Chris@443 1084 }
Chris@443 1085
Chris@443 1086 return results;
Chris@443 1087 }
Chris@443 1088