CTurtle
Loading...
Searching...
No Matches
CTurtle.hpp
1// MIT License
2//
3// Copyright (c) 2021 Jesse W. Walker
4//
5// Permission is hereby granted, free of charge, to any person obtaining a copy
6// of this software and associated documentation files (the "Software"), to deal
7// in the Software without restriction, including without limitation the rights
8// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
9// copies of the Software, and to permit persons to whom the Software is
10// furnished to do so, subject to the following conditions:
11//
12// The above copyright notice and this permission notice shall be included in all
13// copies or substantial portions of the Software.
14//
15// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
18// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
20// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
21// SOFTWARE.
22
110#pragma once
111
112#define CTURTLE_VERSION_MAJOR "1"
113#define CTURTLE_VERSION_MINOR "0"
114#define CTURTLE_VERSION_PATCH "4"
115#define CTURTLE_VERSION "v" CTURTLE_VERSION_MAJOR "." CTURTLE_VERSION_MINOR "." CTURTLE_VERSION_PATCH
116
117#ifdef CTURTLE_HEADLESS
118/*Include GIF utility when compiling headless mode.*/
119
120/* public domain, Simple, Minimalistic GIF writer - http://jonolick.com
121 *
122 * Quick Notes:
123 * Supports only 4 component input, alpha is currently ignored. (RGBX)
124 *
125 * Latest revisions:
126 * 1.00 (2015-11-03) initial release
127 *
128 * Basic usage:
129 * char *frame = new char[128*128*4]; // 4 component. RGBX format, where X is unused
130 * jo_gif_t gif = jo_gif_start("foo.gif", 128, 128, 0, 32);
131 * jo_gif_frame(&gif, frame, 4, false); // frame 1
132 * jo_gif_frame(&gif, frame, 4, false); // frame 2
133 * jo_gif_frame(&gif, frame, 4, false); // frame 3, ...
134 * jo_gif_end(&gif);
135 * */
136
137#ifndef JO_INCLUDE_GIF_H
138#define JO_INCLUDE_GIF_H
139
140#include <stdio.h>
141
142//Header edited to inline all GIF functionality to avoid re-definitions across compilation units
143//otherwise, left the same.
144
145typedef struct {
146 FILE *fp;
147 unsigned char palette[0x300];
148 short width, height, repeat;
149 int numColors, palSize;
150 int frame;
151} jo_gif_t;
152
153// width/height | the same for every frame
154// repeat | 0 = loop forever, 1 = loop once, etc...
155// palSize | must be power of 2 - 1. so, 255 not 256.
156inline jo_gif_t jo_gif_start(const char *filename, short width, short height, short repeat, int palSize);
157
158// gif | the state (returned from jo_gif_start)
159// rgba | the pixels
160// delayCsec | amount of time in between frames (in centiseconds)
161// localPalette | true if you want a unique palette generated for this frame (does not effect future frames)
162inline void jo_gif_frame(jo_gif_t *gif, unsigned char *rgba, short delayCsec, bool localPalette);
163
164// gif | the state (returned from jo_gif_start)
165inline void jo_gif_end(jo_gif_t *gif);
166
167#if defined(_MSC_VER) && _MSC_VER >= 0x1400
168#define _CRT_SECURE_NO_WARNINGS // suppress warnings about fopen()
169#endif
170
171#include <stdlib.h>
172#include <memory.h>
173#include <math.h>
174
175// Based on NeuQuant algorithm
176inline void jo_gif_quantize(unsigned char *rgba, int rgbaSize, int sample, unsigned char *map, int numColors) {
177 // defs for freq and bias
178 const int intbiasshift = 16; /* bias for fractions */
179 const int intbias = (((int) 1) << intbiasshift);
180 const int gammashift = 10; /* gamma = 1024 */
181 const int betashift = 10;
182 const int beta = (intbias >> betashift); /* beta = 1/1024 */
183 const int betagamma = (intbias << (gammashift - betashift));
184
185 // defs for decreasing radius factor
186 const int radiusbiasshift = 6; /* at 32.0 biased by 6 bits */
187 const int radiusbias = (((int) 1) << radiusbiasshift);
188 const int radiusdec = 30; /* factor of 1/30 each cycle */
189
190 // defs for decreasing alpha factor
191 const int alphabiasshift = 10; /* alpha starts at 1.0 */
192 const int initalpha = (((int) 1) << alphabiasshift);
193
194 // radbias and alpharadbias used for radpower calculation
195 const int radbiasshift = 8;
196 const int radbias = (((int) 1) << radbiasshift);
197 const int alpharadbshift = (alphabiasshift + radbiasshift);
198 const int alpharadbias = (((int) 1) << alpharadbshift);
199
200 sample = sample < 1 ? 1 : sample > 30 ? 30 : sample;
201 int network[256][3];
202 int bias[256] = {}, freq[256];
203 for(int i = 0; i < numColors; ++i) {
204 // Put nurons evenly through the luminance spectrum.
205 network[i][0] = network[i][1] = network[i][2] = (i << 12) / numColors;
206 freq[i] = intbias / numColors;
207 }
208 // Learn
209 {
210 const int primes[5] = {499, 491, 487, 503};
211 int step = 4;
212 for(int i = 0; i < 4; ++i) {
213 if(rgbaSize > primes[i] * 4 && (rgbaSize % primes[i])) { // TODO/Error? primes[i]*4?
214 step = primes[i] * 4;
215 }
216 }
217 sample = step == 4 ? 1 : sample;
218
219 int alphadec = 30 + ((sample - 1) / 3);
220 int samplepixels = rgbaSize / (4 * sample);
221 int delta = samplepixels / 100;
222 int alpha = initalpha;
223 delta = delta == 0 ? 1 : delta;
224
225 int radius = (numColors >> 3) * radiusbias;
226 int rad = radius >> radiusbiasshift;
227 rad = rad <= 1 ? 0 : rad;
228 int radSq = rad*rad;
229 int radpower[32];
230 for (int i = 0; i < rad; i++) {
231 radpower[i] = alpha * (((radSq - i * i) * radbias) / radSq);
232 }
233
234 // Randomly walk through the pixels and relax neurons to the "optimal" target.
235 for(int i = 0, pix = 0; i < samplepixels;) {
236 int r = rgba[pix + 0] << 4;
237 int g = rgba[pix + 1] << 4;
238 int b = rgba[pix + 2] << 4;
239 int j = -1;
240 {
241 // finds closest neuron (min dist) and updates freq
242 // finds best neuron (min dist-bias) and returns position
243 // for frequently chosen neurons, freq[k] is high and bias[k] is negative
244 // bias[k] = gamma*((1/numColors)-freq[k])
245
246 int bestd = 0x7FFFFFFF, bestbiasd = 0x7FFFFFFF, bestpos = -1;
247 for (int k = 0; k < numColors; k++) {
248 int *n = network[k];
249 int dist = abs(n[0] - r) + abs(n[1] - g) + abs(n[2] - b);
250 if (dist < bestd) {
251 bestd = dist;
252 bestpos = k;
253 }
254 int biasdist = dist - ((bias[k]) >> (intbiasshift - 4));
255 if (biasdist < bestbiasd) {
256 bestbiasd = biasdist;
257 j = k;
258 }
259 int betafreq = freq[k] >> betashift;
260 freq[k] -= betafreq;
261 bias[k] += betafreq << gammashift;
262 }
263 freq[bestpos] += beta;
264 bias[bestpos] -= betagamma;
265 }
266
267 // Move neuron j towards biased (b,g,r) by factor alpha
268 network[j][0] -= (network[j][0] - r) * alpha / initalpha;
269 network[j][1] -= (network[j][1] - g) * alpha / initalpha;
270 network[j][2] -= (network[j][2] - b) * alpha / initalpha;
271 if (rad != 0) {
272 // Move adjacent neurons by precomputed alpha*(1-((i-j)^2/[r]^2)) in radpower[|i-j|]
273 int lo = j - rad;
274 lo = lo < -1 ? -1 : lo;
275 int hi = j + rad;
276 hi = hi > numColors ? numColors : hi;
277 for(int jj = j+1, m=1; jj < hi; ++jj) {
278 int a = radpower[m++];
279 network[jj][0] -= (network[jj][0] - r) * a / alpharadbias;
280 network[jj][1] -= (network[jj][1] - g) * a / alpharadbias;
281 network[jj][2] -= (network[jj][2] - b) * a / alpharadbias;
282 }
283 for(int k = j-1, m=1; k > lo; --k) {
284 int a = radpower[m++];
285 network[k][0] -= (network[k][0] - r) * a / alpharadbias;
286 network[k][1] -= (network[k][1] - g) * a / alpharadbias;
287 network[k][2] -= (network[k][2] - b) * a / alpharadbias;
288 }
289 }
290
291 pix += step;
292 pix = pix >= rgbaSize ? pix - rgbaSize : pix;
293
294 // every 1% of the image, move less over the following iterations.
295 if(++i % delta == 0) {
296 alpha -= alpha / alphadec;
297 radius -= radius / radiusdec;
298 rad = radius >> radiusbiasshift;
299 rad = rad <= 1 ? 0 : rad;
300 radSq = rad*rad;
301 for (j = 0; j < rad; j++) {
302 radpower[j] = alpha * ((radSq - j * j) * radbias / radSq);
303 }
304 }
305 }
306 }
307 // Unbias network to give byte values 0..255
308 for (int i = 0; i < numColors; i++) {
309 map[i*3+0] = network[i][0] >>= 4;
310 map[i*3+1] = network[i][1] >>= 4;
311 map[i*3+2] = network[i][2] >>= 4;
312 }
313}
314
315typedef struct {
316 FILE *fp;
317 int numBits;
318 unsigned char buf[256];
319 unsigned char idx;
320 unsigned tmp;
321 int outBits;
322 int curBits;
323} jo_gif_lzw_t;
324
325inline void jo_gif_lzw_write(jo_gif_lzw_t *s, int code) {
326 s->outBits |= code << s->curBits;
327 s->curBits += s->numBits;
328 while(s->curBits >= 8) {
329 s->buf[s->idx++] = s->outBits & 255;
330 s->outBits >>= 8;
331 s->curBits -= 8;
332 if (s->idx >= 255) {
333 putc(s->idx, s->fp);
334 fwrite(s->buf, s->idx, 1, s->fp);
335 s->idx = 0;
336 }
337 }
338}
339
340inline void jo_gif_lzw_encode(unsigned char *in, int len, FILE *fp) {
341 jo_gif_lzw_t state = {fp, 9};
342 int maxcode = 511;
343
344 // Note: 30k stack space for dictionary =|
345 const int hashSize = 5003;
346 short codetab[hashSize];
347 int hashTbl[hashSize];
348 memset(hashTbl, 0xFF, sizeof(hashTbl));
349
350 jo_gif_lzw_write(&state, 0x100);
351
352 int free_ent = 0x102;
353 int ent = *in++;
354 CONTINUE:
355 while (--len) {
356 int c = *in++;
357 int fcode = (c << 12) + ent;
358 int key = (c << 4) ^ ent; // xor hashing
359 while(hashTbl[key] >= 0) {
360 if(hashTbl[key] == fcode) {
361 ent = codetab[key];
362 goto CONTINUE;
363 }
364 ++key;
365 key = key >= hashSize ? key - hashSize : key;
366 }
367 jo_gif_lzw_write(&state, ent);
368 ent = c;
369 if(free_ent < 4096) {
370 if(free_ent > maxcode) {
371 ++state.numBits;
372 if(state.numBits == 12) {
373 maxcode = 4096;
374 } else {
375 maxcode = (1<<state.numBits)-1;
376 }
377 }
378 codetab[key] = free_ent++;
379 hashTbl[key] = fcode;
380 } else {
381 memset(hashTbl, 0xFF, sizeof(hashTbl));
382 free_ent = 0x102;
383 jo_gif_lzw_write(&state, 0x100);
384 state.numBits = 9;
385 maxcode = 511;
386 }
387 }
388 jo_gif_lzw_write(&state, ent);
389 jo_gif_lzw_write(&state, 0x101);
390 jo_gif_lzw_write(&state, 0);
391 if(state.idx) {
392 putc(state.idx, fp);
393 fwrite(state.buf, state.idx, 1, fp);
394 }
395}
396
397inline int jo_gif_clamp(int a, int b, int c) { return a < b ? b : a > c ? c : a; }
398
399jo_gif_t jo_gif_start(const char *filename, short width, short height, short repeat, int numColors) {
400 numColors = numColors > 255 ? 255 : numColors < 2 ? 2 : numColors;
401 jo_gif_t gif = {};
402 gif.width = width;
403 gif.height = height;
404 gif.repeat = repeat;
405 gif.numColors = numColors;
406 gif.palSize = log2(numColors);
407
408 gif.fp = fopen(filename, "wb");
409 if(!gif.fp) {
410 printf("Error: Could not WriteGif to %s\n", filename);
411 return gif;
412 }
413
414 fwrite("GIF89a", 6, 1, gif.fp);
415 // Logical Screen Descriptor
416 fwrite(&gif.width, 2, 1, gif.fp);
417 fwrite(&gif.height, 2, 1, gif.fp);
418 putc(0xF0 | gif.palSize, gif.fp);
419 fwrite("\x00\x00", 2, 1, gif.fp); // bg color index (unused), aspect ratio
420 return gif;
421}
422
423inline void jo_gif_frame(jo_gif_t *gif, unsigned char * rgba, short delayCsec, bool localPalette) {
424 if(!gif->fp) {
425 return;
426 }
427 short width = gif->width;
428 short height = gif->height;
429 int size = width * height;
430
431 unsigned char localPalTbl[0x300];
432 unsigned char *palette = gif->frame == 0 || !localPalette ? gif->palette : localPalTbl;
433 if(gif->frame == 0 || localPalette) {
434 jo_gif_quantize(rgba, size*4, 1, palette, gif->numColors);
435 }
436
437 unsigned char *indexedPixels = (unsigned char *)malloc(size);
438 {
439 unsigned char *ditheredPixels = (unsigned char*)malloc(size*4);
440 memcpy(ditheredPixels, rgba, size*4);
441 for(int k = 0; k < size*4; k+=4) {
442 int rgb[3] = { ditheredPixels[k+0], ditheredPixels[k+1], ditheredPixels[k+2] };
443 int bestd = 0x7FFFFFFF, best = -1;
444 // TODO: exhaustive search. do something better.
445 for(int i = 0; i < gif->numColors; ++i) {
446 int bb = palette[i*3+0]-rgb[0];
447 int gg = palette[i*3+1]-rgb[1];
448 int rr = palette[i*3+2]-rgb[2];
449 int d = bb*bb + gg*gg + rr*rr;
450 if(d < bestd) {
451 bestd = d;
452 best = i;
453 }
454 }
455 indexedPixels[k/4] = best;
456 int diff[3] = { ditheredPixels[k+0] - palette[indexedPixels[k/4]*3+0], ditheredPixels[k+1] - palette[indexedPixels[k/4]*3+1], ditheredPixels[k+2] - palette[indexedPixels[k/4]*3+2] };
457 // Floyd-Steinberg Error Diffusion
458 // TODO: Use something better -- http://caca.zoy.org/study/part3.html
459 if(k+4 < size*4) {
460 ditheredPixels[k+4+0] = (unsigned char)jo_gif_clamp(ditheredPixels[k+4+0]+(diff[0]*7/16), 0, 255);
461 ditheredPixels[k+4+1] = (unsigned char)jo_gif_clamp(ditheredPixels[k+4+1]+(diff[1]*7/16), 0, 255);
462 ditheredPixels[k+4+2] = (unsigned char)jo_gif_clamp(ditheredPixels[k+4+2]+(diff[2]*7/16), 0, 255);
463 }
464 if(k+width*4+4 < size*4) {
465 for(int i = 0; i < 3; ++i) {
466 ditheredPixels[k-4+width*4+i] = (unsigned char)jo_gif_clamp(ditheredPixels[k-4+width*4+i]+(diff[i]*3/16), 0, 255);
467 ditheredPixels[k+width*4+i] = (unsigned char)jo_gif_clamp(ditheredPixels[k+width*4+i]+(diff[i]*5/16), 0, 255);
468 ditheredPixels[k+width*4+4+i] = (unsigned char)jo_gif_clamp(ditheredPixels[k+width*4+4+i]+(diff[i]*1/16), 0, 255);
469 }
470 }
471 }
472 free(ditheredPixels);
473 }
474 if(gif->frame == 0) {
475 // Global Color Table
476 fwrite(palette, 3*(1<<(gif->palSize+1)), 1, gif->fp);
477 if(gif->repeat >= 0) {
478 // Netscape Extension
479 fwrite("\x21\xff\x0bNETSCAPE2.0\x03\x01", 16, 1, gif->fp);
480 fwrite(&gif->repeat, 2, 1, gif->fp); // loop count (extra iterations, 0=repeat forever)
481 putc(0, gif->fp); // block terminator
482 }
483 }
484 // Graphic Control Extension
485 fwrite("\x21\xf9\x04\x00", 4, 1, gif->fp);
486 fwrite(&delayCsec, 2, 1, gif->fp); // delayCsec x 1/100 sec
487 fwrite("\x00\x00", 2, 1, gif->fp); // transparent color index (first byte), currently unused
488 // Image Descriptor
489 fwrite("\x2c\x00\x00\x00\x00", 5, 1, gif->fp); // header, x,y
490 fwrite(&width, 2, 1, gif->fp);
491 fwrite(&height, 2, 1, gif->fp);
492 if (gif->frame == 0 || !localPalette) {
493 putc(0, gif->fp);
494 } else {
495 putc(0x80|gif->palSize, gif->fp );
496 fwrite(palette, 3*(1<<(gif->palSize+1)), 1, gif->fp);
497 }
498 putc(8, gif->fp); // block terminator
499 jo_gif_lzw_encode(indexedPixels, size, gif->fp);
500 putc(0, gif->fp); // block terminator
501 ++gif->frame;
502 free(indexedPixels);
503}
504
505inline void jo_gif_end(jo_gif_t *gif) {
506 if(!gif->fp) {
507 return;
508 }
509 putc(0x3b, gif->fp); // gif trailer
510 fclose(gif->fp);
511}
512
513#endif /*JO_INCLUDE_GIF_H*/
514#endif /*CTURTLE_HEADLESS*/
515
516//Automatic linking when operating under MSVC
517//If linking errors occur when compiling on Non-MSVC,
518//Make sure you link X11 and PThread when using Unix-Like environments, when NOT using headless mode.
519#ifndef CTURTLE_MSVC_NO_AUTOLINK
520#ifdef _MSC_VER
521/*Automatically link to the necessary windows libraries while under MSVC.*/
522#pragma comment(lib, "kernel32.lib")
523#pragma comment(lib, "gdi32.lib")
524#endif
525#endif
526
527//MSVC 2017 doesn't seem to like defining M_PI. We define it ourselves
528//when compiling under VisualC++.
529#ifdef _MSC_VER
530#ifndef M_PI
531#define M_PI 3.14159265358979323846264338327950288
532#endif
533#endif
534
535//When using headless, simply pre-define CTURTLE_CONFIG_HEADLESS.
536//This disables the InteractiveTurtleScreen.
537//GIF utility is included at the top of the file when under headless mode.
538
539#ifdef CTURTLE_HEADLESS
540//Optional define to disable HTML Base64 Image output
541 //#define CTURTLE_HEADLESS_NO_HTML
542
543 //Disable CImg Display
544 #define cimg_display 0
545
546 //Define default width and height.
547 #ifndef CTURTLE_HEADLESS_WIDTH
548 #define CTURTLE_HEADLESS_WIDTH 400
549 #endif
550
551 #ifndef CTURTLE_HEADLESS_HEIGHT
552 #define CTURTLE_HEADLESS_HEIGHT 300
553 #endif
554
555 #ifndef CTURTLE_HEADLESS_SAVEDIR
556 #define CTURTLE_HEADLESS_SAVEDIR "./cturtle.gif"
557 #endif
558#endif
559
560#ifdef _MSC_VER
561//Disable MSVC warnings for CImg. Irrelevant to project.
562#include <CodeAnalysis/Warnings.h>
563#pragma warning(push, 0)
564#pragma warning (disable : ALL_CODE_ANALYSIS_WARNINGS)
565#include "CImg.h"
566#pragma warning(pop)
567#else
568#include "CImg.h"
569#endif
570
571#include <memory> //For smart pointers.
572#include <list> //For the ever-useful list data structure.
573#include <unordered_map>//For named color and keys.
574#include <chrono> //For time management for movement animations and callbacks.
575#include <random> //For random color creation.
576#include <functional> //For event function callbacks.
577#include <tuple> //Used for CompoundShapes
578#include <cstring> //For memcpy
579#include <vector> //For Polygon point storage
580#include <cmath> //For rounding, etc
581#include <algorithm> //For std::count
582#include <array> //For Transform storage.
583#include <string> //Strings...
584#include <cstdint> //For well-defined integer types.
585#include <thread> //For the event thread
586#include <mutex> //Mutex object for event thread synchronization.
587#include <stdexcept> //For standard exceptions.
588#include <fstream> //For GIF base-64 encoding to write the file out.
589#include <iostream> //For GIF reading.
590#include <sstream> //used for base64 encoding.
591
592//See https://github.com/mvorbrodt/blog/blob/master/src/base64.hpp for original source.
593//The below has been modified to use unsigned characters to avoid signed->unsigned->signed fiddling.
594namespace base64{
595 static constexpr unsigned char kEncodeLookup[] = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
596 static constexpr unsigned char kPadCharacter = '=';
597
604 inline std::string encode(const std::vector<unsigned char>& input)
605 {
606 std::stringstream encoded;
607 std::uint32_t temp{};
608
609 auto it = input.begin();
610
611 for(std::size_t i = 0; i < input.size() / 3; ++i)
612 {
613 temp = (*it++) << 16;
614 temp += (*it++) << 8;
615 temp += (*it++);
616 encoded << kEncodeLookup[(temp & 0x00FC0000) >> 18];
617 encoded << kEncodeLookup[(temp & 0x0003F000) >> 12];
618 encoded << kEncodeLookup[(temp & 0x00000FC0) >> 6 ];
619 encoded << kEncodeLookup[(temp & 0x0000003F) ];
620 }
621
622 switch(input.size() % 3)
623 {
624 case 1:
625 temp = (*it++) << 16;
626 encoded << kEncodeLookup[(temp & 0x00FC0000) >> 18];
627 encoded << kEncodeLookup[(temp & 0x0003F000) >> 12];
628 encoded << kPadCharacter << kPadCharacter;
629 break;
630 case 2:
631 temp = (*it++) << 16;
632 temp += (*it++) << 8;
633 encoded << kEncodeLookup[(temp & 0x00FC0000) >> 18];
634 encoded << kEncodeLookup[(temp & 0x0003F000) >> 12];
635 encoded << kEncodeLookup[(temp & 0x00000FC0) >> 6 ];
636 encoded << kPadCharacter;
637 break;
638 }
639
640 return encoded.str();
641 }
642}
643
644namespace cturtle {
646 namespace cimg = cimg_library;
648 typedef cimg::CImg<uint8_t> Image;
649
650 namespace detail {
651 // SECTION: COLORS
652 // In an effort to make this package easily distributable,
653 // Colors are defined as packed integers in the header file.
654 //===========================
655 typedef uint32_t color_int_t; //Alpha value is extra, serves as padding
656 typedef uint64_t time_t;
657
665 inline constexpr color_int_t resolveColorInt(uint8_t r, uint8_t g, uint8_t b) noexcept{
666 return ((r & 0xFF) << 16) + ((g & 0xFF) << 8) + (b & 0xFF);
667 }
668
674 inline void resolveColorComp(color_int_t pack, uint8_t& r, uint8_t& g, uint8_t& b) {
675 r = (pack & 0x00FF0000) >> 16; //Red
676 g = (pack & 0x0000FF00) >> 8; //Green
677 b = (pack & 0x000000FF); // >> 0; //Blue
678 }
679
684 inline time_t epochTime() {
685 return std::chrono::system_clock::now().time_since_epoch() / std::chrono::milliseconds(1);
686 }
687
690 inline void sleep(long ms) {
691 if (ms <= 0)
692 return;
693 std::this_thread::sleep_for(std::chrono::milliseconds(ms));
694 }
695
696 namespace col {
697 const detail::color_int_t alice_blue = detail::resolveColorInt(240, 248, 255);
698 const detail::color_int_t AliceBlue = detail::resolveColorInt(240, 248, 255);
699 const detail::color_int_t antique_white = detail::resolveColorInt(250, 235, 215);
700 const detail::color_int_t AntiqueWhite = detail::resolveColorInt(250, 235, 215);
701 const detail::color_int_t AntiqueWhite1 = detail::resolveColorInt(255, 239, 219);
702 const detail::color_int_t AntiqueWhite2 = detail::resolveColorInt(238, 223, 204);
703 const detail::color_int_t AntiqueWhite3 = detail::resolveColorInt(205, 192, 176);
704 const detail::color_int_t AntiqueWhite4 = detail::resolveColorInt(139, 131, 120);
705 const detail::color_int_t aquamarine = detail::resolveColorInt(127, 255, 212);
706 const detail::color_int_t aquamarine1 = detail::resolveColorInt(127, 255, 212);
707 const detail::color_int_t aquamarine2 = detail::resolveColorInt(118, 238, 198);
708 const detail::color_int_t aquamarine3 = detail::resolveColorInt(102, 205, 170);
709 const detail::color_int_t aquamarine4 = detail::resolveColorInt(69, 139, 116);
710 const detail::color_int_t azure = detail::resolveColorInt(240, 255, 255);
711 const detail::color_int_t azure1 = detail::resolveColorInt(240, 255, 255);
712 const detail::color_int_t azure2 = detail::resolveColorInt(224, 238, 238);
713 const detail::color_int_t azure3 = detail::resolveColorInt(193, 205, 205);
714 const detail::color_int_t azure4 = detail::resolveColorInt(131, 139, 139);
715 const detail::color_int_t beige = detail::resolveColorInt(245, 245, 220);
716 const detail::color_int_t bisque = detail::resolveColorInt(255, 228, 196);
717 const detail::color_int_t bisque1 = detail::resolveColorInt(255, 228, 196);
718 const detail::color_int_t bisque2 = detail::resolveColorInt(238, 213, 183);
719 const detail::color_int_t bisque3 = detail::resolveColorInt(205, 183, 158);
720 const detail::color_int_t bisque4 = detail::resolveColorInt(139, 125, 107);
721 const detail::color_int_t black = detail::resolveColorInt(0, 0, 0);
722 const detail::color_int_t blanched_almond = detail::resolveColorInt(255, 235, 205);
723 const detail::color_int_t BlanchedAlmond = detail::resolveColorInt(255, 235, 205);
724 const detail::color_int_t blue = detail::resolveColorInt(0, 0, 255);
725 const detail::color_int_t blue_violet = detail::resolveColorInt(138, 43, 226);
726 const detail::color_int_t blue1 = detail::resolveColorInt(0, 0, 255);
727 const detail::color_int_t blue2 = detail::resolveColorInt(0, 0, 238);
728 const detail::color_int_t blue3 = detail::resolveColorInt(0, 0, 205);
729 const detail::color_int_t blue4 = detail::resolveColorInt(0, 0, 139);
730 const detail::color_int_t BlueViolet = detail::resolveColorInt(138, 43, 226);
731 const detail::color_int_t brown = detail::resolveColorInt(165, 42, 42);
732 const detail::color_int_t brown1 = detail::resolveColorInt(255, 64, 64);
733 const detail::color_int_t brown2 = detail::resolveColorInt(238, 59, 59);
734 const detail::color_int_t brown3 = detail::resolveColorInt(205, 51, 51);
735 const detail::color_int_t brown4 = detail::resolveColorInt(139, 35, 35);
736 const detail::color_int_t burlywood = detail::resolveColorInt(222, 184, 135);
737 const detail::color_int_t burlywood1 = detail::resolveColorInt(255, 211, 155);
738 const detail::color_int_t burlywood2 = detail::resolveColorInt(238, 197, 145);
739 const detail::color_int_t burlywood3 = detail::resolveColorInt(205, 170, 125);
740 const detail::color_int_t burlywood4 = detail::resolveColorInt(139, 115, 85);
741 const detail::color_int_t cadet_blue = detail::resolveColorInt(95, 158, 160);
742 const detail::color_int_t CadetBlue = detail::resolveColorInt(95, 158, 160);
743 const detail::color_int_t CadetBlue1 = detail::resolveColorInt(152, 245, 255);
744 const detail::color_int_t CadetBlue2 = detail::resolveColorInt(142, 229, 238);
745 const detail::color_int_t CadetBlue3 = detail::resolveColorInt(122, 197, 205);
746 const detail::color_int_t CadetBlue4 = detail::resolveColorInt(83, 134, 139);
747 const detail::color_int_t chartreuse = detail::resolveColorInt(127, 255, 0);
748 const detail::color_int_t chartreuse1 = detail::resolveColorInt(127, 255, 0);
749 const detail::color_int_t chartreuse2 = detail::resolveColorInt(118, 238, 0);
750 const detail::color_int_t chartreuse3 = detail::resolveColorInt(102, 205, 0);
751 const detail::color_int_t chartreuse4 = detail::resolveColorInt(69, 139, 0);
752 const detail::color_int_t chocolate = detail::resolveColorInt(210, 105, 30);
753 const detail::color_int_t chocolate1 = detail::resolveColorInt(255, 127, 36);
754 const detail::color_int_t chocolate2 = detail::resolveColorInt(238, 118, 33);
755 const detail::color_int_t chocolate3 = detail::resolveColorInt(205, 102, 29);
756 const detail::color_int_t chocolate4 = detail::resolveColorInt(139, 69, 19);
757 const detail::color_int_t coral = detail::resolveColorInt(255, 127, 80);
758 const detail::color_int_t coral1 = detail::resolveColorInt(255, 114, 86);
759 const detail::color_int_t coral2 = detail::resolveColorInt(238, 106, 80);
760 const detail::color_int_t coral3 = detail::resolveColorInt(205, 91, 69);
761 const detail::color_int_t coral4 = detail::resolveColorInt(139, 62, 47);
762 const detail::color_int_t cornflower_blue = detail::resolveColorInt(100, 149, 237);
763 const detail::color_int_t CornflowerBlue = detail::resolveColorInt(100, 149, 237);
764 const detail::color_int_t cornsilk = detail::resolveColorInt(255, 248, 220);
765 const detail::color_int_t cornsilk1 = detail::resolveColorInt(255, 248, 220);
766 const detail::color_int_t cornsilk2 = detail::resolveColorInt(238, 232, 205);
767 const detail::color_int_t cornsilk3 = detail::resolveColorInt(205, 200, 177);
768 const detail::color_int_t cornsilk4 = detail::resolveColorInt(139, 136, 120);
769 const detail::color_int_t cyan = detail::resolveColorInt(0, 255, 255);
770 const detail::color_int_t cyan1 = detail::resolveColorInt(0, 255, 255);
771 const detail::color_int_t cyan2 = detail::resolveColorInt(0, 238, 238);
772 const detail::color_int_t cyan3 = detail::resolveColorInt(0, 205, 205);
773 const detail::color_int_t cyan4 = detail::resolveColorInt(0, 139, 139);
774 const detail::color_int_t dark_blue = detail::resolveColorInt(0, 0, 139);
775 const detail::color_int_t dark_cyan = detail::resolveColorInt(0, 139, 139);
776 const detail::color_int_t dark_goldenrod = detail::resolveColorInt(184, 134, 11);
777 const detail::color_int_t dark_gray = detail::resolveColorInt(169, 169, 169);
778 const detail::color_int_t dark_green = detail::resolveColorInt(0, 100, 0);
779 const detail::color_int_t dark_grey = detail::resolveColorInt(169, 169, 169);
780 const detail::color_int_t dark_khaki = detail::resolveColorInt(189, 183, 107);
781 const detail::color_int_t dark_magenta = detail::resolveColorInt(139, 0, 139);
782 const detail::color_int_t dark_olive_green = detail::resolveColorInt(85, 107, 47);
783 const detail::color_int_t dark_orange = detail::resolveColorInt(255, 140, 0);
784 const detail::color_int_t dark_orchid = detail::resolveColorInt(153, 50, 204);
785 const detail::color_int_t dark_red = detail::resolveColorInt(139, 0, 0);
786 const detail::color_int_t dark_salmon = detail::resolveColorInt(233, 150, 122);
787 const detail::color_int_t dark_sea_green = detail::resolveColorInt(143, 188, 143);
788 const detail::color_int_t dark_slate_blue = detail::resolveColorInt(72, 61, 139);
789 const detail::color_int_t dark_slate_gray = detail::resolveColorInt(47, 79, 79);
790 const detail::color_int_t dark_slate_grey = detail::resolveColorInt(47, 79, 79);
791 const detail::color_int_t dark_turquoise = detail::resolveColorInt(0, 206, 209);
792 const detail::color_int_t dark_violet = detail::resolveColorInt(148, 0, 211);
793 const detail::color_int_t DarkBlue = detail::resolveColorInt(0, 0, 139);
794 const detail::color_int_t DarkCyan = detail::resolveColorInt(0, 139, 139);
795 const detail::color_int_t DarkGoldenrod = detail::resolveColorInt(184, 134, 11);
796 const detail::color_int_t DarkGoldenrod1 = detail::resolveColorInt(255, 185, 15);
797 const detail::color_int_t DarkGoldenrod2 = detail::resolveColorInt(238, 173, 14);
798 const detail::color_int_t DarkGoldenrod3 = detail::resolveColorInt(205, 149, 12);
799 const detail::color_int_t DarkGoldenrod4 = detail::resolveColorInt(139, 101, 8);
800 const detail::color_int_t DarkGray = detail::resolveColorInt(169, 169, 169);
801 const detail::color_int_t DarkGreen = detail::resolveColorInt(0, 100, 0);
802 const detail::color_int_t DarkGrey = detail::resolveColorInt(169, 169, 169);
803 const detail::color_int_t DarkKhaki = detail::resolveColorInt(189, 183, 107);
804 const detail::color_int_t DarkMagenta = detail::resolveColorInt(139, 0, 139);
805 const detail::color_int_t DarkOliveGreen = detail::resolveColorInt(85, 107, 47);
806 const detail::color_int_t DarkOliveGreen1 = detail::resolveColorInt(202, 255, 112);
807 const detail::color_int_t DarkOliveGreen2 = detail::resolveColorInt(188, 238, 104);
808 const detail::color_int_t DarkOliveGreen3 = detail::resolveColorInt(162, 205, 90);
809 const detail::color_int_t DarkOliveGreen4 = detail::resolveColorInt(110, 139, 61);
810 const detail::color_int_t DarkOrange = detail::resolveColorInt(255, 140, 0);
811 const detail::color_int_t DarkOrange1 = detail::resolveColorInt(255, 127, 0);
812 const detail::color_int_t DarkOrange2 = detail::resolveColorInt(238, 118, 0);
813 const detail::color_int_t DarkOrange3 = detail::resolveColorInt(205, 102, 0);
814 const detail::color_int_t DarkOrange4 = detail::resolveColorInt(139, 69, 0);
815 const detail::color_int_t DarkOrchid = detail::resolveColorInt(153, 50, 204);
816 const detail::color_int_t DarkOrchid1 = detail::resolveColorInt(191, 62, 255);
817 const detail::color_int_t DarkOrchid2 = detail::resolveColorInt(178, 58, 238);
818 const detail::color_int_t DarkOrchid3 = detail::resolveColorInt(154, 50, 205);
819 const detail::color_int_t DarkOrchid4 = detail::resolveColorInt(104, 34, 139);
820 const detail::color_int_t DarkRed = detail::resolveColorInt(139, 0, 0);
821 const detail::color_int_t DarkSalmon = detail::resolveColorInt(233, 150, 122);
822 const detail::color_int_t DarkSeaGreen = detail::resolveColorInt(143, 188, 143);
823 const detail::color_int_t DarkSeaGreen1 = detail::resolveColorInt(193, 255, 193);
824 const detail::color_int_t DarkSeaGreen2 = detail::resolveColorInt(180, 238, 180);
825 const detail::color_int_t DarkSeaGreen3 = detail::resolveColorInt(155, 205, 155);
826 const detail::color_int_t DarkSeaGreen4 = detail::resolveColorInt(105, 139, 105);
827 const detail::color_int_t DarkSlateBlue = detail::resolveColorInt(72, 61, 139);
828 const detail::color_int_t DarkSlateGray = detail::resolveColorInt(47, 79, 79);
829 const detail::color_int_t DarkSlateGray1 = detail::resolveColorInt(151, 255, 255);
830 const detail::color_int_t DarkSlateGray2 = detail::resolveColorInt(141, 238, 238);
831 const detail::color_int_t DarkSlateGray3 = detail::resolveColorInt(121, 205, 205);
832 const detail::color_int_t DarkSlateGray4 = detail::resolveColorInt(82, 139, 139);
833 const detail::color_int_t DarkSlateGrey = detail::resolveColorInt(47, 79, 79);
834 const detail::color_int_t DarkTurquoise = detail::resolveColorInt(0, 206, 209);
835 const detail::color_int_t DarkViolet = detail::resolveColorInt(148, 0, 211);
836 const detail::color_int_t deep_pink = detail::resolveColorInt(255, 20, 147);
837 const detail::color_int_t deep_sky_blue = detail::resolveColorInt(0, 191, 255);
838 const detail::color_int_t DeepPink = detail::resolveColorInt(255, 20, 147);
839 const detail::color_int_t DeepPink1 = detail::resolveColorInt(255, 20, 147);
840 const detail::color_int_t DeepPink2 = detail::resolveColorInt(238, 18, 137);
841 const detail::color_int_t DeepPink3 = detail::resolveColorInt(205, 16, 118);
842 const detail::color_int_t DeepPink4 = detail::resolveColorInt(139, 10, 80);
843 const detail::color_int_t DeepSkyBlue = detail::resolveColorInt(0, 191, 255);
844 const detail::color_int_t DeepSkyBlue1 = detail::resolveColorInt(0, 191, 255);
845 const detail::color_int_t DeepSkyBlue2 = detail::resolveColorInt(0, 178, 238);
846 const detail::color_int_t DeepSkyBlue3 = detail::resolveColorInt(0, 154, 205);
847 const detail::color_int_t DeepSkyBlue4 = detail::resolveColorInt(0, 104, 139);
848 const detail::color_int_t dim_gray = detail::resolveColorInt(105, 105, 105);
849 const detail::color_int_t dim_grey = detail::resolveColorInt(105, 105, 105);
850 const detail::color_int_t DimGray = detail::resolveColorInt(105, 105, 105);
851 const detail::color_int_t DimGrey = detail::resolveColorInt(105, 105, 105);
852 const detail::color_int_t dodger_blue = detail::resolveColorInt(30, 144, 255);
853 const detail::color_int_t DodgerBlue = detail::resolveColorInt(30, 144, 255);
854 const detail::color_int_t DodgerBlue1 = detail::resolveColorInt(30, 144, 255);
855 const detail::color_int_t DodgerBlue2 = detail::resolveColorInt(28, 134, 238);
856 const detail::color_int_t DodgerBlue3 = detail::resolveColorInt(24, 116, 205);
857 const detail::color_int_t DodgerBlue4 = detail::resolveColorInt(16, 78, 139);
858 const detail::color_int_t firebrick = detail::resolveColorInt(178, 34, 34);
859 const detail::color_int_t firebrick1 = detail::resolveColorInt(255, 48, 48);
860 const detail::color_int_t firebrick2 = detail::resolveColorInt(238, 44, 44);
861 const detail::color_int_t firebrick3 = detail::resolveColorInt(205, 38, 38);
862 const detail::color_int_t firebrick4 = detail::resolveColorInt(139, 26, 26);
863 const detail::color_int_t floral_white = detail::resolveColorInt(255, 250, 240);
864 const detail::color_int_t FloralWhite = detail::resolveColorInt(255, 250, 240);
865 const detail::color_int_t forest_green = detail::resolveColorInt(34, 139, 34);
866 const detail::color_int_t ForestGreen = detail::resolveColorInt(34, 139, 34);
867 const detail::color_int_t gainsboro = detail::resolveColorInt(220, 220, 220);
868 const detail::color_int_t ghost_white = detail::resolveColorInt(248, 248, 255);
869 const detail::color_int_t GhostWhite = detail::resolveColorInt(248, 248, 255);
870 const detail::color_int_t gold = detail::resolveColorInt(255, 215, 0);
871 const detail::color_int_t gold1 = detail::resolveColorInt(255, 215, 0);
872 const detail::color_int_t gold2 = detail::resolveColorInt(238, 201, 0);
873 const detail::color_int_t gold3 = detail::resolveColorInt(205, 173, 0);
874 const detail::color_int_t gold4 = detail::resolveColorInt(139, 117, 0);
875 const detail::color_int_t goldenrod = detail::resolveColorInt(218, 165, 32);
876 const detail::color_int_t goldenrod1 = detail::resolveColorInt(255, 193, 37);
877 const detail::color_int_t goldenrod2 = detail::resolveColorInt(238, 180, 34);
878 const detail::color_int_t goldenrod3 = detail::resolveColorInt(205, 155, 29);
879 const detail::color_int_t goldenrod4 = detail::resolveColorInt(139, 105, 20);
880 const detail::color_int_t gray = detail::resolveColorInt(190, 190, 190);
881 const detail::color_int_t gray0 = detail::resolveColorInt(0, 0, 0);
882 const detail::color_int_t gray1 = detail::resolveColorInt(3, 3, 3);
883 const detail::color_int_t gray2 = detail::resolveColorInt(5, 5, 5);
884 const detail::color_int_t gray3 = detail::resolveColorInt(8, 8, 8);
885 const detail::color_int_t gray4 = detail::resolveColorInt(10, 10, 10);
886 const detail::color_int_t gray5 = detail::resolveColorInt(13, 13, 13);
887 const detail::color_int_t gray6 = detail::resolveColorInt(15, 15, 15);
888 const detail::color_int_t gray7 = detail::resolveColorInt(18, 18, 18);
889 const detail::color_int_t gray8 = detail::resolveColorInt(20, 20, 20);
890 const detail::color_int_t gray9 = detail::resolveColorInt(23, 23, 23);
891 const detail::color_int_t gray10 = detail::resolveColorInt(26, 26, 26);
892 const detail::color_int_t gray11 = detail::resolveColorInt(28, 28, 28);
893 const detail::color_int_t gray12 = detail::resolveColorInt(31, 31, 31);
894 const detail::color_int_t gray13 = detail::resolveColorInt(33, 33, 33);
895 const detail::color_int_t gray14 = detail::resolveColorInt(36, 36, 36);
896 const detail::color_int_t gray15 = detail::resolveColorInt(38, 38, 38);
897 const detail::color_int_t gray16 = detail::resolveColorInt(41, 41, 41);
898 const detail::color_int_t gray17 = detail::resolveColorInt(43, 43, 43);
899 const detail::color_int_t gray18 = detail::resolveColorInt(46, 46, 46);
900 const detail::color_int_t gray19 = detail::resolveColorInt(48, 48, 48);
901 const detail::color_int_t gray20 = detail::resolveColorInt(51, 51, 51);
902 const detail::color_int_t gray21 = detail::resolveColorInt(54, 54, 54);
903 const detail::color_int_t gray22 = detail::resolveColorInt(56, 56, 56);
904 const detail::color_int_t gray23 = detail::resolveColorInt(59, 59, 59);
905 const detail::color_int_t gray24 = detail::resolveColorInt(61, 61, 61);
906 const detail::color_int_t gray25 = detail::resolveColorInt(64, 64, 64);
907 const detail::color_int_t gray26 = detail::resolveColorInt(66, 66, 66);
908 const detail::color_int_t gray27 = detail::resolveColorInt(69, 69, 69);
909 const detail::color_int_t gray28 = detail::resolveColorInt(71, 71, 71);
910 const detail::color_int_t gray29 = detail::resolveColorInt(74, 74, 74);
911 const detail::color_int_t gray30 = detail::resolveColorInt(77, 77, 77);
912 const detail::color_int_t gray31 = detail::resolveColorInt(79, 79, 79);
913 const detail::color_int_t gray32 = detail::resolveColorInt(82, 82, 82);
914 const detail::color_int_t gray33 = detail::resolveColorInt(84, 84, 84);
915 const detail::color_int_t gray34 = detail::resolveColorInt(87, 87, 87);
916 const detail::color_int_t gray35 = detail::resolveColorInt(89, 89, 89);
917 const detail::color_int_t gray36 = detail::resolveColorInt(92, 92, 92);
918 const detail::color_int_t gray37 = detail::resolveColorInt(94, 94, 94);
919 const detail::color_int_t gray38 = detail::resolveColorInt(97, 97, 97);
920 const detail::color_int_t gray39 = detail::resolveColorInt(99, 99, 99);
921 const detail::color_int_t gray40 = detail::resolveColorInt(102, 102, 102);
922 const detail::color_int_t gray41 = detail::resolveColorInt(105, 105, 105);
923 const detail::color_int_t gray42 = detail::resolveColorInt(107, 107, 107);
924 const detail::color_int_t gray43 = detail::resolveColorInt(110, 110, 110);
925 const detail::color_int_t gray44 = detail::resolveColorInt(112, 112, 112);
926 const detail::color_int_t gray45 = detail::resolveColorInt(115, 115, 115);
927 const detail::color_int_t gray46 = detail::resolveColorInt(117, 117, 117);
928 const detail::color_int_t gray47 = detail::resolveColorInt(120, 120, 120);
929 const detail::color_int_t gray48 = detail::resolveColorInt(122, 122, 122);
930 const detail::color_int_t gray49 = detail::resolveColorInt(125, 125, 125);
931 const detail::color_int_t gray50 = detail::resolveColorInt(127, 127, 127);
932 const detail::color_int_t gray51 = detail::resolveColorInt(130, 130, 130);
933 const detail::color_int_t gray52 = detail::resolveColorInt(133, 133, 133);
934 const detail::color_int_t gray53 = detail::resolveColorInt(135, 135, 135);
935 const detail::color_int_t gray54 = detail::resolveColorInt(138, 138, 138);
936 const detail::color_int_t gray55 = detail::resolveColorInt(140, 140, 140);
937 const detail::color_int_t gray56 = detail::resolveColorInt(143, 143, 143);
938 const detail::color_int_t gray57 = detail::resolveColorInt(145, 145, 145);
939 const detail::color_int_t gray58 = detail::resolveColorInt(148, 148, 148);
940 const detail::color_int_t gray59 = detail::resolveColorInt(150, 150, 150);
941 const detail::color_int_t gray60 = detail::resolveColorInt(153, 153, 153);
942 const detail::color_int_t gray61 = detail::resolveColorInt(156, 156, 156);
943 const detail::color_int_t gray62 = detail::resolveColorInt(158, 158, 158);
944 const detail::color_int_t gray63 = detail::resolveColorInt(161, 161, 161);
945 const detail::color_int_t gray64 = detail::resolveColorInt(163, 163, 163);
946 const detail::color_int_t gray65 = detail::resolveColorInt(166, 166, 166);
947 const detail::color_int_t gray66 = detail::resolveColorInt(168, 168, 168);
948 const detail::color_int_t gray67 = detail::resolveColorInt(171, 171, 171);
949 const detail::color_int_t gray68 = detail::resolveColorInt(173, 173, 173);
950 const detail::color_int_t gray69 = detail::resolveColorInt(176, 176, 176);
951 const detail::color_int_t gray70 = detail::resolveColorInt(179, 179, 179);
952 const detail::color_int_t gray71 = detail::resolveColorInt(181, 181, 181);
953 const detail::color_int_t gray72 = detail::resolveColorInt(184, 184, 184);
954 const detail::color_int_t gray73 = detail::resolveColorInt(186, 186, 186);
955 const detail::color_int_t gray74 = detail::resolveColorInt(189, 189, 189);
956 const detail::color_int_t gray75 = detail::resolveColorInt(191, 191, 191);
957 const detail::color_int_t gray76 = detail::resolveColorInt(194, 194, 194);
958 const detail::color_int_t gray77 = detail::resolveColorInt(196, 196, 196);
959 const detail::color_int_t gray78 = detail::resolveColorInt(199, 199, 199);
960 const detail::color_int_t gray79 = detail::resolveColorInt(201, 201, 201);
961 const detail::color_int_t gray80 = detail::resolveColorInt(204, 204, 204);
962 const detail::color_int_t gray81 = detail::resolveColorInt(207, 207, 207);
963 const detail::color_int_t gray82 = detail::resolveColorInt(209, 209, 209);
964 const detail::color_int_t gray83 = detail::resolveColorInt(212, 212, 212);
965 const detail::color_int_t gray84 = detail::resolveColorInt(214, 214, 214);
966 const detail::color_int_t gray85 = detail::resolveColorInt(217, 217, 217);
967 const detail::color_int_t gray86 = detail::resolveColorInt(219, 219, 219);
968 const detail::color_int_t gray87 = detail::resolveColorInt(222, 222, 222);
969 const detail::color_int_t gray88 = detail::resolveColorInt(224, 224, 224);
970 const detail::color_int_t gray89 = detail::resolveColorInt(227, 227, 227);
971 const detail::color_int_t gray90 = detail::resolveColorInt(229, 229, 229);
972 const detail::color_int_t gray91 = detail::resolveColorInt(232, 232, 232);
973 const detail::color_int_t gray92 = detail::resolveColorInt(235, 235, 235);
974 const detail::color_int_t gray93 = detail::resolveColorInt(237, 237, 237);
975 const detail::color_int_t gray94 = detail::resolveColorInt(240, 240, 240);
976 const detail::color_int_t gray95 = detail::resolveColorInt(242, 242, 242);
977 const detail::color_int_t gray96 = detail::resolveColorInt(245, 245, 245);
978 const detail::color_int_t gray97 = detail::resolveColorInt(247, 247, 247);
979 const detail::color_int_t gray98 = detail::resolveColorInt(250, 250, 250);
980 const detail::color_int_t gray99 = detail::resolveColorInt(252, 252, 252);
981 const detail::color_int_t gray100 = detail::resolveColorInt(255, 255, 255);
982 const detail::color_int_t green = detail::resolveColorInt(0, 255, 0);
983 const detail::color_int_t green_yellow = detail::resolveColorInt(173, 255, 47);
984 const detail::color_int_t green1 = detail::resolveColorInt(0, 255, 0);
985 const detail::color_int_t green2 = detail::resolveColorInt(0, 238, 0);
986 const detail::color_int_t green3 = detail::resolveColorInt(0, 205, 0);
987 const detail::color_int_t green4 = detail::resolveColorInt(0, 139, 0);
988 const detail::color_int_t GreenYellow = detail::resolveColorInt(173, 255, 47);
989 const detail::color_int_t grey = detail::resolveColorInt(190, 190, 190);
990 const detail::color_int_t grey0 = detail::resolveColorInt(0, 0, 0);
991 const detail::color_int_t grey1 = detail::resolveColorInt(3, 3, 3);
992 const detail::color_int_t grey2 = detail::resolveColorInt(5, 5, 5);
993 const detail::color_int_t grey3 = detail::resolveColorInt(8, 8, 8);
994 const detail::color_int_t grey4 = detail::resolveColorInt(10, 10, 10);
995 const detail::color_int_t grey5 = detail::resolveColorInt(13, 13, 13);
996 const detail::color_int_t grey6 = detail::resolveColorInt(15, 15, 15);
997 const detail::color_int_t grey7 = detail::resolveColorInt(18, 18, 18);
998 const detail::color_int_t grey8 = detail::resolveColorInt(20, 20, 20);
999 const detail::color_int_t grey9 = detail::resolveColorInt(23, 23, 23);
1000 const detail::color_int_t grey10 = detail::resolveColorInt(26, 26, 26);
1001 const detail::color_int_t grey11 = detail::resolveColorInt(28, 28, 28);
1002 const detail::color_int_t grey12 = detail::resolveColorInt(31, 31, 31);
1003 const detail::color_int_t grey13 = detail::resolveColorInt(33, 33, 33);
1004 const detail::color_int_t grey14 = detail::resolveColorInt(36, 36, 36);
1005 const detail::color_int_t grey15 = detail::resolveColorInt(38, 38, 38);
1006 const detail::color_int_t grey16 = detail::resolveColorInt(41, 41, 41);
1007 const detail::color_int_t grey17 = detail::resolveColorInt(43, 43, 43);
1008 const detail::color_int_t grey18 = detail::resolveColorInt(46, 46, 46);
1009 const detail::color_int_t grey19 = detail::resolveColorInt(48, 48, 48);
1010 const detail::color_int_t grey20 = detail::resolveColorInt(51, 51, 51);
1011 const detail::color_int_t grey21 = detail::resolveColorInt(54, 54, 54);
1012 const detail::color_int_t grey22 = detail::resolveColorInt(56, 56, 56);
1013 const detail::color_int_t grey23 = detail::resolveColorInt(59, 59, 59);
1014 const detail::color_int_t grey24 = detail::resolveColorInt(61, 61, 61);
1015 const detail::color_int_t grey25 = detail::resolveColorInt(64, 64, 64);
1016 const detail::color_int_t grey26 = detail::resolveColorInt(66, 66, 66);
1017 const detail::color_int_t grey27 = detail::resolveColorInt(69, 69, 69);
1018 const detail::color_int_t grey28 = detail::resolveColorInt(71, 71, 71);
1019 const detail::color_int_t grey29 = detail::resolveColorInt(74, 74, 74);
1020 const detail::color_int_t grey30 = detail::resolveColorInt(77, 77, 77);
1021 const detail::color_int_t grey31 = detail::resolveColorInt(79, 79, 79);
1022 const detail::color_int_t grey32 = detail::resolveColorInt(82, 82, 82);
1023 const detail::color_int_t grey33 = detail::resolveColorInt(84, 84, 84);
1024 const detail::color_int_t grey34 = detail::resolveColorInt(87, 87, 87);
1025 const detail::color_int_t grey35 = detail::resolveColorInt(89, 89, 89);
1026 const detail::color_int_t grey36 = detail::resolveColorInt(92, 92, 92);
1027 const detail::color_int_t grey37 = detail::resolveColorInt(94, 94, 94);
1028 const detail::color_int_t grey38 = detail::resolveColorInt(97, 97, 97);
1029 const detail::color_int_t grey39 = detail::resolveColorInt(99, 99, 99);
1030 const detail::color_int_t grey40 = detail::resolveColorInt(102, 102, 102);
1031 const detail::color_int_t grey41 = detail::resolveColorInt(105, 105, 105);
1032 const detail::color_int_t grey42 = detail::resolveColorInt(107, 107, 107);
1033 const detail::color_int_t grey43 = detail::resolveColorInt(110, 110, 110);
1034 const detail::color_int_t grey44 = detail::resolveColorInt(112, 112, 112);
1035 const detail::color_int_t grey45 = detail::resolveColorInt(115, 115, 115);
1036 const detail::color_int_t grey46 = detail::resolveColorInt(117, 117, 117);
1037 const detail::color_int_t grey47 = detail::resolveColorInt(120, 120, 120);
1038 const detail::color_int_t grey48 = detail::resolveColorInt(122, 122, 122);
1039 const detail::color_int_t grey49 = detail::resolveColorInt(125, 125, 125);
1040 const detail::color_int_t grey50 = detail::resolveColorInt(127, 127, 127);
1041 const detail::color_int_t grey51 = detail::resolveColorInt(130, 130, 130);
1042 const detail::color_int_t grey52 = detail::resolveColorInt(133, 133, 133);
1043 const detail::color_int_t grey53 = detail::resolveColorInt(135, 135, 135);
1044 const detail::color_int_t grey54 = detail::resolveColorInt(138, 138, 138);
1045 const detail::color_int_t grey55 = detail::resolveColorInt(140, 140, 140);
1046 const detail::color_int_t grey56 = detail::resolveColorInt(143, 143, 143);
1047 const detail::color_int_t grey57 = detail::resolveColorInt(145, 145, 145);
1048 const detail::color_int_t grey58 = detail::resolveColorInt(148, 148, 148);
1049 const detail::color_int_t grey59 = detail::resolveColorInt(150, 150, 150);
1050 const detail::color_int_t grey60 = detail::resolveColorInt(153, 153, 153);
1051 const detail::color_int_t grey61 = detail::resolveColorInt(156, 156, 156);
1052 const detail::color_int_t grey62 = detail::resolveColorInt(158, 158, 158);
1053 const detail::color_int_t grey63 = detail::resolveColorInt(161, 161, 161);
1054 const detail::color_int_t grey64 = detail::resolveColorInt(163, 163, 163);
1055 const detail::color_int_t grey65 = detail::resolveColorInt(166, 166, 166);
1056 const detail::color_int_t grey66 = detail::resolveColorInt(168, 168, 168);
1057 const detail::color_int_t grey67 = detail::resolveColorInt(171, 171, 171);
1058 const detail::color_int_t grey68 = detail::resolveColorInt(173, 173, 173);
1059 const detail::color_int_t grey69 = detail::resolveColorInt(176, 176, 176);
1060 const detail::color_int_t grey70 = detail::resolveColorInt(179, 179, 179);
1061 const detail::color_int_t grey71 = detail::resolveColorInt(181, 181, 181);
1062 const detail::color_int_t grey72 = detail::resolveColorInt(184, 184, 184);
1063 const detail::color_int_t grey73 = detail::resolveColorInt(186, 186, 186);
1064 const detail::color_int_t grey74 = detail::resolveColorInt(189, 189, 189);
1065 const detail::color_int_t grey75 = detail::resolveColorInt(191, 191, 191);
1066 const detail::color_int_t grey76 = detail::resolveColorInt(194, 194, 194);
1067 const detail::color_int_t grey77 = detail::resolveColorInt(196, 196, 196);
1068 const detail::color_int_t grey78 = detail::resolveColorInt(199, 199, 199);
1069 const detail::color_int_t grey79 = detail::resolveColorInt(201, 201, 201);
1070 const detail::color_int_t grey80 = detail::resolveColorInt(204, 204, 204);
1071 const detail::color_int_t grey81 = detail::resolveColorInt(207, 207, 207);
1072 const detail::color_int_t grey82 = detail::resolveColorInt(209, 209, 209);
1073 const detail::color_int_t grey83 = detail::resolveColorInt(212, 212, 212);
1074 const detail::color_int_t grey84 = detail::resolveColorInt(214, 214, 214);
1075 const detail::color_int_t grey85 = detail::resolveColorInt(217, 217, 217);
1076 const detail::color_int_t grey86 = detail::resolveColorInt(219, 219, 219);
1077 const detail::color_int_t grey87 = detail::resolveColorInt(222, 222, 222);
1078 const detail::color_int_t grey88 = detail::resolveColorInt(224, 224, 224);
1079 const detail::color_int_t grey89 = detail::resolveColorInt(227, 227, 227);
1080 const detail::color_int_t grey90 = detail::resolveColorInt(229, 229, 229);
1081 const detail::color_int_t grey91 = detail::resolveColorInt(232, 232, 232);
1082 const detail::color_int_t grey92 = detail::resolveColorInt(235, 235, 235);
1083 const detail::color_int_t grey93 = detail::resolveColorInt(237, 237, 237);
1084 const detail::color_int_t grey94 = detail::resolveColorInt(240, 240, 240);
1085 const detail::color_int_t grey95 = detail::resolveColorInt(242, 242, 242);
1086 const detail::color_int_t grey96 = detail::resolveColorInt(245, 245, 245);
1087 const detail::color_int_t grey97 = detail::resolveColorInt(247, 247, 247);
1088 const detail::color_int_t grey98 = detail::resolveColorInt(250, 250, 250);
1089 const detail::color_int_t grey99 = detail::resolveColorInt(252, 252, 252);
1090 const detail::color_int_t grey100 = detail::resolveColorInt(255, 255, 255);
1091 const detail::color_int_t honeydew = detail::resolveColorInt(240, 255, 240);
1092 const detail::color_int_t honeydew1 = detail::resolveColorInt(240, 255, 240);
1093 const detail::color_int_t honeydew2 = detail::resolveColorInt(224, 238, 224);
1094 const detail::color_int_t honeydew3 = detail::resolveColorInt(193, 205, 193);
1095 const detail::color_int_t honeydew4 = detail::resolveColorInt(131, 139, 131);
1096 const detail::color_int_t hot_pink = detail::resolveColorInt(255, 105, 180);
1097 const detail::color_int_t HotPink = detail::resolveColorInt(255, 105, 180);
1098 const detail::color_int_t HotPink1 = detail::resolveColorInt(255, 110, 180);
1099 const detail::color_int_t HotPink2 = detail::resolveColorInt(238, 106, 167);
1100 const detail::color_int_t HotPink3 = detail::resolveColorInt(205, 96, 144);
1101 const detail::color_int_t HotPink4 = detail::resolveColorInt(139, 58, 98);
1102 const detail::color_int_t indian_red = detail::resolveColorInt(205, 92, 92);
1103 const detail::color_int_t IndianRed = detail::resolveColorInt(205, 92, 92);
1104 const detail::color_int_t IndianRed1 = detail::resolveColorInt(255, 106, 106);
1105 const detail::color_int_t IndianRed2 = detail::resolveColorInt(238, 99, 99);
1106 const detail::color_int_t IndianRed3 = detail::resolveColorInt(205, 85, 85);
1107 const detail::color_int_t IndianRed4 = detail::resolveColorInt(139, 58, 58);
1108 const detail::color_int_t ivory = detail::resolveColorInt(255, 255, 240);
1109 const detail::color_int_t ivory1 = detail::resolveColorInt(255, 255, 240);
1110 const detail::color_int_t ivory2 = detail::resolveColorInt(238, 238, 224);
1111 const detail::color_int_t ivory3 = detail::resolveColorInt(205, 205, 193);
1112 const detail::color_int_t ivory4 = detail::resolveColorInt(139, 139, 131);
1113 const detail::color_int_t khaki = detail::resolveColorInt(240, 230, 140);
1114 const detail::color_int_t khaki1 = detail::resolveColorInt(255, 246, 143);
1115 const detail::color_int_t khaki2 = detail::resolveColorInt(238, 230, 133);
1116 const detail::color_int_t khaki3 = detail::resolveColorInt(205, 198, 115);
1117 const detail::color_int_t khaki4 = detail::resolveColorInt(139, 134, 78);
1118 const detail::color_int_t lavender = detail::resolveColorInt(230, 230, 250);
1119 const detail::color_int_t lavender_blush = detail::resolveColorInt(255, 240, 245);
1120 const detail::color_int_t LavenderBlush = detail::resolveColorInt(255, 240, 245);
1121 const detail::color_int_t LavenderBlush1 = detail::resolveColorInt(255, 240, 245);
1122 const detail::color_int_t LavenderBlush2 = detail::resolveColorInt(238, 224, 229);
1123 const detail::color_int_t LavenderBlush3 = detail::resolveColorInt(205, 193, 197);
1124 const detail::color_int_t LavenderBlush4 = detail::resolveColorInt(139, 131, 134);
1125 const detail::color_int_t lawn_green = detail::resolveColorInt(124, 252, 0);
1126 const detail::color_int_t LawnGreen = detail::resolveColorInt(124, 252, 0);
1127 const detail::color_int_t lemon_chiffon = detail::resolveColorInt(255, 250, 205);
1128 const detail::color_int_t LemonChiffon = detail::resolveColorInt(255, 250, 205);
1129 const detail::color_int_t LemonChiffon1 = detail::resolveColorInt(255, 250, 205);
1130 const detail::color_int_t LemonChiffon2 = detail::resolveColorInt(238, 233, 191);
1131 const detail::color_int_t LemonChiffon3 = detail::resolveColorInt(205, 201, 165);
1132 const detail::color_int_t LemonChiffon4 = detail::resolveColorInt(139, 137, 112);
1133 const detail::color_int_t light_blue = detail::resolveColorInt(173, 216, 230);
1134 const detail::color_int_t light_coral = detail::resolveColorInt(240, 128, 128);
1135 const detail::color_int_t light_cyan = detail::resolveColorInt(224, 255, 255);
1136 const detail::color_int_t light_goldenrod = detail::resolveColorInt(238, 221, 130);
1137 const detail::color_int_t light_goldenrod_yellow = detail::resolveColorInt(250, 250, 210);
1138 const detail::color_int_t light_gray = detail::resolveColorInt(211, 211, 211);
1139 const detail::color_int_t light_green = detail::resolveColorInt(144, 238, 144);
1140 const detail::color_int_t light_grey = detail::resolveColorInt(211, 211, 211);
1141 const detail::color_int_t light_pink = detail::resolveColorInt(255, 182, 193);
1142 const detail::color_int_t light_salmon = detail::resolveColorInt(255, 160, 122);
1143 const detail::color_int_t light_sea_green = detail::resolveColorInt(32, 178, 170);
1144 const detail::color_int_t light_sky_blue = detail::resolveColorInt(135, 206, 250);
1145 const detail::color_int_t light_slate_blue = detail::resolveColorInt(132, 112, 255);
1146 const detail::color_int_t light_slate_gray = detail::resolveColorInt(119, 136, 153);
1147 const detail::color_int_t light_slate_grey = detail::resolveColorInt(119, 136, 153);
1148 const detail::color_int_t light_steel_blue = detail::resolveColorInt(176, 196, 222);
1149 const detail::color_int_t light_yellow = detail::resolveColorInt(255, 255, 224);
1150 const detail::color_int_t LightBlue = detail::resolveColorInt(173, 216, 230);
1151 const detail::color_int_t LightBlue1 = detail::resolveColorInt(191, 239, 255);
1152 const detail::color_int_t LightBlue2 = detail::resolveColorInt(178, 223, 238);
1153 const detail::color_int_t LightBlue3 = detail::resolveColorInt(154, 192, 205);
1154 const detail::color_int_t LightBlue4 = detail::resolveColorInt(104, 131, 139);
1155 const detail::color_int_t LightCoral = detail::resolveColorInt(240, 128, 128);
1156 const detail::color_int_t LightCyan = detail::resolveColorInt(224, 255, 255);
1157 const detail::color_int_t LightCyan1 = detail::resolveColorInt(224, 255, 255);
1158 const detail::color_int_t LightCyan2 = detail::resolveColorInt(209, 238, 238);
1159 const detail::color_int_t LightCyan3 = detail::resolveColorInt(180, 205, 205);
1160 const detail::color_int_t LightCyan4 = detail::resolveColorInt(122, 139, 139);
1161 const detail::color_int_t LightGoldenrod = detail::resolveColorInt(238, 221, 130);
1162 const detail::color_int_t LightGoldenrod1 = detail::resolveColorInt(255, 236, 139);
1163 const detail::color_int_t LightGoldenrod2 = detail::resolveColorInt(238, 220, 130);
1164 const detail::color_int_t LightGoldenrod3 = detail::resolveColorInt(205, 190, 112);
1165 const detail::color_int_t LightGoldenrod4 = detail::resolveColorInt(139, 129, 76);
1166 const detail::color_int_t LightGoldenrodYellow = detail::resolveColorInt(250, 250, 210);
1167 const detail::color_int_t LightGray = detail::resolveColorInt(211, 211, 211);
1168 const detail::color_int_t LightGreen = detail::resolveColorInt(144, 238, 144);
1169 const detail::color_int_t LightGrey = detail::resolveColorInt(211, 211, 211);
1170 const detail::color_int_t LightPink = detail::resolveColorInt(255, 182, 193);
1171 const detail::color_int_t LightPink1 = detail::resolveColorInt(255, 174, 185);
1172 const detail::color_int_t LightPink2 = detail::resolveColorInt(238, 162, 173);
1173 const detail::color_int_t LightPink3 = detail::resolveColorInt(205, 140, 149);
1174 const detail::color_int_t LightPink4 = detail::resolveColorInt(139, 95, 101);
1175 const detail::color_int_t LightSalmon = detail::resolveColorInt(255, 160, 122);
1176 const detail::color_int_t LightSalmon1 = detail::resolveColorInt(255, 160, 122);
1177 const detail::color_int_t LightSalmon2 = detail::resolveColorInt(238, 149, 114);
1178 const detail::color_int_t LightSalmon3 = detail::resolveColorInt(205, 129, 98);
1179 const detail::color_int_t LightSalmon4 = detail::resolveColorInt(139, 87, 66);
1180 const detail::color_int_t LightSeaGreen = detail::resolveColorInt(32, 178, 170);
1181 const detail::color_int_t LightSkyBlue = detail::resolveColorInt(135, 206, 250);
1182 const detail::color_int_t LightSkyBlue1 = detail::resolveColorInt(176, 226, 255);
1183 const detail::color_int_t LightSkyBlue2 = detail::resolveColorInt(164, 211, 238);
1184 const detail::color_int_t LightSkyBlue3 = detail::resolveColorInt(141, 182, 205);
1185 const detail::color_int_t LightSkyBlue4 = detail::resolveColorInt(96, 123, 139);
1186 const detail::color_int_t LightSlateBlue = detail::resolveColorInt(132, 112, 255);
1187 const detail::color_int_t LightSlateGray = detail::resolveColorInt(119, 136, 153);
1188 const detail::color_int_t LightSlateGrey = detail::resolveColorInt(119, 136, 153);
1189 const detail::color_int_t LightSteelBlue = detail::resolveColorInt(176, 196, 222);
1190 const detail::color_int_t LightSteelBlue1 = detail::resolveColorInt(202, 225, 255);
1191 const detail::color_int_t LightSteelBlue2 = detail::resolveColorInt(188, 210, 238);
1192 const detail::color_int_t LightSteelBlue3 = detail::resolveColorInt(162, 181, 205);
1193 const detail::color_int_t LightSteelBlue4 = detail::resolveColorInt(110, 123, 139);
1194 const detail::color_int_t LightYellow = detail::resolveColorInt(255, 255, 224);
1195 const detail::color_int_t LightYellow1 = detail::resolveColorInt(255, 255, 224);
1196 const detail::color_int_t LightYellow2 = detail::resolveColorInt(238, 238, 209);
1197 const detail::color_int_t LightYellow3 = detail::resolveColorInt(205, 205, 180);
1198 const detail::color_int_t LightYellow4 = detail::resolveColorInt(139, 139, 122);
1199 const detail::color_int_t lime_green = detail::resolveColorInt(50, 205, 50);
1200 const detail::color_int_t LimeGreen = detail::resolveColorInt(50, 205, 50);
1201 const detail::color_int_t linen = detail::resolveColorInt(250, 240, 230);
1202 const detail::color_int_t magenta = detail::resolveColorInt(255, 0, 255);
1203 const detail::color_int_t magenta1 = detail::resolveColorInt(255, 0, 255);
1204 const detail::color_int_t magenta2 = detail::resolveColorInt(238, 0, 238);
1205 const detail::color_int_t magenta3 = detail::resolveColorInt(205, 0, 205);
1206 const detail::color_int_t magenta4 = detail::resolveColorInt(139, 0, 139);
1207 const detail::color_int_t maroon = detail::resolveColorInt(176, 48, 96);
1208 const detail::color_int_t maroon1 = detail::resolveColorInt(255, 52, 179);
1209 const detail::color_int_t maroon2 = detail::resolveColorInt(238, 48, 167);
1210 const detail::color_int_t maroon3 = detail::resolveColorInt(205, 41, 144);
1211 const detail::color_int_t maroon4 = detail::resolveColorInt(139, 28, 98);
1212 const detail::color_int_t medium_aquamarine = detail::resolveColorInt(102, 205, 170);
1213 const detail::color_int_t medium_blue = detail::resolveColorInt(0, 0, 205);
1214 const detail::color_int_t medium_orchid = detail::resolveColorInt(186, 85, 211);
1215 const detail::color_int_t medium_purple = detail::resolveColorInt(147, 112, 219);
1216 const detail::color_int_t medium_sea_green = detail::resolveColorInt(60, 179, 113);
1217 const detail::color_int_t medium_slate_blue = detail::resolveColorInt(123, 104, 238);
1218 const detail::color_int_t medium_spring_green = detail::resolveColorInt(0, 250, 154);
1219 const detail::color_int_t medium_turquoise = detail::resolveColorInt(72, 209, 204);
1220 const detail::color_int_t medium_violet_red = detail::resolveColorInt(199, 21, 133);
1221 const detail::color_int_t MediumAquamarine = detail::resolveColorInt(102, 205, 170);
1222 const detail::color_int_t MediumBlue = detail::resolveColorInt(0, 0, 205);
1223 const detail::color_int_t MediumOrchid = detail::resolveColorInt(186, 85, 211);
1224 const detail::color_int_t MediumOrchid1 = detail::resolveColorInt(224, 102, 255);
1225 const detail::color_int_t MediumOrchid2 = detail::resolveColorInt(209, 95, 238);
1226 const detail::color_int_t MediumOrchid3 = detail::resolveColorInt(180, 82, 205);
1227 const detail::color_int_t MediumOrchid4 = detail::resolveColorInt(122, 55, 139);
1228 const detail::color_int_t MediumPurple = detail::resolveColorInt(147, 112, 219);
1229 const detail::color_int_t MediumPurple1 = detail::resolveColorInt(171, 130, 255);
1230 const detail::color_int_t MediumPurple2 = detail::resolveColorInt(159, 121, 238);
1231 const detail::color_int_t MediumPurple3 = detail::resolveColorInt(137, 104, 205);
1232 const detail::color_int_t MediumPurple4 = detail::resolveColorInt(93, 71, 139);
1233 const detail::color_int_t MediumSeaGreen = detail::resolveColorInt(60, 179, 113);
1234 const detail::color_int_t MediumSlateBlue = detail::resolveColorInt(123, 104, 238);
1235 const detail::color_int_t MediumSpringGreen = detail::resolveColorInt(0, 250, 154);
1236 const detail::color_int_t MediumTurquoise = detail::resolveColorInt(72, 209, 204);
1237 const detail::color_int_t MediumVioletRed = detail::resolveColorInt(199, 21, 133);
1238 const detail::color_int_t midnight_blue = detail::resolveColorInt(25, 25, 112);
1239 const detail::color_int_t MidnightBlue = detail::resolveColorInt(25, 25, 112);
1240 const detail::color_int_t mint_cream = detail::resolveColorInt(245, 255, 250);
1241 const detail::color_int_t MintCream = detail::resolveColorInt(245, 255, 250);
1242 const detail::color_int_t misty_rose = detail::resolveColorInt(255, 228, 225);
1243 const detail::color_int_t MistyRose = detail::resolveColorInt(255, 228, 225);
1244 const detail::color_int_t MistyRose1 = detail::resolveColorInt(255, 228, 225);
1245 const detail::color_int_t MistyRose2 = detail::resolveColorInt(238, 213, 210);
1246 const detail::color_int_t MistyRose3 = detail::resolveColorInt(205, 183, 181);
1247 const detail::color_int_t MistyRose4 = detail::resolveColorInt(139, 125, 123);
1248 const detail::color_int_t moccasin = detail::resolveColorInt(255, 228, 181);
1249 const detail::color_int_t navajo_white = detail::resolveColorInt(255, 222, 173);
1250 const detail::color_int_t NavajoWhite = detail::resolveColorInt(255, 222, 173);
1251 const detail::color_int_t NavajoWhite1 = detail::resolveColorInt(255, 222, 173);
1252 const detail::color_int_t NavajoWhite2 = detail::resolveColorInt(238, 207, 161);
1253 const detail::color_int_t NavajoWhite3 = detail::resolveColorInt(205, 179, 139);
1254 const detail::color_int_t NavajoWhite4 = detail::resolveColorInt(139, 121, 94);
1255 const detail::color_int_t navy = detail::resolveColorInt(0, 0, 128);
1256 const detail::color_int_t navy_blue = detail::resolveColorInt(0, 0, 128);
1257 const detail::color_int_t NavyBlue = detail::resolveColorInt(0, 0, 128);
1258 const detail::color_int_t old_lace = detail::resolveColorInt(253, 245, 230);
1259 const detail::color_int_t OldLace = detail::resolveColorInt(253, 245, 230);
1260 const detail::color_int_t olive_drab = detail::resolveColorInt(107, 142, 35);
1261 const detail::color_int_t OliveDrab = detail::resolveColorInt(107, 142, 35);
1262 const detail::color_int_t OliveDrab1 = detail::resolveColorInt(192, 255, 62);
1263 const detail::color_int_t OliveDrab2 = detail::resolveColorInt(179, 238, 58);
1264 const detail::color_int_t OliveDrab3 = detail::resolveColorInt(154, 205, 50);
1265 const detail::color_int_t OliveDrab4 = detail::resolveColorInt(105, 139, 34);
1266 const detail::color_int_t orange = detail::resolveColorInt(255, 165, 0);
1267 const detail::color_int_t orange_red = detail::resolveColorInt(255, 69, 0);
1268 const detail::color_int_t orange1 = detail::resolveColorInt(255, 165, 0);
1269 const detail::color_int_t orange2 = detail::resolveColorInt(238, 154, 0);
1270 const detail::color_int_t orange3 = detail::resolveColorInt(205, 133, 0);
1271 const detail::color_int_t orange4 = detail::resolveColorInt(139, 90, 0);
1272 const detail::color_int_t OrangeRed = detail::resolveColorInt(255, 69, 0);
1273 const detail::color_int_t OrangeRed1 = detail::resolveColorInt(255, 69, 0);
1274 const detail::color_int_t OrangeRed2 = detail::resolveColorInt(238, 64, 0);
1275 const detail::color_int_t OrangeRed3 = detail::resolveColorInt(205, 55, 0);
1276 const detail::color_int_t OrangeRed4 = detail::resolveColorInt(139, 37, 0);
1277 const detail::color_int_t orchid = detail::resolveColorInt(218, 112, 214);
1278 const detail::color_int_t orchid1 = detail::resolveColorInt(255, 131, 250);
1279 const detail::color_int_t orchid2 = detail::resolveColorInt(238, 122, 233);
1280 const detail::color_int_t orchid3 = detail::resolveColorInt(205, 105, 201);
1281 const detail::color_int_t orchid4 = detail::resolveColorInt(139, 71, 137);
1282 const detail::color_int_t pale_goldenrod = detail::resolveColorInt(238, 232, 170);
1283 const detail::color_int_t pale_green = detail::resolveColorInt(152, 251, 152);
1284 const detail::color_int_t pale_turquoise = detail::resolveColorInt(175, 238, 238);
1285 const detail::color_int_t pale_violet_red = detail::resolveColorInt(219, 112, 147);
1286 const detail::color_int_t PaleGoldenrod = detail::resolveColorInt(238, 232, 170);
1287 const detail::color_int_t PaleGreen = detail::resolveColorInt(152, 251, 152);
1288 const detail::color_int_t PaleGreen1 = detail::resolveColorInt(154, 255, 154);
1289 const detail::color_int_t PaleGreen2 = detail::resolveColorInt(144, 238, 144);
1290 const detail::color_int_t PaleGreen3 = detail::resolveColorInt(124, 205, 124);
1291 const detail::color_int_t PaleGreen4 = detail::resolveColorInt(84, 139, 84);
1292 const detail::color_int_t PaleTurquoise = detail::resolveColorInt(175, 238, 238);
1293 const detail::color_int_t PaleTurquoise1 = detail::resolveColorInt(187, 255, 255);
1294 const detail::color_int_t PaleTurquoise2 = detail::resolveColorInt(174, 238, 238);
1295 const detail::color_int_t PaleTurquoise3 = detail::resolveColorInt(150, 205, 205);
1296 const detail::color_int_t PaleTurquoise4 = detail::resolveColorInt(102, 139, 139);
1297 const detail::color_int_t PaleVioletRed = detail::resolveColorInt(219, 112, 147);
1298 const detail::color_int_t PaleVioletRed1 = detail::resolveColorInt(255, 130, 171);
1299 const detail::color_int_t PaleVioletRed2 = detail::resolveColorInt(238, 121, 159);
1300 const detail::color_int_t PaleVioletRed3 = detail::resolveColorInt(205, 104, 127);
1301 const detail::color_int_t PaleVioletRed4 = detail::resolveColorInt(139, 71, 93);
1302 const detail::color_int_t papaya_whip = detail::resolveColorInt(255, 239, 213);
1303 const detail::color_int_t PapayaWhip = detail::resolveColorInt(255, 239, 213);
1304 const detail::color_int_t peach_puff = detail::resolveColorInt(255, 218, 185);
1305 const detail::color_int_t PeachPuff = detail::resolveColorInt(255, 218, 185);
1306 const detail::color_int_t PeachPuff1 = detail::resolveColorInt(255, 218, 185);
1307 const detail::color_int_t PeachPuff2 = detail::resolveColorInt(238, 203, 173);
1308 const detail::color_int_t PeachPuff3 = detail::resolveColorInt(205, 175, 149);
1309 const detail::color_int_t PeachPuff4 = detail::resolveColorInt(139, 119, 101);
1310 const detail::color_int_t peru = detail::resolveColorInt(205, 133, 63);
1311 const detail::color_int_t pink = detail::resolveColorInt(255, 192, 203);
1312 const detail::color_int_t pink1 = detail::resolveColorInt(255, 181, 197);
1313 const detail::color_int_t pink2 = detail::resolveColorInt(238, 169, 184);
1314 const detail::color_int_t pink3 = detail::resolveColorInt(205, 145, 158);
1315 const detail::color_int_t pink4 = detail::resolveColorInt(139, 99, 108);
1316 const detail::color_int_t plum = detail::resolveColorInt(221, 160, 221);
1317 const detail::color_int_t plum1 = detail::resolveColorInt(255, 187, 255);
1318 const detail::color_int_t plum2 = detail::resolveColorInt(238, 174, 238);
1319 const detail::color_int_t plum3 = detail::resolveColorInt(205, 150, 205);
1320 const detail::color_int_t plum4 = detail::resolveColorInt(139, 102, 139);
1321 const detail::color_int_t powder_blue = detail::resolveColorInt(176, 224, 230);
1322 const detail::color_int_t PowderBlue = detail::resolveColorInt(176, 224, 230);
1323 const detail::color_int_t purple = detail::resolveColorInt(160, 32, 240);
1324 const detail::color_int_t purple1 = detail::resolveColorInt(155, 48, 255);
1325 const detail::color_int_t purple2 = detail::resolveColorInt(145, 44, 238);
1326 const detail::color_int_t purple3 = detail::resolveColorInt(125, 38, 205);
1327 const detail::color_int_t purple4 = detail::resolveColorInt(85, 26, 139);
1328 const detail::color_int_t red = detail::resolveColorInt(255, 0, 0);
1329 const detail::color_int_t red1 = detail::resolveColorInt(255, 0, 0);
1330 const detail::color_int_t red2 = detail::resolveColorInt(238, 0, 0);
1331 const detail::color_int_t red3 = detail::resolveColorInt(205, 0, 0);
1332 const detail::color_int_t red4 = detail::resolveColorInt(139, 0, 0);
1333 const detail::color_int_t rosy_brown = detail::resolveColorInt(188, 143, 143);
1334 const detail::color_int_t RosyBrown = detail::resolveColorInt(188, 143, 143);
1335 const detail::color_int_t RosyBrown1 = detail::resolveColorInt(255, 193, 193);
1336 const detail::color_int_t RosyBrown2 = detail::resolveColorInt(238, 180, 180);
1337 const detail::color_int_t RosyBrown3 = detail::resolveColorInt(205, 155, 155);
1338 const detail::color_int_t RosyBrown4 = detail::resolveColorInt(139, 105, 105);
1339 const detail::color_int_t royal_blue = detail::resolveColorInt(65, 105, 225);
1340 const detail::color_int_t RoyalBlue = detail::resolveColorInt(65, 105, 225);
1341 const detail::color_int_t RoyalBlue1 = detail::resolveColorInt(72, 118, 255);
1342 const detail::color_int_t RoyalBlue2 = detail::resolveColorInt(67, 110, 238);
1343 const detail::color_int_t RoyalBlue3 = detail::resolveColorInt(58, 95, 205);
1344 const detail::color_int_t RoyalBlue4 = detail::resolveColorInt(39, 64, 139);
1345 const detail::color_int_t saddle_brown = detail::resolveColorInt(139, 69, 19);
1346 const detail::color_int_t SaddleBrown = detail::resolveColorInt(139, 69, 19);
1347 const detail::color_int_t salmon = detail::resolveColorInt(250, 128, 114);
1348 const detail::color_int_t salmon1 = detail::resolveColorInt(255, 140, 105);
1349 const detail::color_int_t salmon2 = detail::resolveColorInt(238, 130, 98);
1350 const detail::color_int_t salmon3 = detail::resolveColorInt(205, 112, 84);
1351 const detail::color_int_t salmon4 = detail::resolveColorInt(139, 76, 57);
1352 const detail::color_int_t sandy_brown = detail::resolveColorInt(244, 164, 96);
1353 const detail::color_int_t SandyBrown = detail::resolveColorInt(244, 164, 96);
1354 const detail::color_int_t sea_green = detail::resolveColorInt(46, 139, 87);
1355 const detail::color_int_t SeaGreen = detail::resolveColorInt(46, 139, 87);
1356 const detail::color_int_t SeaGreen1 = detail::resolveColorInt(84, 255, 159);
1357 const detail::color_int_t SeaGreen2 = detail::resolveColorInt(78, 238, 148);
1358 const detail::color_int_t SeaGreen3 = detail::resolveColorInt(67, 205, 128);
1359 const detail::color_int_t SeaGreen4 = detail::resolveColorInt(46, 139, 87);
1360 const detail::color_int_t seashell = detail::resolveColorInt(255, 245, 238);
1361 const detail::color_int_t seashell1 = detail::resolveColorInt(255, 245, 238);
1362 const detail::color_int_t seashell2 = detail::resolveColorInt(238, 229, 222);
1363 const detail::color_int_t seashell3 = detail::resolveColorInt(205, 197, 191);
1364 const detail::color_int_t seashell4 = detail::resolveColorInt(139, 134, 130);
1365 const detail::color_int_t sienna = detail::resolveColorInt(160, 82, 45);
1366 const detail::color_int_t sienna1 = detail::resolveColorInt(255, 130, 71);
1367 const detail::color_int_t sienna2 = detail::resolveColorInt(238, 121, 66);
1368 const detail::color_int_t sienna3 = detail::resolveColorInt(205, 104, 57);
1369 const detail::color_int_t sienna4 = detail::resolveColorInt(139, 71, 38);
1370 const detail::color_int_t sky_blue = detail::resolveColorInt(135, 206, 235);
1371 const detail::color_int_t SkyBlue = detail::resolveColorInt(135, 206, 235);
1372 const detail::color_int_t SkyBlue1 = detail::resolveColorInt(135, 206, 255);
1373 const detail::color_int_t SkyBlue2 = detail::resolveColorInt(126, 192, 238);
1374 const detail::color_int_t SkyBlue3 = detail::resolveColorInt(108, 166, 205);
1375 const detail::color_int_t SkyBlue4 = detail::resolveColorInt(74, 112, 139);
1376 const detail::color_int_t slate_blue = detail::resolveColorInt(106, 90, 205);
1377 const detail::color_int_t slate_gray = detail::resolveColorInt(112, 128, 144);
1378 const detail::color_int_t slate_grey = detail::resolveColorInt(112, 128, 144);
1379 const detail::color_int_t SlateBlue = detail::resolveColorInt(106, 90, 205);
1380 const detail::color_int_t SlateBlue1 = detail::resolveColorInt(131, 111, 255);
1381 const detail::color_int_t SlateBlue2 = detail::resolveColorInt(122, 103, 238);
1382 const detail::color_int_t SlateBlue3 = detail::resolveColorInt(105, 89, 205);
1383 const detail::color_int_t SlateBlue4 = detail::resolveColorInt(71, 60, 139);
1384 const detail::color_int_t SlateGray = detail::resolveColorInt(112, 128, 144);
1385 const detail::color_int_t SlateGray1 = detail::resolveColorInt(198, 226, 255);
1386 const detail::color_int_t SlateGray2 = detail::resolveColorInt(185, 211, 238);
1387 const detail::color_int_t SlateGray3 = detail::resolveColorInt(159, 182, 205);
1388 const detail::color_int_t SlateGray4 = detail::resolveColorInt(108, 123, 139);
1389 const detail::color_int_t SlateGrey = detail::resolveColorInt(112, 128, 144);
1390 const detail::color_int_t snow = detail::resolveColorInt(255, 250, 250);
1391 const detail::color_int_t snow1 = detail::resolveColorInt(255, 250, 250);
1392 const detail::color_int_t snow2 = detail::resolveColorInt(238, 233, 233);
1393 const detail::color_int_t snow3 = detail::resolveColorInt(205, 201, 201);
1394 const detail::color_int_t snow4 = detail::resolveColorInt(139, 137, 137);
1395 const detail::color_int_t spring_green = detail::resolveColorInt(0, 255, 127);
1396 const detail::color_int_t SpringGreen = detail::resolveColorInt(0, 255, 127);
1397 const detail::color_int_t SpringGreen1 = detail::resolveColorInt(0, 255, 127);
1398 const detail::color_int_t SpringGreen2 = detail::resolveColorInt(0, 238, 118);
1399 const detail::color_int_t SpringGreen3 = detail::resolveColorInt(0, 205, 102);
1400 const detail::color_int_t SpringGreen4 = detail::resolveColorInt(0, 139, 69);
1401 const detail::color_int_t steel_blue = detail::resolveColorInt(70, 130, 180);
1402 const detail::color_int_t SteelBlue = detail::resolveColorInt(70, 130, 180);
1403 const detail::color_int_t SteelBlue1 = detail::resolveColorInt(99, 184, 255);
1404 const detail::color_int_t SteelBlue2 = detail::resolveColorInt(92, 172, 238);
1405 const detail::color_int_t SteelBlue3 = detail::resolveColorInt(79, 148, 205);
1406 const detail::color_int_t SteelBlue4 = detail::resolveColorInt(54, 100, 139);
1407 const detail::color_int_t tan = detail::resolveColorInt(210, 180, 140);
1408 const detail::color_int_t tan1 = detail::resolveColorInt(255, 165, 79);
1409 const detail::color_int_t tan2 = detail::resolveColorInt(238, 154, 73);
1410 const detail::color_int_t tan3 = detail::resolveColorInt(205, 133, 63);
1411 const detail::color_int_t tan4 = detail::resolveColorInt(139, 90, 43);
1412 const detail::color_int_t thistle = detail::resolveColorInt(216, 191, 216);
1413 const detail::color_int_t thistle1 = detail::resolveColorInt(255, 225, 255);
1414 const detail::color_int_t thistle2 = detail::resolveColorInt(238, 210, 238);
1415 const detail::color_int_t thistle3 = detail::resolveColorInt(205, 181, 205);
1416 const detail::color_int_t thistle4 = detail::resolveColorInt(139, 123, 139);
1417 const detail::color_int_t tomato = detail::resolveColorInt(255, 99, 71);
1418 const detail::color_int_t tomato1 = detail::resolveColorInt(255, 99, 71);
1419 const detail::color_int_t tomato2 = detail::resolveColorInt(238, 92, 66);
1420 const detail::color_int_t tomato3 = detail::resolveColorInt(205, 79, 57);
1421 const detail::color_int_t tomato4 = detail::resolveColorInt(139, 54, 38);
1422 const detail::color_int_t turquoise = detail::resolveColorInt(64, 224, 208);
1423 const detail::color_int_t turquoise1 = detail::resolveColorInt(0, 245, 255);
1424 const detail::color_int_t turquoise2 = detail::resolveColorInt(0, 229, 238);
1425 const detail::color_int_t turquoise3 = detail::resolveColorInt(0, 197, 205);
1426 const detail::color_int_t turquoise4 = detail::resolveColorInt(0, 134, 139);
1427 const detail::color_int_t violet = detail::resolveColorInt(238, 130, 238);
1428 const detail::color_int_t violet_red = detail::resolveColorInt(208, 32, 144);
1429 const detail::color_int_t VioletRed = detail::resolveColorInt(208, 32, 144);
1430 const detail::color_int_t VioletRed1 = detail::resolveColorInt(255, 62, 150);
1431 const detail::color_int_t VioletRed2 = detail::resolveColorInt(238, 58, 140);
1432 const detail::color_int_t VioletRed3 = detail::resolveColorInt(205, 50, 120);
1433 const detail::color_int_t VioletRed4 = detail::resolveColorInt(139, 34, 82);
1434 const detail::color_int_t wheat = detail::resolveColorInt(245, 222, 179);
1435 const detail::color_int_t wheat1 = detail::resolveColorInt(255, 231, 186);
1436 const detail::color_int_t wheat2 = detail::resolveColorInt(238, 216, 174);
1437 const detail::color_int_t wheat3 = detail::resolveColorInt(205, 186, 150);
1438 const detail::color_int_t wheat4 = detail::resolveColorInt(139, 126, 102);
1439 const detail::color_int_t white = detail::resolveColorInt(255, 255, 255);
1440 const detail::color_int_t white_smoke = detail::resolveColorInt(245, 245, 245);
1441 const detail::color_int_t WhiteSmoke = detail::resolveColorInt(245, 245, 245);
1442 const detail::color_int_t yellow = detail::resolveColorInt(255, 255, 0);
1443 const detail::color_int_t yellow_green = detail::resolveColorInt(154, 205, 50);
1444 const detail::color_int_t yellow1 = detail::resolveColorInt(255, 255, 0);
1445 const detail::color_int_t yellow2 = detail::resolveColorInt(238, 238, 0);
1446 const detail::color_int_t yellow3 = detail::resolveColorInt(205, 205, 0);
1447 const detail::color_int_t yellow4 = detail::resolveColorInt(139, 139, 0);
1448 const detail::color_int_t YellowGreen = detail::resolveColorInt(154, 205, 50);
1449 }
1450 }
1451
1460 class Color {
1461 public:
1462 typedef uint8_t component_t;
1463
1464 union {
1465 struct {
1466 component_t r;
1467 component_t g;
1468 component_t b;
1469 };
1470 component_t components[3];
1471 };
1472
1473 Color(detail::color_int_t packedColor) {
1474 detail::resolveColorComp(packedColor, r, g, b);
1475 }
1476
1477 /*\brief Color constructor for unsigned 8-bit RGB values.
1478 \param r Red component.
1479 \param g Green component.
1480 \param b Blue component.*/
1481 Color(component_t r, component_t g, component_t b) :
1482 r(r), g(g), b(b) {
1483 };
1484
1485 /*\brief Copy constructor.
1486 \param other Constant reference to other instance of a color object.*/
1487 Color(const Color& other) :
1488 r(other.r), g(other.g), b(other.b) {
1489 }
1490
1491 /*\brief Name constructor. Takes a literal color name as an input.
1492 \param name The name of the color from which to derive value.
1493 \sa fromName()*/
1494 Color(const std::string& name);
1495
1496 /*\brief Default constructor.
1497 Initializes this color to white. (all components 255)*/
1498 Color() {
1499 r = g = b = 255;
1500 }
1501
1502 Color& operator=(detail::color_int_t pack) {
1503 detail::resolveColorComp(pack, r, g, b);
1504 return *this;
1505 }
1506
1510 const component_t* rgbPtr() const {
1511 return &components[0];
1512 }
1513 };
1514
1515 constexpr char DEFAULT_FONT[] = "default";
1516 constexpr uint32_t DEFAULT_FONT_PIXELS_WIDTH = 256;
1517 constexpr uint32_t DEFAULT_FONT_PIXELS_HEIGHT = 128;
1518 constexpr uint32_t DEFAULT_FONT_GLYPH_WIDTH = 8;
1519 constexpr uint32_t DEFAULT_FONT_GLYPH_HEIGHT = 16;
1520 constexpr uint32_t DEFAULT_FONT_GLYPHS_X = 32;
1521 constexpr uint32_t DEFAULT_FONT_GLYPHS_Y = 8;
1522 constexpr uint32_t DEFAULT_FONT_PIXELS_LEN = 1024;
1523 constexpr uint32_t DEFAULT_FONT_ASCII_OFFSET = 32;
1524
1525 //Proggy clean font, 8x16 glyphs, ASCII offset of 32.
1526 //fairly compact in memory when expressed ike this.
1527 //1-bit encoding per pixel. 1 is opaque white, 0 is transparent/empty.
1528 //Proggy clean can be found HERE: https://proggyfonts.net/
1529 const uint32_t DEFAULT_FONT_PIXELS[] =
1530 { 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1531 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x102814, 0x10443010, 0x18300000, 0x4, 0x38103838, 0x47c187c, 0x38380000, 0x38,
1532 0x102814, 0x3ca84810, 0x10100000, 0x8, 0x44304444, 0xc402004, 0x44440000, 0x44, 0x10007e, 0x50a84800, 0x20081010, 0x8, 0x44500404, 0x14404008, 0x44441020, 0xc006004,
1533 0x100028, 0x50503200, 0x20085410, 0x10, 0x54100818, 0x24787808, 0x38441020, 0x307e1808, 0x100028, 0x38144a00, 0x2008387c, 0x7c0010, 0x54101004, 0x44044410, 0x443c0000, 0xc0000610,
1534 0x1000fc, 0x142a4400, 0x20085410, 0x20, 0x44102004, 0x7e044410, 0x44040000, 0x307e1810, 0x50, 0x142a4400, 0x20081010, 0x20002020, 0x44104044, 0x4444420, 0x44081020, 0xc006000,
1535 0x100050, 0x78443a00, 0x10100000, 0x20002040, 0x387c7c38, 0x4383820, 0x38301020, 0x10, 0x0, 0x10000000, 0x10100000, 0x20000040, 0x0, 0x0, 0x20, 0x0,
1536 0x0, 0x0, 0x8200000, 0x40000000, 0x0, 0x0, 0x40, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1537 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1538 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1539 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x3818781c, 0x787c7c1c, 0x42383842, 0x40c66218, 0x7818783c, 0xfe428282, 0x42827e38, 0x40381000,
1540 0x44184422, 0x44404022, 0x42100844, 0x40c66224, 0x44244442, 0x10428292, 0x42820220, 0x20082800, 0x9a244440, 0x42404040, 0x42100848, 0x40aa5242, 0x44424440, 0x10424492, 0x24440420, 0x20082800,
1541 0xaa247c40, 0x42787840, 0x7e100850, 0x40aa5242, 0x44424430, 0x104244aa, 0x18280820, 0x10084400, 0xaa3c4240, 0x4240404e, 0x42100870, 0x40924a42, 0x7842780c, 0x104228aa, 0x18101020, 0x10084400,
1542 0x9c424240, 0x42404042, 0x42100848, 0x40924a42, 0x40424802, 0x1042286c, 0x24102020, 0x8080000, 0x40424222, 0x44404022, 0x42100844, 0x40824624, 0x40244442, 0x10421044, 0x42104020, 0x8080000,
1543 0x3c427c1c, 0x787c401c, 0x42387042, 0x7c824618, 0x401a423c, 0x103c1044, 0x42107e20, 0x4080000, 0x0, 0x0, 0x0, 0x0, 0x20000, 0x0, 0x20, 0x40800fe,
1544 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x38, 0x380000, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1545 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1546 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1547 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x30004000, 0x4003c00, 0x40100840, 0x30000000, 0x0, 0x20000000, 0x1c, 0x10700000,
1548 0x4000, 0x4002000, 0x40000040, 0x10000000, 0x0, 0x20000000, 0x10, 0x10100000, 0x387838, 0x3c38783c, 0x78301844, 0x10ec7838, 0x783c583c, 0x3c444482, 0x44447c10, 0x10100000,
1549 0x44444, 0x44442044, 0x44100848, 0x10924444, 0x44446440, 0x20444492, 0x28440410, 0x10107200, 0x3c4440, 0x447c2044, 0x44100850, 0x10924444, 0x44444030, 0x20442892, 0x10440860, 0x100c9c00,
1550 0x444440, 0x44402044, 0x44100870, 0x10924444, 0x44444008, 0x204428aa, 0x10441010, 0x10100000, 0x444444, 0x44442044, 0x44100848, 0x10924444, 0x44444004, 0x2044106c, 0x28442010, 0x10100000,
1551 0x3c7838, 0x3c38203c, 0x44100844, 0x10924438, 0x783c4078, 0x1c3c1044, 0x443c7c10, 0x10100000, 0x0, 0x4, 0x800, 0x0, 0x40040000, 0x0, 0x40010, 0x10100000,
1552 0x0, 0x4, 0x7000, 0x0, 0x40040000, 0x0, 0x4000c, 0x10600000, 0x0, 0x38, 0x0, 0x0, 0x40040000, 0x0, 0x380000, 0x0,
1553 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1554 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1555 0xfefe, 0xfefefefe, 0xfefefefe, 0xfe00fe00, 0xfefefe, 0xfefefefe, 0xfefefefe, 0xfe00fefe, 0x8282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282,
1556 0x1c008282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282, 0x22008282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282,
1557 0x78008282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282, 0x20008282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282,
1558 0x78008282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282, 0x22008282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282,
1559 0x1c008282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282, 0x8282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282,
1560 0x8282, 0x82828282, 0x82828282, 0x82008200, 0x828282, 0x82828282, 0x82828282, 0x82008282, 0xfefe, 0xfefefefe, 0xfefefefe, 0xfe00fe00, 0xfefefe, 0xfefefefe, 0xfefefefe, 0xfe00fefe,
1561 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1562 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1563 0x0, 0x0, 0x0, 0xfe, 0x0, 0x0, 0x0, 0x4040400, 0x100018, 0x82107c, 0x287c3000, 0x7c00, 0x78007878, 0x18003e00, 0x607800, 0x4c4cec10,
1564 0x1020, 0x82821040, 0x820800, 0x8200, 0x48101030, 0x7400, 0x204800, 0x48486800, 0x103820, 0x7c441030, 0x9a3814, 0xb200, 0x30102008, 0x447400, 0x204850, 0x50503010,
1565 0x105478, 0x44281028, 0xa24828, 0xaa00, 0x7c7870, 0x443438, 0x703028, 0x101cd010, 0x105020, 0x44100018, 0xa23850, 0x7800b200, 0x100000, 0x441438, 0x14, 0x24222420,
1566 0x105020, 0x447c1004, 0x9a0028, 0x800aa00, 0x100000, 0x441438, 0x28, 0x2c242c40, 0x105440, 0x7c101044, 0x820014, 0x8008200, 0x0, 0x641400, 0x50, 0x5e485e44,
1567 0x10387c, 0x82101038, 0x7c0000, 0x8007c00, 0x7c0000, 0x5a1400, 0x0, 0x444e4438, 0x1000, 0x1000, 0x0, 0x0, 0x0, 0x401400, 0x18000000, 0x0,
1568 0x0, 0x1000, 0x0, 0x0, 0x0, 0x800000, 0x8000000, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x10000000, 0x0,
1569 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1570 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1571 0x10081814, 0x180000, 0x20081000, 0x10081000, 0x141008, 0x18140000, 0x100818, 0x80000, 0x8102428, 0x24241e1c, 0x10102828, 0x8102828, 0x280810, 0x24282400, 0x1a081024, 0x2410407c,
1572 0x18181818, 0x18182822, 0x7c7c7c7c, 0x38383838, 0x78621818, 0x18181800, 0x24424242, 0x42824044, 0x24242424, 0x24242840, 0x40404040, 0x10101010, 0x44522424, 0x24242444, 0x4a424242, 0x42827878,
1573 0x24242424, 0x24244c40, 0x40404040, 0x10101010, 0x42524242, 0x42424228, 0x4a424242, 0x42444444, 0x3c3c3c3c, 0x3c3c7840, 0x78787878, 0x10101010, 0xf24a4242, 0x42424210, 0x52424242, 0x42284442,
1574 0x42424242, 0x42428840, 0x40404040, 0x10101010, 0x424a4242, 0x42424228, 0x52424242, 0x42107842, 0x42424242, 0x42428822, 0x40404040, 0x10101010, 0x44462424, 0x24242444, 0x24424242, 0x42104042,
1575 0x42424242, 0x42428e1c, 0x7c7c7c7c, 0x38383838, 0x78461818, 0x18181800, 0x583c3c3c, 0x3c10407c, 0x0, 0x8, 0x0, 0x0, 0x0, 0x0, 0x0, 0x80,
1576 0x0, 0x8, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x10, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1577 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1578 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1579 0x0, 0x100000, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x1818383c, 0x28380000, 0x30303828, 0x30303828, 0x3c783018, 0x383c2800, 0x303038, 0x28184028,
1580 0x0, 0x0, 0x0, 0x0, 0x14000000, 0x0, 0x4000000, 0x4000, 0x38383838, 0x38386c38, 0x38383838, 0x30303030, 0x4783838, 0x38383810, 0x38444444, 0x44447844,
1581 0x4040404, 0x4041244, 0x44444444, 0x10101010, 0x3c444444, 0x44444400, 0x4c444444, 0x44444444, 0x3c3c3c3c, 0x3c3c7e40, 0x7c7c7c7c, 0x10101010, 0x44444444, 0x4444447c, 0x54444444, 0x44444444,
1582 0x44444444, 0x44449040, 0x40404040, 0x10101010, 0x44444444, 0x44444400, 0x54444444, 0x44444444, 0x44444444, 0x44449244, 0x44444444, 0x10101010, 0x44444444, 0x44444410, 0x64444444, 0x44444444,
1583 0x3c3c3c3c, 0x3c3c7c38, 0x38383838, 0x10101010, 0x38443838, 0x38383800, 0x383c3c3c, 0x3c3c783c, 0x0, 0x10, 0x0, 0x0, 0x0, 0x0, 0x40000000, 0x44004,
1584 0x0, 0x10, 0x0, 0x0, 0x0, 0x0, 0x0, 0x44004, 0x0, 0x20, 0x0, 0x0, 0x0, 0x0, 0x0, 0x380038,
1585 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1586 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1587 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1588 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1589 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1590 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1591 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1592 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0,
1593 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0 };
1594
1595
1596 const std::unordered_map<std::string, detail::color_int_t> NAMED_COLORS = {
1597 {"alice blue", detail::col::alice_blue},
1598 {"AliceBlue", detail::col::AliceBlue},
1599 {"antique white", detail::col::antique_white},
1600 {"AntiqueWhite", detail::col::AntiqueWhite},
1601 {"AntiqueWhite1", detail::col::AntiqueWhite1},
1602 {"AntiqueWhite2", detail::col::AntiqueWhite2},
1603 {"AntiqueWhite3", detail::col::AntiqueWhite3},
1604 {"AntiqueWhite4", detail::col::AntiqueWhite4},
1605 {"aquamarine", detail::col::aquamarine},
1606 {"aquamarine1", detail::col::aquamarine1},
1607 {"aquamarine2", detail::col::aquamarine2},
1608 {"aquamarine3", detail::col::aquamarine3},
1609 {"aquamarine4", detail::col::aquamarine4},
1610 {"azure", detail::col::azure},
1611 {"azure1", detail::col::azure1},
1612 {"azure2", detail::col::azure2},
1613 {"azure3", detail::col::azure3},
1614 {"azure4", detail::col::azure4},
1615 {"beige", detail::col::beige},
1616 {"bisque", detail::col::bisque},
1617 {"bisque1", detail::col::bisque1},
1618 {"bisque2", detail::col::bisque2},
1619 {"bisque3", detail::col::bisque3},
1620 {"bisque4", detail::col::bisque4},
1621 {"black", detail::col::black},
1622 {"blanched almond", detail::col::blanched_almond},
1623 {"BlanchedAlmond", detail::col::BlanchedAlmond},
1624 {"blue", detail::col::blue},
1625 {"blue violet", detail::col::blue_violet},
1626 {"blue1", detail::col::blue1},
1627 {"blue2", detail::col::blue2},
1628 {"blue3", detail::col::blue3},
1629 {"blue4", detail::col::blue4},
1630 {"BlueViolet", detail::col::BlueViolet},
1631 {"brown", detail::col::brown},
1632 {"brown1", detail::col::brown1},
1633 {"brown2", detail::col::brown2},
1634 {"brown3", detail::col::brown3},
1635 {"brown4", detail::col::brown4},
1636 {"burlywood", detail::col::burlywood},
1637 {"burlywood1", detail::col::burlywood1},
1638 {"burlywood2", detail::col::burlywood2},
1639 {"burlywood3", detail::col::burlywood3},
1640 {"burlywood4", detail::col::burlywood4},
1641 {"cadet blue", detail::col::cadet_blue},
1642 {"CadetBlue", detail::col::CadetBlue},
1643 {"CadetBlue1", detail::col::CadetBlue1},
1644 {"CadetBlue2", detail::col::CadetBlue2},
1645 {"CadetBlue3", detail::col::CadetBlue3},
1646 {"CadetBlue4", detail::col::CadetBlue4},
1647 {"chartreuse", detail::col::chartreuse},
1648 {"chartreuse1", detail::col::chartreuse1},
1649 {"chartreuse2", detail::col::chartreuse2},
1650 {"chartreuse3", detail::col::chartreuse3},
1651 {"chartreuse4", detail::col::chartreuse4},
1652 {"chocolate", detail::col::chocolate},
1653 {"chocolate1", detail::col::chocolate1},
1654 {"chocolate2", detail::col::chocolate2},
1655 {"chocolate3", detail::col::chocolate3},
1656 {"chocolate4", detail::col::chocolate4},
1657 {"coral", detail::col::coral},
1658 {"coral1", detail::col::coral1},
1659 {"coral2", detail::col::coral2},
1660 {"coral3", detail::col::coral3},
1661 {"coral4", detail::col::coral4},
1662 {"cornflower blue", detail::col::cornflower_blue},
1663 {"CornflowerBlue", detail::col::CornflowerBlue},
1664 {"cornsilk", detail::col::cornsilk},
1665 {"cornsilk1", detail::col::cornsilk1},
1666 {"cornsilk2", detail::col::cornsilk2},
1667 {"cornsilk3", detail::col::cornsilk3},
1668 {"cornsilk4", detail::col::cornsilk4},
1669 {"cyan", detail::col::cyan},
1670 {"cyan1", detail::col::cyan1},
1671 {"cyan2", detail::col::cyan2},
1672 {"cyan3", detail::col::cyan3},
1673 {"cyan4", detail::col::cyan4},
1674 {"dark blue", detail::col::dark_blue},
1675 {"dark cyan", detail::col::dark_cyan},
1676 {"dark goldenrod", detail::col::dark_goldenrod},
1677 {"dark gray", detail::col::dark_gray},
1678 {"dark green", detail::col::dark_green},
1679 {"dark grey", detail::col::dark_grey},
1680 {"dark khaki", detail::col::dark_khaki},
1681 {"dark magenta", detail::col::dark_magenta},
1682 {"dark olive green", detail::col::dark_olive_green},
1683 {"dark orange", detail::col::dark_orange},
1684 {"dark orchid", detail::col::dark_orchid},
1685 {"dark red", detail::col::dark_red},
1686 {"dark salmon", detail::col::dark_salmon},
1687 {"dark sea green", detail::col::dark_sea_green},
1688 {"dark slate blue", detail::col::dark_slate_blue},
1689 {"dark slate gray", detail::col::dark_slate_gray},
1690 {"dark slate grey", detail::col::dark_slate_grey},
1691 {"dark turquoise", detail::col::dark_turquoise},
1692 {"dark violet", detail::col::dark_violet},
1693 {"DarkBlue", detail::col::DarkBlue},
1694 {"DarkCyan", detail::col::DarkCyan},
1695 {"DarkGoldenrod", detail::col::DarkGoldenrod},
1696 {"DarkGoldenrod1", detail::col::DarkGoldenrod1},
1697 {"DarkGoldenrod2", detail::col::DarkGoldenrod2},
1698 {"DarkGoldenrod3", detail::col::DarkGoldenrod3},
1699 {"DarkGoldenrod4", detail::col::DarkGoldenrod4},
1700 {"DarkGray", detail::col::DarkGray},
1701 {"DarkGreen", detail::col::DarkGreen},
1702 {"DarkGrey", detail::col::DarkGrey},
1703 {"DarkKhaki", detail::col::DarkKhaki},
1704 {"DarkMagenta", detail::col::DarkMagenta},
1705 {"DarkOliveGreen", detail::col::DarkOliveGreen},
1706 {"DarkOliveGreen1", detail::col::DarkOliveGreen1},
1707 {"DarkOliveGreen2", detail::col::DarkOliveGreen2},
1708 {"DarkOliveGreen3", detail::col::DarkOliveGreen3},
1709 {"DarkOliveGreen4", detail::col::DarkOliveGreen4},
1710 {"DarkOrange", detail::col::DarkOrange},
1711 {"DarkOrange1", detail::col::DarkOrange1},
1712 {"DarkOrange2", detail::col::DarkOrange2},
1713 {"DarkOrange3", detail::col::DarkOrange3},
1714 {"DarkOrange4", detail::col::DarkOrange4},
1715 {"DarkOrchid", detail::col::DarkOrchid},
1716 {"DarkOrchid1", detail::col::DarkOrchid1},
1717 {"DarkOrchid2", detail::col::DarkOrchid2},
1718 {"DarkOrchid3", detail::col::DarkOrchid3},
1719 {"DarkOrchid4", detail::col::DarkOrchid4},
1720 {"DarkRed", detail::col::DarkRed},
1721 {"DarkSalmon", detail::col::DarkSalmon},
1722 {"DarkSeaGreen", detail::col::DarkSeaGreen},
1723 {"DarkSeaGreen1", detail::col::DarkSeaGreen1},
1724 {"DarkSeaGreen2", detail::col::DarkSeaGreen2},
1725 {"DarkSeaGreen3", detail::col::DarkSeaGreen3},
1726 {"DarkSeaGreen4", detail::col::DarkSeaGreen4},
1727 {"DarkSlateBlue", detail::col::DarkSlateBlue},
1728 {"DarkSlateGray", detail::col::DarkSlateGray},
1729 {"DarkSlateGray1", detail::col::DarkSlateGray1},
1730 {"DarkSlateGray2", detail::col::DarkSlateGray2},
1731 {"DarkSlateGray3", detail::col::DarkSlateGray3},
1732 {"DarkSlateGray4", detail::col::DarkSlateGray4},
1733 {"DarkSlateGrey", detail::col::DarkSlateGrey},
1734 {"DarkTurquoise", detail::col::DarkTurquoise},
1735 {"DarkViolet", detail::col::DarkViolet},
1736 {"deep pink", detail::col::deep_pink},
1737 {"deep sky blue", detail::col::deep_sky_blue},
1738 {"DeepPink", detail::col::DeepPink},
1739 {"DeepPink1", detail::col::DeepPink1},
1740 {"DeepPink2", detail::col::DeepPink2},
1741 {"DeepPink3", detail::col::DeepPink3},
1742 {"DeepPink4", detail::col::DeepPink4},
1743 {"DeepSkyBlue", detail::col::DeepSkyBlue},
1744 {"DeepSkyBlue1", detail::col::DeepSkyBlue1},
1745 {"DeepSkyBlue2", detail::col::DeepSkyBlue2},
1746 {"DeepSkyBlue3", detail::col::DeepSkyBlue3},
1747 {"DeepSkyBlue4", detail::col::DeepSkyBlue4},
1748 {"dim gray", detail::col::dim_gray},
1749 {"dim grey", detail::col::dim_grey},
1750 {"DimGray", detail::col::DimGray},
1751 {"DimGrey", detail::col::DimGrey},
1752 {"dodger blue", detail::col::dodger_blue},
1753 {"DodgerBlue", detail::col::DodgerBlue},
1754 {"DodgerBlue1", detail::col::DodgerBlue1},
1755 {"DodgerBlue2", detail::col::DodgerBlue2},
1756 {"DodgerBlue3", detail::col::DodgerBlue3},
1757 {"DodgerBlue4", detail::col::DodgerBlue4},
1758 {"firebrick", detail::col::firebrick},
1759 {"firebrick1", detail::col::firebrick1},
1760 {"firebrick2", detail::col::firebrick2},
1761 {"firebrick3", detail::col::firebrick3},
1762 {"firebrick4", detail::col::firebrick4},
1763 {"floral white", detail::col::floral_white},
1764 {"FloralWhite", detail::col::FloralWhite},
1765 {"forest green", detail::col::forest_green},
1766 {"ForestGreen", detail::col::ForestGreen},
1767 {"gainsboro", detail::col::gainsboro},
1768 {"ghost white", detail::col::ghost_white},
1769 {"GhostWhite", detail::col::GhostWhite},
1770 {"gold", detail::col::gold},
1771 {"gold1", detail::col::gold1},
1772 {"gold2", detail::col::gold2},
1773 {"gold3", detail::col::gold3},
1774 {"gold4", detail::col::gold4},
1775 {"goldenrod", detail::col::goldenrod},
1776 {"goldenrod1", detail::col::goldenrod1},
1777 {"goldenrod2", detail::col::goldenrod2},
1778 {"goldenrod3", detail::col::goldenrod3},
1779 {"goldenrod4", detail::col::goldenrod4},
1780 {"gray", detail::col::gray},
1781 {"gray0", detail::col::gray0},
1782 {"gray1", detail::col::gray1},
1783 {"gray2", detail::col::gray2},
1784 {"gray3", detail::col::gray3},
1785 {"gray4", detail::col::gray4},
1786 {"gray5", detail::col::gray5},
1787 {"gray6", detail::col::gray6},
1788 {"gray7", detail::col::gray7},
1789 {"gray8", detail::col::gray8},
1790 {"gray9", detail::col::gray9},
1791 {"gray10", detail::col::gray10},
1792 {"gray11", detail::col::gray11},
1793 {"gray12", detail::col::gray12},
1794 {"gray13", detail::col::gray13},
1795 {"gray14", detail::col::gray14},
1796 {"gray15", detail::col::gray15},
1797 {"gray16", detail::col::gray16},
1798 {"gray17", detail::col::gray17},
1799 {"gray18", detail::col::gray18},
1800 {"gray19", detail::col::gray19},
1801 {"gray20", detail::col::gray20},
1802 {"gray21", detail::col::gray21},
1803 {"gray22", detail::col::gray22},
1804 {"gray23", detail::col::gray23},
1805 {"gray24", detail::col::gray24},
1806 {"gray25", detail::col::gray25},
1807 {"gray26", detail::col::gray26},
1808 {"gray27", detail::col::gray27},
1809 {"gray28", detail::col::gray28},
1810 {"gray29", detail::col::gray29},
1811 {"gray30", detail::col::gray30},
1812 {"gray31", detail::col::gray31},
1813 {"gray32", detail::col::gray32},
1814 {"gray33", detail::col::gray33},
1815 {"gray34", detail::col::gray34},
1816 {"gray35", detail::col::gray35},
1817 {"gray36", detail::col::gray36},
1818 {"gray37", detail::col::gray37},
1819 {"gray38", detail::col::gray38},
1820 {"gray39", detail::col::gray39},
1821 {"gray40", detail::col::gray40},
1822 {"gray41", detail::col::gray41},
1823 {"gray42", detail::col::gray42},
1824 {"gray43", detail::col::gray43},
1825 {"gray44", detail::col::gray44},
1826 {"gray45", detail::col::gray45},
1827 {"gray46", detail::col::gray46},
1828 {"gray47", detail::col::gray47},
1829 {"gray48", detail::col::gray48},
1830 {"gray49", detail::col::gray49},
1831 {"gray50", detail::col::gray50},
1832 {"gray51", detail::col::gray51},
1833 {"gray52", detail::col::gray52},
1834 {"gray53", detail::col::gray53},
1835 {"gray54", detail::col::gray54},
1836 {"gray55", detail::col::gray55},
1837 {"gray56", detail::col::gray56},
1838 {"gray57", detail::col::gray57},
1839 {"gray58", detail::col::gray58},
1840 {"gray59", detail::col::gray59},
1841 {"gray60", detail::col::gray60},
1842 {"gray61", detail::col::gray61},
1843 {"gray62", detail::col::gray62},
1844 {"gray63", detail::col::gray63},
1845 {"gray64", detail::col::gray64},
1846 {"gray65", detail::col::gray65},
1847 {"gray66", detail::col::gray66},
1848 {"gray67", detail::col::gray67},
1849 {"gray68", detail::col::gray68},
1850 {"gray69", detail::col::gray69},
1851 {"gray70", detail::col::gray70},
1852 {"gray71", detail::col::gray71},
1853 {"gray72", detail::col::gray72},
1854 {"gray73", detail::col::gray73},
1855 {"gray74", detail::col::gray74},
1856 {"gray75", detail::col::gray75},
1857 {"gray76", detail::col::gray76},
1858 {"gray77", detail::col::gray77},
1859 {"gray78", detail::col::gray78},
1860 {"gray79", detail::col::gray79},
1861 {"gray80", detail::col::gray80},
1862 {"gray81", detail::col::gray81},
1863 {"gray82", detail::col::gray82},
1864 {"gray83", detail::col::gray83},
1865 {"gray84", detail::col::gray84},
1866 {"gray85", detail::col::gray85},
1867 {"gray86", detail::col::gray86},
1868 {"gray87", detail::col::gray87},
1869 {"gray88", detail::col::gray88},
1870 {"gray89", detail::col::gray89},
1871 {"gray90", detail::col::gray90},
1872 {"gray91", detail::col::gray91},
1873 {"gray92", detail::col::gray92},
1874 {"gray93", detail::col::gray93},
1875 {"gray94", detail::col::gray94},
1876 {"gray95", detail::col::gray95},
1877 {"gray96", detail::col::gray96},
1878 {"gray97", detail::col::gray97},
1879 {"gray98", detail::col::gray98},
1880 {"gray99", detail::col::gray99},
1881 {"gray100", detail::col::gray100},
1882 {"green", detail::col::green},
1883 {"green yellow", detail::col::green_yellow},
1884 {"green1", detail::col::green1},
1885 {"green2", detail::col::green2},
1886 {"green3", detail::col::green3},
1887 {"green4", detail::col::green4},
1888 {"GreenYellow", detail::col::GreenYellow},
1889 {"grey", detail::col::grey},
1890 {"grey0", detail::col::grey0},
1891 {"grey1", detail::col::grey1},
1892 {"grey2", detail::col::grey2},
1893 {"grey3", detail::col::grey3},
1894 {"grey4", detail::col::grey4},
1895 {"grey5", detail::col::grey5},
1896 {"grey6", detail::col::grey6},
1897 {"grey7", detail::col::grey7},
1898 {"grey8", detail::col::grey8},
1899 {"grey9", detail::col::grey9},
1900 {"grey10", detail::col::grey10},
1901 {"grey11", detail::col::grey11},
1902 {"grey12", detail::col::grey12},
1903 {"grey13", detail::col::grey13},
1904 {"grey14", detail::col::grey14},
1905 {"grey15", detail::col::grey15},
1906 {"grey16", detail::col::grey16},
1907 {"grey17", detail::col::grey17},
1908 {"grey18", detail::col::grey18},
1909 {"grey19", detail::col::grey19},
1910 {"grey20", detail::col::grey20},
1911 {"grey21", detail::col::grey21},
1912 {"grey22", detail::col::grey22},
1913 {"grey23", detail::col::grey23},
1914 {"grey24", detail::col::grey24},
1915 {"grey25", detail::col::grey25},
1916 {"grey26", detail::col::grey26},
1917 {"grey27", detail::col::grey27},
1918 {"grey28", detail::col::grey28},
1919 {"grey29", detail::col::grey29},
1920 {"grey30", detail::col::grey30},
1921 {"grey31", detail::col::grey31},
1922 {"grey32", detail::col::grey32},
1923 {"grey33", detail::col::grey33},
1924 {"grey34", detail::col::grey34},
1925 {"grey35", detail::col::grey35},
1926 {"grey36", detail::col::grey36},
1927 {"grey37", detail::col::grey37},
1928 {"grey38", detail::col::grey38},
1929 {"grey39", detail::col::grey39},
1930 {"grey40", detail::col::grey40},
1931 {"grey41", detail::col::grey41},
1932 {"grey42", detail::col::grey42},
1933 {"grey43", detail::col::grey43},
1934 {"grey44", detail::col::grey44},
1935 {"grey45", detail::col::grey45},
1936 {"grey46", detail::col::grey46},
1937 {"grey47", detail::col::grey47},
1938 {"grey48", detail::col::grey48},
1939 {"grey49", detail::col::grey49},
1940 {"grey50", detail::col::grey50},
1941 {"grey51", detail::col::grey51},
1942 {"grey52", detail::col::grey52},
1943 {"grey53", detail::col::grey53},
1944 {"grey54", detail::col::grey54},
1945 {"grey55", detail::col::grey55},
1946 {"grey56", detail::col::grey56},
1947 {"grey57", detail::col::grey57},
1948 {"grey58", detail::col::grey58},
1949 {"grey59", detail::col::grey59},
1950 {"grey60", detail::col::grey60},
1951 {"grey61", detail::col::grey61},
1952 {"grey62", detail::col::grey62},
1953 {"grey63", detail::col::grey63},
1954 {"grey64", detail::col::grey64},
1955 {"grey65", detail::col::grey65},
1956 {"grey66", detail::col::grey66},
1957 {"grey67", detail::col::grey67},
1958 {"grey68", detail::col::grey68},
1959 {"grey69", detail::col::grey69},
1960 {"grey70", detail::col::grey70},
1961 {"grey71", detail::col::grey71},
1962 {"grey72", detail::col::grey72},
1963 {"grey73", detail::col::grey73},
1964 {"grey74", detail::col::grey74},
1965 {"grey75", detail::col::grey75},
1966 {"grey76", detail::col::grey76},
1967 {"grey77", detail::col::grey77},
1968 {"grey78", detail::col::grey78},
1969 {"grey79", detail::col::grey79},
1970 {"grey80", detail::col::grey80},
1971 {"grey81", detail::col::grey81},
1972 {"grey82", detail::col::grey82},
1973 {"grey83", detail::col::grey83},
1974 {"grey84", detail::col::grey84},
1975 {"grey85", detail::col::grey85},
1976 {"grey86", detail::col::grey86},
1977 {"grey87", detail::col::grey87},
1978 {"grey88", detail::col::grey88},
1979 {"grey89", detail::col::grey89},
1980 {"grey90", detail::col::grey90},
1981 {"grey91", detail::col::grey91},
1982 {"grey92", detail::col::grey92},
1983 {"grey93", detail::col::grey93},
1984 {"grey94", detail::col::grey94},
1985 {"grey95", detail::col::grey95},
1986 {"grey96", detail::col::grey96},
1987 {"grey97", detail::col::grey97},
1988 {"grey98", detail::col::grey98},
1989 {"grey99", detail::col::grey99},
1990 {"grey100", detail::col::grey100},
1991 {"honeydew", detail::col::honeydew},
1992 {"honeydew1", detail::col::honeydew1},
1993 {"honeydew2", detail::col::honeydew2},
1994 {"honeydew3", detail::col::honeydew3},
1995 {"honeydew4", detail::col::honeydew4},
1996 {"hot pink", detail::col::hot_pink},
1997 {"HotPink", detail::col::HotPink},
1998 {"HotPink1", detail::col::HotPink1},
1999 {"HotPink2", detail::col::HotPink2},
2000 {"HotPink3", detail::col::HotPink3},
2001 {"HotPink4", detail::col::HotPink4},
2002 {"indian red", detail::col::indian_red},
2003 {"IndianRed", detail::col::IndianRed},
2004 {"IndianRed1", detail::col::IndianRed1},
2005 {"IndianRed2", detail::col::IndianRed2},
2006 {"IndianRed3", detail::col::IndianRed3},
2007 {"IndianRed4", detail::col::IndianRed4},
2008 {"ivory", detail::col::ivory},
2009 {"ivory1", detail::col::ivory1},
2010 {"ivory2", detail::col::ivory2},
2011 {"ivory3", detail::col::ivory3},
2012 {"ivory4", detail::col::ivory4},
2013 {"khaki", detail::col::khaki},
2014 {"khaki1", detail::col::khaki1},
2015 {"khaki2", detail::col::khaki2},
2016 {"khaki3", detail::col::khaki3},
2017 {"khaki4", detail::col::khaki4},
2018 {"lavender", detail::col::lavender},
2019 {"lavender blush", detail::col::lavender_blush},
2020 {"LavenderBlush", detail::col::LavenderBlush},
2021 {"LavenderBlush1", detail::col::LavenderBlush1},
2022 {"LavenderBlush2", detail::col::LavenderBlush2},
2023 {"LavenderBlush3", detail::col::LavenderBlush3},
2024 {"LavenderBlush4", detail::col::LavenderBlush4},
2025 {"lawn green", detail::col::lawn_green},
2026 {"LawnGreen", detail::col::LawnGreen},
2027 {"lemon chiffon", detail::col::lemon_chiffon},
2028 {"LemonChiffon", detail::col::LemonChiffon},
2029 {"LemonChiffon1", detail::col::LemonChiffon1},
2030 {"LemonChiffon2", detail::col::LemonChiffon2},
2031 {"LemonChiffon3", detail::col::LemonChiffon3},
2032 {"LemonChiffon4", detail::col::LemonChiffon4},
2033 {"light blue", detail::col::light_blue},
2034 {"light coral", detail::col::light_coral},
2035 {"light cyan", detail::col::light_cyan},
2036 {"light goldenrod", detail::col::light_goldenrod},
2037 {"light goldenrod yellow", detail::col::light_goldenrod_yellow},
2038 {"light gray", detail::col::light_gray},
2039 {"light green", detail::col::light_green},
2040 {"light grey", detail::col::light_grey},
2041 {"light pink", detail::col::light_pink},
2042 {"light salmon", detail::col::light_salmon},
2043 {"light sea green", detail::col::light_sea_green},
2044 {"light sky blue", detail::col::light_sky_blue},
2045 {"light slate blue", detail::col::light_slate_blue},
2046 {"light slate gray", detail::col::light_slate_gray},
2047 {"light slate grey", detail::col::light_slate_grey},
2048 {"light steel blue", detail::col::light_steel_blue},
2049 {"light yellow", detail::col::light_yellow},
2050 {"LightBlue", detail::col::LightBlue},
2051 {"LightBlue1", detail::col::LightBlue1},
2052 {"LightBlue2", detail::col::LightBlue2},
2053 {"LightBlue3", detail::col::LightBlue3},
2054 {"LightBlue4", detail::col::LightBlue4},
2055 {"LightCoral", detail::col::LightCoral},
2056 {"LightCyan", detail::col::LightCyan},
2057 {"LightCyan1", detail::col::LightCyan1},
2058 {"LightCyan2", detail::col::LightCyan2},
2059 {"LightCyan3", detail::col::LightCyan3},
2060 {"LightCyan4", detail::col::LightCyan4},
2061 {"LightGoldenrod", detail::col::LightGoldenrod},
2062 {"LightGoldenrod1", detail::col::LightGoldenrod1},
2063 {"LightGoldenrod2", detail::col::LightGoldenrod2},
2064 {"LightGoldenrod3", detail::col::LightGoldenrod3},
2065 {"LightGoldenrod4", detail::col::LightGoldenrod4},
2066 {"LightGoldenrodYellow", detail::col::LightGoldenrodYellow},
2067 {"LightGray", detail::col::LightGray},
2068 {"LightGreen", detail::col::LightGreen},
2069 {"LightGrey", detail::col::LightGrey},
2070 {"LightPink", detail::col::LightPink},
2071 {"LightPink1", detail::col::LightPink1},
2072 {"LightPink2", detail::col::LightPink2},
2073 {"LightPink3", detail::col::LightPink3},
2074 {"LightPink4", detail::col::LightPink4},
2075 {"LightSalmon", detail::col::LightSalmon},
2076 {"LightSalmon1", detail::col::LightSalmon1},
2077 {"LightSalmon2", detail::col::LightSalmon2},
2078 {"LightSalmon3", detail::col::LightSalmon3},
2079 {"LightSalmon4", detail::col::LightSalmon4},
2080 {"LightSeaGreen", detail::col::LightSeaGreen},
2081 {"LightSkyBlue", detail::col::LightSkyBlue},
2082 {"LightSkyBlue1", detail::col::LightSkyBlue1},
2083 {"LightSkyBlue2", detail::col::LightSkyBlue2},
2084 {"LightSkyBlue3", detail::col::LightSkyBlue3},
2085 {"LightSkyBlue4", detail::col::LightSkyBlue4},
2086 {"LightSlateBlue", detail::col::LightSlateBlue},
2087 {"LightSlateGray", detail::col::LightSlateGray},
2088 {"LightSlateGrey", detail::col::LightSlateGrey},
2089 {"LightSteelBlue", detail::col::LightSteelBlue},
2090 {"LightSteelBlue1", detail::col::LightSteelBlue1},
2091 {"LightSteelBlue2", detail::col::LightSteelBlue2},
2092 {"LightSteelBlue3", detail::col::LightSteelBlue3},
2093 {"LightSteelBlue4", detail::col::LightSteelBlue4},
2094 {"LightYellow", detail::col::LightYellow},
2095 {"LightYellow1", detail::col::LightYellow1},
2096 {"LightYellow2", detail::col::LightYellow2},
2097 {"LightYellow3", detail::col::LightYellow3},
2098 {"LightYellow4", detail::col::LightYellow4},
2099 {"lime green", detail::col::lime_green},
2100 {"LimeGreen", detail::col::LimeGreen},
2101 {"linen", detail::col::linen},
2102 {"magenta", detail::col::magenta},
2103 {"magenta1", detail::col::magenta1},
2104 {"magenta2", detail::col::magenta2},
2105 {"magenta3", detail::col::magenta3},
2106 {"magenta4", detail::col::magenta4},
2107 {"maroon", detail::col::maroon},
2108 {"maroon1", detail::col::maroon1},
2109 {"maroon2", detail::col::maroon2},
2110 {"maroon3", detail::col::maroon3},
2111 {"maroon4", detail::col::maroon4},
2112 {"medium aquamarine", detail::col::medium_aquamarine},
2113 {"medium blue", detail::col::medium_blue},
2114 {"medium orchid", detail::col::medium_orchid},
2115 {"medium purple", detail::col::medium_purple},
2116 {"medium sea green", detail::col::medium_sea_green},
2117 {"medium slate blue", detail::col::medium_slate_blue},
2118 {"medium spring green", detail::col::medium_spring_green},
2119 {"medium turquoise", detail::col::medium_turquoise},
2120 {"medium violet red", detail::col::medium_violet_red},
2121 {"MediumAquamarine", detail::col::MediumAquamarine},
2122 {"MediumBlue", detail::col::MediumBlue},
2123 {"MediumOrchid", detail::col::MediumOrchid},
2124 {"MediumOrchid1", detail::col::MediumOrchid1},
2125 {"MediumOrchid2", detail::col::MediumOrchid2},
2126 {"MediumOrchid3", detail::col::MediumOrchid3},
2127 {"MediumOrchid4", detail::col::MediumOrchid4},
2128 {"MediumPurple", detail::col::MediumPurple},
2129 {"MediumPurple1", detail::col::MediumPurple1},
2130 {"MediumPurple2", detail::col::MediumPurple2},
2131 {"MediumPurple3", detail::col::MediumPurple3},
2132 {"MediumPurple4", detail::col::MediumPurple4},
2133 {"MediumSeaGreen", detail::col::MediumSeaGreen},
2134 {"MediumSlateBlue", detail::col::MediumSlateBlue},
2135 {"MediumSpringGreen", detail::col::MediumSpringGreen},
2136 {"MediumTurquoise", detail::col::MediumTurquoise},
2137 {"MediumVioletRed", detail::col::MediumVioletRed},
2138 {"midnight blue", detail::col::midnight_blue},
2139 {"MidnightBlue", detail::col::MidnightBlue},
2140 {"mint cream", detail::col::mint_cream},
2141 {"MintCream", detail::col::MintCream},
2142 {"misty rose", detail::col::misty_rose},
2143 {"MistyRose", detail::col::MistyRose},
2144 {"MistyRose1", detail::col::MistyRose1},
2145 {"MistyRose2", detail::col::MistyRose2},
2146 {"MistyRose3", detail::col::MistyRose3},
2147 {"MistyRose4", detail::col::MistyRose4},
2148 {"moccasin", detail::col::moccasin},
2149 {"navajo white", detail::col::navajo_white},
2150 {"NavajoWhite", detail::col::NavajoWhite},
2151 {"NavajoWhite1", detail::col::NavajoWhite1},
2152 {"NavajoWhite2", detail::col::NavajoWhite2},
2153 {"NavajoWhite3", detail::col::NavajoWhite3},
2154 {"NavajoWhite4", detail::col::NavajoWhite4},
2155 {"navy", detail::col::navy},
2156 {"navy blue", detail::col::navy_blue},
2157 {"NavyBlue", detail::col::NavyBlue},
2158 {"old lace", detail::col::old_lace},
2159 {"OldLace", detail::col::OldLace},
2160 {"olive drab", detail::col::olive_drab},
2161 {"OliveDrab", detail::col::OliveDrab},
2162 {"OliveDrab1", detail::col::OliveDrab1},
2163 {"OliveDrab2", detail::col::OliveDrab2},
2164 {"OliveDrab3", detail::col::OliveDrab3},
2165 {"OliveDrab4", detail::col::OliveDrab4},
2166 {"orange", detail::col::orange},
2167 {"orange red", detail::col::orange_red},
2168 {"orange1", detail::col::orange1},
2169 {"orange2", detail::col::orange2},
2170 {"orange3", detail::col::orange3},
2171 {"orange4", detail::col::orange4},
2172 {"OrangeRed", detail::col::OrangeRed},
2173 {"OrangeRed1", detail::col::OrangeRed1},
2174 {"OrangeRed2", detail::col::OrangeRed2},
2175 {"OrangeRed3", detail::col::OrangeRed3},
2176 {"OrangeRed4", detail::col::OrangeRed4},
2177 {"orchid", detail::col::orchid},
2178 {"orchid1", detail::col::orchid1},
2179 {"orchid2", detail::col::orchid2},
2180 {"orchid3", detail::col::orchid3},
2181 {"orchid4", detail::col::orchid4},
2182 {"pale goldenrod", detail::col::pale_goldenrod},
2183 {"pale green", detail::col::pale_green},
2184 {"pale turquoise", detail::col::pale_turquoise},
2185 {"pale violet red", detail::col::pale_violet_red},
2186 {"PaleGoldenrod", detail::col::PaleGoldenrod},
2187 {"PaleGreen", detail::col::PaleGreen},
2188 {"PaleGreen1", detail::col::PaleGreen1},
2189 {"PaleGreen2", detail::col::PaleGreen2},
2190 {"PaleGreen3", detail::col::PaleGreen3},
2191 {"PaleGreen4", detail::col::PaleGreen4},
2192 {"PaleTurquoise", detail::col::PaleTurquoise},
2193 {"PaleTurquoise1", detail::col::PaleTurquoise1},
2194 {"PaleTurquoise2", detail::col::PaleTurquoise2},
2195 {"PaleTurquoise3", detail::col::PaleTurquoise3},
2196 {"PaleTurquoise4", detail::col::PaleTurquoise4},
2197 {"PaleVioletRed", detail::col::PaleVioletRed},
2198 {"PaleVioletRed1", detail::col::PaleVioletRed1},
2199 {"PaleVioletRed2", detail::col::PaleVioletRed2},
2200 {"PaleVioletRed3", detail::col::PaleVioletRed3},
2201 {"PaleVioletRed4", detail::col::PaleVioletRed4},
2202 {"papaya whip", detail::col::papaya_whip},
2203 {"PapayaWhip", detail::col::PapayaWhip},
2204 {"peach puff", detail::col::peach_puff},
2205 {"PeachPuff", detail::col::PeachPuff},
2206 {"PeachPuff1", detail::col::PeachPuff1},
2207 {"PeachPuff2", detail::col::PeachPuff2},
2208 {"PeachPuff3", detail::col::PeachPuff3},
2209 {"PeachPuff4", detail::col::PeachPuff4},
2210 {"peru", detail::col::peru},
2211 {"pink", detail::col::pink},
2212 {"pink1", detail::col::pink1},
2213 {"pink2", detail::col::pink2},
2214 {"pink3", detail::col::pink3},
2215 {"pink4", detail::col::pink4},
2216 {"plum", detail::col::plum},
2217 {"plum1", detail::col::plum1},
2218 {"plum2", detail::col::plum2},
2219 {"plum3", detail::col::plum3},
2220 {"plum4", detail::col::plum4},
2221 {"powder blue", detail::col::powder_blue},
2222 {"PowderBlue", detail::col::PowderBlue},
2223 {"purple", detail::col::purple},
2224 {"purple1", detail::col::purple1},
2225 {"purple2", detail::col::purple2},
2226 {"purple3", detail::col::purple3},
2227 {"purple4", detail::col::purple4},
2228 {"red", detail::col::red},
2229 {"red1", detail::col::red1},
2230 {"red2", detail::col::red2},
2231 {"red3", detail::col::red3},
2232 {"red4", detail::col::red4},
2233 {"rosy brown", detail::col::rosy_brown},
2234 {"RosyBrown", detail::col::RosyBrown},
2235 {"RosyBrown1", detail::col::RosyBrown1},
2236 {"RosyBrown2", detail::col::RosyBrown2},
2237 {"RosyBrown3", detail::col::RosyBrown3},
2238 {"RosyBrown4", detail::col::RosyBrown4},
2239 {"royal blue", detail::col::royal_blue},
2240 {"RoyalBlue", detail::col::RoyalBlue},
2241 {"RoyalBlue1", detail::col::RoyalBlue1},
2242 {"RoyalBlue2", detail::col::RoyalBlue2},
2243 {"RoyalBlue3", detail::col::RoyalBlue3},
2244 {"RoyalBlue4", detail::col::RoyalBlue4},
2245 {"saddle brown", detail::col::saddle_brown},
2246 {"SaddleBrown", detail::col::SaddleBrown},
2247 {"salmon", detail::col::salmon},
2248 {"salmon1", detail::col::salmon1},
2249 {"salmon2", detail::col::salmon2},
2250 {"salmon3", detail::col::salmon3},
2251 {"salmon4", detail::col::salmon4},
2252 {"sandy brown", detail::col::sandy_brown},
2253 {"SandyBrown", detail::col::SandyBrown},
2254 {"sea green", detail::col::sea_green},
2255 {"SeaGreen", detail::col::SeaGreen},
2256 {"SeaGreen1", detail::col::SeaGreen1},
2257 {"SeaGreen2", detail::col::SeaGreen2},
2258 {"SeaGreen3", detail::col::SeaGreen3},
2259 {"SeaGreen4", detail::col::SeaGreen4},
2260 {"seashell", detail::col::seashell},
2261 {"seashell1", detail::col::seashell1},
2262 {"seashell2", detail::col::seashell2},
2263 {"seashell3", detail::col::seashell3},
2264 {"seashell4", detail::col::seashell4},
2265 {"sienna", detail::col::sienna},
2266 {"sienna1", detail::col::sienna1},
2267 {"sienna2", detail::col::sienna2},
2268 {"sienna3", detail::col::sienna3},
2269 {"sienna4", detail::col::sienna4},
2270 {"sky blue", detail::col::sky_blue},
2271 {"SkyBlue", detail::col::SkyBlue},
2272 {"SkyBlue1", detail::col::SkyBlue1},
2273 {"SkyBlue2", detail::col::SkyBlue2},
2274 {"SkyBlue3", detail::col::SkyBlue3},
2275 {"SkyBlue4", detail::col::SkyBlue4},
2276 {"slate blue", detail::col::slate_blue},
2277 {"slate gray", detail::col::slate_gray},
2278 {"slate grey", detail::col::slate_grey},
2279 {"SlateBlue", detail::col::SlateBlue},
2280 {"SlateBlue1", detail::col::SlateBlue1},
2281 {"SlateBlue2", detail::col::SlateBlue2},
2282 {"SlateBlue3", detail::col::SlateBlue3},
2283 {"SlateBlue4", detail::col::SlateBlue4},
2284 {"SlateGray", detail::col::SlateGray},
2285 {"SlateGray1", detail::col::SlateGray1},
2286 {"SlateGray2", detail::col::SlateGray2},
2287 {"SlateGray3", detail::col::SlateGray3},
2288 {"SlateGray4", detail::col::SlateGray4},
2289 {"SlateGrey", detail::col::SlateGrey},
2290 {"snow", detail::col::snow},
2291 {"snow1", detail::col::snow1},
2292 {"snow2", detail::col::snow2},
2293 {"snow3", detail::col::snow3},
2294 {"snow4", detail::col::snow4},
2295 {"spring green", detail::col::spring_green},
2296 {"SpringGreen", detail::col::SpringGreen},
2297 {"SpringGreen1", detail::col::SpringGreen1},
2298 {"SpringGreen2", detail::col::SpringGreen2},
2299 {"SpringGreen3", detail::col::SpringGreen3},
2300 {"SpringGreen4", detail::col::SpringGreen4},
2301 {"steel blue", detail::col::steel_blue},
2302 {"SteelBlue", detail::col::SteelBlue},
2303 {"SteelBlue1", detail::col::SteelBlue1},
2304 {"SteelBlue2", detail::col::SteelBlue2},
2305 {"SteelBlue3", detail::col::SteelBlue3},
2306 {"SteelBlue4", detail::col::SteelBlue4},
2307 {"tan", detail::col::tan},
2308 {"tan1", detail::col::tan1},
2309 {"tan2", detail::col::tan2},
2310 {"tan3", detail::col::tan3},
2311 {"tan4", detail::col::tan4},
2312 {"thistle", detail::col::thistle},
2313 {"thistle1", detail::col::thistle1},
2314 {"thistle2", detail::col::thistle2},
2315 {"thistle3", detail::col::thistle3},
2316 {"thistle4", detail::col::thistle4},
2317 {"tomato", detail::col::tomato},
2318 {"tomato1", detail::col::tomato1},
2319 {"tomato2", detail::col::tomato2},
2320 {"tomato3", detail::col::tomato3},
2321 {"tomato4", detail::col::tomato4},
2322 {"turquoise", detail::col::turquoise},
2323 {"turquoise1", detail::col::turquoise1},
2324 {"turquoise2", detail::col::turquoise2},
2325 {"turquoise3", detail::col::turquoise3},
2326 {"turquoise4", detail::col::turquoise4},
2327 {"violet", detail::col::violet},
2328 {"violet red", detail::col::violet_red},
2329 {"VioletRed", detail::col::VioletRed},
2330 {"VioletRed1", detail::col::VioletRed1},
2331 {"VioletRed2", detail::col::VioletRed2},
2332 {"VioletRed3", detail::col::VioletRed3},
2333 {"VioletRed4", detail::col::VioletRed4},
2334 {"wheat", detail::col::wheat},
2335 {"wheat1", detail::col::wheat1},
2336 {"wheat2", detail::col::wheat2},
2337 {"wheat3", detail::col::wheat3},
2338 {"wheat4", detail::col::wheat4},
2339 {"white", detail::col::white},
2340 {"white smoke", detail::col::white_smoke},
2341 {"WhiteSmoke", detail::col::WhiteSmoke},
2342 {"yellow", detail::col::yellow},
2343 {"yellow green", detail::col::yellow_green},
2344 {"yellow1", detail::col::yellow1},
2345 {"yellow2", detail::col::yellow2},
2346 {"yellow3", detail::col::yellow3},
2347 {"yellow4", detail::col::yellow4},
2348 {"YellowGreen", detail::col::YellowGreen}
2349 };
2350
2355 inline Color randomColor() {
2356 static std::default_random_engine rng(detail::epochTime());
2357 static std::uniform_int_distribution<int> rng_dist(0, 255);
2358 return Color((uint8_t) rng_dist(rng), (uint8_t) rng_dist(rng), (uint8_t) rng_dist(rng));
2359 }
2360
2366 inline Color fromName(const std::string& name) {
2367 if(name == "random")
2368 return randomColor();
2369
2370 if(NAMED_COLORS.count(name))
2371 return NAMED_COLORS.at(name);
2372
2373 throw std::runtime_error("No color by the name \"" + name + "\" exists.");
2374 }
2375
2376 //Define named color constructor after the definition of the named color map.
2377
2378 /*\brief Name constructor.
2379 Takes a literal color name as an input.
2380 \param name The name of the color from which to derive value.
2381 \sa fromName()*/
2382 Color::Color(const std::string& name) {
2383 const Color c = fromName(name);
2384 r = c.r;
2385 g = c.g;
2386 b = c.b;
2387 }
2388
2389 //SECTION: USER IO
2390
2391 /*Mouse event callback type.*/
2392 typedef std::function<void(int, int) > MouseFunc;
2393
2394 /*Keyboard event callback type.*/
2395 typedef std::function<void() > KeyFunc;
2396
2397 /*Timer event callback type.*/
2398 typedef std::function<void(void) > TimerFunc;
2399
2403 enum KeyboardKey {
2404 KEY_ESC = cimg_library::cimg::keyESC,
2405 KEY_F1 = cimg_library::cimg::keyF1,
2406 KEY_F2 = cimg_library::cimg::keyF2,
2407 KEY_F3 = cimg_library::cimg::keyF3,
2408 KEY_F4 = cimg_library::cimg::keyF4,
2409 KEY_F5 = cimg_library::cimg::keyF5,
2410 KEY_F6 = cimg_library::cimg::keyF6,
2411 KEY_F7 = cimg_library::cimg::keyF7,
2412 KEY_F8 = cimg_library::cimg::keyF8,
2413 KEY_F9 = cimg_library::cimg::keyF9,
2414 KEY_F10 = cimg_library::cimg::keyF10,
2415 KEY_F11 = cimg_library::cimg::keyF11,
2416 KEY_F12 = cimg_library::cimg::keyF12,
2417 KEY_PAUSE = cimg_library::cimg::keyPAUSE,
2418 KEY_1 = cimg_library::cimg::key1,
2419 KEY_2 = cimg_library::cimg::key2,
2420 KEY_3 = cimg_library::cimg::key3,
2421 KEY_4 = cimg_library::cimg::key4,
2422 KEY_5 = cimg_library::cimg::key5,
2423 KEY_6 = cimg_library::cimg::key6,
2424 KEY_7 = cimg_library::cimg::key7,
2425 KEY_8 = cimg_library::cimg::key8,
2426 KEY_9 = cimg_library::cimg::key9,
2427 KEY_0 = cimg_library::cimg::key0,
2428 KEY_BACKSPACE = cimg_library::cimg::keyBACKSPACE,
2429 KEY_INSERT = cimg_library::cimg::keyINSERT,
2430 KEY_HOME = cimg_library::cimg::keyHOME,
2431 KEY_PAGEUP = cimg_library::cimg::keyPAGEUP,
2432 KEY_TAB = cimg_library::cimg::keyTAB,
2433 KEY_Q = cimg_library::cimg::keyQ,
2434 KEY_W = cimg_library::cimg::keyW,
2435 KEY_E = cimg_library::cimg::keyE,
2436 KEY_R = cimg_library::cimg::keyR,
2437 KEY_T = cimg_library::cimg::keyT,
2438 KEY_Y = cimg_library::cimg::keyY,
2439 KEY_U = cimg_library::cimg::keyU,
2440 KEY_I = cimg_library::cimg::keyI,
2441 KEY_O = cimg_library::cimg::keyO,
2442 KEY_P = cimg_library::cimg::keyP,
2443 KEY_DELETE = cimg_library::cimg::keyDELETE,
2444 KEY_END = cimg_library::cimg::keyEND,
2445 KEY_PAGEDOWN = cimg_library::cimg::keyPAGEDOWN,
2446 KEY_CAPSLOCK = cimg_library::cimg::keyCAPSLOCK,
2447 KEY_A = cimg_library::cimg::keyA,
2448 KEY_S = cimg_library::cimg::keyS,
2449 KEY_D = cimg_library::cimg::keyD,
2450 KEY_F = cimg_library::cimg::keyF,
2451 KEY_G = cimg_library::cimg::keyG,
2452 KEY_H = cimg_library::cimg::keyH,
2453 KEY_J = cimg_library::cimg::keyJ,
2454 KEY_K = cimg_library::cimg::keyK,
2455 KEY_L = cimg_library::cimg::keyL,
2456 KEY_ENTER = cimg_library::cimg::keyENTER,
2457 KEY_SHIFTLEFT = cimg_library::cimg::keySHIFTLEFT,
2458 KEY_Z = cimg_library::cimg::keyZ,
2459 KEY_X = cimg_library::cimg::keyX,
2460 KEY_C = cimg_library::cimg::keyC,
2461 KEY_V = cimg_library::cimg::keyV,
2462 KEY_B = cimg_library::cimg::keyB,
2463 KEY_N = cimg_library::cimg::keyN,
2464 KEY_M = cimg_library::cimg::keyM,
2465 KEY_SHIFTRIGHT = cimg_library::cimg::keySHIFTRIGHT,
2466 KEY_ARROWUP = cimg_library::cimg::keyARROWUP,
2467 KEY_CTRLLEFT = cimg_library::cimg::keyCTRLLEFT,
2468 KEY_APPLEFT = cimg_library::cimg::keyAPPLEFT,
2469 KEY_ALT = cimg_library::cimg::keyALT,
2470 KEY_SPACE = cimg_library::cimg::keySPACE,
2471 KEY_ALTGR = cimg_library::cimg::keyALTGR,
2472 KEY_APPRIGHT = cimg_library::cimg::keyAPPRIGHT,
2473 KEY_MENU = cimg_library::cimg::keyMENU,
2474 KEY_CTRLRIGHT = cimg_library::cimg::keyCTRLRIGHT,
2475 KEY_ARROWLEFT = cimg_library::cimg::keyARROWLEFT,
2476 KEY_ARROWDOWN = cimg_library::cimg::keyARROWDOWN,
2477 KEY_ARROWRIGHT = cimg_library::cimg::keyARROWRIGHT,
2478 KEY_PAD0 = cimg_library::cimg::keyPAD0,
2479 KEY_PAD1 = cimg_library::cimg::keyPAD1,
2480 KEY_PAD2 = cimg_library::cimg::keyPAD2,
2481 KEY_PAD3 = cimg_library::cimg::keyPAD3,
2482 KEY_PAD4 = cimg_library::cimg::keyPAD4,
2483 KEY_PAD5 = cimg_library::cimg::keyPAD5,
2484 KEY_PAD6 = cimg_library::cimg::keyPAD6,
2485 KEY_PAD7 = cimg_library::cimg::keyPAD7,
2486 KEY_PAD8 = cimg_library::cimg::keyPAD8,
2487 KEY_PAD9 = cimg_library::cimg::keyPAD9,
2488 KEY_PADADD = cimg_library::cimg::keyPADADD,
2489 KEY_PADSUB = cimg_library::cimg::keyPADSUB,
2490 KEY_PADMUL = cimg_library::cimg::keyPADMUL,
2491 KEY_PADDIV = cimg_library::cimg::keyPADDIV
2492 };
2493
2494 const std::unordered_map<std::string, KeyboardKey> NAMED_KEYS = {
2495 {"ESC", KEY_ESC},
2496 {"F1", KEY_F1},
2497 {"F2", KEY_F2},
2498 {"F3", KEY_F3},
2499 {"F4", KEY_F4},
2500 {"F5", KEY_F5},
2501 {"F6", KEY_F6},
2502 {"F7", KEY_F7},
2503 {"F8", KEY_F8},
2504 {"F9", KEY_F9},
2505 {"F10", KEY_F10},
2506 {"F11", KEY_F11},
2507 {"F12", KEY_F12},
2508 {"PAUSE", KEY_PAUSE},
2509 {"1", KEY_1},
2510 {"2", KEY_2},
2511 {"3", KEY_3},
2512 {"4", KEY_4},
2513 {"5", KEY_5},
2514 {"6", KEY_6},
2515 {"7", KEY_7},
2516 {"8", KEY_8},
2517 {"9", KEY_9},
2518 {"0", KEY_0},
2519 {"BACKSPACE", KEY_BACKSPACE},
2520 {"INSERT", KEY_INSERT},
2521 {"HOME", KEY_HOME},
2522 {"PAGEUP", KEY_PAGEUP},
2523 {"TAB", KEY_TAB},
2524 {"Q", KEY_Q},
2525 {"W", KEY_W},
2526 {"E", KEY_E},
2527 {"R", KEY_R},
2528 {"T", KEY_T},
2529 {"Y", KEY_Y},
2530 {"U", KEY_U},
2531 {"I", KEY_I},
2532 {"O", KEY_O},
2533 {"P", KEY_P},
2534 {"DELETE", KEY_DELETE},
2535 {"END", KEY_END},
2536 {"PAGEDOWN", KEY_PAGEDOWN},
2537 {"CAPSLOCK", KEY_CAPSLOCK},
2538 {"A", KEY_A},
2539 {"S", KEY_S},
2540 {"D", KEY_D},
2541 {"F", KEY_F},
2542 {"G", KEY_G},
2543 {"H", KEY_H},
2544 {"J", KEY_J},
2545 {"K", KEY_K},
2546 {"L", KEY_L},
2547 {"ENTER", KEY_ENTER},
2548 {"SHIFTLEFT", KEY_SHIFTLEFT},
2549 {"Z", KEY_Z},
2550 {"X", KEY_X},
2551 {"C", KEY_C},
2552 {"V", KEY_V},
2553 {"B", KEY_B},
2554 {"N", KEY_N},
2555 {"M", KEY_M},
2556 {"SHIFTRIGHT", KEY_SHIFTRIGHT},
2557 {"ARROWUP", KEY_ARROWUP},
2558 {"CTRLLEFT", KEY_CTRLLEFT},
2559 {"APPLEFT", KEY_APPLEFT},
2560 {"ALT", KEY_ALT},
2561 {"SPACE", KEY_SPACE},
2562 {"ALTGR", KEY_ALTGR},
2563 {"APPRIGHT", KEY_APPRIGHT},
2564 {"MENU", KEY_MENU},
2565 {"CTRLRIGHT", KEY_CTRLRIGHT},
2566 {"ARROWLEFT", KEY_ARROWLEFT},
2567 {"ARROWDOWN", KEY_ARROWDOWN},
2568 {"ARROWRIGHT", KEY_ARROWRIGHT},
2569 {"PAD0", KEY_PAD0},
2570 {"PAD1", KEY_PAD1},
2571 {"PAD2", KEY_PAD2},
2572 {"PAD3", KEY_PAD3},
2573 {"PAD4", KEY_PAD4},
2574 {"PAD5", KEY_PAD5},
2575 {"PAD6", KEY_PAD6},
2576 {"PAD7", KEY_PAD7},
2577 {"PAD8", KEY_PAD8},
2578 {"PAD9", KEY_PAD9},
2579 {"PADADD", KEY_PADADD},
2580 {"PADSUB", KEY_PADSUB},
2581 {"PADMUL", KEY_PADMUL},
2582 {"PADDIV", KEY_PADDIV}
2583 };
2584
2588 enum MouseButton {//Stored as bitwise flags from CImgDisplay
2589 MOUSEB_LEFT, //Left Mouse Button
2590 MOUSEB_RIGHT, //Right Mouse Button
2591 MOUSEB_MIDDLE//Middle Mouse Button
2592 };
2593
2598 struct InputEvent {
2599 //True for keyboard, false for mouse
2600 bool type = false;
2601 //mouseX, mouseY
2602 int mX = 0;
2603 int mY = 0;
2604 /*void callback pointer. cast and called when processed.*/
2605 void* cbPointer = nullptr;
2606 };
2607
2613 inline KeyboardKey keyFromName(const std::string& name) {
2614 return NAMED_KEYS.at(name);
2615 }
2616
2620 struct ivec2 {
2621 union{
2622 struct{
2623 int x, y;
2624 };
2625 int data[2];
2626 };
2627
2629 ivec2() : x(0), y(0){}
2630
2634 ivec2(int x, int y) : x(x), y(y) {}
2635
2636 /*Array access operator overload.*/
2637
2641 inline int& operator[](int index){
2642 return data[index];
2643 }
2644
2648 inline int operator[](int index) const{
2649 return data[index];
2650 }
2651
2652 ivec2 operator+(const ivec2& other) const{
2653 return {x + other.x, y + other.y};
2654 }
2655
2656 ivec2& operator+=(const ivec2& other){
2657 x += other.x;
2658 y += other.y;
2659 return *this;
2660 }
2661
2662 ivec2 operator-(const ivec2& other) const{
2663 return {x - other.x, y - other.y};
2664 }
2665
2666 ivec2& operator-=(const ivec2& other){
2667 x -= other.x;
2668 y -= other.y;
2669 return *this;
2670 }
2671
2673 bool operator==(const ivec2& other) const{
2674 return x == other.x && y == other.y;
2675 }
2676 };
2677
2682 inline int distance(const ivec2& a, const ivec2& b) {
2683 return static_cast<int>(std::sqrt(std::pow(b.x - a.x, 2) + std::pow(b.y - a.y, 2)));
2684 }
2685
2690 inline ivec2 middle(const ivec2& a, const ivec2& b) {
2691 return {(a.x + b.x) / 2, (a.y + b.y) / 2};
2692 }
2693
2700 inline ivec2 lerp(const ivec2& a, const ivec2& b, float progress){
2701 if (progress <= 0)
2702 return a;
2703 else if (progress >= 1)
2704 return b;
2705 return {
2706 static_cast<int>(std::round(progress * static_cast<float>(b.x - a.x))) + a.x,
2707 static_cast<int>(std::round(progress * static_cast<float>(b.y - a.y))) + a.y
2708 };
2709 }
2710
2712 typedef ivec2 Point;
2713
2721 public:
2725 identity();
2726 }
2727
2730 Transform(const Transform& other) = default;
2731
2737 Transform(const ivec2& point, float rotation)
2738 : value(){
2739 identity();
2740 setTranslation(point.x, point.y);
2742 }
2743
2751 value.fill(0.0f);
2752 at(0, 0) = at(1, 1) = 1.0f;
2753 rotation = 0;
2754 return *this;
2755 }
2756
2759 bool operator==(const Transform& other) const {
2760 bool eq = true;
2761 for (int i = 0; i < 9; i++) {
2762 if (value[i] != other.value[i]) {
2763 eq = false;
2764 break;
2765 }
2766 }
2767 return eq;
2768 }
2769
2772 float getScaleX() const {
2773 return at(0, 0);
2774 }
2775
2778 float getScaleY() const {
2779 return at(1, 1);
2780 }
2781
2784 float getTranslateX() const {
2785 return at(0, 2);
2786 }
2787
2790 float getTranslateY() const {
2791 return at(1, 2);
2792 }
2793
2796 float getRotation() const {
2797 return rotation;
2798 }
2799
2801 Transform& forward(float distance) {
2802 at(0, 2) += std::cos(rotation) * distance; //x component
2803 at(1, 2) += std::sin(rotation) * distance; //y component
2804 return *this;
2805 }
2806
2807 /*Backwards inline function.
2808 Just negates the input of a forward function call.*/
2809 inline Transform& backward(float distance) {
2810 return forward(-distance);
2811 }
2812
2817 Transform& setTranslation(int x, int y) {
2818 at(0, 2) = static_cast<float>(x);
2819 at(1, 2) = static_cast<float>(y);
2820 return *this;
2821 }
2822
2826 return {(int) std::round(at(0, 2)), (int) std::round(at(1, 2))};
2827 }
2828
2833 at(0, 2) = static_cast<float>(x);
2834 return *this;
2835 }
2836
2841 at(1, 2) = static_cast<float>(y);
2842 return *this;
2843 }
2844
2849 Transform& translate(int x, int y) {
2850 at(0, 2) += static_cast<float>(x) * at(0, 0) + static_cast<float>(y) * at(0, 1);
2851 at(1, 2) += static_cast<float>(x) * at(1, 0) + static_cast<float>(y) * at(1, 1);
2852 return *this;
2853 }
2854
2858 Transform& rotate(float theta) {
2859 //6.28319 is a full rotation in radians. (360 degrees)
2860 constexpr float fullcircle = 6.28319f;
2861
2862 //Much smarter solution than recursive spinning.
2863 //Takes the modulus between what would have been the pre-fix result
2864 //and a full circle, and subtracts the original rotation.
2865 //This gives pretty accurate rotations rather quickly.
2866 //No recursive spinning required! :)
2867 const float origResult = rotation + theta;
2868 if (origResult > fullcircle || origResult < 0)
2869 theta = std::fmod(origResult, fullcircle) - rotation;
2870
2871 const float c = std::cos(theta);
2872 const float s = std::sin(theta);
2873
2874 const float new00 = at(0, 0) * c + at(0, 1) * s;
2875 const float new01 = at(0, 0) * -s + at(0, 1) * c;
2876 const float new10 = at(1, 0) * c + at(1, 1) * s;
2877 const float new11 = at(1, 0) * -s + at(1, 1) * c;
2878
2879 at(0, 0) = new00;//x
2880 at(0, 1) = new01;//y
2881 at(1, 0) = new10;//rotX
2882 at(1, 1) = new11;//rotY
2883
2884 rotation += theta;
2885
2886 return *this;
2887 }
2888
2893 if (val == rotation)
2894 return *this;
2895 if (rotation != 0.0f)
2896 rotate(-rotation);
2897 rotate(val);
2898 return *this;
2899 }
2900
2906 Transform& rotateAround(int x, int y, float theta) {
2907 translate(x, y);
2908 rotate(theta);
2909 translate(-x, -y);
2910 return *this;
2911 }
2912
2916 Transform& scale(float sx, float sy) {
2917 at(0, 0) *= sx;
2918 at(0, 1) *= sy;
2919 at(1, 0) *= sx;
2920 at(1, 1) *= sy;
2921 return *this;
2922 }
2923
2928 const float new00 = at(0, 0) * t.at(0, 0) + at(0, 1) * t.at(1, 0);
2929 const float new01 = at(0, 0) * t.at(0, 1) + at(0, 1) * t.at(1, 1);
2930 const float new02 = at(0, 0) * t.at(0, 2) + at(0, 1) * t.at(1, 2) + at(0, 2);
2931 const float new10 = at(1, 0) * t.at(0, 0) + at(1, 1) * t.at(1, 0);
2932 const float new11 = at(1, 0) * t.at(0, 1) + at(1, 1) * t.at(1, 1);
2933 const float new12 = at(1, 0) * t.at(0, 2) + at(1, 1) * t.at(1, 2) + at(1, 2);
2934
2935 at(0, 0) = new00;
2936 at(0, 1) = new01;
2937 at(0, 2) = new02;
2938 at(1, 0) = new10;
2939 at(1, 1) = new11;
2940 at(1, 2) = new12;
2941 rotation += t.rotation;
2942 return *this;
2943 }
2944
2949 Transform copy;
2950 copy.assign(*this);
2951 copy.concatenate(t);
2952 return copy;
2953 }
2954
2960 Transform lerp(const Transform& t, float progress) const {
2961 if (progress <= 0)
2962 return *this;
2963 else if (progress >= 1)
2964 return t;
2965 Transform result;
2966 for (int i = 0; i < 9; i++) {
2967 result.value[i] = (progress * (t.value[i] - value[i])) + value[i];
2968 }
2969 return result;
2970 }
2971
2974 void assign(const Transform& t) {
2975 value = t.value;
2976 rotation = t.rotation;
2977 }
2978
2984 Point transform(Point in, Point* dst = nullptr) const {
2985 Point temp;
2986 Point* dstPtr = (dst == nullptr) ? &temp : dst;
2987
2988 dstPtr->x = static_cast<int>(
2989 at(0, 0) * (static_cast<float>(in.x)) +
2990 at(0, 1) * (static_cast<float>(in.y)) + at(0, 2));
2991 dstPtr->y = static_cast<int>(
2992 at(1, 0) * (static_cast<float>(in.x)) +
2993 at(1, 1) * (static_cast<float>(in.y)) + at(1, 2));
2994
2995 return *dstPtr;
2996 }
2997
3001 template<typename ITER_T>
3002 void transformSet(ITER_T cur, ITER_T end) const {
3003 while (cur != end) {
3004 transform(&(*cur), &(*cur));
3005 cur++;
3006 }
3007 }
3008
3009 /*Operator overload to transform a single point, for convenience.*/
3010
3013 inline Point operator()(Point in) const {
3014 return transform(in);
3015 }
3016 protected:
3020 typedef std::array<float, 9> mat_t;
3021
3024
3026 float rotation = 0;
3027
3031 inline float& at(int row, int col) {
3032 return value[row * 3 + col];
3033 }
3034
3038 float at(int row, int col) const {
3039 return value[row * 3 + col];
3040 }
3041 };
3042
3048 template<typename T>
3049 inline T toRadians(T val) {
3050 return T(val * (M_PI / 180.0));
3051 }
3052
3058 template<typename T>
3059 inline T toDegrees(T val) {
3060 return std::round(T(val * (180.0 / M_PI)));
3061 }
3062
3063
3072 inline void drawLine(Image& imgRef, int x1, int y1, int x2, int y2, const Color& c, int width = 1) {
3073 if(x1 == x2 && y1 == y2)
3074 return;
3075 else if (width == 1) {
3076 //Just use the built-in bresenham line function
3077 //to draw line with widths of 1.
3078 imgRef.draw_line(x1, y1, x2, y2, c.rgbPtr());
3079 return;
3080 }
3081
3082 const int radius = width / 2;//integer division, be careful here...
3083 cimg::CImg<int> lineGeom(4, 2);
3084
3085 //convert line (p1, p2) to polygon (p1,p2,p3,p4)... huzzah, O(1) implementation!
3086 //start with two transforms (one for each coordinate pair), rotated to face towards one-another,
3087 //with an added 90-degree rotation (1.571~ ish radians).
3088
3089 Transform transforms[2] = {
3090 {{x1, y1}, std::atan2(static_cast<float>(y2 - y1), static_cast<float>(x2 - x1)) + 1.57079633f},
3091 {{x2, y2}, std::atan2(static_cast<float>(y1 - y2), static_cast<float>(x1 - x2)) + 1.57079633f}
3092 };
3093 Point temp[2];
3094
3095 for(int i = 0; i < 2; i++){//for both of the transforms...
3096 Transform& trans = transforms[i];
3097
3098 //move it forward and back, getting the adjacent corners of the polygon line
3099 trans.forward(static_cast<float>(radius));
3100 temp[0] = trans.getTranslation();
3101
3102 trans.backward(static_cast<float>(radius * 2));
3103 temp[1] = trans.getTranslation();
3104
3105 //then, using a loop, copy our temporary points to the point image.
3106 //the first transform (pt a) are indices 0, 1
3107 //the second transform (pt b) are indices 2, 3
3108 //this ensures proper cw/ccw vertex ordering.
3109 for(int j = 0; j < 2; j++){
3110 lineGeom((i * 2) + j, 0) = temp[j][0];
3111 lineGeom((i * 2) + j, 1) = temp[j][1];
3112 }
3113 }
3114
3115 //draw the rounded caps and the fill polygon
3116 imgRef.draw_circle(x1, y1, radius, c.rgbPtr());//circle 1
3117 imgRef.draw_polygon(lineGeom, c.rgbPtr());//line fill
3118 imgRef.draw_circle(x2, y2, radius, c.rgbPtr());//circle 2
3119 }
3120
3127 public:
3136 BitmapFont(const Image& img, int asciiOffs, int glyphWidth, int glyphHeight, int glyphsX, int glyphsY)
3137 : asciiOffset(asciiOffs),
3138 glyphWidth(glyphWidth), glyphHeight(glyphHeight),
3139 glyphsX(glyphsX), glyphsY(glyphsY){
3140
3141 //fill the glyphs vector...
3142 const ivec2 glyphSz = {glyphWidth, glyphHeight};
3143
3144 for(unsigned char c = static_cast<char>(asciiOffset); c < UINT8_MAX; c++){
3145 const ivec2 min = getGlyphPosition(c);
3146 const ivec2 max = (min + glyphSz) - ivec2(1, 1);
3147 glyphs.push_back(img.get_crop(
3148 min.x, min.y,
3149 max.x, max.y));
3150 }
3151 }
3152
3157 const Image& getGlyphImage(unsigned char c) const{
3158 return glyphs.at(c - asciiOffset);
3159 }
3160
3165 inline const Image& operator[](unsigned char c) const{
3166 return getGlyphImage(c);
3167 }
3168
3174 ivec2 getGlyphPosition(unsigned char c) const{
3175 return {((c - asciiOffset) % glyphsX) * glyphWidth,
3176 (static_cast<int>(std::floor(float(c - asciiOffset) / (float)glyphsX))) * glyphHeight
3177 };
3178 }
3179
3185 return {glyphWidth, glyphHeight};
3186 }
3187
3191 int getTotalGlyphs() const{
3192 return glyphsX * glyphsY;
3193 }
3194
3198 bool isValid(char c) const{
3199 return glyphs.size() > (c - asciiOffset);
3200 }
3201
3207 return {glyphsX, glyphsY};
3208 }
3209 private:
3210 std::vector<Image> glyphs;
3211 int asciiOffset, glyphWidth, glyphHeight, glyphsX, glyphsY;
3212 };
3213
3220 public:
3221
3224
3227
3230
3232 virtual ~AbstractDrawableObject() = default;
3233
3236 virtual AbstractDrawableObject* copy() const = 0;
3237
3246 virtual void draw(const Transform& t, Image& imgRef) const = 0;
3247
3248 protected:
3251 };
3252
3257 enum TextAlign{
3258 TEXT_ALIGN_LEFT,
3259 TEXT_ALIGN_RIGHT,
3260 TEXT_ALIGN_CENTER
3261 };
3262
3265 public:
3267 const std::string text;
3268 const BitmapFont& font;
3269 TextAlign alignment;
3270 float scale;
3271
3272 Text(std::string text, const BitmapFont& font, const Color& color, float scale = 1.0f, TextAlign alignment = TEXT_ALIGN_LEFT)
3273 : text(std::move(text)), font(font), scale(scale), alignment(alignment){
3274 fillColor = color;
3275 }
3276
3277 Text(const Text& copy) = default;
3278
3279 AbstractDrawableObject* copy() const override{
3280 return new Text(*this);
3281 }
3282
3283 void draw(const Transform& t, Image& imgRef) const override{
3284 //keep track of the length of the longest line of text...
3285 int longestLine = 0;
3286
3287 //First, split the text into lines.
3288 std::string temp;
3289 std::list<std::string> textLines;
3290 std::stringstream ss(text);
3291 while(std::getline(ss, temp, '\n')){
3292 longestLine = std::max(longestLine, static_cast<int>(temp.size()));
3293 textLines.push_back(temp);
3294 }
3295
3296 //From there we can get some basic metrics.
3297 const ivec2 glyphSz = font.getGlyphExtent();//the size of a single glyph
3298 const int lines = textLines.size();//the total number of lines of text
3299 const int strPixLen = glyphSz.x * longestLine;//the length, in pixels, of the longest line of text
3300
3301 //Create the temporary image.
3302 //we double the temp image size to get a proper final rotation.
3303 Image textImage(strPixLen * 2, (glyphSz.y * lines) * 2);
3304 textImage.channels(0, 3);//force RGBA on temp image
3305
3306 int line = 0;
3307 //blit each character into the temporary image
3308
3309 auto lineIter = textLines.begin();
3310 while(lineIter != textLines.end()){
3311 const std::string& lineText = *lineIter;
3312
3313 //lineText alignment with some relatively simple maths.
3314 int hOffset = 0;
3315
3316 switch(alignment){
3317 case TEXT_ALIGN_LEFT:
3318 break;//left align left needs no horizontal offset...
3319 case TEXT_ALIGN_RIGHT:
3320 hOffset = strPixLen - (static_cast<int>(lineText.size()) * glyphSz.x);
3321 break;
3322 case TEXT_ALIGN_CENTER:
3323 hOffset = (strPixLen / 2) - ((static_cast<int>(lineText.size()) * glyphSz.x) / 2);
3324 break;
3325 }
3326
3327 for(int i = 0; i < lineText.size(); i++) {
3328 const char curChar = lineText[i];
3329 if(curChar == ' ' || !font.isValid(curChar))//skip space or out-of-range characters...
3330 continue;
3331
3332 const Image& tempGlyph = font[curChar];
3333 const ivec2 destPosition = {strPixLen + hOffset + (i * glyphSz.x), line * glyphSz.y};
3334 textImage.draw_image(destPosition.x, destPosition.y,
3335 tempGlyph);
3336
3337 //Apply the fill color with a multiplication filter for the size of each glyph.
3338 //for each pixel in the previously written glyph, multiply its color...
3339 for(int y = destPosition.y; y < destPosition.y + tempGlyph.height(); y++)
3340 for(int x = destPosition.x; x < destPosition.x + tempGlyph.width(); x++)
3341 for(int c = 0; c < 3; c++)
3342 textImage(x,y,c) *= (static_cast<float>(fillColor.components[c]) / float(UINT8_MAX));
3343 }
3344
3345 lineIter++;
3346 line++;
3347 }
3348
3349 //resize image according to scale
3350 textImage.resize(static_cast<int>(std::round(static_cast<float>(textImage.width()) * scale)),
3351 static_cast<int>(std::round(static_cast<float>(textImage.height()) * scale)));
3352
3353 //rotate the image with nearest-neighbor interpolation
3354 textImage.rotate(-toDegrees(t.getRotation()), 1, 0);
3355
3356 const Point translation = t.getTranslation();
3357 //draw the image centered
3358 //rotating a doubly-sized image makes the origin of the rotation essentially halfway through the image
3359 //therefore, to draw at the proper location, we need to center it relative to the transform location.
3360 imgRef.draw_image(
3361 translation.x - (textImage.width() / 2),
3362 translation.y - (textImage.height() / 2),
3363 textImage, textImage.get_shared_channel(3), 1, 255);
3364 }
3365
3366 ~Text() override = default;
3367 };
3368
3372 public:
3379
3381 int width = 1;
3382
3384 Line() = default;
3385
3390 Line(Point a, Point b, const Color& color, int width = 1) : pointA(a), pointB(b), width(width){
3391 fillColor = color;
3392 }
3393
3397 Line(const Line& other) = default;
3398
3399 AbstractDrawableObject* copy() const override{
3400 return new Line(*this);
3401 }
3402
3404 ~Line() override = default;
3405
3406 void draw(const Transform& t, Image& imgRef) const override{
3407 const Point a = t(pointA);
3408 const Point b = t(pointB);
3409 drawLine(imgRef, a.x, a.y, b.x, b.y, fillColor, width);
3410 }
3411 };
3412
3416 public:
3418 int radius = 10;
3421 int steps = 10;
3422
3424 Circle() = default;
3425
3429 Circle(int radius, int steps, const Color& fillColor, int outlineWidth = 0, const Color& outlineColor = Color())
3430 : radius(radius), steps(steps){
3431 this->fillColor = fillColor;
3432 this->outlineWidth = outlineWidth;
3433 this->outlineColor = outlineColor;
3434 }
3435
3438 Circle(const Circle& other) = default;
3439
3440 AbstractDrawableObject* copy() const override{
3441 return new Circle(*this);
3442 }
3443
3444 void draw(const Transform& t, Image& imgRef) const override{
3445 if (steps <= 0)
3446 return; //no step check
3447 cimg::CImg<int> passPts(steps, 2);
3448
3449 for (int i = 0; i < steps; i++) {
3450 Point p;
3451 p.x = int(radius * std::cos(i * (2 * M_PI) / steps));
3452 p.y = int(radius * std::sin(i * (2 * M_PI) / steps));
3453 Point tPoint = t(p);
3454 passPts(i, 0) = tPoint.x;
3455 passPts(i, 1) = tPoint.y;
3456 }
3457
3458 imgRef.draw_polygon(passPts, fillColor.rgbPtr());
3459
3460 if (outlineWidth > 0) {//draw outline using previously generated points.
3461 //LineLoop impl
3462 for (int i = 1; i < steps; i++) {
3463 drawLine(imgRef, passPts(i - 1, 0), passPts(i - 1, 1), passPts(i, 0), passPts(i, 1), outlineColor, outlineWidth);
3464 }
3465 //draw last line between first and last
3466 drawLine(imgRef, passPts(steps - 1, 0), passPts(steps - 1, 1), passPts(0, 0), passPts(0, 1), outlineColor, outlineWidth);
3467 }
3468 }
3469 };
3470
3476 public:
3477 std::vector<Point> points;
3478
3480 Polygon() = default;
3481
3485 Polygon(const std::initializer_list<Point>& init)
3486 : points(init){
3487 this->outlineWidth = outlineWidth;
3488 this->outlineColor = outlineColor;
3489 this->fillColor = fillColor;
3490 }
3491
3494 Polygon(std::vector<Point> copy, const Color& fillColor, int outlineWidth = 0, const Color& outlineColor = Color())
3495 : points(std::move(copy)){
3496 this->outlineWidth = outlineWidth;
3497 this->outlineColor = outlineColor;
3498 this->fillColor = fillColor;
3499 }
3500
3503 Polygon(const Polygon& other) = default;
3504
3509 AbstractDrawableObject* copy() const override{
3510 return new Polygon(*this);
3511 }
3512
3514 ~Polygon() override = default;
3515
3516 void draw(const Transform& t, Image& imgRef) const override{
3517 if (points.empty())
3518 return;
3519 /*CImg requires all polygons to be passed in as an instance of
3520 the image object. Therefore, we can specify an "int" image with
3521 a width of 2 (x,y) and height of the total number of
3522 elements in the point vector.*/
3523 cimg::CImg<int> passPts(static_cast<int>(points.size()), 2);
3524
3525 for (int i = 0; i < points.size(); i++) {
3526 const Point pt = t(points[i]);
3527 passPts(i, 0) = pt.x;
3528 passPts(i, 1) = pt.y;
3529 }
3530
3531 imgRef.draw_polygon(passPts, fillColor.rgbPtr());
3532
3533 if (outlineWidth > 0) {//draw outline using previously generated points.
3534 //LineLoop impl
3535 for (int i = 1; i < points.size(); i++) {
3536 drawLine(imgRef, passPts(i - 1, 0), passPts(i - 1, 1), passPts(i, 0), passPts(i, 1), outlineColor, outlineWidth);
3537 }
3538 //draw last line between first and last
3539 drawLine(imgRef, passPts(int(points.size()) - 1, 0), passPts(int(points.size()) - 1, 1), passPts(0, 0), passPts(0, 1), outlineColor, outlineWidth);
3540 }
3541 }
3542 };
3543
3548 public:
3549 int srcX, srcY, srcW, srcH;
3550 int drawWidth = 0;
3551 int drawHeight = 0;
3552
3553 explicit Sprite(Image& img, int outlineWidth = 0, const Color& outlineColor = Color()) : spriteImg(img) {
3554 srcX = srcY = 0;
3555 srcW = img.width();
3556 srcH = img.height();
3557 }
3558
3559 Sprite(Image& img, int srcX, int srcY, int srcW, int srcH, int outlineWidth = 0, const Color& outlineColor = Color()) : spriteImg(img) {
3560 this->srcX = srcX;
3561 this->srcY = srcY;
3562 this->srcW = srcW;
3563 this->srcH = srcH;
3564 this->outlineWidth = outlineWidth;
3565 this->outlineColor = outlineColor;
3566 }
3567
3568 Sprite(const Sprite& copy) = default;
3569
3570 ~Sprite() override = default;
3571
3572 AbstractDrawableObject* copy() const override{
3573 return new Sprite(*this);
3574 }
3575
3578 void draw(const Transform& t, Image& imgRef) const override{
3579 //Vertex order is as follows for the constructed quad.
3580 // 0--3 3
3581 // | / /|
3582 // |/ / |
3583 // 1 1--2
3584
3585 const int halfW = drawWidth / 2;
3586 const int halfH = drawHeight / 2;
3587
3588 Point destPoints[4] = {
3589 {-halfW, halfH}, //0
3590 {-halfW, -halfH}, //1
3591 {halfW, -halfH}, //2
3592 {halfW, halfH} //3
3593 };
3594
3595 Point texturePoints[4] = {
3596 {srcX, srcY},
3597 {srcX, srcY + srcH},
3598 {srcX + srcW, srcY},
3599 {srcX + srcW, srcY + srcH}
3600 };
3601
3603 for (Point& pt : destPoints)
3604 pt = t(pt);
3605
3606 //Yes, I know this isn't particularly readable.
3607 //But its purpose is described in an above commented illustration.
3608 imgRef.draw_triangle(destPoints[0][0], destPoints[0][1], destPoints[1][0], destPoints[1][1], destPoints[3][0], destPoints[3][1],
3609 spriteImg, texturePoints[0][0], texturePoints[0][1], texturePoints[1][0], texturePoints[1][1], texturePoints[3][0], texturePoints[3][1]);
3610 imgRef.draw_triangle(destPoints[1][0], destPoints[1][1], destPoints[2][0], destPoints[2][1], destPoints[3][0], destPoints[3][1],
3611 spriteImg, texturePoints[1][0], texturePoints[1][1], texturePoints[2][0], texturePoints[2][1], texturePoints[3][0], texturePoints[3][1]);
3612
3613 if (outlineWidth > 0) {//draw outline using previously generated points.
3614 //LineLoop impl
3615 for (int i = 1; i < 4; i++) {
3616 drawLine(imgRef, destPoints[i - 1][0], destPoints[i - 1][1], destPoints[i][0], destPoints[i][1], outlineColor, outlineWidth);
3617 }
3618 //draw last line between first and last
3619 drawLine(imgRef, destPoints[3][0], destPoints[3][1], destPoints[0][0], destPoints[0][1], outlineColor, outlineWidth);
3620 }
3621 }
3622 protected:
3623 Image& spriteImg;
3624 };
3625
3633 public:
3634 //Compound Polygon components are pairs of transforms and unique pointers to other drawable objects..
3635 typedef std::pair<Transform, std::unique_ptr<AbstractDrawableObject>> component_t;
3636
3637 CompoundPolygon() = default;
3638
3641
3642 for(const component_t& component : copy.components)
3643 components.emplace_back(component.first, component.second->copy());
3644 }
3645
3646 ~CompoundPolygon() override = default;
3647
3653 void addcomponent(const AbstractDrawableObject& obj, const Transform& transform = Transform()) {
3654 components.emplace_back(transform, obj.copy());
3655 }
3656
3661 AbstractDrawableObject* copy() const override{
3662 return new CompoundPolygon(*this);
3663 }
3664
3667 void draw(const Transform& t, Image& imgRef) const override{
3668 for (const component_t& comp : components) {
3669 comp.second->draw(t.copyConcatenate(comp.first), imgRef);
3670 }
3671 }
3672 protected:
3673 std::list<component_t> components;
3674 };
3675
3676 //SECTION: TURTLE & TURTLE SCREEN
3677
3680 enum TurtleSpeed {
3682 TS_FASTEST = 0,
3684 TS_FAST = 10,
3686 TS_NORMAL = 6,
3688 TS_SLOW = 3,
3690 TS_SLOWEST = 1
3691 };
3692
3701 std::unique_ptr<AbstractDrawableObject> geom;
3702
3707
3709 bool stamp = false;
3711 int stampid = -1;
3712
3714 SceneObject() = default;
3715
3726 };
3727
3731 struct PenState {
3736 float moveSpeed = TS_NORMAL;
3738 bool tracing = true;
3740 bool angleMode = false;
3742 int penWidth = 1;
3744 bool filling = false;
3751 size_t objectsBefore = 0;
3753 std::unique_ptr<AbstractDrawableObject> cursor = nullptr;
3755 int curStamp = 0;
3757 bool visible = true;
3759 float cursorTilt = 0;
3760
3761 PenState() = default;
3762 PenState(const PenState& copy) {
3763 transform = copy.transform;
3764 moveSpeed = copy.moveSpeed;
3765 tracing = copy.tracing;
3766 angleMode = copy.angleMode;
3767 penWidth = copy.penWidth;
3768 filling = copy.filling;
3769 penColor = copy.penColor;
3770 fillColor = copy.fillColor;
3771 cursor.reset(copy.cursor ? copy.cursor->copy() : nullptr);
3772 curStamp = copy.curStamp;
3773 visible = copy.visible;
3774 cursorTilt = copy.cursorTilt;
3776 }
3777
3778 PenState& operator=(const PenState& copy){
3779 transform = copy.transform;
3780 moveSpeed = copy.moveSpeed;
3781 tracing = copy.tracing;
3782 angleMode = copy.angleMode;
3783 penWidth = copy.penWidth;
3784 filling = copy.filling;
3785 penColor = copy.penColor;
3786 fillColor = copy.fillColor;
3787 cursor.reset(copy.cursor ? copy.cursor->copy() : nullptr);
3788 curStamp = copy.curStamp;
3789 visible = copy.visible;
3790 cursorTilt = copy.cursorTilt;
3792 return *this;
3793 }
3794 };
3795
3799 enum ScreenMode {
3800 SM_STANDARD,
3801 SM_LOGO//,
3802 // SM_WORLD
3803 };
3804 //I'm leaving out SM_WORLD. Adding it would really require more work than I have time for.
3805
3806 //Turtle class prototype so we can go ahead and define abstract turtle screen type.
3807 class Turtle;
3808
3816 public:
3817 virtual ~AbstractTurtleScreen() = default;
3818
3822 virtual void tracer(int countmax, unsigned int delayMS) = 0;
3823
3827 virtual int window_width() const = 0;
3828
3832 virtual int window_height() const = 0;
3833
3837 virtual Color bgcolor() const = 0;
3838
3842 virtual void bgcolor(const Color& c) = 0;
3843
3847 virtual void mode(ScreenMode mode) = 0;
3848
3852 virtual ScreenMode mode() const = 0;
3856 virtual void clearscreen() = 0;
3857
3860 inline void clear() {
3861 clearscreen();
3862 }
3863
3865 virtual void resetscreen() = 0;
3866
3868 inline void reset() {
3869 resetscreen();
3870 }
3871
3877 virtual bool supports_live_animation() const = 0;
3878
3879 virtual ivec2 screensize(Color& bg) = 0;
3880 //code-smell from python->c++, considering separation of functionality
3881
3882 virtual ivec2 screensize() = 0;
3883 virtual void update(bool invalidateDraw, bool processInput) = 0;
3884 virtual void delay(unsigned int ms) = 0;
3885 virtual unsigned int delay() const = 0;
3886
3890 virtual void bye() = 0;
3891
3892 virtual Image& getcanvas() = 0;
3893
3894 virtual bool isclosed() = 0;
3895
3896 virtual void redraw(bool invalidate) = 0;
3897
3901 virtual Transform screentransform() const = 0;
3902
3908 virtual void add(Turtle& turtle) = 0;
3909
3914 virtual void remove(Turtle& turtle) = 0;
3915
3920 virtual std::list<SceneObject>& getScene() = 0;
3921
3927 virtual AbstractDrawableObject& shape(const std::string& name) = 0;
3928
3933 virtual const BitmapFont& font(const std::string& name) const = 0;
3934 protected:
3947 static Image decodeDefaultFont(){
3948 Image img(DEFAULT_FONT_PIXELS_WIDTH, DEFAULT_FONT_PIXELS_HEIGHT);
3949 img.channels(0, 3);//force RGBA
3950 for(int pixId = 0; pixId < DEFAULT_FONT_PIXELS_LEN; pixId++){
3951 const unsigned int decodeVal = DEFAULT_FONT_PIXELS[pixId];
3952 // 8 integers per row of pixels (8*32=256)
3953 const int pixY = pixId / 8;
3954 const int pixOffsX = (pixId % 8) * 32;//offset of every pixel for the current integer.
3955
3956 for(int i = 0; i < 32; i++){ // for every bit in the unsigned integer...
3957 const int pixX = pixOffsX + (31 - i);//31 due to number of bits in unsigned int...
3958 //get i'th pixel in the integer by bitmask and multiply
3959 const uint8_t pixel = ((decodeVal >> i) & 1) * 255;
3960 for(int c = 0; c < 4; c++)
3961 img(pixX, pixY, 0, c) = pixel;
3962 }
3963 }
3964 return img;
3965 }
3966
3967 /*The default shapes that screens initialize with.*/
3968 std::unordered_map<std::string, Polygon> shapes = {
3969 //counterclockwise coordinates.
3970 {"triangle",
3971 Polygon{
3972 {0, 0},
3973 {-5, 5},
3974 {5, 5}}},
3975 {"square",
3976 Polygon{
3977 {-5, -5},
3978 {-5, 5},
3979 {5, 5},
3980 {5, -5}}},
3981 {"indented triangle",
3982 Polygon{
3983 //CCW
3984 {0, 0},
3985 {-5, 10},
3986 {0, 8},
3987 {5, 10}}},
3988 {"arrow",
3989 Polygon{
3990 {0, 0},
3991 {-5, 5},
3992 {-3, 5},
3993 {-3, 10},
3994 {3, 10},
3995 {3, 5},
3996 {5, 5}}}
3997 };
3998
3999 //Abstract class. Private constructor only allows
4000 //for derivative classes to be instantiated.
4001 AbstractTurtleScreen() = default;
4002 };
4003
4014 class Turtle{
4015 public:
4020 screen = &scr;
4021 screen->add(*this);
4022 reset();
4023 }
4024
4028 Turtle(const Turtle&) = delete;
4029
4033 Turtle(Turtle&& mv) = delete;
4034
4038 Turtle& operator=(const Turtle&) = delete;
4039
4043 Turtle& operator=(Turtle&& turtle) = delete;
4044
4047 void forward(int pixels){
4048 if (screen == nullptr)
4049 return;
4050 travelTo(Transform(*transform).forward(static_cast<float>(pixels)));
4051 }
4052
4054 inline void fd(int pixels) {
4055 forward(pixels);
4056 }
4057
4059 void backward(int pixels){
4060 if (screen == nullptr)
4061 return;
4062 travelTo(Transform(*transform).backward(static_cast<float>(pixels)));
4063 }
4064
4066 inline void bk(int pixels) {
4067 backward(pixels);
4068 }
4069
4071 inline void back(int pixels) {
4072 backward(pixels);
4073 }
4074
4082 void right(float amt){
4083 amt = state->angleMode ? -amt : -toRadians(amt);
4084 //Flip angle orientation based on screen mode.
4085 travelTo(Transform(*transform).rotate(amt));
4086 }
4087
4089 inline void rt(float angle) {
4090 right(angle);
4091 }
4092
4100 void left(float amt){
4101 amt = state->angleMode ? amt : toRadians(amt);
4102 //Flip angle orientation based on screen mode.
4103 travelTo(Transform(*transform).rotate(amt));
4104 }
4105
4107 inline void lt(float angle) {
4108 left(angle);
4109 }
4110
4112 void goTo(int x, int y){
4113 travelTo(Transform(*transform).setTranslation(x,y));
4114 }
4115
4117 inline void goTo(const Point& pt){
4118 goTo(pt.x, pt.y);
4119 }
4120
4122 inline void setpos(int x, int y) {
4123 goTo(x, y);
4124 }
4125
4127 inline void setpos(const Point& pt){
4128 goTo(pt.x, pt.y);
4129 }
4130
4132 inline void setposition(int x, int y) {
4133 goTo(x, y);
4134 }
4135
4137 inline void setposition(const Point& pt){
4138 goTo(pt.x, pt.y);
4139 }
4140
4142 void setx(int x){
4143 travelTo(Transform(*transform).setTranslationX(x));
4144 }
4145
4147 void sety(int y){
4148 travelTo(Transform(*transform).setTranslationY(y));
4149 }
4150
4155 int xcor() const{
4156 return static_cast<int>(transform->getTranslateX());
4157 }
4158
4163 int ycor() const{
4164 return static_cast<int>(transform->getTranslateY());
4165 }
4166
4171 Point getpos() const{
4172 return transform->getTranslation();
4173 }
4174
4181 inline void shift(int x, int y) {
4182 goTo(getpos() + Point(x, y));
4183 }
4184
4188 const PenState& penstate() const{
4189 return *state;
4190 }
4191
4199 void setheading(float amt){
4200 //Swap to correct unit if necessary.
4201 amt = state->angleMode ? amt : toRadians(amt);
4202 //Flip angle orientation based on screen mode.
4203 amt = (screen != nullptr) ? screen->mode() == SM_STANDARD ? amt : -amt : amt;
4204 travelTo(Transform(*transform).setRotation(amt));
4205 }
4206
4213 inline void face(int x, int y){
4214 setheading(towards(x,y) - heading());
4215 }
4216
4222 inline void face(const Point& pt){
4223 setheading(towards(pt) - heading());
4224 }
4225
4233 int distance(int x, int y){
4234 return cturtle::distance(transform->getTranslation(), {x,y});
4235 }
4236
4243 int distance(const Point& pt){
4244 return cturtle::distance(transform->getTranslation(), pt);
4245 }
4246
4248 inline void seth(float angle) {
4249 setheading(angle);
4250 }
4251
4258 float towards(int x, int y){
4259 float amt = std::atan2(static_cast<float>(y) - transform->getTranslateY(), static_cast<float>(x) - transform->getTranslateX());
4260
4261 //6.28319 is a full rotation in radians...
4262 if(toDegrees(amt) < 0)
4263 amt = 6.28319f + amt;
4264
4265 //convert to degrees if necessary.
4266 amt = state->angleMode ? amt : toDegrees(amt);
4267 return amt + heading();
4268 }
4269
4275 inline float towards(const Point& pt){
4276 return towards(pt.x, pt.y);
4277 }
4278
4283 inline float heading() {
4284 return state->angleMode ? transform->getRotation() : toDegrees(transform->getRotation());
4285 }
4286
4294 void home(){
4295 travelTo(Transform());//set transform to identity, more-or-less...
4296 }
4297
4302 void circle(int radius, int steps, const Color& color){
4303 pushGeometry(*transform, new Circle(radius, steps, color));
4304 updateParent(false, true);
4305 }
4306
4310 inline void circle(const Color& color) {
4311 circle(30, 15, color);
4312 }
4313
4318 void dot(const Color& color, int size = 10) {
4319 circle(size / 2, 4, color);
4320 }
4321
4326 void fill(bool val){
4327 if(state->filling && !val) {
4328 //Add the fill polygon
4329 screen->getScene().emplace_back(new Polygon(fillAccum.points, state->fillColor), Transform());
4330 objects.push_back(std::prev(screen->getScene().end(), 1));
4331
4332 //Add all trace lines created when tracing out the fill polygon.
4333 if(!fillLines.empty()) {
4334 //for each line we've created when having the pen down, and have been tracing a shape
4335 for(Line& lineInfo : fillLines) {
4336 screen->getScene().emplace_back(lineInfo.copy(), Transform());
4337 objects.push_back(std::prev(screen->getScene().end(), 1));
4338 }
4339 fillLines.clear();
4340 }
4341
4342 fillAccum.points.clear();
4343 updateParent(false, false);
4344 }
4345 state->filling = val;
4346 }
4347
4352 inline bool filling() const{
4353 return penstate().filling;
4354 }
4355
4358 inline void begin_fill() {
4359 fill(true);
4360 }
4361
4364 inline void end_fill() {
4365 fill(false);
4366 }
4367
4370 void fillcolor(const Color& c) {
4371 pushState();
4372 state->fillColor = c;
4373 updateParent(false, false);
4374 }
4375
4379 return state->fillColor;
4380 }
4381
4386 void write(const std::string& text){
4387 if(text.empty())
4388 return;
4389 pushText(*transform, state->fillColor, screen->font(DEFAULT_FONT), text, 1.0f);
4390 updateParent(false, false);
4391 }
4392
4401 void write(const std::string& text, const std::string& font, const Color& color = {"white"}, float scale = 1.0f, TextAlign alignment = TEXT_ALIGN_LEFT){
4402 if(text.empty())
4403 return;
4404 pushText(*transform, color, screen->font(font), text, scale, alignment);
4405 updateParent(false, false);
4406 }
4407
4411 int stamp(){
4412 pushStamp(*transform, state->cursor->copy());
4413 return state->curStamp;
4414 }
4415
4417 void clearstamp(int stampid){
4418 auto iter = objects.begin(); //iterator which holds an iterator to the screen's scene list.
4419
4420 while (iter != objects.end()) {
4421 auto& objIter = *iter;
4422 if (objIter->stamp && objIter->stampid == stampid) {
4423 break;
4424 }
4425 iter++;
4426 }
4427
4428 if (iter != objects.end()) {
4429 objects.erase(iter);
4430
4431 if (screen != nullptr) {
4432 screen->getScene().erase(*iter);
4433 }
4434 }
4435
4436 updateParent(true,false);
4437 }
4438
4441 void clearstamps(int stampid = -1){
4442 typedef decltype(objects.begin()) iter_t;
4443
4444 std::list<iter_t> removals;
4445
4446 auto iter = objects.begin();
4447 while (iter != objects.end()) {
4448 auto& objIter = *iter;
4449 if (stampid < 0 ? objIter->stamp : (objIter->stamp && objIter->stampid <= stampid)) {
4450 removals.push_back(iter);
4451 }
4452 iter++;
4453 }
4454
4455 for (auto& iter : removals) {
4456 screen->getScene().erase(*iter);
4457 objects.erase(iter);
4458 }
4459
4460 updateParent(true, false);
4461 }
4462
4466 pushState();
4467 state->cursor.reset(p.copy());
4468 updateParent(false, false);
4469 }
4470
4473 void shape(const std::string& name){
4474 pushState();
4475 state->cursor.reset((screen->shape(name).copy()));
4476 updateParent(false, false);
4477 }
4478
4481 return *state->cursor;
4482 }
4483
4485 bool undo(bool try_redraw = true){
4486 //total objects on the state stack prior to
4487 const unsigned long int totalBefore = state->objectsBefore;
4488
4489 if (stateStack.size() >= 2)
4490 travelBack(); //Travel back if stack size >= 2
4491
4492 //If we can't pop the state, break early.
4493 if (!popState()) {
4494 return false;
4495 }
4496
4497 auto begin = std::prev(objects.end(), (totalBefore - state->objectsBefore));
4498 auto iter = begin;
4499
4500 while (iter != objects.end()) {
4501 screen->getScene().erase(*iter);
4502 iter++;
4503 }
4504
4505 objects.erase(begin, objects.end());
4506
4507 //Will invalidate the whole screen due to object removal, but we allow the option to not.
4508 updateParent(try_redraw, false);
4509 return true;
4510 }
4511
4514 void setundobuffer(unsigned int size) {
4515 if (size < 1)//clamp lower bound to 1
4516 size = 1;
4517
4518 undoStackSize = size;
4519 while (stateStack.size() > size) {
4520 stateStack.pop_front();
4521 }
4522 }
4523
4525 unsigned int undobufferentries() {
4526 return static_cast<unsigned int>(stateStack.size());
4527 }
4528
4532 void speed(float val) {
4533 pushState();
4534 state->moveSpeed = val;
4535 }
4536
4538 float speed() {
4539 return state->moveSpeed;
4540 }
4541
4543 void tilt(float amt){
4544 amt = state->angleMode ? amt : toRadians(amt);
4545 //Flip angle orientation based on screen mode.
4546 amt = screen->mode() == SM_STANDARD ? amt : -amt;
4547 pushState();
4548 state->cursorTilt += amt;
4549 updateParent(false, false);
4550 }
4551
4554 float tilt() const {
4555 return state->angleMode ? state->cursorTilt : toDegrees(state->cursorTilt);
4556 }
4557
4560 void setshowturtle(bool val){
4561 pushState();
4562 state->visible = val;
4563 updateParent(false, false);
4564 }
4565
4569 inline void showturtle() {
4570 setshowturtle(true);
4571 }
4572
4575 inline void hideturtle() {
4576 setshowturtle(false);
4577 }
4578
4580 void setpenstate(bool down){
4581 pushState();
4582 state->tracing = down;
4583 }
4584
4586 inline void penup() {
4587 setpenstate(false);
4588 }
4589
4591 inline void pendown() {
4592 setpenstate(true);
4593 }
4594
4597 void pencolor(const Color& c) {
4598 pushState();
4599 state->penColor = c;
4600 updateParent(false, false);
4601 }
4602
4605 Color pencolor() const {
4606 return state->penColor;
4607 }
4608
4611 void width(int pixels) {
4612 pushState();
4613 state->penWidth = pixels;
4614 }
4615
4618 int width() const{
4619 return state->penWidth;
4620 }
4621
4625 void draw(const Transform& screenTransform, Image& canvas){
4626 if (this->screen == nullptr)
4627 return;
4628
4629 if (!state->visible)
4630 return;
4631
4632 //Draw all lines queued during filling a shape.
4633 //This is only populated when the turtle moves between a beginfill
4634 //and endfill while the pen is down.
4635 for (const Line& line : fillLines)
4636 line.draw(screenTransform, canvas);
4637
4638 if (traveling && state->tracing) {
4639 //Draw the "Travel-Line" when in the middle of the travelTo func
4640 travelPoints[0] = screenTransform(travelPoints[0]);
4641 travelPoints[1] = screenTransform(travelPoints[1]);
4642 drawLine(canvas, travelPoints[0].x, travelPoints[0].y, travelPoints[1].x, travelPoints[1].y, state->penColor, state->penWidth);
4643 }
4644
4645 //Add the extra rotate to start cursor facing right :)
4646 const float cursorRot = this->screen->mode() == SM_STANDARD ? 1.5708f : -3.1416f;
4647 Transform cursorTransform = screenTransform.copyConcatenate(*transform).rotate(cursorRot + state->cursorTilt);
4648 state->cursor->fillColor = state->fillColor;
4649 state->cursor->outlineWidth = 1;
4650 state->cursor->outlineColor = state->penColor;
4651 state->cursor->draw(cursorTransform, canvas);
4652 }
4653
4656 void degrees() {
4657 pushState();
4658 state->angleMode = false;
4659 }
4660
4663 inline void radians() {
4664 pushState();
4665 state->angleMode = true;
4666 }
4667
4671 void reset(){
4672 //Reset objects, transforms, trace lines, state, etc.
4673
4674 //Note to self, clearing the list, appending a new transform,
4675 //then reassigning the transform reference just didn't want to work.
4676 //I have no idea why. Therefore, we're resetting it in the same
4677 //manner we initially construct it.
4678 stateStack = {PenState()};
4679 state = &stateStack.back();
4680
4681 transform = &state->transform;
4682 const auto numItems = objects.size();
4683
4684 if (screen != nullptr) {
4685 //Re-assign cursor on reset, derived from parent screen.
4686 state->cursor.reset(screen->shape("indented triangle").copy());
4687 //Erase all objects
4688 while (!objects.empty()) {
4689 screen->getScene().erase(objects.front());
4690 objects.pop_front();
4691 }
4692
4693 //Alter cursor tilt and default transform
4694 //when operating under SM_LOGO mode.
4695 //This is to bring it up-to-par with Python's
4696 //implementation of screen modes.
4697 if (screen->mode() == SM_LOGO) {
4698 state->cursorTilt = (-1.5708f);
4699 transform->rotate(1.5708f);
4700 }
4701 }
4702
4703 updateParent(numItems > 0, false);
4704 }
4705
4708 screen = scr;
4709 }
4710
4712 virtual ~Turtle(){
4713 //remove itself from its parent screen...
4714 if(screen != nullptr)
4715 screen->remove(*this);
4716 }
4717 protected:
4718 //a list of iterators that point to the parent screen's scene list.
4719 //this list is in-order as they are created by this instance.
4720 std::list<std::list<SceneObject>::iterator> objects;
4721 //the pen-state stack is, as its name might imply, the previous states
4722 //of the pen owned by this turtle.
4723 std::list<PenState> stateStack = {PenState()};
4724 std::list<Line> fillLines;
4725
4726 //the current transform. This points to the topmost state's transform
4727 //on the pen state stack.
4728 Transform* transform = nullptr;
4729 //the current state. This points directly to the topmost state
4730 //on the pen state stack.
4731 PenState* state = nullptr;
4732
4733 /* These variables are used to draw the "travel" line when
4734 * the turtle is traveling. (e.g, the line between where it's going
4735 * and where it's been)*/
4736 Point travelPoints[2];
4737 bool traveling = false;
4738
4739 /*Undo stack size.*/
4740 unsigned int undoStackSize = 100;
4741
4742 /*Accumulator for fill state*/
4743 Polygon fillAccum;
4744
4745 /*Screen pointer. Assign before calling any other function!*/
4746 AbstractTurtleScreen* screen = nullptr;
4747
4748 /*Pushes a copy of the pen's state on the stack.*/
4749 void pushState(){
4750 if (stateStack.size() + 1 > undoStackSize)
4751 stateStack.pop_front();
4752
4753 stateStack.push_back(stateStack.back()); //Push a copy of the back-most pen state.
4754 state = &stateStack.back();
4755 transform = &state->transform;
4756 state->objectsBefore = objects.size();
4757 }
4758
4759 /*Pops the top of the pen's state stack.*/
4760 bool popState(){
4761 if (stateStack.size() == 1)
4762 return false;
4763 stateStack.pop_back();
4764 state = &stateStack.back();
4765 transform = &state->transform;
4766 return true;
4767 }
4768
4777 if (screen != nullptr) {
4778 pushState();
4779 screen->getScene().emplace_back(geom, t);
4780 objects.push_back(std::prev(screen->getScene().end()));
4781 return true;
4782 }
4783 return false;
4784 }
4785
4791 if (screen != nullptr) {
4792 pushState();
4793 const float cursorRot = this->screen->mode() == SM_STANDARD ? 1.5708f : -3.1416f;
4794
4795 Transform trans(t);
4796 trans.rotate(cursorRot + state->cursorTilt);
4797
4798 geom->outlineWidth = 1;
4799 geom->outlineColor = state->penColor;
4800
4801 screen->getScene().emplace_back(geom, trans, state->curStamp++);
4802 SceneObject& obj = screen->getScene().back();
4803
4804 objects.push_back(std::prev(screen->getScene().end()));
4805 return true;
4806 }
4807 return false;
4808 }
4809
4817 bool pushText(const Transform& t, const Color& color, const BitmapFont& font, const std::string& text, float scale = 1.0f, TextAlign alignment = TEXT_ALIGN_LEFT){
4818 if (screen != nullptr) {
4819 pushState();
4820 screen->getScene().emplace_back(new Text(text, font, color, scale, alignment), t);
4821 objects.push_back(std::prev(screen->getScene().end()));
4822 return true;
4823 }
4824 return false;
4825 }
4826
4833 if (screen != nullptr) {
4834 screen->getScene().emplace_back(new Line(a, b, state->penColor, state->penWidth), Transform());
4835 objects.push_back(std::prev(screen->getScene().end()));
4836 //Trace lines do NOT push a state->
4837 //Their state is encompassed by movement,
4838 //and these lines are only added when moving the turtle
4839 //while the pen is down.
4840 return true;
4841 }
4842 return false;
4843 }
4844
4847 long int getAnimMS() {
4848 if(screen == nullptr)
4849 return 0;
4850
4851 //300 is the "scale" animations adhere to.
4852 //The longest animation is 300 milliseconds, shortest is 0.
4853 //This was an arbitrary choice, trying to match the speed of the Python implementation.
4854 if(!screen->supports_live_animation() || state->moveSpeed < 0)
4855 return 0;//no animation means no time spent animating...
4856 return long((state->moveSpeed / 10.0f) * 300); //<----
4857 }
4858
4860 void updateParent(bool invalidate = false, bool input = true){
4861 if (screen != nullptr)
4862 screen->update(invalidate, input);
4863 }
4864
4870 void travelBetween(Transform src, const Transform& dest, bool doPushState){
4871 if(dest == src)
4872 return;
4873
4874 //Set the "traveling" state for screen drawing. Indicates when to draw travel lines (e.g, when pen is down).
4875 traveling = true;
4876
4877 const auto duration = static_cast<float>(getAnimMS());
4878 if ((screen ? !screen->isclosed() : false) && duration > 0) {//no point in animating with no screen
4879 const unsigned long startTime = detail::epochTime();
4880
4881 float progress = 0;
4882 while (progress < 1.0f) {
4883 const unsigned long curTime = detail::epochTime();
4884
4885 transform->assign(src.lerp(dest, progress));
4886 travelPoints[0] = src.getTranslation();
4887 travelPoints[1] = transform->getTranslation();
4888
4889 updateParent(false, false);
4890
4891 progress = (static_cast<float>(curTime - startTime) / duration);
4892 }
4893 }
4894
4895 if(doPushState){
4896 if (state->tracing && !state->filling) {
4898 } else if (state->filling) {
4899 fillAccum.points.push_back(dest.getTranslation());
4900 if (state->tracing) {
4901 fillLines.emplace_back(src.getTranslation(), dest.getTranslation(), state->penColor, state->penWidth);
4902 }
4903 }
4904
4905 transform->assign(src);
4906 pushState();
4907 }
4908
4909 transform->assign(dest);
4910 traveling = false;
4911 updateParent(false, false);
4912 }
4913
4918 void travelTo(const Transform& dest){
4919 travelBetween(*transform, dest, true);
4920 }
4921
4927 travelBetween(*transform, std::prev(stateStack.end(), 2)->transform, false);
4928 }
4929
4931 Turtle() = default;
4932 };
4933
4934#ifdef CTURTLE_HEADLESS
4935 /*Used to output Base-64 GIF and HTML source for OfflineTurtleScreen.*/
4936 inline std::string encodeFileBase64(const std::string& path){
4937 std::ifstream file(path, std::ios::binary | std::ios::ate);
4938 std::streamsize size = file.tellg();
4939 file.seekg(0, std::ios::beg);
4940
4941 std::vector<unsigned char> buffer(size, 0);
4942 file.read((char*)buffer.data(), size);
4943 return base64::encode(buffer);
4944 }
4945
4946 class OfflineTurtleScreen : public AbstractTurtleScreen{
4947 public:
4948 OfflineTurtleScreen(){
4949 const int width = CTURTLE_HEADLESS_WIDTH;
4950 const int height = CTURTLE_HEADLESS_HEIGHT;
4951 canvas.assign(width, height, 1, 3);
4952 canvas.fill(255);
4953 isClosed = false;
4954 gif = jo_gif_start(CTURTLE_HEADLESS_SAVEDIR, width, height,1, 31);
4955 redraw(true);
4956 }
4957
4958 ~OfflineTurtleScreen(){
4959 bye();
4960 }
4961
4962 void tracer(int countmax, unsigned int delayMS = 10){
4963 redrawCounterMax = countmax;
4964 delay(delayMS);
4965 redraw();
4966 }
4967
4968 int window_width() const{
4969 return canvas.width();
4970 }
4971
4972 int window_height() const{
4973 return canvas.height();
4974 }
4975
4976 Color bgcolor() const{
4977 return backgroundColor;
4978 }
4979
4980 void bgcolor(const Color& c){
4981 backgroundColor = c;
4982 redraw(true);
4983 }
4984
4985 void mode(ScreenMode mode){
4986 //Resets & re-orients all turtles.
4987
4988 curMode = mode;
4989 for (Turtle* t : turtles) {
4990 t->reset();
4991 }
4992 }
4993
4994 ScreenMode mode() const{
4995 return curMode;
4996 }
4997
4998 void clearscreen(){
4999 //1) Delete all drawings and turtles
5000 //2) White background
5001
5002 for (Turtle* turtle : turtles) {
5003 turtle->setScreen(nullptr);
5004 }
5005
5006 turtles.clear();
5007 backgroundColor = Color("white");
5008 curMode = SM_STANDARD;
5009 }
5010
5011 void resetscreen(){
5012 for (Turtle* turtle : turtles)
5013 turtle->reset();
5014 }
5015
5016 ivec2 screensize(Color& bg){
5017 bg = backgroundColor;
5018 return {canvas.width(), canvas.height()};
5019 };
5020 //code-smell from python->c++, considering separation of functionality
5021
5022 ivec2 screensize(){
5023 return {canvas.width(), canvas.height()};
5024 }
5025
5026 void update(bool invalidateDraw = false, bool processInput = false){
5027 redraw(invalidateDraw);
5028 //processInput is ignored. OfflineTurtleScreen does NOT support input.
5029 }
5030
5031 void delay(unsigned int ms){
5032 delayMS = ms;
5033 }
5034
5035 unsigned int delay() const{
5036 return delayMS;
5037 }
5038
5039 void bye() {
5040 if(isClosed)
5041 return;
5042
5043 /*finish up drawing if redraw counter hasn't been met*/
5044 if(redrawCounter > 0 || redrawCounter >= redrawCounterMax){
5045 tracer(1, delayMS);
5046 }
5047
5048 jo_gif_end(&gif);
5049
5050#ifndef CTURTLE_HEADLESS_NO_HTML
5051 /*print base-64 encoding + HTML source*/
5052 std::string imgCode = encodeFileBase64(CTURTLE_HEADLESS_SAVEDIR);
5053
5054 //See the following to understand why this was done:
5055 //https://github.com/ericsonga/APCSAReview/blob/master/_sources/TurtleGraphics/turtleBasics.rst
5056 //HTML can be captured for later output.
5057 std::cout << "<img src=\'data:image/gif;base64,";
5058 std::cout << imgCode;
5059 std::cout << "\'/>";
5060#endif
5061
5062 clearscreen();
5063 isClosed = true;
5064 }
5065
5066 Image& getcanvas(){
5067 return canvas;
5068 }
5069
5070 bool isclosed(){
5071 return isClosed;
5072 }
5073
5074 bool supports_live_animation() const{
5075 return false;
5076 }
5077
5078 virtual void redraw(bool invalidate = false){
5079 if (isclosed())
5080 return;
5081 int fromBack = 0;
5082 bool hasInvalidated = invalidate;
5083
5084 //Handle resizes.
5085
5086 if (lastTotalObjects <= objects.size()) {
5087 fromBack = static_cast<int>(objects.size() - lastTotalObjects);
5088 }
5089
5090 if (hasInvalidated) {
5091 canvas.draw_rectangle(0, 0, canvas.width(), canvas.height(), backgroundColor.rgbPtr());
5092 redrawCounter = 0;//Forced redraw due to canvas invalidation.
5093 } else {
5094 if(redrawCounterMax == 0){
5095 return;
5096 }
5097
5098 redrawCounter++;
5099
5100 if (redrawCounter >= redrawCounterMax) {
5101 redrawCounter = 0;
5102 } else {
5103 return;
5104 }
5105 }
5106
5107 auto latestIter = !hasInvalidated ? std::prev(objects.end(), fromBack) : objects.begin();
5108
5109 Transform screen = screentransform();
5110 while (latestIter != objects.end()) {
5111 SceneObject& object = *latestIter;
5112 const Transform t(screen.copyConcatenate(object.transform));
5113
5114 object.geom->draw(t, canvas);
5115
5116 latestIter++;
5117 }
5118
5119 if (canvas.width() != turtleComposite.width() || canvas.height() != turtleComposite.height()) {
5120 turtleComposite.assign(canvas);
5121 } else {
5122 //This works off the assumption that drawImage is accelerated.
5123 //There might be a more efficient way to do this, however.
5124 turtleComposite.draw_image(0, 0, canvas);
5125 }
5126
5127 for (Turtle* turt : turtles)
5128 turt->draw(screen, turtleComposite);
5129
5130 lastTotalObjects = static_cast<int>(objects.size());
5131
5132 /* The following code takes the place of swapping the display buffer for the canvas,
5133 * which is what the interactive mode does.*/
5134
5135 //This copy is NOT efficient.
5136 //We should be able to take advantage of loop unrolling here
5137 for(int x = 0; x < CTURTLE_HEADLESS_WIDTH; x++){
5138 for(int y = 0; y < CTURTLE_HEADLESS_HEIGHT; y++){
5139 uint8_t* pixel = (&gifWriteBuffer[(y*CTURTLE_HEADLESS_WIDTH+x)*4]);
5140
5141 pixel[0] = turtleComposite(x,y,0);
5142 pixel[1] = turtleComposite(x,y,1);
5143 pixel[2] = turtleComposite(x,y,2);
5144 pixel[3] = 255;
5145 }
5146 }
5147
5148 //GIF frames are measured in centiseconds, thus the /10 on the delayMS...
5149 jo_gif_frame(&gif, gifWriteBuffer, delayMS / 10, true);
5150 }
5151
5152 Transform screentransform() const{
5153 return Transform().translate(canvas.width()/2, canvas.height()/2).scale(1, -1.0f);
5154 }
5155
5156 void add(Turtle& turtle){
5157 turtles.push_back(&turtle);
5158 }
5159
5160 void remove(Turtle& turtle){
5161 turtle.reset();
5162 turtle.setScreen(nullptr);
5163 turtles.remove(&turtle);
5164 }
5165
5166 std::list<SceneObject>& getScene(){
5167 return objects;
5168 }
5169
5170 AbstractDrawableObject& shape(const std::string& name){
5171 return shapes[name];
5172 }
5173
5181 const BitmapFont& font(const std::string& name) const{
5182 return *defaultFont;
5183 }
5184 private:
5185 /*this can be a constant allocated buffer.*/
5186 uint8_t gifWriteBuffer[CTURTLE_HEADLESS_WIDTH * CTURTLE_HEADLESS_HEIGHT*4];
5187 //allocate enough to hold width*height*4 (4 because RGBA).
5188 //this fits into uint32_t type quite nicely. (8+8+8+8 bits, r+g+b+a) = 32
5189
5190 //This struct controls the writing of resulting GIFs.
5191 jo_gif_t gif;
5192
5193 std::list<SceneObject> objects;
5194 std::list<Turtle*> turtles;
5195
5196 bool isClosed = true;
5197 Image canvas;
5198
5199 //The turtle composite image.
5200 //This image copies the canvas and has
5201 //turtles drawn to it to avoid redrawing a "busy" canvas.
5202 //Trace lines are also drawn on this when filling.
5203 Image turtleComposite;
5204
5207 int lastTotalObjects = 0;
5208
5210 Color backgroundColor = Color("white");
5211
5212 //OfflineTurtleScreen has no background image.
5213
5216 ScreenMode curMode = SM_STANDARD;
5217
5219 long int delayMS = 10;
5220
5223 int redrawCounter = 0;
5225 int redrawCounterMax = 1;
5226
5227 std::unique_ptr<BitmapFont> defaultFont =
5228 std::unique_ptr<BitmapFont>(new BitmapFont(
5229 decodeDefaultFont(), DEFAULT_FONT_ASCII_OFFSET,
5230 DEFAULT_FONT_GLYPH_WIDTH, DEFAULT_FONT_GLYPH_HEIGHT,
5231 DEFAULT_FONT_GLYPHS_X, DEFAULT_FONT_GLYPHS_Y));
5232 };
5233
5234 typedef OfflineTurtleScreen TurtleScreen;
5235#else /*NOT DEFINED CTURTLE_HEADLESS*/
5236 constexpr int SCREEN_DEFAULT_WIDTH = 800;
5237 constexpr int SCREEN_DEFAULT_HEIGHT = 600;
5238 constexpr char SCREEN_DEFAULT_TITLE[] = "CTurtle " CTURTLE_VERSION;
5239
5249 public:
5252 InteractiveTurtleScreen() : display(SCREEN_DEFAULT_WIDTH, SCREEN_DEFAULT_HEIGHT, SCREEN_DEFAULT_TITLE, 0) {
5253 canvas.assign(display);
5255 redraw(true);
5256 fonts[DEFAULT_FONT] =
5257 std::unique_ptr<BitmapFont>(new BitmapFont(
5258 decodeDefaultFont(), DEFAULT_FONT_ASCII_OFFSET,
5259 DEFAULT_FONT_GLYPH_WIDTH, DEFAULT_FONT_GLYPH_HEIGHT,
5260 DEFAULT_FONT_GLYPHS_X, DEFAULT_FONT_GLYPHS_Y));
5261 }
5262
5266 explicit InteractiveTurtleScreen(const std::string& title)
5267 : display(SCREEN_DEFAULT_WIDTH, SCREEN_DEFAULT_HEIGHT, title.c_str(), 0) {
5268 canvas.assign(display);
5270 redraw(true);
5271
5272 fonts[DEFAULT_FONT] =
5273 std::unique_ptr<BitmapFont>(new BitmapFont(
5274 decodeDefaultFont(), DEFAULT_FONT_ASCII_OFFSET,
5275 DEFAULT_FONT_GLYPH_WIDTH, DEFAULT_FONT_GLYPH_HEIGHT,
5276 DEFAULT_FONT_GLYPHS_X, DEFAULT_FONT_GLYPHS_Y));
5277 }
5278
5285 InteractiveTurtleScreen(int width, int height, const std::string& title = SCREEN_DEFAULT_TITLE)
5286 : display(width, height) {
5287 display.set_title(title.c_str());
5288 display.set_normalization(0);
5289 canvas.assign(display);
5291 redraw(true);
5292
5293 fonts[DEFAULT_FONT] =
5294 std::unique_ptr<BitmapFont>(new BitmapFont(
5295 decodeDefaultFont(), DEFAULT_FONT_ASCII_OFFSET,
5296 DEFAULT_FONT_GLYPH_WIDTH, DEFAULT_FONT_GLYPH_HEIGHT,
5297 DEFAULT_FONT_GLYPHS_X, DEFAULT_FONT_GLYPHS_Y));
5298 }
5299
5302 bye();
5303 }
5304
5311 void tracer(int countmax, unsigned int delayMS) override{
5312 redrawCounterMax = countmax;
5313 delay(delayMS);
5314 redraw(false);
5315 }
5316
5322 void bgcolor(const Color& color) override{
5323 backgroundColor = color;
5324 redraw(true);
5325 }
5326
5329 Color bgcolor() const override{
5330 return backgroundColor;
5331 }
5332
5337 void bgpic(const Image& img){
5338 backgroundImage.assign(img);
5340 redraw(true);
5341 }
5342
5344 const Image& bgpic(){
5345 return backgroundImage;
5346 }
5347
5348 bool supports_live_animation() const override{
5349 return true;
5350 }
5351
5358 void mode(ScreenMode mode) override{
5359 curMode = mode;
5360 for (Turtle* t : turtles)
5361 t->home();
5362 }
5363
5365 ScreenMode mode() const override{
5366 return curMode;
5367 }
5368
5374 void clearscreen() override{
5375 //1) Delete all drawings and turtles
5376 //2) White background
5377 //3) No background image
5378 //4) No event bindings
5379
5380 while(!turtles.empty())
5381 remove(*turtles.front());
5382
5383 turtles.clear();
5384 backgroundColor = Color("white");
5385 backgroundImage.assign();//assign with no parameters is deleting whatever contents it may have.
5386 curMode = SM_STANDARD;
5387
5388 //Gotta do binding alterations under the cache's mutex lock.
5389 eventCacheMutex.lock();
5390 timerBindings.clear();
5391 keyBindings[0].clear();
5392 keyBindings[1].clear();
5393 for (auto& mouseBinding : mouseBindings)
5394 mouseBinding.clear();
5395 eventCacheMutex.unlock();
5396 }
5397
5399 void resetscreen() override{
5400 for (Turtle* turtle : turtles)
5401 turtle->reset();
5402 }
5403
5406 //code-smell from python->c++, considering separation of functionality
5407 inline ivec2 screensize(Color& bg) override{
5408 bg = backgroundColor;
5409 return {display.width(), display.height()};
5410 }
5411
5413 inline ivec2 screensize() override { //see line above comment about code-smell
5414 return {display.width(), display.height()};
5415 }
5416
5421 void update(bool invalidateDraw, bool processInput) override{
5422 /*Resize canvas when necessary.*/
5423 if (display.is_resized()) {
5424 display.resize();
5425 invalidateDraw = true;
5426 }
5427 redraw(invalidateDraw);
5428
5429 if (processInput && !timerBindings.empty()) {
5430 //Call timer bindings first.
5431 uint64_t curTime = detail::epochTime();
5432 for (auto& timer : timerBindings) {
5433 auto& func = std::get<0>(timer);
5434 uint64_t reqTime = std::get<1>(timer);
5435 uint64_t& lastCalled = std::get<2>(timer);
5436
5437 if (curTime >= lastCalled + reqTime) {
5438 lastCalled = curTime;
5439 func();
5440 }
5441 }
5442 }
5443
5445 if (cachedEvents.empty() || !processInput)
5446 return; //No events to process.
5447
5448 //lock event cache to avoid race conditions
5449 eventCacheMutex.lock();
5450
5451 for (InputEvent& event : cachedEvents) {
5452 if (event.type) {//process keyboard event
5453 KeyFunc& keyFunc = *reinterpret_cast<KeyFunc*> (event.cbPointer);
5454 keyFunc();
5455 } else {//process mouse event
5456 MouseFunc& mFunc = *reinterpret_cast<MouseFunc*> (event.cbPointer);
5457 mFunc(event.mX, event.mY);
5458 }
5459 }
5460
5461 cachedEvents.clear();
5462 eventCacheMutex.unlock();
5463 }
5464
5466 void delay(unsigned int ms) override{
5467 delayMS = ms;
5468 }
5469
5471 unsigned int delay() const override {
5472 return delayMS;
5473 }
5474
5476 int window_width() const override {
5477 return display.window_width();
5478 }
5479
5481 int window_height() const override {
5482 return display.window_height();
5483 }
5484
5487 void save(const std::string& file) {
5488 Image screenshotImg;
5489 display.snapshot(screenshotImg);
5490 screenshotImg.save(file.c_str());
5491 }
5492
5496 void mainloop() {
5497 while (!display.is_closed()) {
5498 update(false, true);
5499 std::this_thread::yield();//Yield repetitive loops on mainloop to avoid high-cpu usage.
5500 detail::sleep(1000/60);
5501 }
5502 }
5503
5505 void bye() override{
5507 tracer(1, delayMS);
5508 }
5509
5510 if (eventThread != nullptr) {
5511 killEventThread = true;
5512 eventThread->join();
5513 eventThread.reset(nullptr);
5514 }
5515
5516 clearscreen();
5517
5518 if (!display.is_closed())
5519 display.close();
5520 }
5521
5523 Image& getcanvas() override{
5524 return canvas;
5525 }
5526
5528 cimg::CImgDisplay& internaldisplay() {
5529 return display;
5530 }
5531
5534 inline bool isclosed() override{
5535 return internaldisplay().is_closed();
5536 }
5537
5539 void redraw(bool invalidate) override{
5540 if (isclosed())
5541 return;
5542 int fromBack = 0;
5543 bool hasInvalidated = invalidate;
5544
5545 //Handle resizes.
5546 if (display.window_width() != canvas.width() || display.window_height() != canvas.height()) {
5547 canvas.resize(display);
5548 hasInvalidated = true;
5549 }
5550
5551 if (lastTotalObjects <= objects.size()) {
5552 fromBack = static_cast<int>(objects.size() - lastTotalObjects);
5553 }
5554
5555 if (hasInvalidated) {
5556 if(!backgroundImage.is_empty()){
5557 const int centerX = (canvas.width() / 2) - (backgroundImage.width() / 2);
5558 const int centerY = (canvas.height() / 2) - (backgroundImage.height() / 2);
5559 canvas.draw_image(centerX, centerY, backgroundImage);
5560 }
5561 else canvas.draw_rectangle(0, 0, canvas.width(), canvas.height(), backgroundColor.rgbPtr());
5562
5563 redrawCounter = 0;//Forced redraw due to canvas invalidation.
5564 } else {
5565 if(redrawCounterMax == 0)//tracer settings may disable rendering for a short time...
5566 return;
5567 redrawCounter++;
5568
5570 redrawCounter = 0;
5571 else return;
5572 }
5573
5574 //get the iterator pointing to the oldest scene object that hasn't been drawn yet
5575 //if the scene has been invalidated, the latest object is the first one in the scene.
5576 auto latestIter = !hasInvalidated ? std::prev(objects.end(), fromBack) : objects.begin();
5577
5578 const Transform screen = screentransform();
5579 while (latestIter != objects.end()) {
5580 SceneObject& object = *latestIter;
5581 const Transform t(screen.copyConcatenate(object.transform));
5582 object.geom->draw(t, canvas);
5583
5584 latestIter++;
5585 }
5586
5587 if (canvas.width() != turtleComposite.width() || canvas.height() != turtleComposite.height()) {
5588 turtleComposite.assign(canvas);
5589 } else {
5590 //This works off the assumption that drawImage is accelerated.
5591 //There might be a more efficient way to do this, however.
5592 turtleComposite.draw_image(0, 0, canvas);
5593 }
5594
5595 for (Turtle* turt : turtles)
5596 turt->draw(screen, turtleComposite);
5597
5598 lastTotalObjects = static_cast<int>(objects.size());
5599 display.display(turtleComposite);
5600 detail::sleep(delayMS);
5601 }
5602
5607 Transform screentransform() const override{
5608 //Scale negatively on Y axis to match
5609 //Python's coordinate system.
5610 //without this scaling, top left is 0,0
5611 //instead of the bottom left (which is 0,Y without the scaling)
5612 return Transform().translate(canvas.width()/2, canvas.height()/2).scale(1, -1.0f);
5613 }
5614
5618 void onkeypress(const KeyFunc& func, KeyboardKey key) {
5619 eventCacheMutex.lock();
5620 //determine if key list exists
5621 if (keyBindings[0].find(key) == keyBindings[0].end()) {
5622 keyBindings[0][key] = std::list<KeyFunc>();
5623 }
5624 //then push it to the end of the list
5625 keyBindings[0][key].push_back(func);
5626 eventCacheMutex.unlock();
5627 }
5628
5632 virtual void onkeyrelease(const KeyFunc& func, KeyboardKey key) {
5633 eventCacheMutex.lock();
5634 //determine if key list exists, if not, make one
5635 if (keyBindings[1].find(key) == keyBindings[1].end()) {
5636 keyBindings[1][key] = std::list<KeyFunc>();
5637 }
5638 //then push it to the end of the list
5639 keyBindings[1][key].push_back(func);
5640 eventCacheMutex.unlock();
5641 }
5642
5645 void presskey(KeyboardKey key) {
5646 if (keyBindings[0].find(key) == keyBindings[0].end())
5647 return;
5648 for (const KeyFunc& func : keyBindings[0][key]) {
5649 func();
5650 }
5651 }
5652
5655 void releasekey(KeyboardKey key) {
5656 if (keyBindings[1].find(key) == keyBindings[1].end())
5657 return;
5658 for (KeyFunc& func : keyBindings[1][key]) {
5659 func();
5660 }
5661 }
5662
5666 void onclick(const MouseFunc& func, MouseButton button = MOUSEB_LEFT) {
5667 eventCacheMutex.lock();
5668 mouseBindings[button].push_back(func);
5669 eventCacheMutex.unlock();
5670 }
5671
5676 void click(int x, int y, MouseButton button) {
5677 eventCacheMutex.lock();
5678 for (MouseFunc& func : mouseBindings[button]) {
5679 func(x, y);
5680 }
5681 eventCacheMutex.unlock();
5682 }
5683
5685 inline void click(const Point& pt, MouseButton button){
5686 click(pt.x, pt.y, button);
5687 }
5688
5692 void ontimer(const TimerFunc& func, unsigned int time) {
5693 timerBindings.emplace_back(std::make_tuple(func, time, detail::epochTime()));
5694 }
5695
5699 //Catch up visually before entering event loop, when necessary.
5701 tracer(1, delayMS);
5702 }
5703
5704 onclick([&](int x, int y) {
5705 display.close();
5706 });
5707 mainloop();
5708 }
5709
5711 void add(Turtle& turtle) override{
5712 turtles.push_back(&turtle);
5713 }
5714
5719 void remove(Turtle& turtle) override {
5720 turtle.reset();
5721 turtle.setScreen(nullptr);
5722 turtles.remove(&turtle);
5723 }
5724
5727 std::list<SceneObject>& getScene() override{
5728 return objects;
5729 }
5730
5736 AbstractDrawableObject& shape(const std::string& name) override{
5737 return shapes[name];
5738 }
5739
5746 void addfont(const std::string& name, const BitmapFont& font){
5747 fonts[name] = std::unique_ptr<BitmapFont>(new BitmapFont(font));
5748 }
5749
5755 const BitmapFont& font(const std::string& name) const override{
5756 return *fonts.at(name);
5757 }
5758 protected:
5760 cimg::CImgDisplay display;
5761
5763 Image canvas;
5764
5765 //The turtle composite image.
5766 //This image copies the canvas and has
5767 //turtles drawn to it to avoid redrawing a "busy" canvas.
5768 //Trace lines are also drawn on this when filling.
5769 Image turtleComposite;
5770
5774
5783 ScreenMode curMode = SM_STANDARD;
5784
5786 long int delayMS = 10;
5787
5793
5800 eventThread.reset(new std::thread([&]() {
5801 //Mouse button states, between updates.
5802 //Keeps track of release/press etc
5803 //states for all three mouse buttons for isDown.
5804 //*importantly, this allows us to avoid repeated events.
5805 bool mButtons[3] = {false, false, false};
5806 //Same thing for keys here.
5807 //(this is a list of keys marked as being in a "down" state)
5808 std::list<KeyboardKey> mKeys;
5809
5810 while (!display.is_closed() && !killEventThread) {
5811 //Updates all input.
5812 if (!display.is_event()) {
5813 std::this_thread::yield();
5814 continue;
5815 }
5816
5817 eventCacheMutex.lock();
5818
5819 Transform mouseOffset = screentransform();
5820 Point mousePos = {
5821 static_cast<int>((static_cast<float>(display.mouse_x()) - mouseOffset.getTranslateX()) * mouseOffset.getScaleX()),
5822 static_cast<int>((static_cast<float>(display.mouse_y()) - mouseOffset.getTranslateY()) * mouseOffset.getScaleY())
5823 };
5824
5825 //Update mouse button input.
5826 const unsigned int button = display.button();
5827 bool buttons[3] = {
5828 static_cast<bool>(button & 1), //left
5829 static_cast<bool>(button & 2), //right
5830 static_cast<bool>(button & 4) //middle
5831 };
5832
5833 for (int i = 0; i < 3; i++) {
5834 if (!(!mButtons[i] && buttons[i]))//is this button state "down"?
5835 continue; //if not, skip its processing loop.
5836
5837 for (MouseFunc& func : mouseBindings[i]) {
5838 //append to the event cache.
5839 InputEvent e;
5840 e.type = false;
5841 e.mX = mousePos.x;
5842 e.mY = mousePos.y;
5843 e.cbPointer = reinterpret_cast<void*> (&func);
5844 cachedEvents.push_back(e);
5845 }
5846 }
5847
5848 const auto& keys = NAMED_KEYS;
5849
5850 //iterate through every key to determine its state,
5851 //then call the appropriate callbacks.
5852 for (const auto& keyPair : keys) {
5853 KeyboardKey key = keyPair.second;
5854 const bool lastDown = std::find(mKeys.begin(), mKeys.end(), key) != mKeys.end();
5855 const bool curDown = display.is_key((unsigned int) key);
5856
5857 int state = -1;
5858 if (!lastDown && curDown) {
5859 //Key down.
5860 state = 0;
5861 mKeys.push_back(key);
5862 } else if (lastDown && !curDown) {
5863 //Key up.
5864 state = 1;
5865 mKeys.remove(key);
5866 } else continue; //skip on case where it was down and is down
5867
5868 try {
5869 //will throw if no bindings available for key,
5870 //and that's perfectly fine, so we just silently catch
5871 auto& bindingList = keyBindings[state][key];
5872 for (auto& cb : bindingList) {
5873 cb();
5874 }
5875 } catch (...) {}
5876 }
5877
5878 mButtons[0] = buttons[0];
5879 mButtons[1] = buttons[1];
5880 mButtons[2] = buttons[2];
5881 eventCacheMutex.unlock();
5882 }
5883 }));
5884 }
5885
5887 std::list<SceneObject> objects;
5888
5890 std::list<Turtle*> turtles;
5891
5894 std::unique_ptr<std::thread> eventThread;
5897 std::list<InputEvent> cachedEvents;
5899 bool killEventThread = false;
5903
5904 //this is an array. 0 for keyDown bindings, 1 for keyUp bindings.
5905 std::unordered_map<KeyboardKey, std::list<KeyFunc>> keyBindings[2] = {
5906 {},
5907 {}
5908 };
5909 //similar, mouseb_left mouseb_middle mouseb_right bindings.
5910 std::list<MouseFunc> mouseBindings[3] = {
5911 {},
5912 {},
5913 {}
5914 };
5915 //timer bindings, one function per originating time and delta.
5916 std::list<std::tuple<TimerFunc, uint64_t, uint64_t>> timerBindings;
5917
5918 //map of fonts. only "default" is initially populated.
5919 std::unordered_map<std::string, std::unique_ptr<BitmapFont>> fonts;
5920 };
5921
5922 typedef InteractiveTurtleScreen TurtleScreen;
5923#endif /*CTURTLE_HEADLESS*/
5924}
AbstractDrawableObject is a base class, intended to be inherited from by all drawable objects....
Definition CTurtle.hpp:3219
Color fillColor
Definition CTurtle.hpp:3223
virtual ~AbstractDrawableObject()=default
Empty– virtual– default de-constructor.
virtual void draw(const Transform &t, Image &imgRef) const =0
This function is intended to draw all applicable geometry in this object to the specified image,...
AbstractDrawableObject()=default
Empty default constructor.
Color outlineColor
Definition CTurtle.hpp:3226
int outlineWidth
Definition CTurtle.hpp:3229
virtual AbstractDrawableObject * copy() const =0
Returns a pointer to a copy of this drawable object, allocated with NEW. Result must be deleted at th...
The AbstractTurtleScreen class is the abstract type for most turtle functionality....
Definition CTurtle.hpp:3815
virtual void tracer(int countmax, unsigned int delayMS)=0
virtual AbstractDrawableObject & shape(const std::string &name)=0
Returns a read-write reference to the registered shape with the specified name.
virtual int window_width() const =0
virtual void resetscreen()=0
virtual std::list< SceneObject > & getScene()=0
Returns a read-write reference to the list of scene objects currently residing on this screen.
virtual Transform screentransform() const =0
Calculates and returns the root-level screen transformation.
virtual void mode(ScreenMode mode)=0
virtual ScreenMode mode() const =0
virtual int window_height() const =0
static Image decodeDefaultFont()
Definition CTurtle.hpp:3947
virtual void add(Turtle &turtle)=0
Adds the specified turtle to this screen. The turtle cannot belong to other screens,...
virtual bool supports_live_animation() const =0
void clear()
Definition CTurtle.hpp:3860
virtual void bgcolor(const Color &c)=0
virtual Color bgcolor() const =0
void reset()
Definition CTurtle.hpp:3868
virtual void clearscreen()=0
virtual const BitmapFont & font(const std::string &name) const =0
Returns a read-only reference to a previously loaded bitmap font.
virtual void remove(Turtle &turtle)=0
Removes the specified turtle from this screen.
The Bitmap Font represents monospaced font image files that covers a range of lower ASCII....
Definition CTurtle.hpp:3126
ivec2 getGlyphExtent() const
Returns the size of a single character glyph, in pixels.
Definition CTurtle.hpp:3184
ivec2 getGlyphAxes() const
Returns the number of glyphs along each axis of the underlying image.
Definition CTurtle.hpp:3206
const Image & operator[](unsigned char c) const
Definition CTurtle.hpp:3165
BitmapFont(const Image &img, int asciiOffs, int glyphWidth, int glyphHeight, int glyphsX, int glyphsY)
Definition CTurtle.hpp:3136
int getTotalGlyphs() const
Definition CTurtle.hpp:3191
const Image & getGlyphImage(unsigned char c) const
Definition CTurtle.hpp:3157
ivec2 getGlyphPosition(unsigned char c) const
Returns the position of the specified character in the font image.
Definition CTurtle.hpp:3174
bool isValid(char c) const
Definition CTurtle.hpp:3198
The Circle class holds a radius and total number of steps, used to generate and draw a circles geomet...
Definition CTurtle.hpp:3415
int radius
Definition CTurtle.hpp:3418
Circle(const Circle &other)=default
Copy constructor.
int steps
Definition CTurtle.hpp:3421
Circle(int radius, int steps, const Color &fillColor, int outlineWidth=0, const Color &outlineColor=Color())
Radius and step assignment constructor.
Definition CTurtle.hpp:3429
AbstractDrawableObject * copy() const override
Returns a pointer to a copy of this drawable object, allocated with NEW. Result must be deleted at th...
Definition CTurtle.hpp:3440
void draw(const Transform &t, Image &imgRef) const override
This function is intended to draw all applicable geometry in this object to the specified image,...
Definition CTurtle.hpp:3444
Circle()=default
Empty constructor.
The primary representation of Color for this library. Represented as a simple RGB color composed of u...
Definition CTurtle.hpp:1460
const component_t * rgbPtr() const
Returns a pointer to the first component of this color. This is useful for functions which require co...
Definition CTurtle.hpp:1510
a Compound Polygon instance is composed from a number of smaller parts, which are each derived from A...
Definition CTurtle.hpp:3632
void draw(const Transform &t, Image &imgRef) const override
Definition CTurtle.hpp:3667
AbstractDrawableObject * copy() const override
Definition CTurtle.hpp:3661
void addcomponent(const AbstractDrawableObject &obj, const Transform &transform=Transform())
Adds a component to this compound polygon.
Definition CTurtle.hpp:3653
The InteractiveTurtleScreen class holds and maintains facilities in relation to displaying \ turtles ...
Definition CTurtle.hpp:5248
void addfont(const std::string &name, const BitmapFont &font)
Definition CTurtle.hpp:5746
void mainloop()
Definition CTurtle.hpp:5496
ScreenMode mode() const override
Definition CTurtle.hpp:5365
Image backgroundImage
Definition CTurtle.hpp:5780
std::list< InputEvent > cachedEvents
Definition CTurtle.hpp:5897
void ontimer(const TimerFunc &func, unsigned int time)
Adds a timer function to be called every N milliseconds.
Definition CTurtle.hpp:5692
long int delayMS
Definition CTurtle.hpp:5786
void tracer(int countmax, unsigned int delayMS) override
Definition CTurtle.hpp:5311
void delay(unsigned int ms) override
Definition CTurtle.hpp:5466
void releasekey(KeyboardKey key)
Simulates a key "on release" event.
Definition CTurtle.hpp:5655
virtual void onkeyrelease(const KeyFunc &func, KeyboardKey key)
Adds an additional "on press" key binding for the specified key.
Definition CTurtle.hpp:5632
std::list< SceneObject > & getScene() override
Definition CTurtle.hpp:5727
void mode(ScreenMode mode) override
Definition CTurtle.hpp:5358
const Image & bgpic()
Definition CTurtle.hpp:5344
int redrawCounter
Definition CTurtle.hpp:5790
InteractiveTurtleScreen()
Definition CTurtle.hpp:5252
Transform screentransform() const override
Definition CTurtle.hpp:5607
Color backgroundColor
Definition CTurtle.hpp:5776
void onclick(const MouseFunc &func, MouseButton button=MOUSEB_LEFT)
Adds an additional "on click" mouse binding for the specified button.
Definition CTurtle.hpp:5666
void presskey(KeyboardKey key)
Simulates a key "on press" event.
Definition CTurtle.hpp:5645
void clearscreen() override
Clears this screen. Deletes all drawings and turtles, Resets the background to plain white,...
Definition CTurtle.hpp:5374
ivec2 screensize(Color &bg) override
Definition CTurtle.hpp:5407
Image canvas
Definition CTurtle.hpp:5763
const BitmapFont & font(const std::string &name) const override
Definition CTurtle.hpp:5755
cimg::CImgDisplay display
Definition CTurtle.hpp:5760
void remove(Turtle &turtle) override
Removes the specified turtle from this screen.
Definition CTurtle.hpp:5719
void add(Turtle &turtle) override
Definition CTurtle.hpp:5711
unsigned int delay() const override
Definition CTurtle.hpp:5471
cimg::CImgDisplay & internaldisplay()
Definition CTurtle.hpp:5528
int window_width() const override
Definition CTurtle.hpp:5476
Image & getcanvas() override
Definition CTurtle.hpp:5523
ScreenMode curMode
Definition CTurtle.hpp:5783
~InteractiveTurtleScreen()
Definition CTurtle.hpp:5301
bool isclosed() override
Definition CTurtle.hpp:5534
void click(const Point &pt, MouseButton button)
Definition CTurtle.hpp:5685
int redrawCounterMax
Definition CTurtle.hpp:5792
InteractiveTurtleScreen(int width, int height, const std::string &title=SCREEN_DEFAULT_TITLE)
Definition CTurtle.hpp:5285
bool killEventThread
Definition CTurtle.hpp:5899
void resetscreen() override
Definition CTurtle.hpp:5399
bool supports_live_animation() const override
Definition CTurtle.hpp:5348
void click(int x, int y, MouseButton button)
Definition CTurtle.hpp:5676
void bye() override
Definition CTurtle.hpp:5505
void initEventThread()
Definition CTurtle.hpp:5799
void redraw(bool invalidate) override
Definition CTurtle.hpp:5539
void save(const std::string &file)
Definition CTurtle.hpp:5487
std::mutex eventCacheMutex
Definition CTurtle.hpp:5902
AbstractDrawableObject & shape(const std::string &name) override
Definition CTurtle.hpp:5736
std::list< SceneObject > objects
Definition CTurtle.hpp:5887
ivec2 screensize() override
Definition CTurtle.hpp:5413
std::unique_ptr< std::thread > eventThread
Definition CTurtle.hpp:5894
Color bgcolor() const override
Definition CTurtle.hpp:5329
void onkeypress(const KeyFunc &func, KeyboardKey key)
Adds an additional "on press" key binding for the specified key.
Definition CTurtle.hpp:5618
InteractiveTurtleScreen(const std::string &title)
Definition CTurtle.hpp:5266
void exitonclick()
Definition CTurtle.hpp:5698
void update(bool invalidateDraw, bool processInput) override
Definition CTurtle.hpp:5421
std::list< Turtle * > turtles
Definition CTurtle.hpp:5890
int window_height() const override
Definition CTurtle.hpp:5481
void bgcolor(const Color &color) override
Definition CTurtle.hpp:5322
void bgpic(const Image &img)
Sets the background image of the display. Sets the background image. Please note that the background ...
Definition CTurtle.hpp:5337
int lastTotalObjects
Definition CTurtle.hpp:5773
The Line class holds two points and the functionality to draw a line between them on a specified canv...
Definition CTurtle.hpp:3371
void draw(const Transform &t, Image &imgRef) const override
This function is intended to draw all applicable geometry in this object to the specified image,...
Definition CTurtle.hpp:3406
Point pointB
Definition CTurtle.hpp:3378
Line()=default
Empty default constructor.
~Line() override=default
Empty de-constructor.
AbstractDrawableObject * copy() const override
Returns a pointer to a copy of this drawable object, allocated with NEW. Result must be deleted at th...
Definition CTurtle.hpp:3399
Line(Point a, Point b, const Color &color, int width=1)
Value constructor. merely assigns value of pointA and pointB to respective A and B.
Definition CTurtle.hpp:3390
Line(const Line &other)=default
Copy constructor. Merely assigns the "to" and "from" points.
Point pointA
Definition CTurtle.hpp:3375
int width
Definition CTurtle.hpp:3381
The polygon class merely holds a vector of points and a function to draw this series to an image....
Definition CTurtle.hpp:3475
Polygon(const std::initializer_list< Point > &init)
Initializer list instructor which assigns the points to the contents of the specified initializer lis...
Definition CTurtle.hpp:3485
~Polygon() override=default
Empty de-constructor.
Polygon()=default
Empty default constructor.
Polygon(std::vector< Point > copy, const Color &fillColor, int outlineWidth=0, const Color &outlineColor=Color())
A copy constructor for another vector of points.
Definition CTurtle.hpp:3494
Polygon(const Polygon &other)=default
A copy constructor for another polygon.
AbstractDrawableObject * copy() const override
Definition CTurtle.hpp:3509
void draw(const Transform &t, Image &imgRef) const override
This function is intended to draw all applicable geometry in this object to the specified image,...
Definition CTurtle.hpp:3516
The Sprite class represents a selection of a larger image. This class ignores color in favor of color...
Definition CTurtle.hpp:3547
void draw(const Transform &t, Image &imgRef) const override
Definition CTurtle.hpp:3578
AbstractDrawableObject * copy() const override
Returns a pointer to a copy of this drawable object, allocated with NEW. Result must be deleted at th...
Definition CTurtle.hpp:3572
The Text class represents a basic string that is drawn on the screen.
Definition CTurtle.hpp:3264
void draw(const Transform &t, Image &imgRef) const override
This function is intended to draw all applicable geometry in this object to the specified image,...
Definition CTurtle.hpp:3283
AbstractDrawableObject * copy() const override
Returns a pointer to a copy of this drawable object, allocated with NEW. Result must be deleted at th...
Definition CTurtle.hpp:3279
const std::string text
Definition CTurtle.hpp:3267
The Transform class provides a myriad of functions to simply transform points. This class it the back...
Definition CTurtle.hpp:2720
float & at(int row, int col)
Returns a reference to the float the specified coordinate.
Definition CTurtle.hpp:3031
Transform(const Transform &other)=default
Copy constructor.
Point getTranslation() const
Returns the translation of this transform as a point.
Definition CTurtle.hpp:2825
float getRotation() const
Returns rotation of this transform, in radians.
Definition CTurtle.hpp:2796
std::array< float, 9 > mat_t
Definition CTurtle.hpp:3020
float getTranslateX() const
Returns the X translation of this transform.
Definition CTurtle.hpp:2784
Transform & setTranslationY(int y)
Set the Y axis translation of this transform.
Definition CTurtle.hpp:2840
Transform & forward(float distance)
Definition CTurtle.hpp:2801
Transform copyConcatenate(const Transform &t) const
Creates a copy of this transform, concatenates the input, and returns it.
Definition CTurtle.hpp:2948
Transform & rotateAround(int x, int y, float theta)
Rotates this transform around a specified point.
Definition CTurtle.hpp:2906
Transform & concatenate(const Transform &t)
Concatenates this Transform with another.
Definition CTurtle.hpp:2927
void assign(const Transform &t)
Assigns the value of this transform to that of another.
Definition CTurtle.hpp:2974
Transform & setRotation(float val)
Sets the rotation of this transform.
Definition CTurtle.hpp:2892
Transform & identity()
Sets this transform to an identity. When you concatenate an identity transform onto another object,...
Definition CTurtle.hpp:2750
mat_t value
Definition CTurtle.hpp:3023
bool operator==(const Transform &other) const
Returns a boolean indicating if this transform is equivalent in value to the one specified.
Definition CTurtle.hpp:2759
float rotation
Definition CTurtle.hpp:3026
float getScaleY() const
Returns the Y scale of this transform.
Definition CTurtle.hpp:2778
Transform()
Definition CTurtle.hpp:2724
Transform & translate(int x, int y)
Translates this transform.
Definition CTurtle.hpp:2849
float getTranslateY() const
Returns the Y translation of this transform.
Definition CTurtle.hpp:2790
Transform & setTranslation(int x, int y)
Sets the translation of this transform.
Definition CTurtle.hpp:2817
Point transform(Point in, Point *dst=nullptr) const
Transforms a point according to this transform.
Definition CTurtle.hpp:2984
Transform & setTranslationX(int x)
Sets the X axis translation of this transform.
Definition CTurtle.hpp:2832
void transformSet(ITER_T cur, ITER_T end) const
Transforms a set of points given a begin and end iterator.
Definition CTurtle.hpp:3002
Transform lerp(const Transform &t, float progress) const
Interpolates between this and the specified transform. Progress is a float in range of 0 to 1.
Definition CTurtle.hpp:2960
Transform & rotate(float theta)
Rotates this transform.
Definition CTurtle.hpp:2858
float getScaleX() const
Returns the X scale of this transform.
Definition CTurtle.hpp:2772
Transform & scale(float sx, float sy)
Applies a scale transformation to this transform.
Definition CTurtle.hpp:2916
Transform(const ivec2 &point, float rotation)
Point and rotation constructor. Initializes a transform with the specified rotation,...
Definition CTurtle.hpp:2737
float at(int row, int col) const
Returns a copy of the float at the specified coordinate.
Definition CTurtle.hpp:3038
Point operator()(Point in) const
Operator overload to transform a single point.
Definition CTurtle.hpp:3013
The Turtle Class Symbolically represents a turtle that runs around a screen that has a paint brush at...
Definition CTurtle.hpp:4014
void face(const Point &pt)
Definition CTurtle.hpp:4222
void setshowturtle(bool val)
Set whether or not the turtle is being shown.
Definition CTurtle.hpp:4560
Point getpos() const
Returns the current location of the turtle, as a Point.
Definition CTurtle.hpp:4171
void setScreen(AbstractTurtleScreen *scr)
Sets this turtles screen.
Definition CTurtle.hpp:4707
void circle(int radius, int steps, const Color &color)
Adds a circle to the screen.
Definition CTurtle.hpp:4302
bool undo(bool try_redraw=true)
Undoes the previous action of this turtle.
Definition CTurtle.hpp:4485
float towards(int x, int y)
Returns the angle between the line of the current turtle transform to the given point.
Definition CTurtle.hpp:4258
void goTo(int x, int y)
Sets the transform location of this turtle.
Definition CTurtle.hpp:4112
void width(int pixels)
Definition CTurtle.hpp:4611
bool pushGeometry(const Transform &t, AbstractDrawableObject *geom)
Internal function used to add geometry to the turtle screen.
Definition CTurtle.hpp:4776
void fd(int pixels)
Moves the turtle forward the specified number of pixels.
Definition CTurtle.hpp:4054
const AbstractDrawableObject & shape()
Returns the shape of this turtle.
Definition CTurtle.hpp:4480
void penup()
Brings the pen up.
Definition CTurtle.hpp:4586
void seth(float angle)
Sets the rotation of this turtle. The unit by which the input is specified is determined by the curre...
Definition CTurtle.hpp:4248
void backward(int pixels)
Moves the turtle backward the specified number of pixels.
Definition CTurtle.hpp:4059
void setheading(float amt)
Sets the rotation of this turtle. The unit by which the input is specified is determined by the curre...
Definition CTurtle.hpp:4199
void right(float amt)
Rotates the turtle the specified number of units to the right. The unit by which the input is specifi...
Definition CTurtle.hpp:4082
void fillcolor(const Color &c)
Sets the fill color of this turtle.
Definition CTurtle.hpp:4370
void dot(const Color &color, int size=10)
Adds a dot to the screen.
Definition CTurtle.hpp:4318
void back(int pixels)
Moves the turtle backward the specified number of pixels.
Definition CTurtle.hpp:4071
int xcor() const
Returns the current x coordinate location of this turtle.
Definition CTurtle.hpp:4155
void bk(int pixels)
Moves the turtle backward the specified number of pixels.
Definition CTurtle.hpp:4066
void forward(int pixels)
Moves the turtle forward the specified number of pixels.
Definition CTurtle.hpp:4047
bool pushText(const Transform &t, const Color &color, const BitmapFont &font, const std::string &text, float scale=1.0f, TextAlign alignment=TEXT_ALIGN_LEFT)
Internal function used to add a text object to the turtle screen.
Definition CTurtle.hpp:4817
void lt(float angle)
Rotates the turtle the specified number of units to the left. The unit by which the input is specifie...
Definition CTurtle.hpp:4107
int distance(int x, int y)
Definition CTurtle.hpp:4233
void shape(const AbstractDrawableObject &p)
Sets the shape of this turtle.
Definition CTurtle.hpp:4465
const PenState & penstate() const
Definition CTurtle.hpp:4188
void rt(float angle)
Rotates the turtle the specified number of units to the right. The unit by which the input is specifi...
Definition CTurtle.hpp:4089
float heading()
Returns the heading of the Turtle (e.g, its current rotation).
Definition CTurtle.hpp:4283
void begin_fill()
Begins filling a polygon.
Definition CTurtle.hpp:4358
void setposition(int x, int y)
Sets the transform location of this turtle.
Definition CTurtle.hpp:4132
bool pushTraceLine(Point a, Point b)
Internal function used to add a trace line object to the turtle screen. Trace lines do NOT push a sta...
Definition CTurtle.hpp:4832
void speed(float val)
Sets the speed of this turtle in range of 0 to 10.
Definition CTurtle.hpp:4532
void left(float amt)
Rotates the turtle the specified number of units to the left. The unit by which the input is specifie...
Definition CTurtle.hpp:4100
void setx(int x)
Sets the X-axis transform location of this turtle.
Definition CTurtle.hpp:4142
void write(const std::string &text)
Definition CTurtle.hpp:4386
virtual ~Turtle()
Empty virtual destructor.
Definition CTurtle.hpp:4712
void setpos(const Point &pt)
Sets the transform location of this turtle.
Definition CTurtle.hpp:4127
Turtle & operator=(const Turtle &)=delete
Turtles are not trivially moved. The copy operator is explicitly disallowed.
void setposition(const Point &pt)
Sets the transform location of this turtle.
Definition CTurtle.hpp:4137
bool pushStamp(const Transform &t, AbstractDrawableObject *geom)
Internal function used to add a stamp object to the turtle screen.
Definition CTurtle.hpp:4790
void pendown()
Brings the pen down.
Definition CTurtle.hpp:4591
int ycor() const
Returns the current y coordinate location of this turtle.
Definition CTurtle.hpp:4163
void travelBack()
Definition CTurtle.hpp:4926
void setundobuffer(unsigned int size)
Set, or disable, the undo buffer.
Definition CTurtle.hpp:4514
void clearstamps(int stampid=-1)
Removes all stamps with an ID less than that which is specified. If the specified stampid is less tha...
Definition CTurtle.hpp:4441
void travelBetween(Transform src, const Transform &dest, bool doPushState)
Definition CTurtle.hpp:4870
Turtle(const Turtle &)=delete
Turtles are not trivially copyable. The copy constructor is explicitly disallowed.
void hideturtle()
Hides the turtle.
Definition CTurtle.hpp:4575
Turtle()=default
void radians()
Definition CTurtle.hpp:4663
void clearstamp(int stampid)
Removes the stamp with the specified ID.
Definition CTurtle.hpp:4417
Color fillcolor()
Returns the fill color of this turtle.
Definition CTurtle.hpp:4378
void degrees()
Definition CTurtle.hpp:4656
int distance(const Point &pt)
Definition CTurtle.hpp:4243
void draw(const Transform &screenTransform, Image &canvas)
Draws this turtle on the specified canvas with the specified transform.
Definition CTurtle.hpp:4625
void end_fill()
Stops filling a polygon.
Definition CTurtle.hpp:4364
Color pencolor() const
Returns the pen color; the color of the lines between movements.
Definition CTurtle.hpp:4605
float towards(const Point &pt)
Returns the angle between the line of the current turtle transform to the given point.
Definition CTurtle.hpp:4275
void home()
Definition CTurtle.hpp:4294
void setpos(int x, int y)
Sets the transform location of this turtle.
Definition CTurtle.hpp:4122
void setpenstate(bool down)
Sets whether or not the pen is down.
Definition CTurtle.hpp:4580
void fill(bool val)
Sets the "filling" state-> If the input is false but the prior state is true, a SceneObject is put on...
Definition CTurtle.hpp:4326
void reset()
Resets this turtle. Moves this turtle home, resets all pen attributes, and removes all previously add...
Definition CTurtle.hpp:4671
bool filling() const
Returns a boolean indicating if this turtle is currently filling a shape.
Definition CTurtle.hpp:4352
void showturtle()
Shows the turtle. Equivalent to calling setshowturtle(true).
Definition CTurtle.hpp:4569
void pencolor(const Color &c)
Sets the pen color.
Definition CTurtle.hpp:4597
void updateParent(bool invalidate=false, bool input=true)
Definition CTurtle.hpp:4860
void tilt(float amt)
Applies a rotation to the turtle's cursor.
Definition CTurtle.hpp:4543
void write(const std::string &text, const std::string &font, const Color &color={"white"}, float scale=1.0f, TextAlign alignment=TEXT_ALIGN_LEFT)
Definition CTurtle.hpp:4401
unsigned int undobufferentries()
Returns the size of the undo stack.
Definition CTurtle.hpp:4525
void shift(int x, int y)
Adds a "naive" translation to the current turtle's transform.\ Does not take into account the rotatio...
Definition CTurtle.hpp:4181
void goTo(const Point &pt)
Sets the transform location of this turtle.
Definition CTurtle.hpp:4117
float tilt() const
Returns the rotation of the cursor. Not the heading, or the angle at which the forward function will ...
Definition CTurtle.hpp:4554
void shape(const std::string &name)
Sets the shape of this turtle from the specified shape name.
Definition CTurtle.hpp:4473
Turtle(Turtle &&mv)=delete
Turtles are not trivially moved. The move constructor is explicitly disallowed.
void sety(int y)
Sets the Y-axis transform location of this turtle.
Definition CTurtle.hpp:4147
float speed()
Returns the speed of this turtle.
Definition CTurtle.hpp:4538
int width() const
Definition CTurtle.hpp:4618
long int getAnimMS()
Definition CTurtle.hpp:4847
Turtle & operator=(Turtle &&turtle)=delete
Turtles are not trivially moved. The move operator is explicitly disallowed.
int stamp()
Puts the current shape of this turtle on the screen with the current fill color and the outline of th...
Definition CTurtle.hpp:4411
Turtle(AbstractTurtleScreen &scr)
The turtle constructor. Turtles are attached to whatever screen they are constructed with.
Definition CTurtle.hpp:4019
void travelTo(const Transform &dest)
Definition CTurtle.hpp:4918
void circle(const Color &color)
Adds a circle to the screen. Default parameters are circle with a radius of 30 with 15 steps.
Definition CTurtle.hpp:4310
void face(int x, int y)
Definition CTurtle.hpp:4213
The internally-used representation of an Input Event. Contains information pertaining to keyboard and...
Definition CTurtle.hpp:2598
The Pen State structure Holds all pen attributes, which are grouped in this way to allow stack-based ...
Definition CTurtle.hpp:3731
Color penColor
Definition CTurtle.hpp:3746
int curStamp
Definition CTurtle.hpp:3755
int penWidth
Definition CTurtle.hpp:3742
float cursorTilt
Definition CTurtle.hpp:3759
size_t objectsBefore
Definition CTurtle.hpp:3751
Color fillColor
Definition CTurtle.hpp:3748
bool visible
Definition CTurtle.hpp:3757
bool tracing
Definition CTurtle.hpp:3738
bool filling
Definition CTurtle.hpp:3744
Transform transform
Definition CTurtle.hpp:3734
bool angleMode
Definition CTurtle.hpp:3740
float moveSpeed
Definition CTurtle.hpp:3736
std::unique_ptr< AbstractDrawableObject > cursor
Definition CTurtle.hpp:3753
Turtles append Scene Objects to a list to keep track of what it has drawn (a history)....
Definition CTurtle.hpp:3698
bool stamp
Definition CTurtle.hpp:3709
std::unique_ptr< AbstractDrawableObject > geom
Definition CTurtle.hpp:3701
SceneObject(AbstractDrawableObject *geom, const Transform &t, int stampid=-1)
Definition CTurtle.hpp:3723
Transform transform
Definition CTurtle.hpp:3706
int stampid
Definition CTurtle.hpp:3711
Represents a coordinate pair (e.g, x & y) This class is represented as a low-precision point,...
Definition CTurtle.hpp:2620
int & operator[](int index)
Array access operator overload. Useful for convenience.
Definition CTurtle.hpp:2641
ivec2(int x, int y)
Assignment cons tructor.
Definition CTurtle.hpp:2634
ivec2()
Empty constructor. Initializes X and Y to both equal 0.
Definition CTurtle.hpp:2629
int operator[](int index) const
Array access operator overload. Useful for convenience.
Definition CTurtle.hpp:2648
bool operator==(const ivec2 &other) const
Comparison operator between this vector and the other specified.
Definition CTurtle.hpp:2673