hb-ot-color.cc 8.1 KB
Newer Older
S
Sascha Brawer 已提交
1 2
/*
 * Copyright © 2016  Google, Inc.
3
 * Copyright © 2018  Ebrahim Byagowi
S
Sascha Brawer 已提交
4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27
 *
 *  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): Sascha Brawer
 */

28
#include "hb-open-type.hh"
29
#include "hb-ot-color-cbdt-table.hh"
30 31
#include "hb-ot-color-colr-table.hh"
#include "hb-ot-color-cpal-table.hh"
32 33
#include "hb-ot-color-sbix-table.hh"
#include "hb-ot-color-svg-table.hh"
K
Khaled Hosny 已提交
34
#include "hb-ot-face.hh"
S
Sascha Brawer 已提交
35 36 37 38 39
#include "hb-ot.h"

#include <stdlib.h>
#include <string.h>

40
#include "hb-ot-layout.hh"
S
Sascha Brawer 已提交
41 42


E
Ebrahim Byagowi 已提交
43 44 45
static inline const OT::COLR&
_get_colr (hb_face_t *face)
{
B
Behdad Esfahbod 已提交
46
  if (unlikely (!hb_ot_shaper_face_data_ensure (face))) return Null(OT::COLR);
K
Khaled Hosny 已提交
47
  return *(hb_ot_face_data (face)->COLR.get ());
E
Ebrahim Byagowi 已提交
48 49
}

S
Sascha Brawer 已提交
50 51 52
static inline const OT::CPAL&
_get_cpal (hb_face_t *face)
{
B
Behdad Esfahbod 已提交
53
  if (unlikely (!hb_ot_shaper_face_data_ensure (face))) return Null(OT::CPAL);
K
Khaled Hosny 已提交
54
  return *(hb_ot_face_data (face)->CPAL.get ());
S
Sascha Brawer 已提交
55 56
}

57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79
#if 0
static inline const OT::CBDT_accelerator_t&
_get_cbdt (hb_face_t *face)
{
  if (unlikely (!hb_ot_shaper_face_data_ensure (face))) return Null(OT::CBDT_accelerator_t);
  return *(hb_ot_face_data (face)->CBDT.get ());
}

static inline const OT::sbix&
_get_sbix (hb_face_t *face)
{
  if (unlikely (!hb_ot_shaper_face_data_ensure (face))) return Null(OT::sbix);
  return *(hb_ot_face_data (face)->sbix.get ());
}

static inline const OT::SVG&
_get_svg (hb_face_t *face)
{
  if (unlikely (!hb_ot_shaper_face_data_ensure (face))) return Null(OT::SVG);
  return *(hb_ot_face_data (face)->SVG.get ());
}
#endif

B
Behdad Esfahbod 已提交
80 81
/*
 * CPAL
S
Sascha Brawer 已提交
82
 */
B
Behdad Esfahbod 已提交
83

S
Sascha Brawer 已提交
84 85

/**
B
Behdad Esfahbod 已提交
86
 * hb_ot_color_has_palettes:
S
Sascha Brawer 已提交
87 88
 * @face: a font face.
 *
B
Behdad Esfahbod 已提交
89
 * Returns: whether CPAL table is available.
S
Sascha Brawer 已提交
90
 *
91
 * Since: REPLACEME
S
Sascha Brawer 已提交
92
 */
93
hb_bool_t
B
Behdad Esfahbod 已提交
94
hb_ot_color_has_palettes (hb_face_t *face)
S
Sascha Brawer 已提交
95
{
B
Behdad Esfahbod 已提交
96
  return _get_cpal (face).has_data ();
S
Sascha Brawer 已提交
97 98 99
}

/**
100 101
 * hb_ot_color_get_palette_count:
 * @face: a font face.
S
Sascha Brawer 已提交
102
 *
K
Khaled Hosny 已提交
103 104
 * Returns: the number of color palettes in @face, or zero if @face has
 * no colors.
S
Sascha Brawer 已提交
105
 *
106
 * Since: REPLACEME
S
Sascha Brawer 已提交
107
 */
