2011-08-11 11:54:31 +02:00
|
|
|
/*
|
2011-09-19 22:41:17 +02:00
|
|
|
* Copyright © 2010 Behdad Esfahbod
|
2012-05-16 05:10:39 +02:00
|
|
|
* Copyright © 2011,2012 Google, Inc.
|
2011-08-11 11:54:31 +02:00
|
|
|
*
|
|
|
|
* 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): Behdad Esfahbod
|
|
|
|
*/
|
|
|
|
|
2012-05-16 05:10:39 +02:00
|
|
|
#include "main-font-text.hh"
|
|
|
|
#include "shape-consumer.hh"
|
2011-09-19 22:41:17 +02:00
|
|
|
|
2012-06-02 18:13:08 +02:00
|
|
|
struct output_buffer_t
|
2011-09-19 22:41:17 +02:00
|
|
|
{
|
|
|
|
output_buffer_t (option_parser_t *parser)
|
2014-03-19 23:38:02 +01:00
|
|
|
: options (parser, hb_buffer_serialize_list_formats ()),
|
2013-06-10 20:39:51 +02:00
|
|
|
format (parser),
|
2017-10-15 12:11:08 +02:00
|
|
|
gs (nullptr),
|
2013-06-10 20:39:51 +02:00
|
|
|
line_no (0),
|
2017-10-15 12:11:08 +02:00
|
|
|
font (nullptr),
|
2017-02-08 03:17:48 +01:00
|
|
|
output_format (HB_BUFFER_SERIALIZE_FORMAT_INVALID),
|
|
|
|
format_flags (HB_BUFFER_SERIALIZE_FLAG_DEFAULT) {}
|
2011-09-19 22:41:17 +02:00
|
|
|
|
2017-07-19 04:14:19 +02:00
|
|
|
void init (hb_buffer_t *buffer, const font_options_t *font_opts)
|
2012-05-16 05:10:39 +02:00
|
|
|
{
|
2012-06-02 18:13:08 +02:00
|
|
|
options.get_file_handle ();
|
2017-10-15 12:11:08 +02:00
|
|
|
gs = g_string_new (nullptr);
|
2012-05-16 05:10:39 +02:00
|
|
|
line_no = 0;
|
|
|
|
font = hb_font_reference (font_opts->get_font ());
|
2012-11-15 21:14:09 +01:00
|
|
|
|
|
|
|
if (!options.output_format)
|
|
|
|
output_format = HB_BUFFER_SERIALIZE_FORMAT_TEXT;
|
|
|
|
else
|
|
|
|
output_format = hb_buffer_serialize_format_from_string (options.output_format, -1);
|
2014-07-09 02:02:29 +02:00
|
|
|
/* An empty "output_format" parameter basically skips output generating.
|
|
|
|
* Useful for benchmarking. */
|
2014-07-09 23:00:48 +02:00
|
|
|
if ((!options.output_format || *options.output_format) &&
|
2014-07-09 02:02:29 +02:00
|
|
|
!hb_buffer_serialize_format_to_string (output_format))
|
2012-11-15 22:29:51 +01:00
|
|
|
{
|
2012-12-21 22:01:52 +01:00
|
|
|
if (options.explicit_output_format)
|
|
|
|
fail (false, "Unknown output format `%s'; supported formats are: %s",
|
2014-03-19 23:38:02 +01:00
|
|
|
options.output_format,
|
|
|
|
g_strjoinv ("/", const_cast<char**> (options.supported_formats)));
|
2012-12-21 22:01:52 +01:00
|
|
|
else
|
|
|
|
/* Just default to TEXT if not explicitly requested and the
|
|
|
|
* file extension is not recognized. */
|
|
|
|
output_format = HB_BUFFER_SERIALIZE_FORMAT_TEXT;
|
2012-11-15 22:29:51 +01:00
|
|
|
}
|
2012-11-15 21:14:09 +01:00
|
|
|
|
2013-08-27 02:39:00 +02:00
|
|
|
unsigned int flags = HB_BUFFER_SERIALIZE_FLAG_DEFAULT;
|
2012-11-15 21:14:09 +01:00
|
|
|
if (!format.show_glyph_names)
|
|
|
|
flags |= HB_BUFFER_SERIALIZE_FLAG_NO_GLYPH_NAMES;
|
|
|
|
if (!format.show_clusters)
|
|
|
|
flags |= HB_BUFFER_SERIALIZE_FLAG_NO_CLUSTERS;
|
|
|
|
if (!format.show_positions)
|
|
|
|
flags |= HB_BUFFER_SERIALIZE_FLAG_NO_POSITIONS;
|
2018-01-10 02:20:14 +01:00
|
|
|
if (!format.show_advances)
|
|
|
|
flags |= HB_BUFFER_SERIALIZE_FLAG_NO_ADVANCES;
|
2015-08-24 14:49:55 +02:00
|
|
|
if (format.show_extents)
|
|
|
|
flags |= HB_BUFFER_SERIALIZE_FLAG_GLYPH_EXTENTS;
|
2016-05-02 14:47:45 +02:00
|
|
|
if (format.show_flags)
|
|
|
|
flags |= HB_BUFFER_SERIALIZE_FLAG_GLYPH_FLAGS;
|
2012-11-15 21:14:09 +01:00
|
|
|
format_flags = (hb_buffer_serialize_flags_t) flags;
|
2017-07-19 04:14:19 +02:00
|
|
|
|
2017-07-20 02:20:55 +02:00
|
|
|
if (format.trace)
|
2017-10-15 12:11:08 +02:00
|
|
|
hb_buffer_set_message_func (buffer, message_func, this, nullptr);
|
2012-05-16 05:10:39 +02:00
|
|
|
}
|
2018-12-17 19:01:01 +01:00
|
|
|
void new_line () { line_no++; }
|
2012-06-02 18:13:08 +02:00
|
|
|
void consume_text (hb_buffer_t *buffer,
|
2011-09-19 22:41:17 +02:00
|
|
|
const char *text,
|
2012-04-17 00:08:20 +02:00
|
|
|
unsigned int text_len,
|
2012-05-16 05:10:39 +02:00
|
|
|
hb_bool_t utf8_clusters)
|
|
|
|
{
|
|
|
|
g_string_set_size (gs, 0);
|
2012-11-15 21:14:09 +01:00
|
|
|
format.serialize_buffer_of_text (buffer, line_no, text, text_len, font, gs);
|
2012-06-02 18:13:08 +02:00
|
|
|
fprintf (options.fp, "%s", gs->str);
|
|
|
|
}
|
2017-08-12 00:12:25 +02:00
|
|
|
void error (const char *message)
|
2012-06-02 18:13:08 +02:00
|
|
|
{
|
|
|
|
g_string_set_size (gs, 0);
|
2017-08-16 02:30:18 +02:00
|
|
|
format.serialize_message (line_no, "error", message, gs);
|
2012-06-02 18:13:08 +02:00
|
|
|
fprintf (options.fp, "%s", gs->str);
|
|
|
|
}
|
|
|
|
void consume_glyphs (hb_buffer_t *buffer,
|
|
|
|
const char *text,
|
|
|
|
unsigned int text_len,
|
|
|
|
hb_bool_t utf8_clusters)
|
|
|
|
{
|
|
|
|
g_string_set_size (gs, 0);
|
2012-11-15 21:14:09 +01:00
|
|
|
format.serialize_buffer_of_glyphs (buffer, line_no, text, text_len, font,
|
|
|
|
output_format, format_flags, gs);
|
2012-06-02 18:13:08 +02:00
|
|
|
fprintf (options.fp, "%s", gs->str);
|
2012-05-16 05:10:39 +02:00
|
|
|
}
|
2017-07-19 04:14:19 +02:00
|
|
|
void finish (hb_buffer_t *buffer, const font_options_t *font_opts)
|
2012-05-16 05:10:39 +02:00
|
|
|
{
|
2017-10-15 12:11:08 +02:00
|
|
|
hb_buffer_set_message_func (buffer, nullptr, nullptr, nullptr);
|
2012-05-16 05:10:39 +02:00
|
|
|
hb_font_destroy (font);
|
2012-06-06 02:35:40 +02:00
|
|
|
g_string_free (gs, true);
|
2017-10-15 12:11:08 +02:00
|
|
|
gs = nullptr;
|
|
|
|
font = nullptr;
|
2012-05-16 05:10:39 +02:00
|
|
|
}
|
2011-09-19 22:41:17 +02:00
|
|
|
|
2017-07-19 04:14:19 +02:00
|
|
|
static hb_bool_t
|
|
|
|
message_func (hb_buffer_t *buffer,
|
|
|
|
hb_font_t *font,
|
|
|
|
const char *message,
|
|
|
|
void *user_data)
|
|
|
|
{
|
|
|
|
output_buffer_t *that = (output_buffer_t *) user_data;
|
2017-08-16 02:30:18 +02:00
|
|
|
that->trace (buffer, font, message);
|
2017-07-19 04:14:19 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2017-08-16 02:30:18 +02:00
|
|
|
trace (hb_buffer_t *buffer,
|
|
|
|
hb_font_t *font,
|
|
|
|
const char *message)
|
2017-07-19 04:14:19 +02:00
|
|
|
{
|
|
|
|
g_string_set_size (gs, 0);
|
|
|
|
format.serialize_line_no (line_no, gs);
|
2017-08-16 02:30:18 +02:00
|
|
|
g_string_append_printf (gs, "trace: %s buffer: ", message);
|
2020-09-17 17:49:59 +02:00
|
|
|
format.serialize (buffer, font, output_format, format_flags, gs);
|
2017-07-19 04:14:19 +02:00
|
|
|
g_string_append_c (gs, '\n');
|
|
|
|
fprintf (options.fp, "%s", gs->str);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-19 22:41:17 +02:00
|
|
|
protected:
|
2012-06-02 18:13:08 +02:00
|
|
|
output_options_t options;
|
2012-05-16 05:10:39 +02:00
|
|
|
format_options_t format;
|
|
|
|
|
2011-09-19 22:41:17 +02:00
|
|
|
GString *gs;
|
2012-01-19 18:46:18 +01:00
|
|
|
unsigned int line_no;
|
2012-05-16 05:10:39 +02:00
|
|
|
hb_font_t *font;
|
2012-11-15 21:14:09 +01:00
|
|
|
hb_buffer_serialize_format_t output_format;
|
|
|
|
hb_buffer_serialize_flags_t format_flags;
|
2011-09-19 22:41:17 +02:00
|
|
|
};
|
2011-08-11 11:54:31 +02:00
|
|
|
|
2021-08-01 16:14:59 +02:00
|
|
|
template <int eol = '\n'>
|
|
|
|
using driver_t = main_font_text_t<shape_consumer_t<output_buffer_t>, FONT_SIZE_UPEM, 0, eol>;
|
|
|
|
|
2011-09-19 22:41:17 +02:00
|
|
|
int
|
|
|
|
main (int argc, char **argv)
|
|
|
|
{
|
2018-10-30 08:50:18 +01:00
|
|
|
if (argc == 2 && !strcmp (argv[1], "--batch"))
|
|
|
|
{
|
|
|
|
unsigned int ret = 0;
|
2018-10-30 09:35:58 +01:00
|
|
|
char buf[4092];
|
|
|
|
while (fgets (buf, sizeof (buf), stdin))
|
2018-10-30 08:50:18 +01:00
|
|
|
{
|
|
|
|
size_t l = strlen (buf);
|
|
|
|
if (l && buf[l - 1] == '\n') buf[l - 1] = '\0';
|
2021-08-01 16:14:59 +02:00
|
|
|
|
2018-10-30 08:50:18 +01:00
|
|
|
char *args[32];
|
|
|
|
argc = 0;
|
|
|
|
char *p = buf, *e;
|
|
|
|
args[argc++] = p;
|
2020-04-05 20:21:58 +02:00
|
|
|
unsigned start_offset = 0;
|
|
|
|
while ((e = strchr (p + start_offset, ':')) && argc < (int) ARRAY_LENGTH (args))
|
2018-10-30 08:50:18 +01:00
|
|
|
{
|
|
|
|
*e++ = '\0';
|
2020-04-05 20:21:58 +02:00
|
|
|
while (*e == ':')
|
2018-10-30 08:50:18 +01:00
|
|
|
e++;
|
|
|
|
args[argc++] = p = e;
|
2020-04-05 20:21:58 +02:00
|
|
|
/* Skip 2 first bytes on first argument if is Windows path, "C:\..." */
|
|
|
|
start_offset = argc == 2 && p[0] != '\0' && p[0] != ':' && p[1] == ':' && (p[2] == '\\' || p[2] == '/') ? 2 : 0;
|
2018-10-30 08:50:18 +01:00
|
|
|
}
|
2021-08-01 16:14:59 +02:00
|
|
|
|
|
|
|
driver_t<EOF> driver;
|
2018-10-30 08:50:18 +01:00
|
|
|
ret |= driver.main (argc, args);
|
2021-08-01 16:14:59 +02:00
|
|
|
|
2018-10-30 08:50:18 +01:00
|
|
|
fflush (stdout);
|
|
|
|
|
|
|
|
if (ret)
|
2019-08-24 15:27:14 +02:00
|
|
|
break;
|
2018-10-30 08:50:18 +01:00
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
2021-08-01 16:14:59 +02:00
|
|
|
|
|
|
|
driver_t<> driver;
|
2012-05-16 05:10:39 +02:00
|
|
|
return driver.main (argc, argv);
|
2011-08-11 11:54:31 +02:00
|
|
|
}
|