annotate transform/TransformFactory.cpp @ 481:a82645e788fc

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