2019-08-23 10:41:23 -04:00
|
|
|
/*
|
|
|
|
* Copyright (C) 2019 Robin Gareus <robin@gareus.org>
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU General Public License as published by
|
|
|
|
* the Free Software Foundation; either version 2, or (at your option)
|
|
|
|
* any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <cmath>
|
|
|
|
#include <cstdint>
|
|
|
|
#include <cstdlib>
|
|
|
|
#include <getopt.h>
|
|
|
|
#include <iostream>
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
|
|
#include <cairo/cairo.h>
|
|
|
|
|
|
|
|
#include "pbd/xml++.h"
|
|
|
|
#include "gtkmm2ext/colors.h"
|
|
|
|
#include "widgets/ardour_icon.h"
|
|
|
|
|
|
|
|
using namespace ArdourWidgets;
|
|
|
|
|
2019-08-27 12:00:56 -04:00
|
|
|
static int wh = 64;
|
|
|
|
static int sq = 1;
|
2020-02-22 11:18:20 -05:00
|
|
|
static int bd = 0;
|
2019-08-27 12:00:56 -04:00
|
|
|
static bool grid = false;
|
2019-08-23 10:41:23 -04:00
|
|
|
|
|
|
|
static uint32_t bg_color = 0x3d3d3dff; // gtk_background
|
|
|
|
static uint32_t fg_color = 0xeeeeecff; // gtk_foreground
|
|
|
|
|
|
|
|
static XMLNode*
|
|
|
|
find_named_node (const XMLNode& node, std::string name)
|
|
|
|
{
|
|
|
|
XMLNodeList nlist = node.children ();
|
|
|
|
for (XMLNodeConstIterator niter = nlist.begin (); niter != nlist.end (); ++niter) {
|
|
|
|
XMLNode* child = *niter;
|
|
|
|
if (child->name () == name) {
|
|
|
|
return child;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::string
|
|
|
|
find_color_alias (const XMLNode& node, std::string colorname)
|
|
|
|
{
|
|
|
|
XMLNodeList nlist = node.children ();
|
|
|
|
for (XMLNodeConstIterator niter = nlist.begin (); niter != nlist.end (); ++niter) {
|
|
|
|
XMLNode* child = *niter;
|
|
|
|
if (child->name () != "ColorAlias") {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
XMLProperty const* name = child->property ("name");
|
|
|
|
XMLProperty const* alias = child->property ("alias");
|
|
|
|
if (!name || !alias) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (name->value () == colorname) {
|
|
|
|
return alias->value ();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return "";
|
|
|
|
}
|
|
|
|
|
|
|
|
static uint32_t
|
|
|
|
lookup_aliased_color (const XMLNode& node, std::string aliasname)
|
|
|
|
{
|
|
|
|
XMLNodeList nlist = node.children ();
|
|
|
|
for (XMLNodeConstIterator niter = nlist.begin (); niter != nlist.end (); ++niter) {
|
|
|
|
XMLNode* child = *niter;
|
|
|
|
if (child->name () != "Color") {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
XMLProperty const* name = child->property ("name");
|
|
|
|
XMLProperty const* color = child->property ("value");
|
|
|
|
if (!name || !color) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (name->value () == aliasname) {
|
|
|
|
return strtoul (color->value ().c_str (), 0, 16);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool
|
|
|
|
load_colors (const char* path)
|
|
|
|
{
|
|
|
|
XMLTree tree;
|
|
|
|
if (!tree.read (path)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
XMLNode* colors = find_named_node (*tree.root (), "Colors");
|
|
|
|
XMLNode* aliases = find_named_node (*tree.root (), "ColorAliases");
|
|
|
|
|
|
|
|
if (!colors || !aliases) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bg_color = lookup_aliased_color (*colors, find_color_alias (*aliases, "gtk_background"));
|
|
|
|
fg_color = lookup_aliased_color (*colors, find_color_alias (*aliases, "gtk_foreground"));
|
|
|
|
|
|
|
|
printf ("Theme colors bg:0x%x fg:0x%x\n", bg_color, fg_color);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
draw_icon (cairo_t* cr, int pos, const enum ArdourIcon::Icon icon, const Gtkmm2ext::ActiveState state)
|
|
|
|
{
|
|
|
|
int col = pos % sq;
|
|
|
|
int row = pos / sq;
|
|
|
|
cairo_save (cr);
|
2020-02-22 11:18:20 -05:00
|
|
|
cairo_translate (cr, bd + col * wh, bd + row * wh);
|
2019-08-27 12:00:56 -04:00
|
|
|
if (grid) {
|
|
|
|
cairo_rectangle (cr, .5, .5, wh - 1, wh - 1);
|
|
|
|
cairo_set_line_width (cr, 1);
|
|
|
|
cairo_set_source_rgba (cr, 0, 0, 0, 1.0);
|
|
|
|
cairo_stroke (cr);
|
|
|
|
cairo_move_to (cr, wh * .5, 0);
|
|
|
|
cairo_line_to (cr, wh * .5, wh);
|
|
|
|
cairo_move_to (cr, 0, wh * .5);
|
|
|
|
cairo_line_to (cr, wh, wh * .5);
|
|
|
|
cairo_stroke (cr);
|
|
|
|
}
|
2019-08-23 10:41:23 -04:00
|
|
|
ArdourIcon::render (cr, icon, wh, wh, state, fg_color);
|
|
|
|
cairo_restore (cr);
|
|
|
|
}
|
|
|
|
|
|
|
|
int
|
|
|
|
main (int argc, char** argv)
|
|
|
|
{
|
|
|
|
const char* fn = "/tmp/ardour_icons.png";
|
2020-02-22 11:18:20 -05:00
|
|
|
bool border = false;
|
2019-08-23 10:41:23 -04:00
|
|
|
|
|
|
|
int c = 0;
|
2020-02-22 11:18:20 -05:00
|
|
|
while (EOF != (c = getopt (argc, argv, "bgo:s:t:"))) {
|
2019-08-23 10:41:23 -04:00
|
|
|
switch (c) {
|
2020-02-22 11:18:20 -05:00
|
|
|
case 'b':
|
|
|
|
border = true;
|
|
|
|
break;
|
2019-08-27 12:00:56 -04:00
|
|
|
case 'g':
|
|
|
|
grid = true;
|
|
|
|
break;
|
2019-08-23 10:41:23 -04:00
|
|
|
case 't':
|
|
|
|
if (!load_colors (optarg)) {
|
|
|
|
std::cerr << "Error: failed to load color theme.\n";
|
|
|
|
::exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case 'o':
|
|
|
|
fn = optarg;
|
|
|
|
break;
|
|
|
|
case 's':
|
|
|
|
wh = atoi (optarg);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
std::cerr << "Error: unrecognized option.\n";
|
|
|
|
::exit (EXIT_FAILURE);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (optind < argc) {
|
|
|
|
std::cerr << "Error: Extra commandline argument.\n";
|
|
|
|
::exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (wh <= 0 || wh > 256) {
|
|
|
|
wh = 64;
|
|
|
|
}
|
|
|
|
|
2020-02-22 11:18:20 -05:00
|
|
|
int fs = std::max (9, wh / 3);
|
|
|
|
|
|
|
|
if (border) {
|
|
|
|
bd = wh / 4;
|
|
|
|
cairo_surface_t* cs = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, 1, 1);
|
|
|
|
cairo_t* cr = cairo_create (cs);
|
|
|
|
cairo_set_font_size (cr, fs);
|
|
|
|
cairo_text_extents_t extents;
|
|
|
|
cairo_text_extents (cr, "A8", &extents);
|
|
|
|
cairo_destroy (cr);
|
|
|
|
cairo_surface_destroy (cs);
|
|
|
|
bd = std::max (bd, 2 + (int)ceil(extents.height));
|
|
|
|
bd = std::max (bd, 2 + (int)ceil(extents.width));
|
|
|
|
}
|
|
|
|
|
2019-08-23 10:41:23 -04:00
|
|
|
sq = ceil (sqrt (ArdourIcon::NoIcon + 3));
|
|
|
|
|
2024-04-24 20:32:01 -04:00
|
|
|
int w = sq;
|
|
|
|
int h = ceil ((ArdourIcon::NoIcon + 3.0) / sq);
|
|
|
|
|
|
|
|
cairo_surface_t* cs = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, 2 * bd + wh * w, 2 * bd + wh * h);
|
2019-08-23 10:41:23 -04:00
|
|
|
cairo_t* cr = cairo_create (cs);
|
|
|
|
|
|
|
|
Gtkmm2ext::set_source_rgba (cr, bg_color);
|
|
|
|
cairo_paint (cr);
|
|
|
|
|
|
|
|
int pos = 0;
|
|
|
|
|
|
|
|
draw_icon (cr, pos++, ArdourIcon::RecButton, Gtkmm2ext::Off);
|
|
|
|
draw_icon (cr, pos++, ArdourIcon::RecButton, Gtkmm2ext::ImplicitActive);
|
|
|
|
|
|
|
|
for (int i = 0; i < ArdourIcon::NoIcon; ++i) {
|
|
|
|
draw_icon (cr, pos++, ArdourIcon::Icon (i), Gtkmm2ext::ExplicitActive);
|
|
|
|
}
|
|
|
|
|
2020-02-22 11:18:20 -05:00
|
|
|
if (bd > 0) {
|
|
|
|
Gtkmm2ext::set_source_rgba (cr, fg_color);
|
|
|
|
cairo_set_line_width (cr, 1);
|
2024-04-24 20:32:01 -04:00
|
|
|
cairo_rectangle (cr, bd - .5, bd - .5, 1 + wh * w, 1 + wh * h);
|
2020-02-22 11:18:20 -05:00
|
|
|
cairo_stroke (cr);
|
|
|
|
cairo_set_font_size (cr, fs);
|
|
|
|
for (int rc = 0; rc < sq; ++rc) {
|
|
|
|
char tmp[8];
|
|
|
|
snprintf (tmp, sizeof(tmp), "%c", rc + 'A');
|
|
|
|
cairo_text_extents_t extents;
|
|
|
|
cairo_text_extents (cr, tmp, &extents);
|
|
|
|
cairo_move_to (cr,
|
|
|
|
bd + wh * rc + rint (.5 * (wh - extents.width)),
|
|
|
|
rint (.5 * (bd + extents.height))
|
|
|
|
);
|
|
|
|
cairo_show_text (cr, tmp);
|
|
|
|
|
2024-04-24 20:32:01 -04:00
|
|
|
if (rc >= h) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2020-02-22 11:18:20 -05:00
|
|
|
snprintf (tmp, sizeof(tmp), "%d", rc + 1);
|
|
|
|
cairo_text_extents (cr, tmp, &extents);
|
|
|
|
cairo_move_to (cr,
|
|
|
|
rint (.5 * (bd - extents.width)),
|
|
|
|
bd + wh * rc + rint (.5 * (wh + extents.height))
|
|
|
|
);
|
|
|
|
cairo_show_text (cr, tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-23 10:41:23 -04:00
|
|
|
if (CAIRO_STATUS_SUCCESS != cairo_surface_write_to_png (cs, fn)) {
|
|
|
|
std::cerr << "Error: Failed to write to '" << fn << "'.\n";
|
|
|
|
::exit (EXIT_FAILURE);
|
|
|
|
}
|
|
|
|
cairo_destroy (cr);
|
|
|
|
cairo_surface_destroy (cs);
|
|
|
|
return 0;
|
|
|
|
}
|