HarfBuzz text shaping engine
http://harfbuzz.github.io/
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
173 lines
4.5 KiB
173 lines
4.5 KiB
/* |
|
* Copyright © 2010 Behdad Esfahbod |
|
* Copyright © 2011,2012 Google, Inc. |
|
* |
|
* This is part of HarfBuzz, a text shaping library. |
|
* |
|
* Permission is hereby granted, without written agreement and without |
|
* license or royalty fees, to use, copy, modify, and distribute this |
|
* software and its documentation for any purpose, provided that the |
|
* above copyright notice and the following two paragraphs appear in |
|
* all copies of this software. |
|
* |
|
* IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR |
|
* DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES |
|
* ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN |
|
* IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH |
|
* DAMAGE. |
|
* |
|
* THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, |
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND |
|
* FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS |
|
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO |
|
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. |
|
* |
|
* Google Author(s): Garret Rieger, Rod Sheeter |
|
*/ |
|
|
|
#include <stdio.h> |
|
|
|
#include "main-font-text.hh" |
|
#include "hb-subset.h" |
|
|
|
// XXX Remove eventually |
|
const unsigned DEFAULT_FONT_SIZE = FONT_SIZE_NONE; |
|
const unsigned SUBPIXEL_BITS = 0; |
|
|
|
/* |
|
* Command line interface to the harfbuzz font subsetter. |
|
*/ |
|
|
|
struct subset_consumer_t |
|
{ |
|
void add_options (option_parser_t *parser) |
|
{ |
|
output_options.add_options (parser); |
|
subset_options.add_options (parser); |
|
} |
|
|
|
void init (const face_options_t *face_opts) |
|
{ |
|
face = hb_face_reference (face_opts->get_face ()); |
|
input = hb_subset_input_reference (subset_options.get_input ()); |
|
} |
|
|
|
void consume_line (const char *text, |
|
unsigned int text_len, |
|
const char *text_before, |
|
const char *text_after) |
|
{ |
|
// TODO does this only get called with at least 1 codepoint? |
|
hb_set_t *codepoints = hb_subset_input_unicode_set (input); |
|
if (0 == strcmp (text, "*")) |
|
{ |
|
hb_face_collect_unicodes (face, codepoints); |
|
return; |
|
} |
|
|
|
gchar *c = (gchar *)text; |
|
do { |
|
gunichar cp = g_utf8_get_char(c); |
|
hb_codepoint_t hb_cp = cp; |
|
hb_set_add (codepoints, hb_cp); |
|
} while ((c = g_utf8_find_next_char(c, text + text_len))); |
|
} |
|
|
|
hb_bool_t |
|
write_file (const char *output_file, hb_blob_t *blob) { |
|
unsigned int size; |
|
const char* data = hb_blob_get_data (blob, &size); |
|
|
|
if (!output_file) |
|
fail (true, "No output file was specified"); |
|
|
|
FILE *fp = fopen(output_file, "wb"); |
|
if (!fp) |
|
fail (false, "Cannot open output file `%s': %s", |
|
g_filename_display_name (output_file), strerror (errno)); |
|
|
|
while (size) { |
|
size_t ret = fwrite (data, 1, size, fp); |
|
size -= ret; |
|
data += ret; |
|
if (size && ferror (fp)) |
|
fail (false, "Failed to write output: %s", strerror (errno)); |
|
} |
|
|
|
fclose (fp); |
|
|
|
return true; |
|
} |
|
|
|
void finish (const face_options_t *face_opts) |
|
{ |
|
hb_face_t *new_face = nullptr; |
|
for (unsigned i = 0; i < subset_options.num_iterations; i++) |
|
{ |
|
hb_face_destroy (new_face); |
|
new_face = hb_subset_or_fail (face, input); |
|
} |
|
|
|
failed = !new_face; |
|
if (!failed) |
|
{ |
|
hb_blob_t *result = hb_face_reference_blob (new_face); |
|
write_file (output_options.output_file, result); |
|
hb_blob_destroy (result); |
|
} |
|
|
|
hb_subset_input_destroy (input); |
|
hb_face_destroy (new_face); |
|
hb_face_destroy (face); |
|
} |
|
|
|
public: |
|
bool failed = false; |
|
|
|
private: |
|
output_options_t output_options; |
|
subset_options_t subset_options; |
|
|
|
hb_face_t *face = nullptr; |
|
hb_subset_input_t *input = nullptr; |
|
}; |
|
|
|
int |
|
main (int argc, char **argv) |
|
{ |
|
using driver_t = main_font_text_t<subset_consumer_t, face_options_t>; |
|
|
|
if (argc == 2 && !strcmp (argv[1], "--batch")) |
|
{ |
|
unsigned int ret = 0; |
|
char buf[4092]; |
|
while (fgets (buf, sizeof (buf), stdin)) |
|
{ |
|
size_t l = strlen (buf); |
|
if (l && buf[l - 1] == '\n') buf[l - 1] = '\0'; |
|
|
|
char *args[32]; |
|
argc = 0; |
|
char *p = buf, *e; |
|
args[argc++] = p; |
|
unsigned start_offset = 0; |
|
while ((e = strchr (p + start_offset, ';')) && argc < (int) ARRAY_LENGTH (args)) |
|
{ |
|
*e++ = '\0'; |
|
while (*e == ':') |
|
e++; |
|
args[argc++] = p = e; |
|
} |
|
|
|
driver_t driver; |
|
int result = driver.main (argc, args, EOF); |
|
fprintf (stdout, result == 0 ? "success\n" : "failure\n"); |
|
fflush (stdout); |
|
ret |= result; |
|
} |
|
return ret; |
|
} |
|
|
|
driver_t driver; |
|
return driver.main (argc, argv); |
|
}
|
|
|