skia2/tools/fonts/create_test_font.cpp

427 lines
15 KiB
C++
Raw Normal View History

/*
* Copyright 2014 Google Inc.
*
* Use of this source code is governed by a BSD-style license that can be
* found in the LICENSE file.
*/
// Running create_test_font generates ./tools/fonts/test_font_index.inc
// and ./tools/fonts/test_font_<generic name>.inc which are read by
// ./tools/fonts/SkTestFontMgr.cpp
#include "SkFontStyle.h"
#include "SkOSFile.h"
#include "SkOSPath.h"
#include "SkPaint.h"
#include "SkPath.h"
#include "SkSpan.h"
#include "SkStream.h"
#include "SkTArray.h"
#include "SkTSort.h"
#include "SkTypeface.h"
#include "SkUTF.h"
#include <stdio.h>
namespace {
struct NamedFontStyle {
char const * const fName;
char const * const fIdentifierName;
SkFontStyle const fStyle;
};
struct FontDesc {
NamedFontStyle const fNamedStyle;
char const * const fFile;
};
struct FontFamilyDesc {
char const * const fGenericName;
char const * const fFamilyName;
char const * const fIdentifierName;
SkSpan<const FontDesc> const fFonts;
};
} // namespace
static FILE* font_header(const char* family) {
SkString outPath(SkOSPath::Join(".", "tools"));
outPath = SkOSPath::Join(outPath.c_str(), "fonts");
outPath = SkOSPath::Join(outPath.c_str(), "test_font_");
SkString fam(family);
do {
int dashIndex = fam.find("-");
if (dashIndex < 0) {
break;
}
fam.writable_str()[dashIndex] = '_';
} while (true);
outPath.append(fam);
outPath.append(".inc");
FILE* out = fopen(outPath.c_str(), "w");
static const char kHeader[] =
"/*\n"
" * Copyright 2015 Google Inc.\n"
" *\n"
" * Use of this source code is governed by a BSD-style license that can be\n"
" * found in the LICENSE file.\n"
" */\n"
"\n"
"// Auto-generated by ";
fprintf(out, "%s%s\n\n", kHeader, SkOSPath::Basename(__FILE__).c_str());
return out;
}
enum {
kMaxLineLength = 80,
};
static ptrdiff_t last_line_length(const SkString& str) {
const char* first = str.c_str();
const char* last = first + str.size();
const char* ptr = last;
while (ptr > first && *--ptr != '\n')
;
return last - ptr - 1;
}
static void output_fixed(SkScalar num, int emSize, SkString* out) {
int hex = (int) (num * 65536 / emSize);
out->appendf("0x%08x,", hex);
*out += (int) last_line_length(*out) >= kMaxLineLength ? '\n' : ' ';
}
static void output_scalar(SkScalar num, int emSize, SkString* out) {
num /= emSize;
if (num == (int) num) {
out->appendS32((int) num);
} else {
SkString str;
str.printf("%1.6g", num);
int width = (int) str.size();
const char* cStr = str.c_str();
while (cStr[width - 1] == '0') {
--width;
}
str.remove(width, str.size() - width);
out->appendf("%sf", str.c_str());
}
*out += ',';
*out += (int) last_line_length(*out) >= kMaxLineLength ? '\n' : ' ';
}
static int output_points(const SkPoint* pts, int emSize, int count, SkString* ptsOut) {
for (int index = 0; index < count; ++index) {
output_scalar(pts[index].fX, emSize, ptsOut);
output_scalar(pts[index].fY, emSize, ptsOut);
}
return count;
}
static void output_path_data(const SkPaint& paint,
int emSize, SkString* ptsOut, SkTDArray<SkPath::Verb>* verbs,
SkTDArray<unsigned>* charCodes, SkTDArray<SkScalar>* widths) {
for (SkUnichar index = 0x00; index < 0x7f; ++index) {
uint16_t utf16[2];
size_t utf16Bytes = sizeof(uint16_t) * SkUTF::ToUTF16(index, utf16);
SkPath path;
SkASSERT(paint.getTextEncoding() == SkPaint::kUTF16_TextEncoding);
paint.getTextPath(utf16, utf16Bytes, 0, 0, &path);
SkPath::RawIter iter(path);
SkPath::Verb verb;
SkPoint pts[4];
while ((verb = iter.next(pts)) != SkPath::kDone_Verb) {
*verbs->append() = verb;
switch (verb) {
case SkPath::kMove_Verb:
output_points(&pts[0], emSize, 1, ptsOut);
break;
case SkPath::kLine_Verb:
output_points(&pts[1], emSize, 1, ptsOut);
break;
case SkPath::kQuad_Verb:
output_points(&pts[1], emSize, 2, ptsOut);
break;
case SkPath::kCubic_Verb:
output_points(&pts[1], emSize, 3, ptsOut);
break;
case SkPath::kClose_Verb:
break;
default:
SkDEBUGFAIL("bad verb");
SkASSERT(0);
}
}
*verbs->append() = SkPath::kDone_Verb;
*charCodes->append() = index;
SkScalar width;
SkDEBUGCODE(int charCount =) paint.getTextWidths(utf16, utf16Bytes, &width);
SkASSERT(charCount == 1);
// SkASSERT(floor(width) == width); // not true for Hiragino Maru Gothic Pro
*widths->append() = width;
if (0 == index) {
index = 0x1f; // skip the rest of the control codes
}
}
}
static int offset_str_len(unsigned num) {
if (num == (unsigned) -1) {
return 10;
}
unsigned result = 1;
unsigned ref = 10;
while (ref <= num) {
++result;
ref *= 10;
}
return result;
}
static SkString strip_final(const SkString& str) {
SkString result(str);
if (result.endsWith("\n")) {
result.remove(result.size() - 1, 1);
}
if (result.endsWith(" ")) {
result.remove(result.size() - 1, 1);
}
if (result.endsWith(",")) {
result.remove(result.size() - 1, 1);
}
return result;
}
static void output_font(sk_sp<SkTypeface> face, const char* identifier, FILE* out) {
int emSize = face->getUnitsPerEm() * 2;
SkPaint paint;
paint.setAntiAlias(true);
paint.setTextAlign(SkPaint::kLeft_Align);
paint.setTextEncoding(SkPaint::kUTF16_TextEncoding);
paint.setTextSize(emSize);
paint.setTypeface(std::move(face));
SkTDArray<SkPath::Verb> verbs;
SkTDArray<unsigned> charCodes;
SkTDArray<SkScalar> widths;
SkString ptsOut;
output_path_data(paint, emSize, &ptsOut, &verbs, &charCodes, &widths);
fprintf(out, "const SkScalar %sPoints[] = {\n", identifier);
ptsOut = strip_final(ptsOut);
fprintf(out, "%s", ptsOut.c_str());
fprintf(out, "\n};\n\n");
fprintf(out, "const unsigned char %sVerbs[] = {\n", identifier);
int verbCount = verbs.count();
int outChCount = 0;
for (int index = 0; index < verbCount;) {
SkPath::Verb verb = verbs[index];
SkASSERT(verb >= SkPath::kMove_Verb && verb <= SkPath::kDone_Verb);
SkASSERT(SkTFitsIn<uint8_t>(verb));
fprintf(out, "%u", verb);
if (++index < verbCount) {
outChCount += 3;
fprintf(out, "%c", ',');
if (outChCount >= kMaxLineLength) {
outChCount = 0;
fprintf(out, "%c", '\n');
} else {
fprintf(out, "%c", ' ');
}
}
}
fprintf(out, "\n};\n\n");
// all fonts are now 0x00, 0x20 - 0xFE
// don't need to generate or output character codes?
fprintf(out, "const SkUnichar %sCharCodes[] = {\n", identifier);
int offsetCount = charCodes.count();
for (int index = 0; index < offsetCount;) {
unsigned offset = charCodes[index];
fprintf(out, "%u", offset);
if (++index < offsetCount) {
outChCount += offset_str_len(offset) + 2;
fprintf(out, "%c", ',');
if (outChCount >= kMaxLineLength) {
outChCount = 0;
fprintf(out, "%c", '\n');
} else {
fprintf(out, "%c", ' ');
}
}
}
fprintf(out, "\n};\n\n");
SkString widthsStr;
fprintf(out, "const SkFixed %sWidths[] = {\n", identifier);
for (int index = 0; index < offsetCount; ++index) {
output_fixed(widths[index], emSize, &widthsStr);
}
widthsStr = strip_final(widthsStr);
fprintf(out, "%s\n};\n\n", widthsStr.c_str());
fprintf(out, "const size_t %sCharCodesCount = SK_ARRAY_COUNT(%sCharCodes);\n\n",
identifier, identifier);
SkPaint::FontMetrics metrics;
paint.getFontMetrics(&metrics);
fprintf(out, "const SkPaint::FontMetrics %sMetrics = {\n", identifier);
SkString metricsStr;
metricsStr.printf("0x%08x, ", metrics.fFlags);
output_scalar(metrics.fTop, emSize, &metricsStr);
output_scalar(metrics.fAscent, emSize, &metricsStr);
output_scalar(metrics.fDescent, emSize, &metricsStr);
output_scalar(metrics.fBottom, emSize, &metricsStr);
output_scalar(metrics.fLeading, emSize, &metricsStr);
output_scalar(metrics.fAvgCharWidth, emSize, &metricsStr);
output_scalar(metrics.fMaxCharWidth, emSize, &metricsStr);
output_scalar(metrics.fXMin, emSize, &metricsStr);
output_scalar(metrics.fXMax, emSize, &metricsStr);
output_scalar(metrics.fXHeight, emSize, &metricsStr);
output_scalar(metrics.fCapHeight, emSize, &metricsStr);
output_scalar(metrics.fUnderlineThickness, emSize, &metricsStr);
output_scalar(metrics.fUnderlinePosition, emSize, &metricsStr);
output_scalar(metrics.fStrikeoutThickness, emSize, &metricsStr);
output_scalar(metrics.fStrikeoutPosition, emSize, &metricsStr);
metricsStr = strip_final(metricsStr);
fprintf(out, "%s\n};\n\n", metricsStr.c_str());
}
static SkString identifier(const FontFamilyDesc& family, const FontDesc& font) {
SkString id(family.fIdentifierName);
id.append(font.fNamedStyle.fIdentifierName);
return id;
}
static void generate_fonts(const char* basepath, const SkSpan<const FontFamilyDesc>& families) {
FILE* out = nullptr;
for (const FontFamilyDesc& family : families) {
out = font_header(family.fGenericName);
for (const FontDesc& font : family.fFonts) {
SkString filepath(SkOSPath::Join(basepath, font.fFile));
SkASSERTF(sk_exists(filepath.c_str()), "The file %s does not exist.", filepath.c_str());
sk_sp<SkTypeface> resourceTypeface = SkTypeface::MakeFromFile(filepath.c_str());
SkASSERTF(resourceTypeface, "The file %s is not a font.", filepath.c_str());
output_font(std::move(resourceTypeface), identifier(family, font).c_str(), out);
}
fclose(out);
}
}
static const char* slant_to_string(SkFontStyle::Slant slant) {
switch (slant) {
case SkFontStyle::kUpright_Slant: return "SkFontStyle::kUpright_Slant";
case SkFontStyle::kItalic_Slant : return "SkFontStyle::kItalic_Slant" ;
case SkFontStyle::kOblique_Slant: return "SkFontStyle::kOblique_Slant";
default: SK_ABORT("Unknown slant"); return "";
}
}
static void generate_index(const SkSpan<const FontFamilyDesc>& families,
const FontDesc* defaultFont)
{
FILE* out = font_header("index");
fprintf(out, "static SkTestFontData gTestFonts[] = {\n");
for (const FontFamilyDesc& family : families) {
for (const FontDesc& font : family.fFonts) {
SkString identifierStr = identifier(family, font);
const char* identifier = identifierStr.c_str();
const SkFontStyle& style = font.fNamedStyle.fStyle;
fprintf(out,
" { %sPoints, %sVerbs,\n"
" %sCharCodes, %sCharCodesCount, %sWidths,\n"
" %sMetrics, \"Toy %s\", SkFontStyle(%d,%d,%s)\n"
" },\n",
identifier, identifier,
identifier, identifier, identifier,
identifier, family.fFamilyName,
style.weight(), style.width(), slant_to_string(style.slant()));
}
}
fprintf(out, "};\n\n");
fprintf(out,
"struct SubFont {\n"
" const char* fFamilyName;\n"
" const char* fStyleName;\n"
" SkFontStyle fStyle;\n"
" SkTestFontData& fFont;\n"
" const char* fFile;\n"
"};\n\n"
"const SubFont gSubFonts[] = {\n");
int defaultIndex = -1;
int testFontsIndex = 0;
for (const FontFamilyDesc& family : families) {
for (const FontDesc& font : family.fFonts) {
if (&font == defaultFont) {
defaultIndex = testFontsIndex;
}
const SkFontStyle& style = font.fNamedStyle.fStyle;
fprintf(out,
" { \"%s\", \"%s\", SkFontStyle(%d,%d,%s), gTestFonts[%d], \"%s\" },\n",
family.fGenericName, font.fNamedStyle.fName,
style.weight(), style.width(), slant_to_string(style.slant()),
testFontsIndex, font.fFile);
testFontsIndex++;
}
}
testFontsIndex = 0;
for (const FontFamilyDesc& family : families) {
for (const FontDesc& font : family.fFonts) {
fprintf(out,
" { \"Toy %s\", \"%s\", SkFontStyle(%d,%d,%s), gTestFonts[%d], \"%s\" },\n",
family.fFamilyName, font.fNamedStyle.fName,
font.fNamedStyle.fStyle.weight(), font.fNamedStyle.fStyle.width(),
slant_to_string(font.fNamedStyle.fStyle.slant()), testFontsIndex, font.fFile);
testFontsIndex++;
}
}
fprintf(out, "};\n\n");
SkASSERT(defaultIndex >= 0);
fprintf(out, "const size_t gDefaultFontIndex = %d;\n", defaultIndex);
fclose(out);
}
int main(int , char * const []) {
constexpr NamedFontStyle normal = {"Normal", "Normal", SkFontStyle::Normal() };
constexpr NamedFontStyle bold = {"Bold", "Bold", SkFontStyle::Bold() };
constexpr NamedFontStyle italic = {"Italic", "Italic", SkFontStyle::Italic() };
constexpr NamedFontStyle bolditalic = {"Bold Italic", "BoldItalic", SkFontStyle::BoldItalic()};
static constexpr FontDesc kMonoFonts[] = {
{normal, "LiberationMono-Regular.ttf"},
{bold, "LiberationMono-Bold.ttf"},
{italic, "LiberationMono-Italic.ttf"},
{bolditalic, "LiberationMono-BoldItalic.ttf"},
};
static constexpr FontDesc kSansFonts[] = {
{normal, "LiberationSans-Regular.ttf"},
{bold, "LiberationSans-Bold.ttf"},
{italic, "LiberationSans-Italic.ttf"},
{bolditalic, "LiberationSans-BoldItalic.ttf"},
};
static constexpr FontDesc kSerifFonts[] = {
{normal, "LiberationSerif-Regular.ttf"},
{bold, "LiberationSerif-Bold.ttf"},
{italic, "LiberationSerif-Italic.ttf"},
{bolditalic, "LiberationSerif-BoldItalic.ttf"},
};
static constexpr FontFamilyDesc kFamiliesData[] = {
{"monospace", "Liberation Mono", "LiberationMono", { kMonoFonts }},
{"sans-serif", "Liberation Sans", "LiberationSans", { kSansFonts }},
{"serif", "Liberation Serif", "LiberationSerif", { kSerifFonts }},
};
static constexpr SkSpan<const FontFamilyDesc> kFamilies(kFamiliesData);
#ifdef SK_BUILD_FOR_UNIX
generate_fonts("/usr/share/fonts/truetype/liberation/", kFamilies);
#else
generate_fonts("/Library/Fonts/", kFamilies);
#endif
generate_index(kFamilies, &kFamilies[1].fFonts[0]);
return 0;
}