annotate layer/ColourMapper.cpp @ 947:e53a87a5efb2

Allow layers to be loaded without models if their layer class explicitly says it's OK (otherwise default template won't load, as it has an empty waveform layer)
author Chris Cannam
date Mon, 20 Apr 2015 10:10:26 +0100
parents 6e2a034f7117
children 2a85ab180d08
rev   line source
Chris@376 1 /* -*- c-basic-offset: 4 indent-tabs-mode: nil -*- vi:set ts=8 sts=4 sw=4: */
Chris@376 2
Chris@376 3 /*
Chris@376 4 Sonic Visualiser
Chris@376 5 An audio file viewer and annotation editor.
Chris@376 6 Centre for Digital Music, Queen Mary, University of London.
Chris@376 7 This file copyright 2006-2007 Chris Cannam and QMUL.
Chris@376 8
Chris@376 9 This program is free software; you can redistribute it and/or
Chris@376 10 modify it under the terms of the GNU General Public License as
Chris@376 11 published by the Free Software Foundation; either version 2 of the
Chris@376 12 License, or (at your option) any later version. See the file
Chris@376 13 COPYING included with this distribution for more information.
Chris@376 14 */
Chris@376 15
Chris@376 16 #include "ColourMapper.h"
Chris@376 17
Chris@376 18 #include <iostream>
Chris@376 19
Chris@376 20 #include <cmath>
Chris@376 21
Chris@682 22 #include "base/Debug.h"
Chris@682 23
Chris@902 24 ColourMapper::ColourMapper(int map, double min, double max) :
Chris@376 25 QObject(),
Chris@376 26 m_map(map),
Chris@376 27 m_min(min),
Chris@376 28 m_max(max)
Chris@376 29 {
Chris@376 30 if (m_min == m_max) {
Chris@682 31 cerr << "WARNING: ColourMapper: min == max (== " << m_min
Chris@682 32 << "), adjusting" << endl;
Chris@376 33 m_max = m_min + 1;
Chris@376 34 }
Chris@376 35 }
Chris@376 36
Chris@376 37 ColourMapper::~ColourMapper()
Chris@376 38 {
Chris@376 39 }
Chris@376 40
Chris@376 41 int
Chris@376 42 ColourMapper::getColourMapCount()
Chris@376 43 {
Chris@536 44 return 12;
Chris@376 45 }
Chris@376 46
Chris@376 47 QString
Chris@376 48 ColourMapper::getColourMapName(int n)
Chris@376 49 {
Chris@376 50 if (n >= getColourMapCount()) return tr("<unknown>");
Chris@376 51 StandardMap map = (StandardMap)n;
Chris@376 52
Chris@376 53 switch (map) {
Chris@376 54 case DefaultColours: return tr("Default");
Chris@376 55 case WhiteOnBlack: return tr("White on Black");
Chris@376 56 case BlackOnWhite: return tr("Black on White");
Chris@376 57 case RedOnBlue: return tr("Red on Blue");
Chris@376 58 case YellowOnBlack: return tr("Yellow on Black");
Chris@376 59 case BlueOnBlack: return tr("Blue on Black");
Chris@376 60 case Sunset: return tr("Sunset");
Chris@376 61 case FruitSalad: return tr("Fruit Salad");
Chris@376 62 case Banded: return tr("Banded");
Chris@376 63 case Highlight: return tr("Highlight");
Chris@376 64 case Printer: return tr("Printer");
Chris@536 65 case HighGain: return tr("High Gain");
Chris@376 66 }
Chris@376 67
Chris@376 68 return tr("<unknown>");
Chris@376 69 }
Chris@376 70
Chris@376 71 QColor
Chris@902 72 ColourMapper::map(double value) const
Chris@376 73 {
Chris@902 74 double norm = (value - m_min) / (m_max - m_min);
Chris@902 75 if (norm < 0.0) norm = 0.0;
Chris@902 76 if (norm > 1.0) norm = 1.0;
Chris@376 77
Chris@902 78 double h = 0.0, s = 0.0, v = 0.0, r = 0.0, g = 0.0, b = 0.0;
Chris@376 79 bool hsv = true;
Chris@376 80
Chris@911 81 double blue = 0.6666, pieslice = 0.3333;
Chris@376 82
Chris@376 83 if (m_map >= getColourMapCount()) return Qt::black;
Chris@376 84 StandardMap map = (StandardMap)m_map;
Chris@376 85
Chris@376 86 switch (map) {
Chris@376 87
Chris@376 88 case DefaultColours:
Chris@902 89 h = blue - norm * 2.0 * pieslice;
Chris@902 90 s = 0.5f + norm/2.0;
Chris@376 91 v = norm;
Chris@376 92 break;
Chris@376 93
Chris@376 94 case WhiteOnBlack:
Chris@376 95 r = g = b = norm;
Chris@376 96 hsv = false;
Chris@376 97 break;
Chris@376 98
Chris@376 99 case BlackOnWhite:
Chris@902 100 r = g = b = 1.0 - norm;
Chris@376 101 hsv = false;
Chris@376 102 break;
Chris@376 103
Chris@376 104 case RedOnBlue:
Chris@902 105 h = blue - pieslice/4.0 + norm * (pieslice + pieslice/4.0);
Chris@902 106 s = 1.0;
Chris@376 107 v = norm;
Chris@376 108 break;
Chris@376 109
Chris@376 110 case YellowOnBlack:
Chris@902 111 h = 0.15;
Chris@902 112 s = 1.0;
Chris@376 113 v = norm;
Chris@376 114 break;
Chris@376 115
Chris@376 116 case BlueOnBlack:
Chris@376 117 h = blue;
Chris@902 118 s = 1.0;
Chris@902 119 v = norm * 2.0;
Chris@902 120 if (v > 1.0) {
Chris@902 121 v = 1.0;
Chris@904 122 s = 1.0 - (sqrt(norm) - 0.707) * 3.413;
Chris@902 123 if (s < 0.0) s = 0.0;
Chris@902 124 if (s > 1.0) s = 1.0;
Chris@376 125 }
Chris@376 126 break;
Chris@376 127
Chris@376 128 case Sunset:
Chris@902 129 r = (norm - 0.24) * 2.38;
Chris@902 130 if (r > 1.0) r = 1.0;
Chris@902 131 if (r < 0.0) r = 0.0;
Chris@902 132 g = (norm - 0.64) * 2.777;
Chris@902 133 if (g > 1.0) g = 1.0;
Chris@902 134 if (g < 0.0) g = 0.0;
Chris@376 135 b = (3.6f * norm);
Chris@902 136 if (norm > 0.277) b = 2.0 - b;
Chris@902 137 if (b > 1.0) b = 1.0;
Chris@902 138 if (b < 0.0) b = 0.0;
Chris@376 139 hsv = false;
Chris@376 140 break;
Chris@376 141
Chris@376 142 case FruitSalad:
Chris@902 143 h = blue + (pieslice/6.0) - norm;
Chris@902 144 if (h < 0.0) h += 1.0;
Chris@902 145 s = 1.0;
Chris@902 146 v = 1.0;
Chris@376 147 break;
Chris@376 148
Chris@376 149 case Banded:
Chris@376 150 if (norm < 0.125) return Qt::darkGreen;
Chris@376 151 else if (norm < 0.25) return Qt::green;
Chris@376 152 else if (norm < 0.375) return Qt::darkBlue;
Chris@376 153 else if (norm < 0.5) return Qt::blue;
Chris@376 154 else if (norm < 0.625) return Qt::darkYellow;
Chris@376 155 else if (norm < 0.75) return Qt::yellow;
Chris@376 156 else if (norm < 0.875) return Qt::darkRed;
Chris@376 157 else return Qt::red;
Chris@376 158 break;
Chris@376 159
Chris@376 160 case Highlight:
Chris@376 161 if (norm > 0.99) return Qt::white;
Chris@376 162 else return Qt::darkBlue;
Chris@376 163
Chris@376 164 case Printer:
Chris@376 165 if (norm > 0.8) {
Chris@902 166 r = 1.0;
Chris@376 167 } else if (norm > 0.7) {
Chris@902 168 r = 0.9;
Chris@376 169 } else if (norm > 0.6) {
Chris@902 170 r = 0.8;
Chris@376 171 } else if (norm > 0.5) {
Chris@902 172 r = 0.7;
Chris@376 173 } else if (norm > 0.4) {
Chris@902 174 r = 0.6;
Chris@376 175 } else if (norm > 0.3) {
Chris@902 176 r = 0.5;
Chris@376 177 } else if (norm > 0.2) {
Chris@902 178 r = 0.4;
Chris@376 179 } else {
Chris@902 180 r = 0.0;
Chris@376 181 }
Chris@902 182 r = g = b = 1.0 - r;
Chris@376 183 hsv = false;
Chris@376 184 break;
Chris@536 185
Chris@536 186 case HighGain:
Chris@902 187 if (norm <= 1.0 / 256.0) {
Chris@902 188 norm = 0.0;
Chris@536 189 } else {
Chris@904 190 norm = 0.1f + (pow(((norm - 0.5) * 2.0), 3.0) + 1.0) / 2.081;
Chris@536 191 }
Chris@536 192 // now as for Sunset
Chris@902 193 r = (norm - 0.24) * 2.38;
Chris@902 194 if (r > 1.0) r = 1.0;
Chris@902 195 if (r < 0.0) r = 0.0;
Chris@902 196 g = (norm - 0.64) * 2.777;
Chris@902 197 if (g > 1.0) g = 1.0;
Chris@902 198 if (g < 0.0) g = 0.0;
Chris@536 199 b = (3.6f * norm);
Chris@902 200 if (norm > 0.277) b = 2.0 - b;
Chris@902 201 if (b > 1.0) b = 1.0;
Chris@902 202 if (b < 0.0) b = 0.0;
Chris@536 203 hsv = false;
Chris@536 204 /*
Chris@902 205 if (r > 1.0) r = 1.0;
Chris@902 206 r = g = b = 1.0 - r;
Chris@536 207 hsv = false;
Chris@536 208 */
Chris@536 209 break;
Chris@376 210 }
Chris@376 211
Chris@376 212 if (hsv) {
Chris@376 213 return QColor::fromHsvF(h, s, v);
Chris@376 214 } else {
Chris@376 215 return QColor::fromRgbF(r, g, b);
Chris@376 216 }
Chris@376 217 }
Chris@376 218
Chris@376 219 QColor
Chris@376 220 ColourMapper::getContrastingColour() const
Chris@376 221 {
Chris@376 222 if (m_map >= getColourMapCount()) return Qt::white;
Chris@376 223 StandardMap map = (StandardMap)m_map;
Chris@376 224
Chris@376 225 switch (map) {
Chris@376 226
Chris@376 227 case DefaultColours:
Chris@376 228 return QColor(255, 150, 50);
Chris@376 229
Chris@376 230 case WhiteOnBlack:
Chris@376 231 return Qt::red;
Chris@376 232
Chris@376 233 case BlackOnWhite:
Chris@376 234 return Qt::darkGreen;
Chris@376 235
Chris@376 236 case RedOnBlue:
Chris@376 237 return Qt::green;
Chris@376 238
Chris@376 239 case YellowOnBlack:
Chris@376 240 return QColor::fromHsv(240, 255, 255);
Chris@376 241
Chris@376 242 case BlueOnBlack:
Chris@376 243 return Qt::red;
Chris@376 244
Chris@376 245 case Sunset:
Chris@376 246 return Qt::white;
Chris@376 247
Chris@376 248 case FruitSalad:
Chris@376 249 return Qt::white;
Chris@376 250
Chris@376 251 case Banded:
Chris@376 252 return Qt::cyan;
Chris@376 253
Chris@376 254 case Highlight:
Chris@376 255 return Qt::red;
Chris@376 256
Chris@376 257 case Printer:
Chris@376 258 return Qt::red;
Chris@536 259
Chris@536 260 case HighGain:
Chris@536 261 return Qt::red;
Chris@376 262 }
Chris@376 263
Chris@376 264 return Qt::white;
Chris@376 265 }
Chris@376 266
Chris@376 267 bool
Chris@376 268 ColourMapper::hasLightBackground() const
Chris@376 269 {
Chris@376 270 if (m_map >= getColourMapCount()) return false;
Chris@376 271 StandardMap map = (StandardMap)m_map;
Chris@376 272
Chris@376 273 switch (map) {
Chris@376 274
Chris@376 275 case BlackOnWhite:
Chris@376 276 case Printer:
Chris@536 277 case HighGain:
Chris@376 278 return true;
Chris@376 279
Chris@805 280 case DefaultColours:
Chris@805 281 case Sunset:
Chris@805 282 case WhiteOnBlack:
Chris@805 283 case RedOnBlue:
Chris@805 284 case YellowOnBlack:
Chris@805 285 case BlueOnBlack:
Chris@805 286 case FruitSalad:
Chris@805 287 case Banded:
Chris@805 288 case Highlight:
Chris@805 289
Chris@376 290 default:
Chris@376 291 return false;
Chris@376 292 }
Chris@376 293 }
Chris@376 294
Chris@376 295