108 109
unsigned int
hb_ot_color_get_palette_count (hb_face_t *face)
S
Sascha Brawer 已提交
110
{
111
  return _get_cpal (face).get_palette_count ();
S
Sascha Brawer 已提交
112 113
}

114 115
/**
 * hb_ot_color_get_palette_name_id:
E
Minor  
Ebrahim Byagowi 已提交
116
 * @face:    a font face.
117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140
 * @palette: the index of the color palette whose name is being requested.
 *
 * Retrieves the name id of a color palette. For example, a color font can
 * have themed palettes like "Spring", "Summer", "Fall", and "Winter".
 *
 * Returns: an identifier within @face's `name` table.
 * If the requested palette has no name, or if @face has no colors,
 * or if @palette is not between 0 and hb_ot_color_get_palette_count(),
 * the result is 0xFFFF. The implementation does not check whether
 * the returned palette name id is actually in @face's `name` table.
 *
 * Since: REPLACEME
 */
hb_name_id_t
hb_ot_color_get_palette_name_id (hb_face_t *face, unsigned int palette)
{
  return _get_cpal (face).get_palette_name_id (palette);
}

/**
 * hb_ot_color_get_palette_entry_name_id:
 * @face: a font face.
 * @palette_entry:
 *
E
Minor  
Ebrahim Byagowi 已提交
141
 * Returns: Name ID associated with a palette entry, e.g. eye color
142 143 144 145 146 147 148 149 150
 *
 * Since: REPLACEME
 */
hb_name_id_t
hb_ot_color_get_palette_entry_name_id (hb_face_t *face, unsigned int palette_entry)
{
  return _get_cpal (face).get_palette_entry_name_id (palette_entry);
}

S
Sascha Brawer 已提交
151 152 153 154 155 156 157 158 159
/**
 * hb_ot_color_get_palette_colors:
 * @face:         a font face.
 * @palette:      the index of the color palette whose colors
 *                are being requested.
 * @start_offset: the index of the first color being requested.
 * @color_count:  (inout) (optional): on input, how many colors
 *                can be maximally stored into the @colors array;
 *                on output, how many colors were actually stored.
K
Khaled Hosny 已提交
160
 * @colors: (array length=color_count) (out) (optional):
161
 *                an array of #hb_color_t records. After calling
S
Sascha Brawer 已提交
162 163 164 165 166 167 168 169 170 171 172 173 174 175
 *                this function, @colors will be filled with
 *                the palette colors. If @colors is NULL, the function
 *                will just return the number of total colors
 *                without storing any actual colors; this can be used
 *                for allocating a buffer of suitable size before calling
 *                hb_ot_color_get_palette_colors() a second time.
 *
 * Retrieves the colors in a color palette.
 *
 * Returns: the total number of colors in the palette. All palettes in
 * a font have the same number of colors. If @face has no colors, or if
 * @palette is not between 0 and hb_ot_color_get_palette_count(),
 * the result is zero.
 *
176
 * Since: REPLACEME
S
Sascha Brawer 已提交
177 178
 */
unsigned int
K
Khaled Hosny 已提交
179
hb_ot_color_get_palette_colors (hb_face_t      *face,
180
				unsigned int    palette,      /* default=0 */
K
Khaled Hosny 已提交
181
				unsigned int    start_offset,
E
Minor  
Ebrahim Byagowi 已提交
182
				unsigned int   *colors_count  /* IN/OUT.  May be NULL. */,
183
				hb_color_t     *colors        /* OUT.     May be NULL. */)
