annotate transform/Transform.cpp @ 490:c3fb8258e34d

* Make it possible to import an entire session from an RDF document. However, at the moment the timings of events appear to be constrained by how far the audio decoder has got through its audio file at the time the event is queried -- need to investigate.
author Chris Cannam
date Fri, 21 Nov 2008 18:03:14 +0000
parents 3012af787e4a
children 1b8c748fd7ea
rev   line source
Chris@320 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@320 2
Chris@320 3 /*
Chris@320 4 Sonic Visualiser
Chris@320 5 An audio file viewer and annotation editor.
Chris@320 6 Centre for Digital Music, Queen Mary, University of London.
Chris@328 7 This file copyright 2006-2007 Chris Cannam and QMUL.
Chris@320 8
Chris@320 9 This program is free software; you can redistribute it and/or
Chris@320 10 modify it under the terms of the GNU General Public License as
Chris@320 11 published by the Free Software Foundation; either version 2 of the
Chris@320 12 License, or (at your option) any later version. See the file
Chris@320 13 COPYING included with this distribution for more information.
Chris@320 14 */
Chris@320 15
Chris@320 16 #include "Transform.h"
Chris@320 17
Chris@328 18 #include "plugin/PluginIdentifier.h"
Chris@328 19
Chris@332 20 #include "plugin/FeatureExtractionPluginFactory.h"
Chris@332 21
Chris@350 22 #include <QXmlAttributes>
Chris@350 23
Chris@350 24 #include <QDomDocument>
Chris@350 25 #include <QDomElement>
Chris@350 26 #include <QDomNamedNodeMap>
Chris@350 27 #include <QDomAttr>
Chris@350 28
Chris@350 29 #include <QTextStream>
Chris@350 30
Chris@350 31 #include <iostream>
Chris@350 32
Chris@328 33 Transform::Transform() :
Chris@328 34 m_stepSize(0),
Chris@328 35 m_blockSize(0),
Chris@328 36 m_windowType(HanningWindow),
Chris@328 37 m_sampleRate(0)
Chris@320 38 {
Chris@320 39 }
Chris@320 40
Chris@350 41 Transform::Transform(QString xml) :
Chris@350 42 m_stepSize(0),
Chris@350 43 m_blockSize(0),
Chris@350 44 m_windowType(HanningWindow),
Chris@350 45 m_sampleRate(0)
Chris@350 46 {
Chris@350 47 QDomDocument doc;
Chris@350 48
Chris@350 49 QString error;
Chris@350 50 int errorLine;
Chris@350 51 int errorColumn;
Chris@350 52
Chris@350 53 if (!doc.setContent(xml, false, &error, &errorLine, &errorColumn)) {
Chris@350 54 std::cerr << "Transform::Transform: Error in parsing XML: "
Chris@350 55 << error.toStdString() << " at line " << errorLine
Chris@350 56 << ", column " << errorColumn << std::endl;
Chris@350 57 std::cerr << "Input follows:" << std::endl;
Chris@350 58 std::cerr << xml.toStdString() << std::endl;
Chris@350 59 std::cerr << "Input ends." << std::endl;
Chris@350 60 return;
Chris@350 61 }
Chris@350 62
Chris@350 63 QDomElement transformElt = doc.firstChildElement("transform");
Chris@350 64 QDomNamedNodeMap attrNodes = transformElt.attributes();
Chris@350 65 QXmlAttributes attrs;
Chris@350 66
Chris@350 67 for (unsigned int i = 0; i < attrNodes.length(); ++i) {
Chris@350 68 QDomAttr attr = attrNodes.item(i).toAttr();
Chris@350 69 if (!attr.isNull()) attrs.append(attr.name(), "", "", attr.value());
Chris@350 70 }
Chris@350 71
Chris@350 72 setFromXmlAttributes(attrs);
Chris@350 73
Chris@350 74 for (QDomElement paramElt = transformElt.firstChildElement("parameter");
Chris@350 75 !paramElt.isNull();
Chris@350 76 paramElt = paramElt.nextSiblingElement("parameter")) {
Chris@350 77
Chris@350 78 QDomNamedNodeMap paramAttrs = paramElt.attributes();
Chris@350 79
Chris@350 80 QDomAttr nameAttr = paramAttrs.namedItem("name").toAttr();
Chris@350 81 if (nameAttr.isNull() || nameAttr.value() == "") continue;
Chris@350 82
Chris@350 83 QDomAttr valueAttr = paramAttrs.namedItem("value").toAttr();
Chris@350 84 if (valueAttr.isNull() || valueAttr.value() == "") continue;
Chris@350 85
Chris@350 86 setParameter(nameAttr.value(), valueAttr.value().toFloat());
Chris@350 87 }
Chris@350 88
Chris@350 89 for (QDomElement configElt = transformElt.firstChildElement("configuration");
Chris@350 90 !configElt.isNull();
Chris@350 91 configElt = configElt.nextSiblingElement("configuration")) {
Chris@350 92
Chris@350 93 QDomNamedNodeMap configAttrs = configElt.attributes();
Chris@350 94
Chris@350 95 QDomAttr nameAttr = configAttrs.namedItem("name").toAttr();
Chris@350 96 if (nameAttr.isNull() || nameAttr.value() == "") continue;
Chris@350 97
Chris@350 98 QDomAttr valueAttr = configAttrs.namedItem("value").toAttr();
Chris@350 99 if (valueAttr.isNull() || valueAttr.value() == "") continue;
Chris@350 100
Chris@350 101 setConfigurationValue(nameAttr.value(), valueAttr.value());
Chris@350 102 }
Chris@350 103 }
Chris@350 104
Chris@320 105 Transform::~Transform()
Chris@320 106 {
Chris@320 107 }
Chris@320 108
Chris@350 109 bool
Chris@400 110 Transform::operator==(const Transform &t) const
Chris@350 111 {
Chris@350 112 return
Chris@350 113 m_id == t.m_id &&
Chris@350 114 m_parameters == t.m_parameters &&
Chris@350 115 m_configuration == t.m_configuration &&
Chris@350 116 m_program == t.m_program &&
Chris@350 117 m_stepSize == t.m_stepSize &&
Chris@350 118 m_blockSize == t.m_blockSize &&
Chris@350 119 m_windowType == t.m_windowType &&
Chris@350 120 m_startTime == t.m_startTime &&
Chris@350 121 m_duration == t.m_duration &&
Chris@350 122 m_sampleRate == t.m_sampleRate;
Chris@350 123 }
Chris@350 124
Chris@400 125 bool
Chris@400 126 Transform::operator<(const Transform &t) const
Chris@400 127 {
Chris@400 128 if (m_id != t.m_id) {
Chris@400 129 return m_id < t.m_id;
Chris@400 130 }
Chris@400 131 if (m_parameters != t.m_parameters) {
Chris@400 132 return mapLessThan<QString, float>(m_parameters, t.m_parameters);
Chris@400 133 }
Chris@400 134 if (m_configuration != t.m_configuration) {
Chris@400 135 return mapLessThan<QString, QString>(m_configuration, t.m_configuration);
Chris@400 136 }
Chris@400 137 if (m_program != t.m_program) {
Chris@400 138 return m_program < t.m_program;
Chris@400 139 }
Chris@400 140 if (m_stepSize != t.m_stepSize) {
Chris@400 141 return m_stepSize < t.m_stepSize;
Chris@400 142 }
Chris@400 143 if (m_blockSize != t.m_blockSize) {
Chris@400 144 return m_blockSize < t.m_blockSize;
Chris@400 145 }
Chris@400 146 if (m_windowType != t.m_windowType) {
Chris@400 147 return m_windowType < t.m_windowType;
Chris@400 148 }
Chris@400 149 if (m_startTime != t.m_startTime) {
Chris@400 150 return m_startTime < t.m_startTime;
Chris@400 151 }
Chris@400 152 if (m_duration != t.m_duration) {
Chris@400 153 return m_duration < t.m_duration;
Chris@400 154 }
Chris@400 155 if (m_sampleRate != t.m_sampleRate) {
Chris@400 156 return m_sampleRate < t.m_sampleRate;
Chris@400 157 }
Chris@400 158 return false;
Chris@400 159 }
Chris@400 160
Chris@350 161 void
Chris@350 162 Transform::setIdentifier(TransformId id)
Chris@350 163 {
Chris@350 164 m_id = id;
Chris@350 165 }
Chris@350 166
Chris@350 167 TransformId
Chris@350 168 Transform::getIdentifier() const
Chris@350 169 {
Chris@350 170 return m_id;
Chris@350 171 }
Chris@350 172
Chris@328 173 QString
Chris@328 174 Transform::createIdentifier(QString type, QString soName, QString label,
Chris@328 175 QString output)
Chris@328 176 {
Chris@328 177 QString pluginId = PluginIdentifier::createIdentifier(type, soName, label);
Chris@328 178 return pluginId + ":" + output;
Chris@328 179 }
Chris@328 180
Chris@328 181 void
Chris@328 182 Transform::parseIdentifier(QString identifier,
Chris@328 183 QString &type, QString &soName,
Chris@328 184 QString &label, QString &output)
Chris@328 185 {
Chris@328 186 output = identifier.section(':', 3);
Chris@328 187 PluginIdentifier::parseIdentifier(identifier.section(':', 0, 2),
Chris@328 188 type, soName, label);
Chris@328 189 }
Chris@328 190
Chris@328 191 Transform::Type
Chris@328 192 Transform::getType() const
Chris@328 193 {
Chris@332 194 if (FeatureExtractionPluginFactory::instanceFor(getPluginIdentifier())) {
Chris@332 195 return FeatureExtraction;
Chris@332 196 } else {
Chris@332 197 // We don't have an unknown/invalid return value, so always
Chris@332 198 // return this
Chris@332 199 return RealTimeEffect;
Chris@332 200 }
Chris@328 201 }
Chris@328 202
Chris@328 203 QString
Chris@328 204 Transform::getPluginIdentifier() const
Chris@328 205 {
Chris@328 206 return m_id.section(':', 0, 2);
Chris@328 207 }
Chris@328 208
Chris@328 209 QString
Chris@328 210 Transform::getOutput() const
Chris@328 211 {
Chris@328 212 return m_id.section(':', 3);
Chris@328 213 }
Chris@328 214
Chris@353 215 void
Chris@353 216 Transform::setPluginIdentifier(QString pluginIdentifier)
Chris@353 217 {
Chris@353 218 m_id = pluginIdentifier + ':' + getOutput();
Chris@353 219 }
Chris@353 220
Chris@353 221 void
Chris@353 222 Transform::setOutput(QString output)
Chris@353 223 {
Chris@353 224 m_id = getPluginIdentifier() + ':' + output;
Chris@353 225 }
Chris@353 226
Chris@353 227 TransformId
Chris@353 228 Transform::getIdentifierForPluginOutput(QString pluginIdentifier,
Chris@353 229 QString output)
Chris@353 230 {
Chris@353 231 return pluginIdentifier + ':' + output;
Chris@353 232 }
Chris@353 233
Chris@350 234 const Transform::ParameterMap &
Chris@350 235 Transform::getParameters() const
Chris@350 236 {
Chris@350 237 return m_parameters;
Chris@350 238 }
Chris@350 239
Chris@328 240 void
Chris@350 241 Transform::setParameters(const ParameterMap &pm)
Chris@328 242 {
Chris@350 243 m_parameters = pm;
Chris@350 244 }
Chris@350 245
Chris@350 246 void
Chris@350 247 Transform::setParameter(QString name, float value)
Chris@350 248 {
Chris@403 249 // std::cerr << "Transform::setParameter(" << name.toStdString()
Chris@403 250 // << ") -> " << value << std::endl;
Chris@350 251 m_parameters[name] = value;
Chris@350 252 }
Chris@350 253
Chris@350 254 const Transform::ConfigurationMap &
Chris@350 255 Transform::getConfiguration() const
Chris@350 256 {
Chris@350 257 return m_configuration;
Chris@350 258 }
Chris@350 259
Chris@350 260 void
Chris@350 261 Transform::setConfiguration(const ConfigurationMap &cm)
Chris@350 262 {
Chris@350 263 m_configuration = cm;
Chris@350 264 }
Chris@350 265
Chris@350 266 void
Chris@350 267 Transform::setConfigurationValue(QString name, QString value)
Chris@350 268 {
Chris@350 269 std::cerr << "Transform::setConfigurationValue(" << name.toStdString()
Chris@350 270 << ") -> " << value.toStdString() << std::endl;
Chris@350 271 m_configuration[name] = value;
Chris@350 272 }
Chris@350 273
Chris@350 274 QString
Chris@366 275 Transform::getPluginVersion() const
Chris@366 276 {
Chris@366 277 return m_pluginVersion;
Chris@366 278 }
Chris@366 279
Chris@366 280 void
Chris@366 281 Transform::setPluginVersion(QString version)
Chris@366 282 {
Chris@366 283 m_pluginVersion = version;
Chris@366 284 }
Chris@366 285
Chris@366 286 QString
Chris@350 287 Transform::getProgram() const
Chris@350 288 {
Chris@350 289 return m_program;
Chris@350 290 }
Chris@350 291
Chris@350 292 void
Chris@350 293 Transform::setProgram(QString program)
Chris@350 294 {
Chris@350 295 m_program = program;
Chris@350 296 }
Chris@350 297
Chris@328 298
Chris@350 299 size_t
Chris@350 300 Transform::getStepSize() const
Chris@350 301 {
Chris@350 302 return m_stepSize;
Chris@328 303 }
Chris@350 304
Chris@350 305 void
Chris@350 306 Transform::setStepSize(size_t s)
Chris@350 307 {
Chris@350 308 m_stepSize = s;
Chris@350 309 }
Chris@350 310
Chris@350 311 size_t
Chris@350 312 Transform::getBlockSize() const
Chris@350 313 {
Chris@350 314 return m_blockSize;
Chris@350 315 }
Chris@350 316
Chris@350 317 void
Chris@350 318 Transform::setBlockSize(size_t s)
Chris@350 319 {
Chris@350 320 m_blockSize = s;
Chris@350 321 }
Chris@350 322
Chris@350 323 WindowType
Chris@350 324 Transform::getWindowType() const
Chris@350 325 {
Chris@350 326 return m_windowType;
Chris@350 327 }
Chris@350 328
Chris@350 329 void
Chris@350 330 Transform::setWindowType(WindowType type)
Chris@350 331 {
Chris@350 332 m_windowType = type;
Chris@350 333 }
Chris@350 334
Chris@350 335 RealTime
Chris@350 336 Transform::getStartTime() const
Chris@350 337 {
Chris@350 338 return m_startTime;
Chris@350 339 }
Chris@350 340
Chris@350 341 void
Chris@350 342 Transform::setStartTime(RealTime t)
Chris@350 343 {
Chris@350 344 m_startTime = t;
Chris@350 345 }
Chris@350 346
Chris@350 347 RealTime
Chris@350 348 Transform::getDuration() const
Chris@350 349 {
Chris@350 350 return m_duration;
Chris@350 351 }
Chris@350 352
Chris@350 353 void
Chris@350 354 Transform::setDuration(RealTime d)
Chris@350 355 {
Chris@350 356 m_duration = d;
Chris@350 357 }
Chris@350 358
Chris@350 359 float
Chris@350 360 Transform::getSampleRate() const
Chris@350 361 {
Chris@350 362 return m_sampleRate;
Chris@350 363 }
Chris@350 364
Chris@350 365 void
Chris@350 366 Transform::setSampleRate(float rate)
Chris@350 367 {
Chris@350 368 m_sampleRate = rate;
Chris@350 369 }
Chris@350 370
Chris@350 371 void
Chris@350 372 Transform::toXml(QTextStream &out, QString indent, QString extraAttributes) const
Chris@350 373 {
Chris@350 374 out << indent;
Chris@350 375
Chris@350 376 bool haveContent = true;
Chris@350 377 if (m_parameters.empty() && m_configuration.empty()) haveContent = false;
Chris@350 378
Chris@396 379 out << QString("<transform\n id=\"%1\"\n pluginVersion=\"%2\"\n program=\"%3\"\n stepSize=\"%4\"\n blockSize=\"%5\"\n windowType=\"%6\"\n startTime=\"%7\"\n duration=\"%8\"\n sampleRate=\"%9\"")
Chris@350 380 .arg(encodeEntities(m_id))
Chris@366 381 .arg(encodeEntities(m_pluginVersion))
Chris@350 382 .arg(encodeEntities(m_program))
Chris@350 383 .arg(m_stepSize)
Chris@350 384 .arg(m_blockSize)
Chris@350 385 .arg(encodeEntities(Window<float>::getNameForType(m_windowType).c_str()))
Chris@350 386 .arg(encodeEntities(m_startTime.toString().c_str()))
Chris@350 387 .arg(encodeEntities(m_duration.toString().c_str()))
Chris@366 388 .arg(m_sampleRate);
Chris@366 389
Chris@366 390 if (extraAttributes != "") {
Chris@366 391 out << " " << extraAttributes;
Chris@366 392 }
Chris@350 393
Chris@350 394 if (haveContent) {
Chris@350 395
Chris@350 396 out << ">\n";
Chris@350 397
Chris@350 398 for (ParameterMap::const_iterator i = m_parameters.begin();
Chris@350 399 i != m_parameters.end(); ++i) {
Chris@350 400 out << indent << " "
Chris@350 401 << QString("<parameter name=\"%1\" value=\"%2\"/>\n")
Chris@350 402 .arg(encodeEntities(i->first))
Chris@350 403 .arg(i->second);
Chris@350 404 }
Chris@350 405
Chris@350 406 for (ConfigurationMap::const_iterator i = m_configuration.begin();
Chris@350 407 i != m_configuration.end(); ++i) {
Chris@350 408 out << indent << " "
Chris@350 409 << QString("<configuration name=\"%1\" value=\"%2\"/>\n")
Chris@350 410 .arg(encodeEntities(i->first))
Chris@350 411 .arg(encodeEntities(i->second));
Chris@350 412 }
Chris@350 413
Chris@350 414 out << indent << "</transform>\n";
Chris@350 415
Chris@350 416 } else {
Chris@350 417
Chris@350 418 out << "/>\n";
Chris@350 419 }
Chris@350 420 }
Chris@350 421
Chris@350 422 void
Chris@350 423 Transform::setFromXmlAttributes(const QXmlAttributes &attrs)
Chris@350 424 {
Chris@350 425 if (attrs.value("id") != "") {
Chris@350 426 setIdentifier(attrs.value("id"));
Chris@350 427 }
Chris@350 428
Chris@366 429 if (attrs.value("pluginVersion") != "") {
Chris@366 430 setPluginVersion(attrs.value("pluginVersion"));
Chris@366 431 }
Chris@366 432
Chris@350 433 if (attrs.value("program") != "") {
Chris@350 434 setProgram(attrs.value("program"));
Chris@350 435 }
Chris@350 436
Chris@350 437 if (attrs.value("stepSize") != "") {
Chris@350 438 setStepSize(attrs.value("stepSize").toInt());
Chris@350 439 }
Chris@350 440
Chris@350 441 if (attrs.value("blockSize") != "") {
Chris@350 442 setBlockSize(attrs.value("blockSize").toInt());
Chris@350 443 }
Chris@350 444
Chris@350 445 if (attrs.value("windowType") != "") {
Chris@350 446 setWindowType(Window<float>::getTypeForName
Chris@350 447 (attrs.value("windowType").toStdString()));
Chris@350 448 }
Chris@350 449
Chris@350 450 if (attrs.value("startTime") != "") {
Chris@350 451 setStartTime(RealTime::fromString(attrs.value("startTime").toStdString()));
Chris@350 452 }
Chris@350 453
Chris@350 454 if (attrs.value("duration") != "") {
Chris@350 455 setStartTime(RealTime::fromString(attrs.value("duration").toStdString()));
Chris@350 456 }
Chris@350 457
Chris@350 458 if (attrs.value("sampleRate") != "") {
Chris@350 459 setSampleRate(attrs.value("sampleRate").toFloat());
Chris@350 460 }
Chris@350 461 }
Chris@350 462