comparison layer/ColourMapper.cpp @ 1362:d79e21855aef

Add mechanism for saving/loading colour maps by name/id rather than by numerical index, for future compatibility when adding to or changing the supported colour maps. Add two new colour maps (and one old one). Write out backward-compatible numerical indices for use when reloading in older versions. Also add a mechanism to invert the colour map, though I don't think it turns out useful enough to include in the UI.
author Chris Cannam
date Thu, 18 Oct 2018 13:21:56 +0100
parents 6e724c81f18f
children c8a6fd3f9dff
comparison
equal deleted inserted replaced
1361:2e3b3fadba27 1362:d79e21855aef
25 25
26 #include <QPainter> 26 #include <QPainter>
27 27
28 using namespace std; 28 using namespace std;
29 29
30 static vector<QColor> convertStrings(const vector<QString> &strs) 30 static vector<QColor> convertStrings(const vector<QString> &strs,
31 bool reversed)
31 { 32 {
32 vector<QColor> converted; 33 vector<QColor> converted;
33 for (const auto &s: strs) converted.push_back(QColor(s)); 34 for (const auto &s: strs) converted.push_back(QColor(s));
34 reverse(converted.begin(), converted.end()); 35 if (reversed) {
36 reverse(converted.begin(), converted.end());
37 }
35 return converted; 38 return converted;
36 } 39 }
37 40
38 static vector<QColor> ice = convertStrings({ 41 static vector<QColor> ice = convertStrings({
39 // Based on ColorBrewer ylGnBu 42 // Based on ColorBrewer ylGnBu
40 "#ffffff", "#ffff00", "#f7fcf0", "#e0f3db", "#ccebc5", "#a8ddb5", 43 "#ffffff", "#ffff00", "#f7fcf0", "#e0f3db", "#ccebc5", "#a8ddb5",
41 "#7bccc4", "#4eb3d3", "#2b8cbe", "#0868ac", "#084081", "#042040" 44 "#7bccc4", "#4eb3d3", "#2b8cbe", "#0868ac", "#084081", "#042040"
42 }); 45 },
46 true);
43 47
44 static vector<QColor> cherry = convertStrings({ 48 static vector<QColor> cherry = convertStrings({
45 "#f7f7f7", "#fddbc7", "#f4a582", "#d6604d", "#b2182b", "#dd3497", 49 "#f7f7f7", "#fddbc7", "#f4a582", "#d6604d", "#b2182b", "#dd3497",
46 "#ae017e", "#7a0177", "#49006a" 50 "#ae017e", "#7a0177", "#49006a"
47 }); 51 },
48 52 true);
53
54 static vector<QColor> magma = convertStrings({
55 "#FCFFB2", "#FCDF96", "#FBC17D", "#FBA368", "#FA8657", "#F66B4D",
56 "#ED504A", "#E03B50", "#C92D59", "#B02363", "#981D69", "#81176D",
57 "#6B116F", "#57096E", "#43006A", "#300060", "#1E0848", "#110B2D",
58 "#080616", "#000005"
59 },
60 true);
61
62 static vector<QColor> cividis = convertStrings({
63 "#00204c", "#00204e", "#002150", "#002251", "#002353", "#002355",
64 "#002456", "#002558", "#00265a", "#00265b", "#00275d", "#00285f",
65 "#002861", "#002963", "#002a64", "#002a66", "#002b68", "#002c6a",
66 "#002d6c", "#002d6d", "#002e6e", "#002e6f", "#002f6f", "#002f6f",
67 "#00306f", "#00316f", "#00316f", "#00326e", "#00336e", "#00346e",
68 "#00346e", "#01356e", "#06366e", "#0a376d", "#0e376d", "#12386d",
69 "#15396d", "#17396d", "#1a3a6c", "#1c3b6c", "#1e3c6c", "#203c6c",
70 "#223d6c", "#243e6c", "#263e6c", "#273f6c", "#29406b", "#2b416b",
71 "#2c416b", "#2e426b", "#2f436b", "#31446b", "#32446b", "#33456b",
72 "#35466b", "#36466b", "#37476b", "#38486b", "#3a496b", "#3b496b",
73 "#3c4a6b", "#3d4b6b", "#3e4b6b", "#404c6b", "#414d6b", "#424e6b",
74 "#434e6b", "#444f6b", "#45506b", "#46506b", "#47516b", "#48526b",
75 "#49536b", "#4a536b", "#4b546b", "#4c556b", "#4d556b", "#4e566b",
76 "#4f576c", "#50586c", "#51586c", "#52596c", "#535a6c", "#545a6c",
77 "#555b6c", "#565c6c", "#575d6d", "#585d6d", "#595e6d", "#5a5f6d",
78 "#5b5f6d", "#5c606d", "#5d616e", "#5e626e", "#5f626e", "#5f636e",
79 "#60646e", "#61656f", "#62656f", "#63666f", "#64676f", "#65676f",
80 "#666870", "#676970", "#686a70", "#686a70", "#696b71", "#6a6c71",
81 "#6b6d71", "#6c6d72", "#6d6e72", "#6e6f72", "#6f6f72", "#6f7073",
82 "#707173", "#717273", "#727274", "#737374", "#747475", "#757575",
83 "#757575", "#767676", "#777776", "#787876", "#797877", "#7a7977",
84 "#7b7a77", "#7b7b78", "#7c7b78", "#7d7c78", "#7e7d78", "#7f7e78",
85 "#807e78", "#817f78", "#828078", "#838178", "#848178", "#858278",
86 "#868378", "#878478", "#888578", "#898578", "#8a8678", "#8b8778",
87 "#8c8878", "#8d8878", "#8e8978", "#8f8a78", "#908b78", "#918c78",
88 "#928c78", "#938d78", "#948e78", "#958f78", "#968f77", "#979077",
89 "#989177", "#999277", "#9a9377", "#9b9377", "#9c9477", "#9d9577",
90 "#9e9676", "#9f9776", "#a09876", "#a19876", "#a29976", "#a39a75",
91 "#a49b75", "#a59c75", "#a69c75", "#a79d75", "#a89e74", "#a99f74",
92 "#aaa074", "#aba174", "#aca173", "#ada273", "#aea373", "#afa473",
93 "#b0a572", "#b1a672", "#b2a672", "#b4a771", "#b5a871", "#b6a971",
94 "#b7aa70", "#b8ab70", "#b9ab70", "#baac6f", "#bbad6f", "#bcae6e",
95 "#bdaf6e", "#beb06e", "#bfb16d", "#c0b16d", "#c1b26c", "#c2b36c",
96 "#c3b46c", "#c5b56b", "#c6b66b", "#c7b76a", "#c8b86a", "#c9b869",
97 "#cab969", "#cbba68", "#ccbb68", "#cdbc67", "#cebd67", "#d0be66",
98 "#d1bf66", "#d2c065", "#d3c065", "#d4c164", "#d5c263", "#d6c363",
99 "#d7c462", "#d8c561", "#d9c661", "#dbc760", "#dcc860", "#ddc95f",
100 "#deca5e", "#dfcb5d", "#e0cb5d", "#e1cc5c", "#e3cd5b", "#e4ce5b",
101 "#e5cf5a", "#e6d059", "#e7d158", "#e8d257", "#e9d356", "#ebd456",
102 "#ecd555", "#edd654", "#eed753", "#efd852", "#f0d951", "#f1da50",
103 "#f3db4f", "#f4dc4e", "#f5dd4d", "#f6de4c", "#f7df4b", "#f9e049",
104 "#fae048", "#fbe147", "#fce246", "#fde345", "#ffe443", "#ffe542",
105 "#ffe642", "#ffe743", "#ffe844", "#ffe945"
106 },
107 false);
108
49 static void 109 static void
50 mapDiscrete(double norm, vector<QColor> &colours, double &r, double &g, double &b) 110 mapDiscrete(double norm, vector<QColor> &colours, double &r, double &g, double &b)
51 { 111 {
52 int n = int(colours.size()); 112 int n = int(colours.size());
53 double m = norm * (n-1); 113 double m = norm * (n-1);
59 r = c0.redF() * prop0 + c1.redF() * prop1; 119 r = c0.redF() * prop0 + c1.redF() * prop1;
60 g = c0.greenF() * prop0 + c1.greenF() * prop1; 120 g = c0.greenF() * prop0 + c1.greenF() * prop1;
61 b = c0.blueF() * prop0 + c1.blueF() * prop1; 121 b = c0.blueF() * prop0 + c1.blueF() * prop1;
62 } 122 }
63 123
64 ColourMapper::ColourMapper(int map, double min, double max) : 124 ColourMapper::ColourMapper(int map, bool inverted, double min, double max) :
65 m_map(map), 125 m_map(map),
126 m_inverted(inverted),
66 m_min(min), 127 m_min(min),
67 m_max(max) 128 m_max(max)
68 { 129 {
69 if (m_min == m_max) { 130 if (m_min == m_max) {
70 SVCERR << "WARNING: ColourMapper: min == max (== " << m_min 131 SVCERR << "WARNING: ColourMapper: min == max (== " << m_min
78 } 139 }
79 140
80 int 141 int
81 ColourMapper::getColourMapCount() 142 ColourMapper::getColourMapCount()
82 { 143 {
83 return 12; 144 return 15;
84 } 145 }
85 146
86 QString 147 QString
87 ColourMapper::getColourMapName(int n) 148 ColourMapper::getColourMapLabel(int n)
88 { 149 {
150 // When adding a map, be sure to also update getColourMapCount()
151
89 if (n >= getColourMapCount()) return QObject::tr("<unknown>"); 152 if (n >= getColourMapCount()) return QObject::tr("<unknown>");
90 StandardMap map = (StandardMap)n; 153 ColourMap map = (ColourMap)n;
91 154
92 switch (map) { 155 switch (map) {
93 case Green: return QObject::tr("Green"); 156 case Green: return QObject::tr("Green");
94 case WhiteOnBlack: return QObject::tr("White on Black"); 157 case WhiteOnBlack: return QObject::tr("White on Black");
95 case BlackOnWhite: return QObject::tr("Black on White"); 158 case BlackOnWhite: return QObject::tr("Black on White");
100 case FruitSalad: return QObject::tr("Fruit Salad"); 163 case FruitSalad: return QObject::tr("Fruit Salad");
101 case Banded: return QObject::tr("Banded"); 164 case Banded: return QObject::tr("Banded");
102 case Highlight: return QObject::tr("Highlight"); 165 case Highlight: return QObject::tr("Highlight");
103 case Printer: return QObject::tr("Printer"); 166 case Printer: return QObject::tr("Printer");
104 case HighGain: return QObject::tr("High Gain"); 167 case HighGain: return QObject::tr("High Gain");
168 case BlueOnBlack: return QObject::tr("Blue on Black");
169 case Cividis: return QObject::tr("Cividis");
170 case Magma: return QObject::tr("Magma");
105 } 171 }
106 172
107 return QObject::tr("<unknown>"); 173 return QObject::tr("<unknown>");
174 }
175
176 QString
177 ColourMapper::getColourMapId(int n)
178 {
179 if (n >= getColourMapCount()) return "<unknown>";
180 ColourMap map = (ColourMap)n;
181
182 switch (map) {
183 case Green: return "Green";
184 case WhiteOnBlack: return "White on Black";
185 case BlackOnWhite: return "Black on White";
186 case Cherry: return "Cherry";
187 case Wasp: return "Wasp";
188 case Ice: return "Ice";
189 case Sunset: return "Sunset";
190 case FruitSalad: return "Fruit Salad";
191 case Banded: return "Banded";
192 case Highlight: return "Highlight";
193 case Printer: return "Printer";
194 case HighGain: return "High Gain";
195 case BlueOnBlack: return "Blue on Black";
196 case Cividis: return "Cividis";
197 case Magma: return "Magma";
198 }
199
200 return "<unknown>";
201 }
202
203 int
204 ColourMapper::getColourMapById(QString id)
205 {
206 ColourMap map = (ColourMap)getColourMapCount();
207
208 if (id == "Green") { map = Green; }
209 else if (id == "White on Black") { map = WhiteOnBlack; }
210 else if (id == "Black on White") { map = BlackOnWhite; }
211 else if (id == "Cherry") { map = Cherry; }
212 else if (id == "Wasp") { map = Wasp; }
213 else if (id == "Ice") { map = Ice; }
214 else if (id == "Sunset") { map = Sunset; }
215 else if (id == "Fruit Salad") { map = FruitSalad; }
216 else if (id == "Banded") { map = Banded; }
217 else if (id == "Highlight") { map = Highlight; }
218 else if (id == "Printer") { map = Printer; }
219 else if (id == "High Gain") { map = HighGain; }
220 else if (id == "Blue on Black") { map = BlueOnBlack; }
221 else if (id == "Cividis") { map = Cividis; }
222 else if (id == "Magma") { map = Magma; }
223
224 if (map == (ColourMap)getColourMapCount()) {
225 return -1;
226 } else {
227 return int(map);
228 }
229 }
230
231 int
232 ColourMapper::getBackwardCompatibilityColourMap(int n)
233 {
234 /* Returned value should be an index into the series
235 * (Default/Green, Sunset, WhiteOnBlack, BlackOnWhite, RedOnBlue,
236 * YellowOnBlack, BlueOnBlack, FruitSalad, Banded, Highlight,
237 * Printer, HighGain). Minimum 0, maximum 11.
238 */
239
240 if (n >= getColourMapCount()) return 0;
241 ColourMap map = (ColourMap)n;
242
243 switch (map) {
244 case Green: return 0;
245 case WhiteOnBlack: return 2;
246 case BlackOnWhite: return 3;
247 case Cherry: return 4;
248 case Wasp: return 5;
249 case Ice: return 6;
250 case Sunset: return 1;
251 case FruitSalad: return 7;
252 case Banded: return 8;
253 case Highlight: return 9;
254 case Printer: return 10;
255 case HighGain: return 11;
256 case BlueOnBlack: return 6;
257 case Cividis: return 6;
258 case Magma: return 1;
259 }
260
261 return 0;
108 } 262 }
109 263
110 QColor 264 QColor
111 ColourMapper::map(double value) const 265 ColourMapper::map(double value) const
112 { 266 {
113 double norm = (value - m_min) / (m_max - m_min); 267 double norm = (value - m_min) / (m_max - m_min);
114 if (norm < 0.0) norm = 0.0; 268 if (norm < 0.0) norm = 0.0;
115 if (norm > 1.0) norm = 1.0; 269 if (norm > 1.0) norm = 1.0;
270
271 if (m_inverted) {
272 norm = 1.0 - norm;
273 }
116 274
117 double h = 0.0, s = 0.0, v = 0.0, r = 0.0, g = 0.0, b = 0.0; 275 double h = 0.0, s = 0.0, v = 0.0, r = 0.0, g = 0.0, b = 0.0;
118 bool hsv = true; 276 bool hsv = true;
119 277
120 double blue = 0.6666, pieslice = 0.3333; 278 double blue = 0.6666, pieslice = 0.3333;
121 279
122 if (m_map >= getColourMapCount()) return Qt::black; 280 if (m_map >= getColourMapCount()) return Qt::black;
123 StandardMap map = (StandardMap)m_map; 281 ColourMap map = (ColourMap)m_map;
124 282
125 switch (map) { 283 switch (map) {
126 284
127 case Green: 285 case Green:
128 h = blue - norm * 2.0 * pieslice; 286 h = blue - norm * 2.0 * pieslice;
147 305
148 case Wasp: 306 case Wasp:
149 h = 0.15; 307 h = 0.15;
150 s = 1.0; 308 s = 1.0;
151 v = norm; 309 v = norm;
310 break;
311
312 case BlueOnBlack:
313 h = blue;
314 s = 1.0;
315 v = norm * 2.0;
316 if (v > 1.0) {
317 v = 1.0;
318 s = 1.0 - (sqrt(norm) - 0.707) * 3.413;
319 if (s < 0.0) s = 0.0;
320 if (s > 1.0) s = 1.0;
321 }
152 break; 322 break;
153 323
154 case Sunset: 324 case Sunset:
155 r = (norm - 0.24) * 2.38; 325 r = (norm - 0.24) * 2.38;
156 if (r > 1.0) r = 1.0; 326 if (r > 1.0) r = 1.0;
235 break; 405 break;
236 406
237 case Ice: 407 case Ice:
238 hsv = false; 408 hsv = false;
239 mapDiscrete(norm, ice, r, g, b); 409 mapDiscrete(norm, ice, r, g, b);
410 break;
411
412 case Cividis:
413 hsv = false;
414 mapDiscrete(norm, cividis, r, g, b);
415 break;
416
417 case Magma:
418 hsv = false;
419 mapDiscrete(norm, magma, r, g, b);
420 break;
240 } 421 }
241 422
242 if (hsv) { 423 if (hsv) {
243 return QColor::fromHsvF(h, s, v); 424 return QColor::fromHsvF(h, s, v);
244 } else { 425 } else {
248 429
249 QColor 430 QColor
250 ColourMapper::getContrastingColour() const 431 ColourMapper::getContrastingColour() const
251 { 432 {
252 if (m_map >= getColourMapCount()) return Qt::white; 433 if (m_map >= getColourMapCount()) return Qt::white;
253 StandardMap map = (StandardMap)m_map; 434 ColourMap map = (ColourMap)m_map;
254 435
255 switch (map) { 436 switch (map) {
256 437
257 case Green: 438 case Green:
258 return QColor(255, 150, 50); 439 return QColor(255, 150, 50);
287 case Printer: 468 case Printer:
288 return Qt::red; 469 return Qt::red;
289 470
290 case HighGain: 471 case HighGain:
291 return Qt::red; 472 return Qt::red;
473
474 case BlueOnBlack:
475 return Qt::red;
476
477 case Cividis:
478 return Qt::white;
479
480 case Magma:
481 return Qt::white;
292 } 482 }
293 483
294 return Qt::white; 484 return Qt::white;
295 } 485 }
296 486
297 bool 487 bool
298 ColourMapper::hasLightBackground() const 488 ColourMapper::hasLightBackground() const
299 { 489 {
300 if (m_map >= getColourMapCount()) return false; 490 if (m_map >= getColourMapCount()) return false;
301 StandardMap map = (StandardMap)m_map; 491 ColourMap map = (ColourMap)m_map;
302 492
303 switch (map) { 493 switch (map) {
304 494
305 case BlackOnWhite: 495 case BlackOnWhite:
306 case Printer: 496 case Printer:
314 case Wasp: 504 case Wasp:
315 case Ice: 505 case Ice:
316 case FruitSalad: 506 case FruitSalad:
317 case Banded: 507 case Banded:
318 case Highlight: 508 case Highlight:
509 case BlueOnBlack:
510 case Cividis:
511 case Magma:
319 512
320 default: 513 default:
321 return false; 514 return false;
322 } 515 }
323 } 516 }