S
Sascha Brawer 已提交
184 185
{
  const OT::CPAL& cpal = _get_cpal(face);
K
Khaled Hosny 已提交
186
  if (unlikely (palette >= cpal.get_palette_count ()))
S
Sascha Brawer 已提交
187
  {
E
Minor  
Ebrahim Byagowi 已提交
188
    if (colors_count) *colors_count = 0;
S
Sascha Brawer 已提交
189 190 191
    return 0;
  }

K
Khaled Hosny 已提交
192
  unsigned int num_results = 0;
E
Minor  
Ebrahim Byagowi 已提交
193
  if (colors_count)
K
Khaled Hosny 已提交
194
  {
E
Minor  
Ebrahim Byagowi 已提交
195 196 197
    unsigned int platte_count;
    platte_count = MIN<unsigned int>(*colors_count,
				     cpal.get_palette_entries_count () - start_offset);
K
Khaled Hosny 已提交
198 199 200
    for (unsigned int i = 0; i < platte_count; i++)
    {
      if (cpal.get_color_record_argb(start_offset + i, palette, &colors[num_results]))
E
Minor  
Ebrahim Byagowi 已提交
201
	++num_results;
K
Khaled Hosny 已提交
202 203
    }
  }
S
Sascha Brawer 已提交
204

E
Minor  
Ebrahim Byagowi 已提交
205
  if (likely (colors_count)) *colors_count = num_results;
K
Khaled Hosny 已提交
206 207 208
  return cpal.get_palette_entries_count ();
}

B
Behdad Esfahbod 已提交
209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245
/**
 * hb_ot_color_get_palette_flags:
 * @face:    a font face
 * @palette: the index of the color palette whose flags are being requested
 *
 * Returns: the flags for the requested color palette.  If @face has no colors,
 * or if @palette is not between 0 and hb_ot_color_get_palette_count(),
 * the result is #HB_OT_COLOR_PALETTE_FLAG_DEFAULT.
 *
 * Since: REPLACEME
 */
hb_ot_color_palette_flags_t
hb_ot_color_get_palette_flags (hb_face_t *face, unsigned int palette)
{
  const OT::CPAL& cpal = _get_cpal(face);
  return cpal.get_palette_flags (palette);
}


/*
 * COLR
 */

/**
 * hb_ot_color_has_layers:
 * @face: a font face.
 *
 * Returns: whether COLR table is available.
 *
 * Since: REPLACEME
 */
hb_bool_t
hb_ot_color_has_layers (hb_face_t *face)
{
  return _get_colr (face).has_data ();
}

246
/**
B
Behdad Esfahbod 已提交
247
 * hb_ot_color_glyph_get_layers:
E
Minor  
Ebrahim Byagowi 已提交
248
 * @face: a font face.
B
Behdad Esfahbod 已提交
249
 * @glyph:
250 251
 * @start_offset:
 * @count:  (inout) (optional):
B
Behdad Esfahbod 已提交
252
 * @layers: (array length=count) (out) (optional):
253 254 255 256 257
 *
 * Returns:
 *
 * Since: REPLACEME
 */
K
Khaled Hosny 已提交
258
unsigned int
B
Behdad Esfahbod 已提交
259 260 261 262 263
hb_ot_color_glyph_get_layers (hb_face_t           *face,
			      hb_codepoint_t       glyph,
			      unsigned int         start_offset,
			      unsigned int        *count, /* IN/OUT.  May be NULL. */
			      hb_ot_color_layer_t *layers /* OUT.     May be NULL. */)
K
Khaled Hosny 已提交
264 265
{
  const OT::COLR& colr = _get_colr (face);
S
Sascha Brawer 已提交
266
  unsigned int num_results = 0;
K
Khaled Hosny 已提交
267
  unsigned int start_layer_index, num_layers = 0;
B
Behdad Esfahbod 已提交
268
  if (colr.get_base_glyph_record (glyph, &start_layer_index, &num_layers))
S
Sascha Brawer 已提交
269
  {
K
Khaled Hosny 已提交
270
    if (count)
S
Sascha Brawer 已提交
271
    {
K
Khaled Hosny 已提交
272 273 274 275
      unsigned int layer_count = MIN<unsigned int>(*count, num_layers - start_offset);
      for (unsigned int i = 0; i < layer_count; i++)
      {
	if (colr.get_layer_record (start_layer_index + start_offset + i,
B
Behdad Esfahbod 已提交
276
				   &layers[num_results].glyph, &layers[num_results].color_index))
K
Khaled Hosny 已提交
277 278
	  ++num_results;
      }
S
Sascha Brawer 已提交
279 280 281
    }
  }

K
Khaled Hosny 已提交
282 283
  if (likely (count)) *count = num_results;
  return num_layers;
S
Sascha Brawer 已提交
284
}